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

spire.algebra.partial.PartialAction.scala Maven / Gradle / Ivy

package spire.algebra
package partial

import spire.util.Opt

/**
  * A left partial action of a semigroupoid `G` on `P` is the implementation of
  * a method `partialActl(g, p)`, or `g ?|+|> p` returning `Opt[P]`, such that:
  *
  * 1. for all `g`, `h` in `G`, `p` in `P` such that `g |+|? h` and `h ?|+|> p` are defined,
  *
  *    `((g |+|? h).get ?|+|> p).get === (g ?|+|> (h ?|+|> p).get).get` with all operations
  *    defined.
  *
  * In addition, if `G` is a partial monoid, the following relation holds:
  *
  * 2. for all `g` in `G` and `p` in `P` such that `g ?|+|> p` is defined:
  *
  * `(g.rightId ?|+|> p).get === p`, the operation `?|+|>` being defined.
  */
trait LeftPartialAction[P, G] extends Any {
  def actlIsDefined(g: G, p: P): Boolean = partialActl(g, p).nonEmpty
  def partialActl(g: G, p: P): Opt[P]
}

object LeftPartialAction {
  @inline final def apply[P, G](implicit G: LeftPartialAction[P, G]): LeftPartialAction[P, G] = G

  implicit def fromLeftAction[P, G](implicit G: LeftAction[P, G]): LeftPartialAction[P, G] =
    new LeftPartialAction[P, G] {
      override def actlIsDefined(g: G, p: P): Boolean = true
      def partialActl(g: G, p: P): Opt[P] = Opt(G.actl(g, p))
    }
}

/**
  * A right partial action of a semigroupoid `G` on `P` is the implementation of
  * a method `partialActr(p, g)`, or `p <|+|? g` returning `Opt[P]`, such that:
  *
  * 1. for all `g`, `h` in `G`, `p` in `P` such that `g |+|? h` and `p <|+|? g` are defined,
  *
  *    `(p <|+|? (g |+|? h).get).get === ((p <|+|? g).get |+|? h).get`,
  *    and all operations are defined.
  *
  * In addition, if `G` is a partial monoid, the following relation holds:
  *
  * 2. for all `g` in `G` and `p` in `P` such that `p <|+|? g` is defined:
  *
  * `(p <|+|? g.leftId).get === p`, the operation `<|+|?` being defined.
  */

trait RightPartialAction[P, G] extends Any {
  def actrIsDefined(p: P, g: G): Boolean = partialActr(p, g).nonEmpty
  def partialActr(p: P, g: G): Opt[P]
}

object RightPartialAction {
  @inline final def apply[P, G](implicit G: RightPartialAction[P, G]): RightPartialAction[P, G] = G

  implicit def fromRightAction[P, G](implicit G: RightAction[P, G]): RightPartialAction[P, G] =
    new RightPartialAction[P, G] {
      override def actrIsDefined(p: P, g: G): Boolean = true
      def partialActr(p: P, g: G): Opt[P] = Opt(G.actr(p, g))
    }
}

/**
  * A partial action is the combination of left and right partial actions, providing:
  *
  * - a method `partialActl(g, p)`, or `g ?|+|> p` returning `Opt[P]`, such that:
  *
  * 1. for all `g`, `h` in `G`, `p` in `P` such that `g |+|? h` and `h ?|+|> p` are defined,
  *
  *    `((g |+|? h).get ?|+|> p).get === (g ?|+|> (h ?|+|> p).get).get` with all operations
  *    defined.
  *
  *  - a method `partialActr(p, g)`, or `p <|+|? g` returning `Opt[P]`, such that:
  *
  * 2. for all `g`, `h` in `G`, `p` in `P` such that `g |+|? h` and `p <|+|? g` are defined,
  *
  *    `(p <|+|? (g |+|? h).get).get === ((p <|+|? g).get |+|? h).get`,
  *    and all operations are defined.
  *
  * In addition, if `G` is a groupoid, the following relations holds:
  *
  * 3. for all `g` in `G` and `p` in `P` such that `g ?|+|> p` is defined:
  *
  * `(g.rightId ?|+|> p).get === p`, the operation `?|+|>` being defined.
  *
  * 4. for all `g` in `G` and `p` in `P` such that `p <|+|? g` is defined:
  *
  * `(p <|+|? g.leftId).get === p`, the operation `<|+|?` being defined.
  *
  * 5. for all `g` in `G` and `p` in `P` such that `g ?|+|> p` is defined:
  *
  * `(g ?|+|> p).get === (p <|+|? g.inverse).get`
  *
  */
trait PartialAction[P, G] extends Any
    with LeftPartialAction[P, G] with RightPartialAction[P, G]

object PartialAction {
  @inline final def apply[P, G](implicit G: PartialAction[P, G]): PartialAction[P, G] = G

  implicit def fromAction[P, G](implicit G: Action[P, G]): PartialAction[P, G] =
    new PartialAction[P, G] {
      override def actlIsDefined(g: G, p: P): Boolean = true
      def partialActl(g: G, p: P): Opt[P] = Opt(G.actl(g, p))
      override def actrIsDefined(p: P, g: G): Boolean = true
      def partialActr(p: P, g: G): Opt[P] = Opt(G.actr(p, g))
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy