All Downloads are FREE. Search and download functionalities are using the official Maven repository.

crystal.react.syntax.effect.scala Maven / Gradle / Ivy

// Copyright (c) 2016-2023 Association of Universities for Research in Astronomy, Inc. (AURA)
// For license information see LICENSE or https://opensource.org/licenses/BSD-3-Clause

package crystal.react.syntax

import cats.MonadError
import cats.effect.Async
import cats.effect.Sync
import cats.syntax.all.*
import crystal.*
import crystal.react.hooks.UseSerialState
import japgolly.scalajs.react.*
import japgolly.scalajs.react.component.Generic.MountedSimple
import japgolly.scalajs.react.util.DefaultEffects.{Async => DefaultA}
import japgolly.scalajs.react.util.DefaultEffects.{Sync => DefaultS}
import japgolly.scalajs.react.util.Effect
import japgolly.scalajs.react.util.Effect.UnsafeSync
import monocle.Lens
import org.typelevel.log4cats.Logger

import scala.util.control.NonFatal

trait effect {
  extension [A](self: DefaultS[A])
    inline def to[F[_]](using F: Sync[F]): F[A]       =
      F.delay(summon[UnsafeSync[DefaultS]].runSync(self))
    inline def toStream[F[_]: Sync]: fs2.Stream[F, A] =
      fs2.Stream.eval(self.to[F])
    inline def toAsync: DefaultA[A]                   =
      to[DefaultA]
    inline def toAsyncStream: fs2.Stream[DefaultA, A] =
      toStream[DefaultA]

  extension [S, P](self: MountedSimple[DefaultS, DefaultA, P, S])
    def propsIn[F[_]: Sync]: F[P] = self.props.to[F]

  extension [S](self: StateAccess[DefaultS, DefaultA, S])
    /** Provides access to state `S` in an `F` */
    def stateIn[F[_]: Sync]: F[S] = self.state.to[F]

  extension [S](self: StateAccess.Write[DefaultS, DefaultA, S])
    def setStateIn[F[_]: Sync](s: S): F[Unit]      = self.setState(s).to[F]
    def modStateIn[F[_]: Sync](f: S => S): F[Unit] = self.modState(f).to[F]

    /**
     * Like `setState` but completes with a `Unit` value *after* the state modification has been
     * completed. In contrast, `setState(mod).to[F]` completes with a unit once the state
     * modification has been enqueued.
     *
     * Provides access only to state.
     */
    def setStateAsyncIn[F[_]](s: S)(using F: Async[F], dispatch: UnsafeSync[DefaultS]): F[Unit] =
      F.async(cb =>
        self
          .setState(s, DefaultS.delay(cb(Right(()))))
          .maybeHandleError { case NonFatal(t) => DefaultS.delay(cb(Left(t))) }
          .to[F]
          .as(F.unit.some)
      )

    /**
     * Like `modState` but completes with a `Unit` value *after* the state modification has been
     * completed. In contrast, `modState(mod).to[F]` completes with a unit once the state
     * modification has been enqueued.
     *
     * Provides access only to state.
     */
    def modStateAsyncIn[F[_]](
      mod: S => S
    )(using F: Async[F], dispatch: UnsafeSync[DefaultS]): F[Unit] =
      F.async(asyncCB =>
        self
          .modState(mod, DefaultS.delay(asyncCB(Right(()))))
          .maybeHandleError { case NonFatal(t) => DefaultS.delay(asyncCB(Left(t))) }
          .to[F]
          .as(F.unit.some)
      )

    def setStateLIn[F[_]]: SetStateLApplied[F, S] =
      new SetStateLApplied[F, S](self)

    def modStateLIn[F[_]]: ModStateLApplied[F, S] =
      new ModStateLApplied[F, S](self)

  extension [S, P](self: StateAccess.WriteWithProps[DefaultS, DefaultA, P, S])
    /**
     * Like `modState` but completes with a `Unit` value *after* the state modification has been
     * completed. In contrast, `modState(mod).to[F]` completes with a unit once the state
     * modification has been enqueued.
     *
     * Provides access to both state and props.
     */
    def modStateWithPropsIn[F[_]](
      mod: (S, P) => S
    )(using F: Async[F], dispatch: UnsafeSync[DefaultS]): F[Unit] =
      F.async(cb =>
        self
          .modState(mod, DefaultS.delay(cb(Right(()))))
          .maybeHandleError { case NonFatal(t) => DefaultS.delay(cb(Left(t))) }
          .to[F]
          .as(F.unit.some)
      )

  extension [S](self: Hooks.UseState[S])
    inline def setStateAsync: Reusable[S => DefaultA[Unit]] =
      self.setState.map(f => s => f(s).to[DefaultA])

    inline def modStateAsync: Reusable[(S => S) => DefaultA[Unit]] =
      self.modState.map(f => g => f(g).to[DefaultA])

    inline def withReusableInputsAsync: WithReusableInputsAsync[S] =
      new WithReusableInputsAsync[S](self)

  extension [S](self: Hooks.UseStateWithReuse[S])
    inline def setStateAsync: Reusable[S => Reusable[DefaultA[Unit]]] =
      self.setState.map(f => s => f(s).map(_.to[DefaultA]))

    inline def modStateAsync(f: S => S): Reusable[DefaultA[Unit]] =
      self.modState(f).map(_.to[DefaultA])

