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

com.raquo.laminar.keys.EventProcessor.scala Maven / Gradle / Ivy

package com.raquo.laminar.keys

import com.raquo.airstream.core.{EventStream, Observable, Signal, Sink}
import com.raquo.airstream.flatten.SwitchingStrategy
import com.raquo.airstream.status.Status
import com.raquo.laminar.DomApi
import com.raquo.laminar.api.UnitArrowsFeature
import com.raquo.laminar.modifiers.EventListener
import org.scalajs.dom

/**
  * This class represents a sequence of transformations that are applied to events feeding into an [[EventListener]]
  * EventProcessor-s are immutable, so can be reused by multiple setters.
  *
  * Example syntax: `input(onChange().preventDefault.mapTo(true) --> myBooleanWriteBus)`
  *
  * Note: Params are protected to avoid confusing autocomplete options (e.g. "useCapture")
  *
  * @param shouldUseCapture (false means using bubble mode)
  *                         See `useCapture` docs here: https://developer.mozilla.org/en-US/docs/Web/API/EventTarget/addEventListener
  *
  * @param processor        Processes incoming events before they're passed to the next processor or to the listening EventBus.
  *                         Returns an Option of the processed value. If None, the value should not passed down the chain.
  */
class EventProcessor[Ev <: dom.Event, V](
  protected val eventProp: EventProp[Ev],
  protected val shouldUseCapture: Boolean = false,
  protected val shouldBePassive: Boolean = false,
  protected val processor: Ev => Option[V]
) {

  @inline def -->(sink: Sink[V]): EventListener[Ev, V] = {
    this --> (sink.toObserver.onNext(_))
  }

  @inline def -->(onNext: V => Unit): EventListener[Ev, V] = {
    new EventListener[Ev, V](this, onNext)
  }

  @inline def -->(onNext: => Unit)(implicit evidence: UnitArrowsFeature): EventListener[Ev, V] = {
    new EventListener[Ev, V](this, _ => onNext)
  }

  /** Use capture mode
    *
    * Note that unlike `preventDefault` config which applies to individual events,
    * useCapture is used to install the listener onto the DOM node in the first place.
    *
    * See `useCapture` docs here: https://developer.mozilla.org/en-US/docs/Web/API/EventTarget/addEventListener#usecapture
    */
  def useCapture: EventProcessor[Ev, V] = {
    new EventProcessor(eventProp, shouldUseCapture = true, shouldBePassive = shouldBePassive, processor = processor)
  }

  /** Use standard bubble propagation mode.
    * You don't need to call this unless you set `useCapture` previously, and want to revert to bubbling.
    *
    * See `useCapture` docs here: https://developer.mozilla.org/en-US/docs/Web/API/EventTarget/addEventListener#usecapture
    */
  def useBubbleMode: EventProcessor[Ev, V] = {
    new EventProcessor(eventProp, shouldUseCapture = false, shouldBePassive = shouldBePassive, processor = processor)
  }

  /** Use a passive event listener
   *
   * Note that unlike `preventDefault` config which applies to individual events,
   * `passive` is used to install the listener onto the DOM node in the first place.
   *
   * See `passive` docs here: https://developer.mozilla.org/en-US/docs/Web/API/EventTarget/addEventListener#passive
   */
  def passive: EventProcessor[Ev, V] = {
    new EventProcessor(eventProp, shouldUseCapture = shouldUseCapture, shouldBePassive = true, processor = processor)
  }

  /** Use a standard non-passive listener.
   * You don't need to call this unless you set `passive` previously, and want to revert to non-passive.
   *
   * See `passive` docs here: https://developer.mozilla.org/en-US/docs/Web/API/EventTarget/addEventListener#passive
   */
  def nonPassive: EventProcessor[Ev, V] = {
    new EventProcessor(eventProp, shouldUseCapture = shouldUseCapture, shouldBePassive = false, processor = processor)
  }

  /** Prevent default browser action for the given event (e.g. following the link when it is clicked)
    * https://developer.mozilla.org/en-US/docs/Web/API/Event/preventDefault
    *
    * Note: this is just a standard processor, so it will be fired in whatever order you have applied it.
    * So for example, you can [[filter]] events before applying this, preventing default action only for certain events.
    *
    * Example: `input(onKeyUp().filter(ev => ev.keyCode == KeyCode.Tab).preventDefault --> tabKeyUpBus)`
    */
  def preventDefault: EventProcessor[Ev, V] = {
    withNewProcessor { ev =>
      processor(ev).map { value =>
        ev.preventDefault()
        value
      }
    }
  }

  /** Propagation here refers to DOM Event bubbling or capture propagation.
    * https://developer.mozilla.org/en-US/docs/Web/API/Event/stopPropagation
    *
    * Note: this is just a standard processor, so it will be fired in whatever order you have applied it.
    * So for example, you can [[filter]] events before applying this, propagation will be stopped only for certain events.
    *
    * Example: `div(onClick.filter(isGoodClick).stopPropagation --> goodClickBus)`
    */
  def stopPropagation: EventProcessor[Ev, V] = {
    withNewProcessor { ev =>
      processor(ev).map { value =>
        ev.stopPropagation()
        value
      }
    }
  }

  /** This method prevents other listeners of the same event from being called.
    * If several listeners are attached to the same element for the same event type,
    * they are called in the order in which they were added. If stopImmediatePropagation()
    * is invoked during one such call, no remaining listeners will be called.
    *
    * MDN https://developer.mozilla.org/en-US/docs/Web/API/Event/stopImmediatePropagation
    *
    * Note: this is just a standard processor, so it will be fired in whatever order you have applied it.
    * So for example, you can [[filter]] events before applying this, propagation will be stopped only for certain events.
    *
    * Example: `div(onClick.filter(isGoodClick).stopImmediatePropagation --> goodClickBus)`
    */
  def stopImmediatePropagation: EventProcessor[Ev, V] = {
    withNewProcessor { ev =>
      processor(ev).map { value =>
        ev.stopImmediatePropagation()
        value
      }
    }
  }

  /** Values that do not pass the test will not propagate down the chain and into the emitter. */
  def filter(passes: V => Boolean): EventProcessor[Ev, V] = {
    withNewProcessor(ev => processor(ev).filter(passes))
  }

  /** Values that pass the test will not propagate down the chain and into the emitter. */
  def filterNot(skip: V => Boolean): EventProcessor[Ev, V] = filter(!skip(_))

  /** Filter events by `event.target`
    *
    * For example, discard clicks on child  links with something like:
    *
    *     div(
    *       onClick.filterByTarget {
    *         case dom.html.Anchor => false
    *         case _ => true
    *       } --> observer,
    *       "A bunch of clickable stuff",
    *       a("Some link", href("..."))
    *     )
    * */
  def filterByTarget(passes: dom.EventTarget => Boolean): EventProcessor[Ev, V] = {
    withNewProcessor(ev => if (passes(ev.target)) processor(ev) else None)
  }

  def collect[V2](pf: PartialFunction[V, V2]): EventProcessor[Ev, V2] = {
    withNewProcessor(ev => processor(ev).collect(pf))
  }

  def map[V2](project: V => V2): EventProcessor[Ev, V2] = {
    withNewProcessor(ev => processor(ev).map(project))
  }

  /** Same as `map(_ => value)`
    *
    * Note: `value` will be re-evaluated every time the event is fired
    */
  def mapTo[V2](value: => V2): EventProcessor[Ev, V2] = {
    withNewProcessor(ev => processor(ev).map(_ => value))
  }

  /** Like mapTo, but with strict evaluation of the value */
  def mapToStrict[V2](value: V2): EventProcessor[Ev, V2] = {
    withNewProcessor(ev => processor(ev).map(_ => value))
  }

  def mapToUnit: EventProcessor[Ev, Unit] = mapToStrict(())

  /** Get the original event. You might want to call this in a chain, after some other processing. */
  def mapToEvent: EventProcessor[Ev, Ev] = {
    withNewProcessor(ev => processor(ev).map(_ => ev))
  }

  /** Get the value of `event.target.value` */
  def mapToValue: EventProcessor[Ev, String] = {
    withNewProcessor { ev =>
      processor(ev).map { _ =>
        // @TODO[Warn] Throw or console.log a warning here if using on the wrong element type
        DomApi.getValue(ev.target.asInstanceOf[dom.Element]).getOrElse("")
      }
    }
  }

  /** Get the value of `event.target.checked` */
  def mapToChecked: EventProcessor[Ev, Boolean] = {
    withNewProcessor { ev =>
      processor(ev).map { _ =>
        // @TODO[Warn] Throw or console.log a warning here if using on the wrong element type
        DomApi.getChecked(ev.target.asInstanceOf[dom.Element]).getOrElse(false)
      }
    }
  }

  /** Get the value of `event.target.files`
   *
   * @see https://developer.mozilla.org/en-US/docs/Web/API/File_API/Using_files_from_web_applications
   */
  def mapToFiles: EventProcessor[Ev, List[dom.File]] = {
    withNewProcessor { ev =>
      processor(ev).map { _ =>
        DomApi.getFiles(ev.target.asInstanceOf[dom.Element]).getOrElse(Nil)
      }
    }
  }

  /** Unsafe – Get the value of `event.target`, cast to a certain element type
   *
   * You should generally avoid this in favor of other helpers like
   * `mapToValue` or `inContext { thisNode => ... }`.
   */
  def mapToTargetAs[Ref <: dom.EventTarget]: EventProcessor[Ev, Ref] = {
    withNewProcessor { ev =>
      processor(ev).map { _ =>
        ev.target.asInstanceOf[Ref]
      }
    }
  }

  /** Execute a side effecting callback every time the event emits.
    *
    * Note: Do not provide a callback that returns a LAZY value such as EventStream,
    * it will not be started. Use `compose` or `flatMap` methods for that kind of thing.
    *
    * Note: You still need to bind this event processor to an observer with
    * the `-->` method. Generally you should put "side effects" in the observer, but
    * strictly speaking this is not required, and you can use `--> Observer.empty`.
    *
    * Note: This method is called `tapEach` for consistency with Scala collections.
    */
  def tapEach[U](f: V => U): EventProcessor[Ev, V] = map { v => f(v); v }

  /** Like [[tapEach]] but provides the original event instead of the processed value. */
  def tapEachEvent[U](f: Ev => U): EventProcessor[Ev, V] = {
    withNewProcessor { ev => f(ev); processor(ev) }
  }

  /** Similar to the Airstream `compose` operator.
    *
    * Use this when you need to apply stream operators on this element's events, e.g.:
    *
    *     div(onScroll.compose(_.throttle(100)) --> observer)
    *
    *     a(onClick.preventDefault.compose(_.delay(100)) --> observer)
    *
    * Note: can also use with more compact `apply` alias:
    *
    *     div(onScroll(_.throttle(100)) --> observer)
    *
    *     a(onClick.preventDefault(_.delay(100)) --> observer)
    *
    * Note: This method is not chainable. Put all the operations you need inside the `operator` callback.
    *
    */
  def compose[Out](
    operator: EventStream[V] => Observable[Out]
  ): LockedEventKey[Ev, V, Out] = {
    new LockedEventKey(this, operator)
  }

  /** Alias for [[compose]] */
  @inline def apply[Out](
    composer: EventStream[V] => Observable[Out]
  ): LockedEventKey[Ev, V, Out] = {
    compose(composer)
  }

  /** Similar to the Airstream `flatMap` operator.
    *
    * Use this when you want to create a new stream or signal on every event, e.g.:
    *
    * {{{
    * button(onClick.preventDefault.flatMap(_ => makeAjaxRequest()) --> observer)
    * }}}
    *
    * #TODO[IDE] IntelliJ (2022.3.2) shows false errors when using this flatMap implementation,
    *  at least with Scala 2, making it annoying. Use flatMapStream or flatMapSignal to get around that.
    *  https://youtrack.jetbrains.com/issue/SCL-21836/Kind-context-bound-callback-argument-causes-false-positive-type-mismatch-error
    *
    * Note: This method is not chainable. Put all the operations you need inside the `operator` callback,
    *       or use the `compose` method instead for more flexibility
    */
  def flatMap[Out, Obs[_] <: Observable[_]](
    operator: V => Obs[Out]
  )(
    implicit strategy: SwitchingStrategy[EventStream, Obs, Observable]
  ): LockedEventKey[Ev, V, Out] = {
    new LockedEventKey[Ev, V, Out](
      this,
      eventStream => eventStream.flatMapSwitch(operator)(strategy)
    )
  }

  /** Equivalent to `flatMap(_ => observable)`
    *
    * Note: `observable` will be re-evaluated every time the event is fired.
    */
  @inline def flatMapTo[Out, Obs[_] <: Observable[_]](
    observable: => Obs[Out]
  )(
    implicit strategy: SwitchingStrategy[EventStream, Obs, Observable]
  ): LockedEventKey[Ev, V, Out] = {
    flatMap(_ => observable)(strategy)
  }

  /** Similar to `flatMap`, but restricted to streams only. */
  @inline def flatMapStream[Out](
    operator: V => EventStream[Out]
  )(
    implicit strategy: SwitchingStrategy[EventStream, EventStream, Observable]
  ): LockedEventKey[Ev, V, Out] = {
    flatMap(operator)(strategy)
  }

  /** Similar to `flatMap`, but restricted to signals only. */
  @inline def flatMapSignal[Out](
    operator: V => Signal[Out]
  )(
    implicit strategy: SwitchingStrategy[EventStream, Signal, Observable]
  ): LockedEventKey[Ev, V, Out] = {
    flatMap(operator)(strategy)
  }

  /** Similar to Airstream `flatMapWithStatus` operator.
    *
    * Use this when you want to flatMapSwitch and get a status indicating
    * whether the input has been processed by the inner stream, e.g.:
    *
    * {{{
    * button(onClick.flatMapWithStatus(ev => AjaxStream.get(ev, ...)) --> observer
    * }}}
    */
  def flatMapWithStatus[Out](
    operator: V => EventStream[Out]
  ): LockedEventKey[Ev, V, Status[V, Out]] = {
    new LockedEventKey[Ev, V, Status[V, Out]](
      this,
      eventStream => eventStream.flatMapWithStatus(operator)
    )
  }

  /** Similar to Airstream `flatMapWithStatus` operator.
    *
    * Use this when you want to flatMapSwitch and get a status indicating
    * whether the input has been processed by the inner stream, e.g.:
    *
    * {{{
    * button(onClick.flatMapWithStatus(AjaxStream.get(...)) --> observer
    * }}}
    */
  def flatMapWithStatus[Out](
    innerStream: => EventStream[Out]
  ): LockedEventKey[Ev, V, Status[V, Out]] = {
    new LockedEventKey[Ev, V, Status[V, Out]](
      this,
      eventStream => eventStream.flatMapWithStatus(innerStream)
    )
  }

  /** Evaluate `f` if the value was filtered out up the chain. For example:
    *
    *     onClick.filter(isRightClick).orElseEval(_.preventDefault()) --> observer
    *
    * This observer will fire only on right clicks, and for events that aren't right
    * clicks, ev.preventDefault() will be called instead.
    */
  def orElseEval(f: Ev => Unit): EventProcessor[Ev, V] = {
    withNewProcessor { ev =>
      val result = processor(ev)
      if (result.isEmpty) {
        f(ev)
      }
      result
    }
  }

  /** (originalEvent, accumulatedValue) => newAccumulatedValue
    *
    * Unlike other processors, this one will fire regardless of .filter-s up the chain.
    * Instead, if the event was filtered, project will receive None as accumulatedValue.
    * The output of project should be Some(newValue), or None if you want to filter out this event.
    */
  def mapRaw[V2](project: (Ev, Option[V]) => Option[V2]): EventProcessor[Ev, V2] = {
    withNewProcessor { ev =>
      project(ev, processor(ev))
    }
  }

  /**
   * Write a custom string into `event.target.value`.
   * You can only do this on elements that have a value property - input, textarea, select
   */
  def setValue(nextValue: String): EventProcessor[Ev, V] = {
    withNewProcessor { ev =>
      processor(ev).map { result =>
        // @TODO[Warn] Console.log a warning here if using on the wrong element type
        DomApi.setValue(ev.target.asInstanceOf[dom.Element], nextValue)
        result
      }
    }
  }

  /**
    * Write the resulting string into `event.target.value`.
    * You can only do this on elements that have a value property - input, textarea, select
    */
  def setAsValue(implicit stringEvidence: V <:< String): EventProcessor[Ev, V] = {
    withNewProcessor { ev =>
      processor(ev).map { value =>
        val nextInputValue = stringEvidence(value)
        // @TODO[Warn] Console.log a warning here if using on the wrong element type
        DomApi.setValue(ev.target.asInstanceOf[dom.Element], nextInputValue)
        value
      }
    }
  }

  /**
   * Write a custom boolean into `event.target.checked`.
   * You can only do this on checkbox or radio button elements.
   *
   * Warning: if using this, do not use preventDefault. The browser may override the value you set here.
   */
  def setChecked(nextChecked: Boolean): EventProcessor[Ev, V] = {
    withNewProcessor { ev =>
      processor(ev).map { result =>
        // @TODO[Warn] Console.log a warning here if using on the wrong element type
        DomApi.setChecked(ev.target.asInstanceOf[dom.Element], nextChecked)
        result
      }
    }
  }

  /** Write the resulting boolean into `event.target.checked`.
    * You can only do this on checkbox or radio button elements.
    *
    * Warning: if using this, do not use preventDefault. The browser may override the value you set here.
    */
  def setAsChecked(implicit boolEvidence: V <:< Boolean): EventProcessor[Ev, V] = {
    withNewProcessor { ev =>
      processor(ev).map { value =>
        val nextChecked = boolEvidence(value)
        // @TODO[Warn] Console.log a warning here if using on the wrong element type
        DomApi.setChecked(ev.target.asInstanceOf[dom.Element], nextChecked)
        value
      }
    }
  }

  private def withNewProcessor[V2](newProcessor: Ev => Option[V2]): EventProcessor[Ev, V2] = {
    new EventProcessor[Ev, V2](eventProp, shouldUseCapture, shouldBePassive, newProcessor)
  }
}

object EventProcessor {

  def empty[Ev <: dom.Event](eventProp: EventProp[Ev], shouldUseCapture: Boolean = false, shouldBePassive: Boolean = false): EventProcessor[Ev, Ev] = {
    new EventProcessor(eventProp, shouldUseCapture, shouldBePassive, Some(_))
  }

  // These methods are only exposed publicly via companion object
  // to avoid polluting autocomplete when chaining EventProcessor-s

  @inline def eventProp[Ev <: dom.Event](prop: EventProcessor[Ev, _]): EventProp[Ev] = prop.eventProp

  @inline def shouldUseCapture(prop: EventProcessor[_, _]): Boolean = prop.shouldUseCapture

  @inline def shouldBePassive(prop: EventProcessor[_, _]): Boolean = prop.shouldBePassive

  @inline def processor[Ev <: dom.Event, Out](prop: EventProcessor[Ev, Out]): Ev => Option[Out] = prop.processor
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy