Many changes
Updated store code - wrapped fold function in Task Added JSON logging functionality to store middleware Initial attempt at creating filter combinator for fx observables Made ListStore code use Effects Made a router using the store pattern Misc updates to fx monix implicits
This commit is contained in:
parent
857fd03bf1
commit
935ca358e6
build.sbt
src/main/scala/nova/monadic_sfx
@ -23,16 +23,16 @@ libraryDependencies ++= Seq(
|
|||||||
"com.softwaremill.sttp.client" %% "core" % "2.2.9",
|
"com.softwaremill.sttp.client" %% "core" % "2.2.9",
|
||||||
"com.softwaremill.sttp.client" %% "monix" % "2.2.9",
|
"com.softwaremill.sttp.client" %% "monix" % "2.2.9",
|
||||||
"com.softwaremill.sttp.client" %% "circe" % "2.2.9",
|
"com.softwaremill.sttp.client" %% "circe" % "2.2.9",
|
||||||
"com.softwaremill.sttp.client" %% "async-http-client-backend-monix" % "2.2.9",
|
|
||||||
"com.softwaremill.sttp.client" %% "httpclient-backend-monix" % "2.2.9",
|
"com.softwaremill.sttp.client" %% "httpclient-backend-monix" % "2.2.9",
|
||||||
"com.softwaremill.quicklens" %% "quicklens" % "1.6.1",
|
"com.softwaremill.quicklens" %% "quicklens" % "1.6.1",
|
||||||
"com.github.valskalla" %% "odin-monix" % "0.8.1",
|
|
||||||
"com.typesafe.akka" %% "akka-actor-typed" % "2.6.8",
|
"com.typesafe.akka" %% "akka-actor-typed" % "2.6.8",
|
||||||
"com.softwaremill.macwire" %% "util" % "2.3.7",
|
"com.softwaremill.macwire" %% "util" % "2.3.7",
|
||||||
"com.softwaremill.macwire" %% "macros" % "2.3.6" % "provided",
|
"com.softwaremill.macwire" %% "macros" % "2.3.6" % "provided",
|
||||||
"com.softwaremill.macwire" %% "macrosakka" % "2.3.6" % "provided",
|
"com.softwaremill.macwire" %% "macrosakka" % "2.3.6" % "provided",
|
||||||
"com.github.valskalla" %% "odin-slf4j" % "0.8.1",
|
"com.github.valskalla" %% "odin-monix" % "0.9.1",
|
||||||
|
"com.github.valskalla" %% "odin-slf4j" % "0.9.1",
|
||||||
"com.github.valskalla" %% "odin-json" % "0.9.1",
|
"com.github.valskalla" %% "odin-json" % "0.9.1",
|
||||||
|
"com.github.valskalla" %% "odin-extras" % "0.9.1",
|
||||||
"com.typesafe.scala-logging" %% "scala-logging" % "3.9.2",
|
"com.typesafe.scala-logging" %% "scala-logging" % "3.9.2",
|
||||||
"com.jfoenix" % "jfoenix" % "9.0.10",
|
"com.jfoenix" % "jfoenix" % "9.0.10",
|
||||||
"org.kordamp.ikonli" % "ikonli-core" % "12.0.0",
|
"org.kordamp.ikonli" % "ikonli-core" % "12.0.0",
|
||||||
|
@ -1,16 +1,12 @@
|
|||||||
package nova.monadic_sfx
|
package nova.monadic_sfx
|
||||||
|
|
||||||
import scala.concurrent.duration._
|
|
||||||
|
|
||||||
import _root_.monix.bio.BIOApp
|
import _root_.monix.bio.BIOApp
|
||||||
import _root_.monix.bio.Task
|
import _root_.monix.bio.Task
|
||||||
import _root_.monix.bio.UIO
|
import _root_.monix.bio.UIO
|
||||||
import cats.effect.ExitCode
|
import cats.effect.ExitCode
|
||||||
import cats.effect.Resource
|
import cats.effect.Resource
|
||||||
import cats.implicits._
|
|
||||||
import com.softwaremill.macwire._
|
import com.softwaremill.macwire._
|
||||||
import io.odin._
|
import io.odin._
|
||||||
import io.odin.syntax._
|
|
||||||
import nova.monadic_sfx.executors._
|
import nova.monadic_sfx.executors._
|
||||||
// import nova.monadic_sfx.util.IOUtils._
|
// import nova.monadic_sfx.util.IOUtils._
|
||||||
// import sttp.client.httpclient.monix.HttpClientMonixBackend
|
// import sttp.client.httpclient.monix.HttpClientMonixBackend
|
||||||
@ -18,10 +14,7 @@ object Main extends MainModule with BIOApp {
|
|||||||
|
|
||||||
def appResource(startTime: Long) =
|
def appResource(startTime: Long) =
|
||||||
for {
|
for {
|
||||||
implicit0(logger: Logger[Task]) <-
|
implicit0(logger: Logger[Task]) <- makeLogger
|
||||||
consoleLogger().withAsync(timeWindow = 1.millis) |+| fileLogger(
|
|
||||||
"application.log"
|
|
||||||
).withAsync()
|
|
||||||
schedulers = new Schedulers()
|
schedulers = new Schedulers()
|
||||||
// backend <- Resource.make(
|
// backend <- Resource.make(
|
||||||
// toIO(HttpClientMonixBackend()(schedulers.async))
|
// toIO(HttpClientMonixBackend()(schedulers.async))
|
||||||
|
@ -7,6 +7,8 @@ import nova.monadic_sfx.executors.Schedulers
|
|||||||
import nova.monadic_sfx.implicits.JFXButton
|
import nova.monadic_sfx.implicits.JFXButton
|
||||||
import nova.monadic_sfx.implicits.JavaFXMonixObservables._
|
import nova.monadic_sfx.implicits.JavaFXMonixObservables._
|
||||||
import nova.monadic_sfx.ui.MyFxApp
|
import nova.monadic_sfx.ui.MyFxApp
|
||||||
|
import nova.monadic_sfx.ui.components.router.BrainNotWorking
|
||||||
|
import nova.monadic_sfx.ui.components.router.FXRouter
|
||||||
import nova.monadic_sfx.ui.components.todo.TodoListStore
|
import nova.monadic_sfx.ui.components.todo.TodoListStore
|
||||||
import nova.monadic_sfx.ui.components.todo.TodoListView
|
import nova.monadic_sfx.ui.components.todo.TodoListView
|
||||||
import org.gerweck.scalafx.util._
|
import org.gerweck.scalafx.util._
|
||||||
@ -33,7 +35,7 @@ class MainApp(
|
|||||||
text = "Add"
|
text = "Add"
|
||||||
}
|
}
|
||||||
|
|
||||||
lazy val addTodoObs = addTodoButton.observableAction()
|
lazy val addTodoObs = addTodoButton.observableAction
|
||||||
|
|
||||||
// lazy val todoListView = TodoListView.defaultListView
|
// lazy val todoListView = TodoListView.defaultListView
|
||||||
|
|
||||||
@ -46,14 +48,14 @@ class MainApp(
|
|||||||
fill = Color.DeepSkyBlue
|
fill = Color.DeepSkyBlue
|
||||||
}
|
}
|
||||||
children ++= Seq(
|
children ++= Seq(
|
||||||
new JFXButton {
|
// new JFXButton {
|
||||||
text = "DummyButton"
|
// text = "DummyButton"
|
||||||
},
|
// },
|
||||||
new JFXButton {
|
// new JFXButton {
|
||||||
text = "DummyButton2"
|
// text = "DummyButton2"
|
||||||
},
|
// },
|
||||||
addTodoButton,
|
// addTodoButton,
|
||||||
Test.ttv
|
// Test.ttv
|
||||||
// todoListView
|
// todoListView
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
@ -62,7 +64,7 @@ class MainApp(
|
|||||||
private lazy val stage = new PrimaryStage {
|
private lazy val stage = new PrimaryStage {
|
||||||
title = "Simple ScalaFX App"
|
title = "Simple ScalaFX App"
|
||||||
scene = _scene
|
scene = _scene
|
||||||
width = 800
|
width = 1000
|
||||||
height = 400
|
height = 400
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -85,6 +87,30 @@ class MainApp(
|
|||||||
// .startAndForget
|
// .startAndForget
|
||||||
// )
|
// )
|
||||||
_ <- createTodoComponent.executeOn(schedulers.fx)
|
_ <- createTodoComponent.executeOn(schedulers.fx)
|
||||||
|
router <- Task(BrainNotWorking.router)
|
||||||
|
routerStore <- router.store(BrainNotWorking.Page.Home, logger)
|
||||||
|
routerNode <- for {
|
||||||
|
node <-
|
||||||
|
Task
|
||||||
|
.deferAction(implicit s =>
|
||||||
|
Task(new HBox {
|
||||||
|
children <-- router
|
||||||
|
.render(BrainNotWorking.resolver)(routerStore)
|
||||||
|
.map(_.delegate)
|
||||||
|
})
|
||||||
|
)
|
||||||
|
.executeOn(schedulers.fx)
|
||||||
|
_ <- Task.deferFuture(
|
||||||
|
routerStore.onNext(FXRouter.Replace(BrainNotWorking.Page.UserHome(1)))
|
||||||
|
)
|
||||||
|
|
||||||
|
} yield node
|
||||||
|
// _ <-
|
||||||
|
// BrainNotWorking
|
||||||
|
// .routerTask(logger)
|
||||||
|
// .flatMap(node => Task(_scene.getChildren += node))
|
||||||
|
// .executeOn(schedulers.fx)
|
||||||
|
_ <- Task(_scene.getChildren += routerNode).executeOn(schedulers.fx)
|
||||||
_ <- logger.info(
|
_ <- logger.info(
|
||||||
s"Application started in ${(System.currentTimeMillis() - startTime) / 1000f} seconds"
|
s"Application started in ${(System.currentTimeMillis() - startTime) / 1000f} seconds"
|
||||||
)
|
)
|
||||||
|
@ -1,7 +1,41 @@
|
|||||||
package nova.monadic_sfx
|
package nova.monadic_sfx
|
||||||
|
|
||||||
|
import scala.concurrent.duration._
|
||||||
|
|
||||||
|
import _root_.monix.bio.Task
|
||||||
|
import cats.implicits._
|
||||||
|
import io.odin._
|
||||||
|
import io.odin.config._
|
||||||
|
import io.odin.syntax._
|
||||||
import nova.monadic_sfx.actors.ActorModule
|
import nova.monadic_sfx.actors.ActorModule
|
||||||
import nova.monadic_sfx.http.HttpModule
|
import nova.monadic_sfx.http.HttpModule
|
||||||
import nova.monadic_sfx.ui.UiModule
|
import nova.monadic_sfx.ui.UiModule
|
||||||
|
import nova.monadic_sfx.util.reactive.Middlewares
|
||||||
|
|
||||||
trait MainModule extends ActorModule with UiModule with HttpModule
|
trait MainModule extends ActorModule with UiModule with HttpModule {
|
||||||
|
def routerLogger(defaultLogger: Logger[Task], storeLogger: Logger[Task]) =
|
||||||
|
enclosureRouting[Task](
|
||||||
|
"nova.monadic_sfx.util.reactive.Middlewares" -> storeLogger,
|
||||||
|
"nova.monadic_sfx.util.reactive.Store" -> storeLogger
|
||||||
|
)
|
||||||
|
.withFallback(defaultLogger)
|
||||||
|
.withAsync()
|
||||||
|
|
||||||
|
def makeLogger =
|
||||||
|
for {
|
||||||
|
defaultLogger <- consoleLogger[Task]()
|
||||||
|
.withAsync(timeWindow = 1.millis) |+| fileLogger[Task](
|
||||||
|
"application.log"
|
||||||
|
).withAsync()
|
||||||
|
middlewareLogger <-
|
||||||
|
consoleLogger[
|
||||||
|
Task
|
||||||
|
](formatter = Middlewares.format)
|
||||||
|
.withMinimalLevel(Level.Trace)
|
||||||
|
.withAsync() |+| fileLogger[Task](
|
||||||
|
"stores.log",
|
||||||
|
formatter = Middlewares.format
|
||||||
|
).withAsync()
|
||||||
|
routerLogger <- routerLogger(defaultLogger, middlewareLogger)
|
||||||
|
} yield (routerLogger)
|
||||||
|
}
|
||||||
|
@ -5,7 +5,7 @@ import java.nio.ByteBuffer
|
|||||||
import monix.eval.Task
|
import monix.eval.Task
|
||||||
import monix.reactive.Observable
|
import monix.reactive.Observable
|
||||||
import sttp.client.SttpBackend
|
import sttp.client.SttpBackend
|
||||||
import sttp.client.asynchttpclient.WebSocketHandler
|
import sttp.client.httpclient.WebSocketHandler
|
||||||
|
|
||||||
trait AppTypes {
|
trait AppTypes {
|
||||||
type HttpBackend =
|
type HttpBackend =
|
||||||
|
@ -7,9 +7,11 @@ import javafx.event.EventHandler
|
|||||||
import javafx.scene.{input => jfxsi}
|
import javafx.scene.{input => jfxsi}
|
||||||
import javafx.{event => jfxe}
|
import javafx.{event => jfxe}
|
||||||
import monix.bio.Task
|
import monix.bio.Task
|
||||||
|
import monix.eval.Coeval
|
||||||
import monix.execution.Ack
|
import monix.execution.Ack
|
||||||
import monix.execution.Cancelable
|
import monix.execution.Cancelable
|
||||||
import monix.execution.Scheduler
|
import monix.execution.Scheduler
|
||||||
|
import monix.execution.cancelables.SingleAssignCancelable
|
||||||
import monix.reactive.Observable
|
import monix.reactive.Observable
|
||||||
import monix.reactive.Observer
|
import monix.reactive.Observer
|
||||||
import monix.reactive.OverflowStrategy
|
import monix.reactive.OverflowStrategy
|
||||||
@ -19,8 +21,8 @@ import org.gerweck.scalafx.util._
|
|||||||
import scalafx.Includes._
|
import scalafx.Includes._
|
||||||
import scalafx.beans.property.Property
|
import scalafx.beans.property.Property
|
||||||
import scalafx.beans.property.ReadOnlyProperty
|
import scalafx.beans.property.ReadOnlyProperty
|
||||||
import scalafx.beans.value.ObservableValue
|
|
||||||
import scalafx.collections.ObservableBuffer
|
import scalafx.collections.ObservableBuffer
|
||||||
|
import scalafx.event.subscriptions.Subscription
|
||||||
import scalafx.scene.Scene
|
import scalafx.scene.Scene
|
||||||
import scalafx.scene.control.ButtonBase
|
import scalafx.scene.control.ButtonBase
|
||||||
import scalafx.scene.control.MenuItem
|
import scalafx.scene.control.MenuItem
|
||||||
@ -84,17 +86,20 @@ object JavaFXMonixObservables {
|
|||||||
}
|
}
|
||||||
|
|
||||||
def <--(obs: Observable[T])(implicit s: Scheduler) = {
|
def <--(obs: Observable[T])(implicit s: Scheduler) = {
|
||||||
obs.doOnNext(v => me.Task(prop.value = v)).subscribe()
|
obs.doOnNextF(v => Coeval(prop.value = v)).subscribe()
|
||||||
}
|
}
|
||||||
|
|
||||||
def asOption = prop.map(Option(_))
|
def asOption = prop.map(Option(_))
|
||||||
|
|
||||||
def observableChange[J1 >: J](): Observable[J1] = {
|
def observableChange[J1 >: J]: Observable[J1] = {
|
||||||
import monix.execution.cancelables.SingleAssignCancelable
|
import monix.execution.cancelables.SingleAssignCancelable
|
||||||
Observable.create(OverflowStrategy.Unbounded) { sub =>
|
Observable.create(OverflowStrategy.Unbounded) { sub =>
|
||||||
val c = SingleAssignCancelable()
|
val c = SingleAssignCancelable()
|
||||||
|
|
||||||
val canc = prop.onChange((a, b, c) => sub.onNext(c))
|
val canc =
|
||||||
|
prop.onChange((a, b, c1) =>
|
||||||
|
if (c1 != null && sub.onNext(c1) == Ack.Stop) c.cancel()
|
||||||
|
)
|
||||||
|
|
||||||
c := Cancelable(() => canc.cancel())
|
c := Cancelable(() => canc.cancel())
|
||||||
c
|
c
|
||||||
@ -113,16 +118,17 @@ object JavaFXMonixObservables {
|
|||||||
}
|
}
|
||||||
|
|
||||||
def <--(obs: Observable[A])(implicit s: Scheduler) = {
|
def <--(obs: Observable[A])(implicit s: Scheduler) = {
|
||||||
obs.doOnNext(v => me.Task(prop() = v)).subscribe()
|
obs.doOnNextF(v => Coeval(prop() = v)).subscribe()
|
||||||
}
|
}
|
||||||
|
|
||||||
def observableChange[J1 >: A]()
|
def observableChange[J1 >: A]: Observable[J1] = {
|
||||||
: Observable[(ObservableValue[A, A], J1, J1)] = {
|
|
||||||
import monix.execution.cancelables.SingleAssignCancelable
|
import monix.execution.cancelables.SingleAssignCancelable
|
||||||
Observable.create(OverflowStrategy.Unbounded) { sub =>
|
Observable.create(OverflowStrategy.Unbounded) { sub =>
|
||||||
val c = SingleAssignCancelable()
|
val c = SingleAssignCancelable()
|
||||||
|
|
||||||
val canc = prop.onChange((a, b, c) => sub.onNext((a, b, c)))
|
val canc = prop.onChange((_, _, c1) =>
|
||||||
|
if (c1 != null && sub.onNext(c1) == Ack.Stop) c.cancel()
|
||||||
|
)
|
||||||
|
|
||||||
c := Cancelable(() => canc.cancel())
|
c := Cancelable(() => canc.cancel())
|
||||||
c
|
c
|
||||||
@ -130,6 +136,59 @@ object JavaFXMonixObservables {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
implicit final class ObservableListExt[A](
|
||||||
|
private val buffer: ObservableList[A]
|
||||||
|
) extends AnyVal {
|
||||||
|
|
||||||
|
// def -->(sub: Observer[A]) =
|
||||||
|
// buffer.onChange((a, b, c) => if (c != null) sub.onNext(c))
|
||||||
|
|
||||||
|
// def -->(op: Property[A, A]) = {
|
||||||
|
// buffer.onChange((a, b, c) => if (c != null) op() = c)
|
||||||
|
// }
|
||||||
|
|
||||||
|
def <--(obs: Observable[A])(implicit s: Scheduler) = {
|
||||||
|
obs
|
||||||
|
.doOnNextF(v =>
|
||||||
|
for {
|
||||||
|
_ <- Coeval(buffer.clear())
|
||||||
|
_ <- Coeval(buffer += v)
|
||||||
|
} yield ()
|
||||||
|
)
|
||||||
|
.subscribe()
|
||||||
|
}
|
||||||
|
|
||||||
|
def observableChange[J1 >: A]: Observable[J1] = {
|
||||||
|
import monix.execution.cancelables.SingleAssignCancelable
|
||||||
|
Observable.create(OverflowStrategy.Unbounded) { sub =>
|
||||||
|
val c = SingleAssignCancelable()
|
||||||
|
|
||||||
|
implicit val s = sub.scheduler
|
||||||
|
|
||||||
|
val canc =
|
||||||
|
buffer.onChange((buf, _) =>
|
||||||
|
loop(sub, buf.toIterable.iterator, c).runToFuture
|
||||||
|
)
|
||||||
|
|
||||||
|
c := Cancelable(() => canc.cancel())
|
||||||
|
c
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private def loop(
|
||||||
|
sub: Observer[A],
|
||||||
|
it: Iterator[A],
|
||||||
|
c: Cancelable
|
||||||
|
): Task[Unit] =
|
||||||
|
if (it.hasNext) {
|
||||||
|
val next = it.next()
|
||||||
|
Task.deferFuture(sub.onNext(next)).flatMap {
|
||||||
|
case Ack.Continue => loop(sub, it, c)
|
||||||
|
case Ack.Stop => Task(c.cancel())
|
||||||
|
}
|
||||||
|
} else Task.unit
|
||||||
|
}
|
||||||
|
|
||||||
implicit final class BindObs3[T, J](private val prop: ReadOnlyProperty[T, J])
|
implicit final class BindObs3[T, J](private val prop: ReadOnlyProperty[T, J])
|
||||||
extends AnyVal {
|
extends AnyVal {
|
||||||
def -->(op: Observer[T]) = {
|
def -->(op: Observer[T]) = {
|
||||||
@ -140,12 +199,14 @@ object JavaFXMonixObservables {
|
|||||||
op <== prop
|
op <== prop
|
||||||
}
|
}
|
||||||
|
|
||||||
def observableChange[J1 >: J](): Observable[J1] = {
|
def observableChange[J1 >: J]: Observable[J1] = {
|
||||||
import monix.execution.cancelables.SingleAssignCancelable
|
import monix.execution.cancelables.SingleAssignCancelable
|
||||||
Observable.create(OverflowStrategy.Unbounded) { sub =>
|
Observable.create(OverflowStrategy.Unbounded) { sub =>
|
||||||
val c = SingleAssignCancelable()
|
val c = SingleAssignCancelable()
|
||||||
|
|
||||||
val canc = prop.onChange((a, b, c) => sub.onNext(c))
|
val canc = prop.onChange((a, b, c1) =>
|
||||||
|
if (c1 != null && sub.onNext(c1) == Ack.Stop) c.cancel()
|
||||||
|
)
|
||||||
|
|
||||||
c := Cancelable(() => canc.cancel())
|
c := Cancelable(() => canc.cancel())
|
||||||
c
|
c
|
||||||
@ -160,15 +221,18 @@ object JavaFXMonixObservables {
|
|||||||
obs.doOnNext(v => me.Task(prop() = ObservableBuffer.from(v))).subscribe()
|
obs.doOnNext(v => me.Task(prop() = ObservableBuffer.from(v))).subscribe()
|
||||||
}
|
}
|
||||||
|
|
||||||
def -->(sub: Observer[A])(implicit s: Scheduler) =
|
def -->(sub: Observer[A])(implicit s: Scheduler) = {
|
||||||
prop.onChange((a, b, c) =>
|
val c = SingleAssignCancelable()
|
||||||
if (c != null)
|
val subs: Subscription = prop.onChange((a, b, c1) =>
|
||||||
|
if (c1 != null)
|
||||||
Iterant[Task]
|
Iterant[Task]
|
||||||
.fromIterable(c.toIterable)
|
.fromIterable(c1.toIterable)
|
||||||
.consume
|
.consume
|
||||||
.use(consume(sub, _))
|
.use(consume(sub, c, _))
|
||||||
.runToFuture
|
.runToFuture
|
||||||
)
|
)
|
||||||
|
c := Cancelable(() => subs.cancel())
|
||||||
|
}
|
||||||
|
|
||||||
private def loop(sub: Observer[A], it: Iterator[A]): Task[Unit] =
|
private def loop(sub: Observer[A], it: Iterator[A]): Task[Unit] =
|
||||||
if (it.hasNext) {
|
if (it.hasNext) {
|
||||||
@ -181,14 +245,15 @@ object JavaFXMonixObservables {
|
|||||||
|
|
||||||
private def consume(
|
private def consume(
|
||||||
sub: Observer[A],
|
sub: Observer[A],
|
||||||
|
c: Cancelable,
|
||||||
consumer: Iterant.Consumer[Task, A]
|
consumer: Iterant.Consumer[Task, A]
|
||||||
): Task[Unit] =
|
): Task[Unit] =
|
||||||
consumer.pull.flatMap {
|
consumer.pull.flatMap {
|
||||||
case Left(value) => Task.unit
|
case Left(value) => Task.unit
|
||||||
case Right(value) =>
|
case Right(value) =>
|
||||||
Task.deferFuture(sub.onNext(value)).flatMap {
|
Task.deferFuture(sub.onNext(value)).flatMap {
|
||||||
case Ack.Continue => consume(sub, consumer)
|
case Ack.Continue => consume(sub, c, consumer)
|
||||||
case Ack.Stop => Task.unit
|
case Ack.Stop => Task(c.cancel())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -212,7 +277,7 @@ object JavaFXMonixObservables {
|
|||||||
private val button: ButtonBase
|
private val button: ButtonBase
|
||||||
) extends AnyVal {
|
) extends AnyVal {
|
||||||
|
|
||||||
def observableAction(): Observable[jfxe.ActionEvent] = {
|
def observableAction: Observable[jfxe.ActionEvent] = {
|
||||||
import monix.execution.cancelables.SingleAssignCancelable
|
import monix.execution.cancelables.SingleAssignCancelable
|
||||||
Observable.create(OverflowStrategy.Unbounded) { sub =>
|
Observable.create(OverflowStrategy.Unbounded) { sub =>
|
||||||
val c = SingleAssignCancelable()
|
val c = SingleAssignCancelable()
|
||||||
@ -238,7 +303,7 @@ object JavaFXMonixObservables {
|
|||||||
private val item: MenuItem
|
private val item: MenuItem
|
||||||
) extends AnyVal {
|
) extends AnyVal {
|
||||||
|
|
||||||
def observableAction(): Observable[jfxe.ActionEvent] = {
|
def observableAction: Observable[jfxe.ActionEvent] = {
|
||||||
import monix.execution.cancelables.SingleAssignCancelable
|
import monix.execution.cancelables.SingleAssignCancelable
|
||||||
Observable.create(OverflowStrategy.Unbounded) { sub =>
|
Observable.create(OverflowStrategy.Unbounded) { sub =>
|
||||||
val c = SingleAssignCancelable()
|
val c = SingleAssignCancelable()
|
||||||
|
@ -5,6 +5,5 @@ import scalafx.scene.{control => sfxc}
|
|||||||
import JavaFXMonixObservables._
|
import JavaFXMonixObservables._
|
||||||
|
|
||||||
class MenuItem extends sfxc.MenuItem {
|
class MenuItem extends sfxc.MenuItem {
|
||||||
def obsAction =
|
def obsAction = new ActionObservableBuilder(this.observableAction)
|
||||||
new ActionObservableBuilder(this.observableAction())
|
|
||||||
}
|
}
|
||||||
|
@ -21,14 +21,14 @@ class MyFxApp(val schedulers: Schedulers)(implicit logger: Logger[Task]) {
|
|||||||
|
|
||||||
// def stage_=(stage: PrimaryStage) = Task(internal.stage = stage)
|
// def stage_=(stage: PrimaryStage) = Task(internal.stage = stage)
|
||||||
|
|
||||||
def useInternal[T](f: JFXApp => Task[T]): Task[T] =
|
// def useInternal[T](f: JFXApp => Task[T]): Task[T] =
|
||||||
for {
|
// for {
|
||||||
_ <- logger.debug("Request for using internal value")
|
// _ <- logger.debug("Request for using internal value")
|
||||||
res <- f(internal).executeOn(schedulers.fx)
|
// res <- f(internal).executeOn(schedulers.fx)
|
||||||
_ <- logger.debug(s"Result was ${res.toString()}")
|
// _ <- logger.debug(s"Result was ${res.toString()}")
|
||||||
} yield (res)
|
// } yield (res)
|
||||||
|
|
||||||
def init(stage: => PrimaryStage, delay: FiniteDuration = 2000.millis) =
|
def init(stage: => PrimaryStage, delay: FiniteDuration = 200.millis) =
|
||||||
for {
|
for {
|
||||||
_ <- logger.info("Starting FX App")
|
_ <- logger.info("Starting FX App")
|
||||||
fib <- Task(internal.main(Array.empty)).start
|
fib <- Task(internal.main(Array.empty)).start
|
||||||
|
@ -0,0 +1,107 @@
|
|||||||
|
package nova.monadic_sfx.ui.components.router
|
||||||
|
|
||||||
|
import enumeratum._
|
||||||
|
import io.circe.Encoder
|
||||||
|
import io.circe.generic.JsonCodec
|
||||||
|
import io.odin.Logger
|
||||||
|
import monix.bio.Task
|
||||||
|
import nova.monadic_sfx.util.IOUtils
|
||||||
|
import nova.monadic_sfx.util.reactive.Reducer
|
||||||
|
import nova.monadic_sfx.util.reactive.Store
|
||||||
|
import scalafx.scene.Parent
|
||||||
|
import scalafx.scene.control.Label
|
||||||
|
|
||||||
|
object FXRouter {
|
||||||
|
|
||||||
|
final case class State[P](page: P)
|
||||||
|
|
||||||
|
@JsonCodec
|
||||||
|
sealed abstract class Action[T]
|
||||||
|
// final case object Init extends Action
|
||||||
|
final case class Replace[T](p: T) extends Action[T]
|
||||||
|
|
||||||
|
type FXStore[P] = Store[Action[P], State[P]]
|
||||||
|
|
||||||
|
// def resolver2 = resolver.lift.andThen(_.getOrElse(notFound))
|
||||||
|
|
||||||
|
// def resolver: PartialFunction[P <: Enum[P]][P, Parent] = {
|
||||||
|
// case Home => new TextField
|
||||||
|
// }
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
class FXRouter[P <: EnumEntry](
|
||||||
|
)(implicit E: Encoder[P]) {
|
||||||
|
import FXRouter._
|
||||||
|
|
||||||
|
def store(initialPage: P, logger: Logger[Task]) =
|
||||||
|
Task.deferAction(implicit s =>
|
||||||
|
Store.createL[Action[P], State[P]](
|
||||||
|
Replace(initialPage),
|
||||||
|
State(initialPage),
|
||||||
|
Reducer.withOptionalEffects[Task, Action[P], State[P]](reducer _)
|
||||||
|
// Seq(actionLoggerMiddleware(logger, "RouterStore"))
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
def reducer(
|
||||||
|
state: State[P],
|
||||||
|
action: Action[P]
|
||||||
|
): (State[P], Option[Task[Action[P]]]) =
|
||||||
|
action match {
|
||||||
|
// case Init => (state, None)
|
||||||
|
case Replace(p) =>
|
||||||
|
(state.copy(page = p), None)
|
||||||
|
}
|
||||||
|
|
||||||
|
def render(
|
||||||
|
resolver: P => Task[Parent]
|
||||||
|
)(implicit store: FXStore[P]) =
|
||||||
|
store.mapEval { case (_, FXRouter.State(p)) => IOUtils.toTask(resolver(p)) }
|
||||||
|
|
||||||
|
def link(
|
||||||
|
page: P,
|
||||||
|
store: FXStore[P]
|
||||||
|
) = {
|
||||||
|
store.onNext(Replace(page))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
object BrainNotWorking {
|
||||||
|
|
||||||
|
@JsonCodec
|
||||||
|
sealed trait Page extends EnumEntry
|
||||||
|
object Page extends Enum[Page] {
|
||||||
|
val values = findValues
|
||||||
|
final case object Home extends Page
|
||||||
|
final case class UserHome(id: Int) extends Page
|
||||||
|
}
|
||||||
|
|
||||||
|
def resolver: PartialFunction[Page, Task[Parent]] = {
|
||||||
|
case Page.Home =>
|
||||||
|
Task(new Label {
|
||||||
|
text = "HomePage"
|
||||||
|
})
|
||||||
|
case Page.UserHome(id0) =>
|
||||||
|
Task(new Label {
|
||||||
|
text = s"User Home, Id = $id0"
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
val router = new FXRouter[Page]
|
||||||
|
}
|
||||||
|
|
||||||
|
// case class State()
|
||||||
|
// object RouterStore {
|
||||||
|
|
||||||
|
// sealed trait Action
|
||||||
|
// case object Init extends Action
|
||||||
|
|
||||||
|
// def reducer(state: State, action: Action) =
|
||||||
|
// action match {
|
||||||
|
// case Init => state
|
||||||
|
|
||||||
|
// }
|
||||||
|
// def apply() =
|
||||||
|
// Store.createL[Action, State](Init, State(), Reducer(reducer _), Seq.empty)
|
||||||
|
// }
|
@ -13,40 +13,77 @@ case class Todo(id: Int, content: String)
|
|||||||
object TodoListStore {
|
object TodoListStore {
|
||||||
|
|
||||||
@JsonCodec
|
@JsonCodec
|
||||||
sealed trait Command
|
sealed trait Action
|
||||||
case object Init extends Command
|
case object Init extends Action
|
||||||
case class Add(content: String) extends Command
|
case class Add(content: String) extends Action
|
||||||
case class Edit(id: Int, content: String) extends Command
|
case class Edit(id: Int, content: String) extends Action
|
||||||
case class Delete(id: Int) extends Command
|
case class Delete(id: Int) extends Action
|
||||||
|
|
||||||
|
private case class InternalAdd(content: String) extends Action
|
||||||
|
private case object End extends Action
|
||||||
|
|
||||||
case class State(todos: Vector[Todo], counter: Int)
|
case class State(todos: Vector[Todo], counter: Int)
|
||||||
|
|
||||||
def reducer(
|
def reducer(logger: Logger[Task])(
|
||||||
state: State,
|
state: State,
|
||||||
action: Command
|
action: Action
|
||||||
) =
|
): (State, Option[Task[Action]]) =
|
||||||
action match {
|
action match {
|
||||||
case Init => state
|
case Init => (state, None)
|
||||||
case Add(content) =>
|
case Add(content) =>
|
||||||
state.copy(
|
val nextAction = Some(for {
|
||||||
|
// _ <- logger.debug(s"Received $content")
|
||||||
|
res <- Task.pure(InternalAdd(content))
|
||||||
|
} yield res)
|
||||||
|
(state, nextAction)
|
||||||
|
case Edit(_id, content) =>
|
||||||
|
val condition: Todo => Boolean = _.id == _id
|
||||||
|
val nextState = state
|
||||||
|
.modify(_.todos.eachWhere(condition))
|
||||||
|
.using(_.copy(content = content))
|
||||||
|
(nextState, None)
|
||||||
|
case Delete(id) =>
|
||||||
|
(state.copy(state.todos.filterNot(_.id == id)), None)
|
||||||
|
|
||||||
|
case InternalAdd(content) =>
|
||||||
|
val nextState = state.copy(
|
||||||
todos = state.todos :+ Todo(state.counter, content),
|
todos = state.todos :+ Todo(state.counter, content),
|
||||||
counter = state.counter + 1
|
counter = state.counter + 1
|
||||||
)
|
)
|
||||||
case Edit(_id, content) =>
|
(nextState, Some(logger.debug(s"Received $content") >> Task.pure(End)))
|
||||||
val condition: Todo => Boolean = _.id == _id
|
case End => (state, None)
|
||||||
state
|
|
||||||
.modify(_.todos.eachWhere(condition))
|
|
||||||
.using(_.copy(content = content))
|
|
||||||
case Delete(id) =>
|
|
||||||
state.copy(state.todos.filterNot(_.id == id))
|
|
||||||
}
|
}
|
||||||
|
|
||||||
def apply(logger: Logger[Task]) =
|
def apply(logger: Logger[Task]) =
|
||||||
|
Task.deferAction(implicit s =>
|
||||||
|
for {
|
||||||
|
logMware <- actionLoggerMiddleware[Action, State](logger, "TodoStore")
|
||||||
|
store <-
|
||||||
Store
|
Store
|
||||||
.createL[Command, State](
|
.createL[Action, State](
|
||||||
Init,
|
Init,
|
||||||
State(Vector.empty[Todo], 0),
|
State(Vector.empty[Todo], 0),
|
||||||
Reducer(reducer _),
|
Reducer.withOptionalEffects(reducer(logger) _),
|
||||||
Seq(actionLoggerMiddleware(logger))
|
Seq(
|
||||||
|
// actionLoggerMiddleware(logger, "TodoStore2")
|
||||||
|
logMware
|
||||||
|
)
|
||||||
|
)
|
||||||
|
} yield (store)
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Task.deferAction(implicit s =>
|
||||||
|
// Store
|
||||||
|
// .createJsonL[Action, State](
|
||||||
|
// Init,
|
||||||
|
// State(Vector.empty[Todo], 0),
|
||||||
|
// Reducer.withOptionalEffects(reducer(logger) _),
|
||||||
|
// "TodoStore",
|
||||||
|
// logger
|
||||||
|
// // Seq(
|
||||||
|
// // actionLoggerMiddleware(logger, "TodoStore")
|
||||||
|
// // // actionLoggerMiddleware(logger, "TodoStore2")
|
||||||
|
// // )
|
||||||
|
// )
|
||||||
|
// )
|
||||||
|
@ -2,6 +2,7 @@ package nova.monadic_sfx.ui.components.todo
|
|||||||
|
|
||||||
import monix.bio.Task
|
import monix.bio.Task
|
||||||
import monix.execution.cancelables.CompositeCancelable
|
import monix.execution.cancelables.CompositeCancelable
|
||||||
|
import monix.{eval => me}
|
||||||
import nova.monadic_sfx.implicits.FontIcon
|
import nova.monadic_sfx.implicits.FontIcon
|
||||||
import nova.monadic_sfx.implicits.IconLiteral
|
import nova.monadic_sfx.implicits.IconLiteral
|
||||||
import nova.monadic_sfx.implicits.JFXButton
|
import nova.monadic_sfx.implicits.JFXButton
|
||||||
@ -21,13 +22,12 @@ import scalafx.scene.control.ListCell
|
|||||||
import scalafx.scene.control.SelectionMode
|
import scalafx.scene.control.SelectionMode
|
||||||
import scalafx.scene.layout.HBox
|
import scalafx.scene.layout.HBox
|
||||||
import scalafx.scene.text.Text
|
import scalafx.scene.text.Text
|
||||||
import monix.{eval => me}
|
|
||||||
|
|
||||||
object TodoListView {
|
object TodoListView {
|
||||||
def apply(
|
def apply(
|
||||||
store: MonixProSubject[
|
store: MonixProSubject[
|
||||||
TodoListStore.Command,
|
TodoListStore.Action,
|
||||||
(TodoListStore.Command, TodoListStore.State)
|
(TodoListStore.Action, TodoListStore.State)
|
||||||
]
|
]
|
||||||
): Task[Node] =
|
): Task[Node] =
|
||||||
Task.deferAction(implicit s =>
|
Task.deferAction(implicit s =>
|
||||||
@ -48,6 +48,9 @@ object TodoListView {
|
|||||||
new JFXListView[Todo] {
|
new JFXListView[Todo] {
|
||||||
def selectedItems = selectionModel().selectedItems.view
|
def selectedItems = selectionModel().selectedItems.view
|
||||||
|
|
||||||
|
selectionModel().selectionMode = SelectionMode.Multiple
|
||||||
|
selectionModel().selectedItems.observableSeqValue ==> _selectedItems
|
||||||
|
|
||||||
cc += items <-- todos
|
cc += items <-- todos
|
||||||
|
|
||||||
val emptyCell = ObjectProperty(new HBox)
|
val emptyCell = ObjectProperty(new HBox)
|
||||||
@ -78,9 +81,6 @@ object TodoListView {
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
selectionModel().selectionMode = SelectionMode.Multiple
|
|
||||||
selectionModel().selectedItems.observableSeqValue ==> _selectedItems
|
|
||||||
|
|
||||||
contextMenu = new ContextMenu {
|
contextMenu = new ContextMenu {
|
||||||
items ++= Seq(
|
items ++= Seq(
|
||||||
new MenuItem {
|
new MenuItem {
|
||||||
@ -94,6 +94,12 @@ object TodoListView {
|
|||||||
.map(todo => TodoListStore.Delete(todo.id))
|
.map(todo => TodoListStore.Delete(todo.id))
|
||||||
.toList
|
.toList
|
||||||
) --> store
|
) --> store
|
||||||
|
|
||||||
|
// obsAction.split(
|
||||||
|
// _.useLazyEval(me.Task(TodoListStore.Delete(0))) --> store,
|
||||||
|
// _.useLazyEval(me.Task(TodoListStore.Delete(0))) --> store,
|
||||||
|
// _.useLazyEval(me.Task(TodoListStore.Delete(0))) --> store
|
||||||
|
// )
|
||||||
},
|
},
|
||||||
new MenuItem {
|
new MenuItem {
|
||||||
text = "Edit"
|
text = "Edit"
|
||||||
@ -103,7 +109,7 @@ object TodoListView {
|
|||||||
},
|
},
|
||||||
new JFXButton {
|
new JFXButton {
|
||||||
text = "Add"
|
text = "Add"
|
||||||
disable <== _selectedItems.map(_.length > 0)
|
// disable <== _selectedItems.map(_.length > 0)
|
||||||
obsAction
|
obsAction
|
||||||
.useLazyEval(me.Task(TodoListStore.Add(_content()))) --> store
|
.useLazyEval(me.Task(TodoListStore.Add(_content()))) --> store
|
||||||
},
|
},
|
||||||
|
63
src/main/scala/nova/monadic_sfx/util/Misc.scala
Normal file
63
src/main/scala/nova/monadic_sfx/util/Misc.scala
Normal file
@ -0,0 +1,63 @@
|
|||||||
|
package nova.monadic_sfx.util
|
||||||
|
|
||||||
|
import scalafx.beans.property.ObjectProperty
|
||||||
|
import scalafx.beans.property.ReadOnlyObjectProperty
|
||||||
|
import scalafx.beans.value.ObservableValue
|
||||||
|
|
||||||
|
object Misc {
|
||||||
|
|
||||||
|
implicit final class MyRichObservable[A, C](val self: ObservableValue[A, C])
|
||||||
|
extends AnyVal {
|
||||||
|
def filter(f: A => Boolean): ReadOnlyObjectProperty[A] =
|
||||||
|
Method.filter(self)(f)
|
||||||
|
def filterNull: ReadOnlyObjectProperty[A] = Method.filterNull(self)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
object Method {
|
||||||
|
type Observable[A] = ObservableValue[A, _]
|
||||||
|
|
||||||
|
def filter[B](
|
||||||
|
a: Observable[B]
|
||||||
|
)(f: B => Boolean): ReadOnlyObjectProperty[B] = {
|
||||||
|
|
||||||
|
val prop = ObjectProperty[B](a.value)
|
||||||
|
|
||||||
|
def changeHandler() =
|
||||||
|
prop.synchronized {
|
||||||
|
if (f(a.value)) {
|
||||||
|
prop.value = a.value
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
a onChange changeHandler()
|
||||||
|
prop
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Simply creates a new observable that mirrors the source observable but
|
||||||
|
* doesn't emit null values. JavaFX likes to work with null values in scene
|
||||||
|
* nodes/properties (shrugs) and observables by default emit null values
|
||||||
|
* that can cause crashes. ScalaFX does not offer any *fixes* for this
|
||||||
|
*
|
||||||
|
* @param a
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
|
def filterNull[B](
|
||||||
|
a: Observable[B]
|
||||||
|
): ReadOnlyObjectProperty[B] = {
|
||||||
|
|
||||||
|
val prop = ObjectProperty[B](a.value)
|
||||||
|
|
||||||
|
def changeHandler() =
|
||||||
|
prop.synchronized {
|
||||||
|
if (a.value != null) {
|
||||||
|
prop.value = a.value
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
a onChange changeHandler()
|
||||||
|
prop
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -1,27 +1,75 @@
|
|||||||
package nova.monadic_sfx.util.reactive
|
package nova.monadic_sfx.util.reactive
|
||||||
|
|
||||||
|
import java.time.LocalDateTime
|
||||||
|
|
||||||
|
import io.circe.Encoder
|
||||||
|
import io.circe.Printer
|
||||||
|
import io.circe.generic.JsonCodec
|
||||||
|
import io.circe.syntax._
|
||||||
import io.odin.Logger
|
import io.odin.Logger
|
||||||
|
import io.odin.LoggerMessage
|
||||||
|
import io.odin.formatter.options.PositionFormat
|
||||||
|
import io.odin.formatter.options.ThrowableFormat
|
||||||
|
import io.odin.meta.Render
|
||||||
import monix.bio.Task
|
import monix.bio.Task
|
||||||
import monix.reactive.Observable
|
import monix.reactive.Observable
|
||||||
import nova.monadic_sfx.util.IOUtils._
|
|
||||||
|
|
||||||
// object Middleware {
|
// object Middleware {
|
||||||
// def apply[A,M,T](ob: Observable[(A,M)], cb: (A,M) => T): Observable[(A,M)] = ob
|
// def apply[A,M,T](ob: Observable[(A,M)], cb: (A,M) => T): Observable[(A,M)] = ob
|
||||||
// }
|
// }
|
||||||
object Middlewares {
|
|
||||||
def actionStateLoggerMiddleware[A, M](
|
|
||||||
logger: Logger[Task]
|
|
||||||
): Middleware[A, M] =
|
|
||||||
(obs: Observable[(A, M)]) =>
|
|
||||||
obs.doOnNext {
|
|
||||||
case (a, m) => toTask(logger.debug(s"Received action $a with state $m"))
|
|
||||||
}
|
|
||||||
|
|
||||||
def actionLoggerMiddleware[A, M](
|
@JsonCodec
|
||||||
logger: Logger[Task]
|
final case class StoreInfo[A](
|
||||||
): Middleware[A, M] =
|
name: String,
|
||||||
(obs: Observable[(A, M)]) =>
|
action: A,
|
||||||
obs.doOnNext {
|
time: LocalDateTime = LocalDateTime.now()
|
||||||
case (a, _) => toTask(logger.debug(s"Received action $a "))
|
)
|
||||||
|
|
||||||
|
object StoreInfo {
|
||||||
|
val printer = Printer.noSpaces
|
||||||
|
implicit def render[T: Encoder]: Render[StoreInfo[T]] =
|
||||||
|
new Render[StoreInfo[T]] {
|
||||||
|
override def render(m: StoreInfo[T]): String = printer.print(m.asJson)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
object Middlewares {
|
||||||
|
|
||||||
|
// val encoder: Encoder[LoggerMessage] =
|
||||||
|
// Encoder.forProduct1("message")(m => m.message.value)
|
||||||
|
|
||||||
|
val format = create(ThrowableFormat.Default, PositionFormat.Full)
|
||||||
|
|
||||||
|
def create(
|
||||||
|
throwableFormat: ThrowableFormat,
|
||||||
|
positionFormat: PositionFormat
|
||||||
|
): io.odin.formatter.Formatter = {
|
||||||
|
// val encoder: Encoder[LoggerMessage] =
|
||||||
|
// Encoder.forProduct1("message")(m => m.message.value)
|
||||||
|
(msg: LoggerMessage) => msg.message.value
|
||||||
|
}
|
||||||
|
|
||||||
|
def actionStateLoggerMiddleware[A, M](
|
||||||
|
logger: Logger[Task]
|
||||||
|
): Task[Middleware[A, M]] =
|
||||||
|
Task.deferAction(implicit s =>
|
||||||
|
Task((obs: Observable[(A, M)]) =>
|
||||||
|
obs.doOnNextF {
|
||||||
|
case (a, m) =>
|
||||||
|
logger.debug(s"Received action $a with state $m")
|
||||||
|
}
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
def actionLoggerMiddleware[A: Encoder, M](
|
||||||
|
logger: Logger[Task],
|
||||||
|
name: String
|
||||||
|
): Task[Middleware[A, M]] =
|
||||||
|
Task.deferAction(implicit s =>
|
||||||
|
Task((obs: Observable[(A, M)]) =>
|
||||||
|
obs.doOnNextF {
|
||||||
|
case (a, _) =>
|
||||||
|
logger.debug(StoreInfo(name, a))
|
||||||
|
}
|
||||||
|
)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
@ -1,9 +1,8 @@
|
|||||||
package nova.monadic_sfx.util.reactive
|
package nova.monadic_sfx.util.reactive
|
||||||
|
|
||||||
import cats.effect.Sync
|
import io.circe.Encoder
|
||||||
|
import io.odin.Logger
|
||||||
import monix.bio.Task
|
import monix.bio.Task
|
||||||
import monix.execution.Scheduler
|
|
||||||
import monix.reactive.Observable
|
|
||||||
import monix.reactive.OverflowStrategy
|
import monix.reactive.OverflowStrategy
|
||||||
import monix.reactive.subjects.ConcurrentSubject
|
import monix.reactive.subjects.ConcurrentSubject
|
||||||
|
|
||||||
@ -15,7 +14,7 @@ object Store {
|
|||||||
middlewares: Seq[Middleware[A, M]] = Seq.empty,
|
middlewares: Seq[Middleware[A, M]] = Seq.empty,
|
||||||
overflowStrategy: OverflowStrategy.Synchronous[A] =
|
overflowStrategy: OverflowStrategy.Synchronous[A] =
|
||||||
OverflowStrategy.DropOld(50)
|
OverflowStrategy.DropOld(50)
|
||||||
) =
|
): Task[Store[A, M]] =
|
||||||
Task.deferAction { implicit s =>
|
Task.deferAction { implicit s =>
|
||||||
Task {
|
Task {
|
||||||
val subject = ConcurrentSubject.publish[A](overflowStrategy)
|
val subject = ConcurrentSubject.publish[A](overflowStrategy)
|
||||||
@ -35,7 +34,7 @@ object Store {
|
|||||||
.behavior(initialAction -> initialState)
|
.behavior(initialAction -> initialState)
|
||||||
.refCount
|
.refCount
|
||||||
|
|
||||||
val res = middlewares.view.reverse.foldLeft(obs) {
|
val res = middlewares.foldLeft(obs) {
|
||||||
case (obs, middleware) => middleware(obs)
|
case (obs, middleware) => middleware(obs)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -46,16 +45,23 @@ object Store {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
def create[F[_], A, M](
|
def createJsonL[A: Encoder, M](
|
||||||
initialAction: A,
|
initialAction: A,
|
||||||
initialState: M,
|
initialState: M,
|
||||||
reducer: Reducer[A, M]
|
reducer: Reducer[A, M],
|
||||||
)(implicit s: Scheduler, F: Sync[F]): F[Observable[(A, M)]] =
|
storeName: String,
|
||||||
F.delay {
|
logger: Logger[Task],
|
||||||
val subject = ConcurrentSubject.publish[A]
|
middlewares: Seq[Middleware[A, M]] = Seq.empty,
|
||||||
|
overflowStrategy: OverflowStrategy.Synchronous[A] =
|
||||||
|
OverflowStrategy.DropOld(50)
|
||||||
|
): Task[Store[A, M]] =
|
||||||
|
Task.deferAction { implicit s =>
|
||||||
|
Task {
|
||||||
|
val subject = ConcurrentSubject.publish[A](overflowStrategy)
|
||||||
|
|
||||||
val fold: ((A, M), A) => (A, M) = {
|
val fold: ((A, M), A) => Task[(A, M)] = {
|
||||||
case ((_, state), action) => {
|
case ((_, state), action) =>
|
||||||
|
Task {
|
||||||
val (newState, effects) = reducer(state, action)
|
val (newState, effects) = reducer(state, action)
|
||||||
|
|
||||||
effects.subscribe(subject.onNext _)
|
effects.subscribe(subject.onNext _)
|
||||||
@ -64,10 +70,28 @@ object Store {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
subject
|
val obs = subject
|
||||||
.scan[(A, M)](initialAction -> initialState)(fold)
|
.doOnNextF(action =>
|
||||||
|
logger.debug(
|
||||||
|
StoreInfo(storeName, action)
|
||||||
|
) // .executeOn(Scheduler.global)
|
||||||
|
)
|
||||||
|
// .doOnNextF(action => Coeval(println(action)))
|
||||||
|
.scanEvalF[Task, (A, M)](Task.pure(initialAction -> initialState))(
|
||||||
|
fold
|
||||||
|
)
|
||||||
.behavior(initialAction -> initialState)
|
.behavior(initialAction -> initialState)
|
||||||
.refCount
|
.refCount
|
||||||
|
|
||||||
|
// val res = middlewares.foldLeft(obs) {
|
||||||
|
// case (obs, middleware) => middleware(obs)
|
||||||
|
// }
|
||||||
|
|
||||||
|
MonixProSubject.from(
|
||||||
|
subject,
|
||||||
|
obs
|
||||||
|
)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -6,6 +6,7 @@ import monix.reactive.Observer
|
|||||||
package object reactive {
|
package object reactive {
|
||||||
type MonixProSubject[-I, +O] = Observable[O] with Observer[I]
|
type MonixProSubject[-I, +O] = Observable[O] with Observer[I]
|
||||||
type Middleware[A, M] = Observable[(A, M)] => Observable[(A, M)]
|
type Middleware[A, M] = Observable[(A, M)] => Observable[(A, M)]
|
||||||
|
type Store[A, M] = MonixProSubject[A, (A, M)]
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* A Function that applies an Action onto the Stores current state.
|
* A Function that applies an Action onto the Stores current state.
|
||||||
|
Loading…
Reference in New Issue
Block a user