UNPKG

@simplux/core

Version:

The core package of simplux. Contains everything to manage your application state in a simple way.

91 lines (90 loc) 15.9 kB
/** * Helper symbol used for identifying simplux selector objects. * * @public */ // should really be a symbol, but as of TypeScript 4.1 there is a bug // that causes the symbol to not be properly re-exported in type // definitions when spreading a select object onto an export, which can // cause issues with composite builds export const SIMPLUX_SELECTOR = '[SIMPLUX_SELECTOR]'; /** * Create new selectors for the module. A selector is a function * that takes the module state and optionally additional parameters * and returns some selected value. * * The selector must be a pure function. Its result is memoized * for the latest state and parameters. * * @param simpluxModule - the module to create selectors for * @param selectorDefinitions - the selectors to create * * @returns an object that contains a function for each provided * selector * * @public */ export function createSelectors(simpluxModule, selectorDefinitions) { const module = simpluxModule; const internals = module.$simplux; const resolvedSelectors = Object.keys(selectorDefinitions).reduce((acc, selectorName) => { const definition = selectorDefinitions[selectorName]; const memoizedDefinition = memoize(definition); const selectorId = (internals.lastSelectorId || 0) + 1; internals.lastSelectorId = selectorId; const namedSelector = nameFunction(selectorName, (...args) => { var _a; const mock = (_a = internals.selectorMocks) === null || _a === void 0 ? void 0 : _a[selectorId]; if (mock) { return mock(...args); } return memoizedDefinition(internals.getState(), ...args); }); acc[selectorName] = namedSelector; const extras = namedSelector; extras.withState = memoizedDefinition; extras.selectorId = selectorId; extras.selectorName = selectorName; extras.owningModule = module; extras[SIMPLUX_SELECTOR] = ''; return acc; }, {}); return resolvedSelectors; // this helper function allows creating a function with a dynamic name (only works with ES6+) function nameFunction(name, body) { return { [name](...args) { return body(...args); }, }[name]; } function memoize(fn) { let memoizedArgs; let memoizedResult; function memoizedFunction(...args) { const memoizedResultNeedsToBeRefreshed = !memoizedArgs || memoizedArgs.length !== args.length || !memoizedArgs.every((a, idx) => a === args[idx]); if (memoizedResultNeedsToBeRefreshed) { memoizedResult = fn(...args); memoizedArgs = args; } return memoizedResult; } return memoizedFunction; } } /** * Checks if an object is a simplux selector. * * @param object - the object to check * * @returns true if the object is a simplux selector * * @internal */ export function _isSimpluxSelector(object) { var _a; return ((_a = object) === null || _a === void 0 ? void 0 : _a[SIMPLUX_SELECTOR]) === ''; } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"selectors.js","sources":["@simplux/core/src/selectors.ts"],"sourcesContent":["import type { SimpluxModule, SimpluxModuleMarker } from './module.js'\r\nimport type { Immutable } from './types.js'\r\n\r\n/**\r\n * Helper symbol used for identifying simplux selector objects.\r\n *\r\n * @public\r\n */\r\n// should really be a symbol, but as of TypeScript 4.1 there is a bug\r\n// that causes the symbol to not be properly re-exported in type\r\n// definitions when spreading a select object onto an export, which can\r\n// cause issues with composite builds\r\nexport const SIMPLUX_SELECTOR = '[SIMPLUX_SELECTOR]'\r\n\r\n/**\r\n * A function to turn into a selector.\r\n *\r\n * @public\r\n */\r\nexport type SelectorDefinition<TState, TReturn> = (\r\n  state: Immutable<TState>,\r\n  ...args: any\r\n) => TReturn\r\n\r\n/**\r\n * The functions to turn into selectors.\r\n *\r\n * @public\r\n */\r\nexport interface SelectorDefinitions<TState> {\r\n  [name: string]: SelectorDefinition<TState, any>\r\n}\r\n\r\n/**\r\n * Interface for efficiently identifying simplux selector objects at compile time.\r\n *\r\n * @public\r\n */\r\nexport interface SimpluxSelectorMarker<TState, TArgs extends any[], TReturn> {\r\n  /**\r\n   * A symbol that allows efficient compile-time and run-time identification\r\n   * of simplux selector objects.\r\n   *\r\n   * This property will have an `undefined` value at runtime.\r\n   *\r\n   * @public\r\n   */\r\n  readonly [SIMPLUX_SELECTOR]: [TState, TArgs, TReturn]\r\n}\r\n\r\n/**\r\n * A simplux selector is a function that projects a module's state to some value.\r\n *\r\n * @public\r\n */\r\nexport interface SimpluxSelector<TState, TArgs extends any[], TReturn>\r\n  extends SimpluxSelectorMarker<TState, TArgs, TReturn> {\r\n  // this signature does not have a TSDoc comment on purpose to allow\r\n  // consumers to define their own docs for their selectors (which would\r\n  // be overwritten if this signature had a TSDoc comment)\r\n  (...args: TArgs): TReturn\r\n\r\n  /**\r\n   * The ID that identifies this selector uniquely on the owning module.\r\n   *\r\n   * @internal\r\n   */\r\n  readonly selectorId: number\r\n\r\n  /**\r\n   * The name of this selector.\r\n   */\r\n  readonly selectorName: string\r\n\r\n  /**\r\n   * By default a selector is evaluated with the module's latest state.\r\n   * This function evaluates the selector with the given state instead.\r\n   *\r\n   * @param state - the state to use when evaluating the selector\r\n   * @param args - the arguments for the selector\r\n   *\r\n   * @returns the selected value\r\n   */\r\n  readonly withState: (state: Immutable<TState>, ...args: TArgs) => TReturn\r\n\r\n  /**\r\n   * The module this selector belongs to.\r\n   *\r\n   * @internal\r\n   */\r\n  readonly owningModule: SimpluxModule<TState>\r\n}\r\n\r\ntype Mutable<T> = { -readonly [prop in keyof T]: T[prop] }\r\n\r\n/**\r\n * A simplux selector is a function that projects a module's state to some value.\r\n * {@link SimpluxSelector}\r\n *\r\n * @public\r\n */\r\nexport type ResolvedSelector<\r\n  TState,\r\n  TSelectorDefinition extends SelectorDefinition<\r\n    TState,\r\n    ReturnType<TSelectorDefinition>\r\n  >\r\n> = TSelectorDefinition extends (\r\n  state: Immutable<TState>,\r\n  ...args: infer TArgs\r\n) => infer TReturn\r\n  ? SimpluxSelector<TState, TArgs, TReturn>\r\n  : never\r\n\r\n/**\r\n * A collection of simplux selectors.\r\n *\r\n * @public\r\n */\r\nexport type SimpluxSelectors<\r\n  TState,\r\n  TSelectorDefinitions extends SelectorDefinitions<TState>\r\n> = {\r\n  readonly [name in keyof TSelectorDefinitions]: ResolvedSelector<\r\n    TState,\r\n    TSelectorDefinitions[name]\r\n  >\r\n}\r\n\r\n/**\r\n * Create new selectors for the module. A selector is a function\r\n * that takes the module state and optionally additional parameters\r\n * and returns some selected value.\r\n *\r\n * The selector must be a pure function. Its result is memoized\r\n * for the latest state and parameters.\r\n *\r\n * @param simpluxModule - the module to create selectors for\r\n * @param selectorDefinitions - the selectors to create\r\n *\r\n * @returns an object that contains a function for each provided\r\n * selector\r\n *\r\n * @public\r\n */\r\nexport function createSelectors<\r\n  TState,\r\n  TSelectorDefinitions extends SelectorDefinitions<TState>\r\n>(\r\n  simpluxModule: SimpluxModuleMarker<TState>,\r\n  selectorDefinitions: TSelectorDefinitions,\r\n): SimpluxSelectors<TState, TSelectorDefinitions> {\r\n  const module = simpluxModule as SimpluxModule<TState>\r\n  const internals = module.$simplux\r\n\r\n  const resolvedSelectors = Object.keys(selectorDefinitions).reduce(\r\n    (acc, selectorName: keyof TSelectorDefinitions) => {\r\n      const definition = selectorDefinitions[selectorName as string]!\r\n      const memoizedDefinition = memoize(definition)\r\n\r\n      const selectorId = (internals.lastSelectorId || 0) + 1\r\n      internals.lastSelectorId = selectorId\r\n\r\n      const namedSelector = nameFunction(\r\n        selectorName as string,\r\n        (...args: any[]) => {\r\n          const mock = internals.selectorMocks?.[selectorId]\r\n          if (mock) {\r\n            return mock(...args)\r\n          }\r\n\r\n          return memoizedDefinition(internals.getState(), ...args)\r\n        },\r\n      ) as ResolvedSelector<TState, TSelectorDefinitions[typeof selectorName]>\r\n\r\n      acc[selectorName] = namedSelector\r\n\r\n      const extras = namedSelector as Mutable<typeof namedSelector>\r\n\r\n      extras.withState = memoizedDefinition\r\n\r\n      extras.selectorId = selectorId\r\n      extras.selectorName = selectorName as string\r\n      extras.owningModule = module\r\n      extras[SIMPLUX_SELECTOR] = '' as any\r\n\r\n      return acc\r\n    },\r\n    {} as Mutable<SimpluxSelectors<TState, TSelectorDefinitions>>,\r\n  )\r\n\r\n  return resolvedSelectors\r\n\r\n  // this helper function allows creating a function with a dynamic name (only works with ES6+)\r\n  function nameFunction<T extends (...args: any[]) => any>(\r\n    name: string,\r\n    body: T,\r\n  ): T {\r\n    return {\r\n      [name](...args: any[]) {\r\n        return body(...args)\r\n      },\r\n    }[name] as T\r\n  }\r\n\r\n  function memoize<TFunction extends Function>(fn: TFunction): TFunction {\r\n    let memoizedArgs: any[] | undefined\r\n    let memoizedResult: any\r\n\r\n    function memoizedFunction(...args: any[]) {\r\n      const memoizedResultNeedsToBeRefreshed =\r\n        !memoizedArgs ||\r\n        memoizedArgs.length !== args.length ||\r\n        !memoizedArgs.every((a, idx) => a === args[idx])\r\n\r\n      if (memoizedResultNeedsToBeRefreshed) {\r\n        memoizedResult = fn(...args)\r\n        memoizedArgs = args\r\n      }\r\n\r\n      return memoizedResult\r\n    }\r\n\r\n    return (memoizedFunction as unknown) as TFunction\r\n  }\r\n}\r\n\r\n/**\r\n * Checks if an object is a simplux selector.\r\n *\r\n * @param object - the object to check\r\n *\r\n * @returns true if the object is a simplux selector\r\n *\r\n * @internal\r\n */\r\nexport function _isSimpluxSelector<\r\n  TState,\r\n  TArgs extends any[],\r\n  TReturn,\r\n  TOther\r\n>(\r\n  object: SimpluxSelectorMarker<TState, TArgs, TReturn> | TOther,\r\n): object is SimpluxSelector<TState, TArgs, TReturn> {\r\n  return (object as any)?.[SIMPLUX_SELECTOR] === ''\r\n}\r\n"],"names":[],"mappings":"AAGA;;;;GAIG;AACH,qEAAqE;AACrE,gEAAgE;AAChE,uEAAuE;AACvE,qCAAqC;AACrC,MAAM,CAAC,MAAM,gBAAgB,GAAG,oBAAoB,CAAA;AAqHpD;;;;;;;;;;;;;;;GAeG;AACH,MAAM,UAAU,eAAe,CAI7B,aAA0C,EAC1C,mBAAyC;IAEzC,MAAM,MAAM,GAAG,aAAsC,CAAA;IACrD,MAAM,SAAS,GAAG,MAAM,CAAC,QAAQ,CAAA;IAEjC,MAAM,iBAAiB,GAAG,MAAM,CAAC,IAAI,CAAC,mBAAmB,CAAC,CAAC,MAAM,CAC/D,CAAC,GAAG,EAAE,YAAwC,EAAE,EAAE;QAChD,MAAM,UAAU,GAAG,mBAAmB,CAAC,YAAsB,CAAE,CAAA;QAC/D,MAAM,kBAAkB,GAAG,OAAO,CAAC,UAAU,CAAC,CAAA;QAE9C,MAAM,UAAU,GAAG,CAAC,SAAS,CAAC,cAAc,IAAI,CAAC,CAAC,GAAG,CAAC,CAAA;QACtD,SAAS,CAAC,cAAc,GAAG,UAAU,CAAA;QAErC,MAAM,aAAa,GAAG,YAAY,CAChC,YAAsB,EACtB,CAAC,GAAG,IAAW,EAAE,EAAE;;YACjB,MAAM,IAAI,GAAG,MAAA,SAAS,CAAC,aAAa,0CAAG,UAAU,CAAC,CAAA;YAClD,IAAI,IAAI,EAAE;gBACR,OAAO,IAAI,CAAC,GAAG,IAAI,CAAC,CAAA;aACrB;YAED,OAAO,kBAAkB,CAAC,SAAS,CAAC,QAAQ,EAAE,EAAE,GAAG,IAAI,CAAC,CAAA;QAC1D,CAAC,CACqE,CAAA;QAExE,GAAG,CAAC,YAAY,CAAC,GAAG,aAAa,CAAA;QAEjC,MAAM,MAAM,GAAG,aAA8C,CAAA;QAE7D,MAAM,CAAC,SAAS,GAAG,kBAAkB,CAAA;QAErC,MAAM,CAAC,UAAU,GAAG,UAAU,CAAA;QAC9B,MAAM,CAAC,YAAY,GAAG,YAAsB,CAAA;QAC5C,MAAM,CAAC,YAAY,GAAG,MAAM,CAAA;QAC5B,MAAM,CAAC,gBAAgB,CAAC,GAAG,EAAS,CAAA;QAEpC,OAAO,GAAG,CAAA;IACZ,CAAC,EACD,EAA6D,CAC9D,CAAA;IAED,OAAO,iBAAiB,CAAA;IAExB,6FAA6F;IAC7F,SAAS,YAAY,CACnB,IAAY,EACZ,IAAO;QAEP,OAAO;YACL,CAAC,IAAI,CAAC,CAAC,GAAG,IAAW;gBACnB,OAAO,IAAI,CAAC,GAAG,IAAI,CAAC,CAAA;YACtB,CAAC;SACF,CAAC,IAAI,CAAM,CAAA;IACd,CAAC;IAED,SAAS,OAAO,CAA6B,EAAa;QACxD,IAAI,YAA+B,CAAA;QACnC,IAAI,cAAmB,CAAA;QAEvB,SAAS,gBAAgB,CAAC,GAAG,IAAW;YACtC,MAAM,gCAAgC,GACpC,CAAC,YAAY;gBACb,YAAY,CAAC,MAAM,KAAK,IAAI,CAAC,MAAM;gBACnC,CAAC,YAAY,CAAC,KAAK,CAAC,CAAC,CAAC,EAAE,GAAG,EAAE,EAAE,CAAC,CAAC,KAAK,IAAI,CAAC,GAAG,CAAC,CAAC,CAAA;YAElD,IAAI,gCAAgC,EAAE;gBACpC,cAAc,GAAG,EAAE,CAAC,GAAG,IAAI,CAAC,CAAA;gBAC5B,YAAY,GAAG,IAAI,CAAA;aACpB;YAED,OAAO,cAAc,CAAA;QACvB,CAAC;QAED,OAAQ,gBAAyC,CAAA;IACnD,CAAC;AACH,CAAC;AAED;;;;;;;;GAQG;AACH,MAAM,UAAU,kBAAkB,CAMhC,MAA8D;;IAE9D,OAAO,CAAA,MAAC,MAAc,0CAAG,gBAAgB,CAAC,MAAK,EAAE,CAAA;AACnD,CAAC"}