Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add plus operator idea from concat #192

Merged
merged 12 commits into from
Nov 11, 2023
5 changes: 1 addition & 4 deletions .idea/vcs.xml

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

1 change: 1 addition & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -29,6 +29,7 @@

- Add `Flow.zipWithNext()` operator, it is an alias to `Flow.pairwise()` operator.
- Add `Flow.zipWithNext(transform)` operator, it is an alias to `Flow.pairwise(transform)` operator.
- Add `Flow.plus` operator, it is an alias to `concatWith` operator.

## [0.7.2] - Oct 7, 2023

Expand Down
23 changes: 21 additions & 2 deletions README.md
hoangchungk53qx1 marked this conversation as resolved.
Show resolved Hide resolved
Original file line number Diff line number Diff line change
Expand Up @@ -112,7 +112,7 @@ dependencies {
## Table of contents

- Create
- [`concat`](#concat)
- [`concat`](#concat--plus)
- [`defer`](#defer)
- [`flowFromSuspend`](#flowfromsuspend)
- [`interval`](#interval)
Expand All @@ -121,6 +121,7 @@ dependencies {
- [`amb`](#race--amb)
- [`range`](#range)
- [`timer`](#timer)
- [`plus`](#concat--plus)

- Intermediate operators
- [`bufferCount`](#buffercount--chunked)
Expand Down Expand Up @@ -200,19 +201,27 @@ bufferCount: [8, 9]

----

#### concat
#### concat / plus

- Similar to [RxJS concat](https://rxjs.dev/api/index/function/concat)
- Similar
to [RxJava concat](http://reactivex.io/RxJava/3.x/javadoc/io/reactivex/rxjava3/core/Flowable.html#concat-java.lang.Iterable-)

Creates an output `Flow` which sequentially emits all values from the first given `Flow` and then moves on to the next.
Note, `plus` is an alias to `concat`.

```kotlin
concat(
flow1 = flowOf(1, 2, 3),
flow2 = flowOf(4, 5, 6)
).collect { println("concat: $it") }

println("---")

val flow1 = flowOf(1, 2, 3)
val flow2 = flowOf(4, 5, 6)

(flow1 + flow2).collect { println("plus: $it") }
```

Output:
Expand All @@ -224,6 +233,13 @@ concat: 3
concat: 4
concat: 5
concat: 6
---
plus: 1
plus: 2
plus: 3
plus: 4
plus: 5
plus: 6
```

----
Expand Down Expand Up @@ -1303,6 +1319,9 @@ withLatestFrom: (2, 3)
withLatestFrom: (3, 4)
withLatestFrom: (4, 6)
```
#### plus


hoc081098 marked this conversation as resolved.
Show resolved Hide resolved

... and more, please check out [Docs 0.x](https://hoc081098.github.io/FlowExt/docs/0.x)/[Docs
snapshot](https://hoc081098.github.io/FlowExt/docs/latest).
Expand Down
1 change: 1 addition & 0 deletions api/FlowExt.api
Original file line number Diff line number Diff line change
Expand Up @@ -33,6 +33,7 @@ public final class com/hoc081098/flowext/ConcatKt {
public static final fun concatWith (Lkotlinx/coroutines/flow/Flow;Lkotlinx/coroutines/flow/Flow;Lkotlinx/coroutines/flow/Flow;Lkotlinx/coroutines/flow/Flow;)Lkotlinx/coroutines/flow/Flow;
public static final fun concatWith (Lkotlinx/coroutines/flow/Flow;Lkotlinx/coroutines/flow/Flow;Lkotlinx/coroutines/flow/Flow;Lkotlinx/coroutines/flow/Flow;Lkotlinx/coroutines/flow/Flow;)Lkotlinx/coroutines/flow/Flow;
public static final fun concatWith (Lkotlinx/coroutines/flow/Flow;Lkotlinx/coroutines/flow/Flow;[Lkotlinx/coroutines/flow/Flow;)Lkotlinx/coroutines/flow/Flow;
public static final fun plus (Lkotlinx/coroutines/flow/Flow;Lkotlinx/coroutines/flow/Flow;)Lkotlinx/coroutines/flow/Flow;
public static final fun startWith (Lkotlinx/coroutines/flow/Flow;Ljava/lang/Iterable;)Lkotlinx/coroutines/flow/Flow;
public static final fun startWith (Lkotlinx/coroutines/flow/Flow;Ljava/lang/Object;)Lkotlinx/coroutines/flow/Flow;
public static final fun startWith (Lkotlinx/coroutines/flow/Flow;Ljava/lang/Object;Ljava/lang/Object;)Lkotlinx/coroutines/flow/Flow;
Expand Down
16 changes: 16 additions & 0 deletions src/commonMain/kotlin/com/hoc081098/flowext/concat.kt
Original file line number Diff line number Diff line change
Expand Up @@ -267,3 +267,19 @@ public fun <T> Flow<T>.startWith(others: Sequence<T>): Flow<T> = concat(others.a
* Returns a [Flow] that emits the items in a specified [Flow] before it begins to emit items emitted by the current [Flow].
*/
public fun <T> Flow<T>.startWith(other: Flow<T>): Flow<T> = concat(other, this)

/**
* Combines two [Flow]s of the same base type [T] into a single [Flow] by concatenating their elements.
* @param other The [Flow] to concatenate with the current [Flow].
* @return A new [Flow] containing the concatenated elements of both the current and the [other] [Flow]s.
*
* Example:
*
* ``` kotlin
* val flow1 = flowOf(1, 2, 3)
* val flow2 = flowOf(4, 5, 6)
* val plusFlow = flow1 + flow2 //1, 2, 3, 4, 5, 6
* ```
*/

hoc081098 marked this conversation as resolved.
Show resolved Hide resolved
public operator fun <T, R : T> Flow<T>.plus(other: Flow<R>): Flow<T> = concat(this, other)
57 changes: 57 additions & 0 deletions src/commonTest/kotlin/com/hoc081098/flowext/ConcatTest.kt
hoangchungk53qx1 marked this conversation as resolved.
Show resolved Hide resolved
Original file line number Diff line number Diff line change
Expand Up @@ -286,6 +286,63 @@ class ConcatTest : BaseTest() {
),
).test(null, expectation)
}

@Test
fun plusTwoFlow() = runTest {
val flow1 = flowOf(1, 2, 3)
val flow2 = flowOf(4, 5, 6)

(flow1 + flow2)
.test(
listOf(
Event.Value(1),
Event.Value(2),
Event.Value(3),
Event.Value(4),
Event.Value(5),
Event.Value(6),
Event.Complete,
),
)
}

@Test
fun plusTwoFlow2() = runTest {
(flowOf("a", 2, 3) + flowOf(4, 5, 6))
.test(
listOf(
Event.Value("a"),
Event.Value(2),
Event.Value(3),
Event.Value(4),
Event.Value(5),
Event.Value(6),
Event.Complete,
),
)
}

@Test
fun testPlus_firstFailureUpstream() = runTest {
val flow = flowOf(1, 2, 3)
val failureFlow = kotlinx.coroutines.flow.flow<Nothing> { throw TestException("Crash!") }
val expectation: suspend (List<Event<Int>>) -> Unit = { events ->
val message = assertIs<TestException>(events.single().errorOrThrow()).message
assertEquals("Crash!", message)
}

(failureFlow + flow)
.test(null, expectation)

(failureFlow + flow + flow)
.test(null, expectation)

(failureFlow + flow + flow + flow)
.test(null, expectation)

(failureFlow + flow + flow + flow + flow)
.test(null, expectation)
}
}

private operator fun <T> Iterable<T>.times(times: Int): List<T> = (0 until times).flatMap { this }