  extension [A](self: Hooks.UseRef[A])
    inline def setIn[F[_]: Sync](a: A): F[Unit]      = self.set(a).to[F]
    inline def modIn[F[_]: Sync](f: A => A): F[Unit] = self.mod(f).to[F]
    inline def getIn[F[_]: Sync]: F[A]              = self.get.to[F]
    inline def setAsync: A => DefaultA[Unit]        = setIn[DefaultA](_)
    inline def modAsync: (A => A) => DefaultA[Unit] = modIn[DefaultA](_)
    inline def getAsync: DefaultA[A]                = getIn[DefaultA]

  extension [S](self: UseSerialState[S])
    inline def setStateAsync: Reusable[S => DefaultA[Unit]] =
      self.setState.map(f => s => f(s).to[DefaultA])

    inline def modStateAsync: Reusable[(S => S) => DefaultA[Unit]] =
      self.modState.map(f => g => f(g).to[DefaultA])

  extension [F[_], A](self: F[A])
    /**
     * Return a `DefaultS[Unit]` that will run the effect `F[A]` asynchronously.
     *
     * @param cb
     *   Result handler returning a `F[Unit]`.
     */
    def runAsync(
      cb: Either[Throwable, A] => F[Unit]
    )(using F: MonadError[F, Throwable], dispatcher: Effect.Dispatch[F]): DefaultS[Unit] =
      DefaultS.delay(dispatcher.dispatch(self.attempt.flatMap(cb)))

    /**
     * Return a `DefaultS[Unit]` that will run the effect `F[A]` asynchronously.
     *
     * @param cb
     *   Result handler returning a `DefaultS[Unit]`.
     */
    def runAsyncAndThen(
      cb:          Either[Throwable, A] => DefaultS[Unit]
    )(using
      F:           Sync[F],
      dispatcherF: Effect.Dispatch[F],
      dispatchS:   UnsafeSync[DefaultS]
    ): DefaultS[Unit] =
      runAsync(cb.andThen(c => F.delay(dispatchS.runSync(c))))

    /**
     * Return a `DefaultS[Unit]` that will run the effect `F[A]` asynchronously and discard the
     * result or errors.
     */
    def runAsyncAndForget(using
      F:           MonadError[F, Throwable],
      dispatcherF: Effect.Dispatch[F]
    ): DefaultS[Unit] =
      self.runAsync(_ => F.unit)

  extension [F[_]](self: F[Unit])
    /**
     * Return a `DefaultS[Unit]` that will run the effect `F[Unit]` asynchronously and log possible
     * errors.
     *
     * @param cb
     *   `F[Unit]` to run in case of success.
     */
    def runAsyncAndThenF(
      cb:         F[Unit],
      errorMsg:   String = "Error in F[Unit].runAsyncAndThenF"
    )(using
      F:          MonadError[F, Throwable],
      dispatcher: Effect.Dispatch[F],
      logger:     Logger[F]
    ): DefaultS[Unit] =
      // new EffectAOps(self).runAsync {
      self.runAsync {
        case Right(()) => cb
        case Left(t)   => logger.error(t)(errorMsg)
      }

    /**
     * Return a `DefaultS[Unit]` that will run the effect `F[Unit]` asynchronously and log possible
     * errors.
     *
     * @param cb
     *   `DefaultS[Unit]` to run in case of success.
     */
    def runAsyncAndThen(
      cb:          DefaultS[Unit],
      errorMsg:    String = "Error in F[Unit].runAsyncAndThen"
    )(using
      F:           Sync[F],
      dispatcherF: Effect.Dispatch[F],
      logger:      Logger[F],
      dispatchS:   UnsafeSync[DefaultS]
    ): DefaultS[Unit] =
      runAsyncAndThenF(F.delay(dispatchS.runSync(cb)), errorMsg)

    /**
     * Return a `DefaultS[Unit]` that will run the effect F[Unit] asynchronously and log possible
     * errors.
     */
    def runAsync(
      errorMsg:   String = "Error in F[Unit].runAsync"
    )(using
      F:          MonadError[F, Throwable],
      dispatcher: Effect.Dispatch[F],
      logger:     Logger[F]
    ): DefaultS[Unit] =
      runAsyncAndThenF(F.unit, errorMsg)

    def runAsync(using
      F:          MonadError[F, Throwable],
      dispatcher: Effect.Dispatch[F],
      logger:     Logger[F]
    ): DefaultS[Unit] =
      runAsync()

  protected final class SetStateLApplied[F[_], S](
    self: StateAccess.Write[DefaultS, DefaultA, S]
  ) {
    inline def apply[A, B](lens: Lens[S, B])(a: A)(using conv: A => B, F: Sync[F]): F[Unit] =
      self.modStateIn(lens.replace(conv(a)))
  }

  protected final class ModStateLApplied[F[_], S](
    self: StateAccess.Write[DefaultS, DefaultA, S]
  ) {
    inline def apply[A](lens: Lens[S, A])(f: A => A)(using F: Sync[F]): F[Unit] =
      self.modStateIn(lens.modify(f))
  }

  protected final class WithReusableInputsAsync[S](self: Hooks.UseStateF[DefaultS, S]) {
    inline def setState: Reusable[Reusable[S] => Reusable[DefaultA[Unit]]] =
      self.withReusableInputs.setState.map(f => s => f(s).map(_.to[DefaultA]))

    inline def modState: Reusable[Reusable[S => S] => Reusable[DefaultA[Unit]]] =
      self.withReusableInputs.modState.map(f => g => f(g).map(_.to[DefaultA]))
  }
}

object effect extends effect




© 2015 - 2025 Weber Informatics LLC | Privacy Policy