UNPKG

@simplux/core

Version:

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

114 lines (113 loc) 22.3 kB
import { createImmerReducer } from './immer.js'; import { createModuleReducer } from './reducer.js'; import { createSelectors } from './selectors.js'; import { simpluxStore } from './store.js'; /** * Helper symbol used for identifying simplux module 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 module object onto an export, which can // cause issues with composite builds export const SIMPLUX_MODULE = '[SIMPLUX_MODULE]'; /** * This is part of the simplux internal API and should not be accessed * except by simplux extensions. * * @internal */ export function createModule(store, config) { if (process.env.NODE_ENV !== 'production') { if (!config.name) { throw new Error('you must provide a module name!'); } } const { getState, dispatch, subscribe, setReducer } = store; const internals = { name: config.name, mutations: {}, dispatch, getReducer: () => simpluxStore.getReducer(config.name), getState: getModuleState, }; function getModuleState() { return internals.mockStateValue || getState()[config.name]; } function setModuleState(state) { dispatch({ type: `@simplux/${config.name}/setState`, state, }); } let unsubscribeFromStore; let latestModuleState = config.initialState; const handlers = []; const defaultStateChangeHandlerOptions = { shouldSkipInitialInvocation: false, }; const subscribeToStateChanges = (handler, options = {}) => { const fullOptions = Object.assign(Object.assign({}, defaultStateChangeHandlerOptions), options); handlers.push(handler); if (handlers.length === 1) { latestModuleState = getModuleState(); unsubscribeFromStore = subscribe(() => { const moduleState = getModuleState(); if (moduleState !== latestModuleState) { const previousModuleState = latestModuleState; latestModuleState = moduleState; for (const handler of handlers) { handler(moduleState, previousModuleState); } } }); } if (!fullOptions.shouldSkipInitialInvocation) { handler(latestModuleState, latestModuleState); } const unsubscribe = () => { const idx = handlers.indexOf(handler); if (idx >= 0) { handlers.splice(idx, 1); } if (handlers.length === 0 && unsubscribeFromStore) { unsubscribeFromStore(); unsubscribeFromStore = undefined; } }; return { unsubscribe, handler: handler, }; }; const moduleReducer = createModuleReducer(config.name, config.initialState, internals.mutations); const moduleReducerWithImmerSupport = createImmerReducer(moduleReducer); setReducer(config.name, moduleReducerWithImmerSupport); const result = { state: undefined, setState: setModuleState, subscribeToStateChanges, $simplux: internals, [SIMPLUX_MODULE]: '', }; const selectors = createSelectors(result, { state: (s) => s, }); result.state = selectors.state; return result; } /** * Checks if an object is a simplux module. * * @param object - the object to check * * @returns true if the object is a simplux module * * @internal */ export function _isSimpluxModule(object) { var _a; return ((_a = object) === null || _a === void 0 ? void 0 : _a[SIMPLUX_MODULE]) === ''; } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"module.js","sources":["@simplux/core/src/module.ts"],"sourcesContent":["import type { AnyAction, Reducer } from 'redux'\r\nimport { createImmerReducer } from './immer.js'\r\nimport type { MutationDefinitions } from './mutations.js'\r\nimport { createModuleReducer } from './reducer.js'\r\nimport { createSelectors, SimpluxSelector } from './selectors.js'\r\nimport { simpluxStore, _SimpluxStore } from './store.js'\r\nimport type { Immutable } from './types.js'\r\n\r\n/**\r\n * Helper symbol used for identifying simplux module 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 module object onto an export, which can\r\n// cause issues with composite builds\r\nexport const SIMPLUX_MODULE = '[SIMPLUX_MODULE]'\r\n\r\n/**\r\n * Configuration object for creating simplux modules.\r\n *\r\n * @public\r\n */\r\nexport interface SimpluxModuleConfig<TState> {\r\n  readonly name: string\r\n  readonly initialState: TState\r\n}\r\n\r\n/**\r\n * A function that can be registered in a module to be notified of\r\n * module state changes.\r\n *\r\n * @param state - the current state of the module\r\n * @param previousState - the previous state of the module\r\n *\r\n * @public\r\n */\r\nexport type StateChangeHandler<TState> = (\r\n  state: Immutable<TState>,\r\n  previousState: Immutable<TState>,\r\n) => void\r\n\r\n/**\r\n * Configuration object for registering state change handlers.\r\n *\r\n * @public\r\n */\r\nexport interface StateChangeHandlerOptions {\r\n  /**\r\n   * By default a state change handler will be called with the module's\r\n   * current state immediately after being registered. Setting this\r\n   * property to true will skip that invocation and will only call the\r\n   * handler as soon as the state changes.\r\n   *\r\n   * @defaultValue `false`\r\n   */\r\n  readonly shouldSkipInitialInvocation?: boolean\r\n}\r\n\r\n/**\r\n * This type exists to get the concrete type of the handler, i.e. to return\r\n * a handler with the correct number or parameters\r\n *\r\n * @public\r\n */\r\nexport type ResolvedStateChangeHandler<TState, THandler> = THandler extends (\r\n  state: Immutable<TState>,\r\n) => void\r\n  ? (state: Immutable<TState>) => void\r\n  : StateChangeHandler<TState>\r\n\r\n/**\r\n * An object that can be used to unsubscribe from a subscription (e.g. for a\r\n * state change handler).\r\n *\r\n * @public\r\n */\r\nexport interface Subscription {\r\n  /**\r\n   * Unsubscribe from the subscription.\r\n   */\r\n  readonly unsubscribe: () => void\r\n}\r\n\r\n/**\r\n * A subscription for a state change handler.\r\n *\r\n * @public\r\n */\r\nexport interface StateChangeSubscription<\r\n  TState,\r\n  THandler extends StateChangeHandler<TState>\r\n> extends Subscription {\r\n  /**\r\n   * The handler function. Useful for testing.\r\n   */\r\n  readonly handler: ResolvedStateChangeHandler<TState, THandler>\r\n}\r\n\r\n/**\r\n * Subscribe to state changes.\r\n *\r\n * @param handler - the handler function to be called whenever the state changes\r\n * @param options - configuration for the subscription\r\n *\r\n * @returns a subscription that can be used to unsubscribe from state changes\r\n *\r\n * @public\r\n */\r\nexport type SubscribeToStateChanges<TState> = <\r\n  THandler extends StateChangeHandler<TState>\r\n>(\r\n  handler: THandler,\r\n  options?: StateChangeHandlerOptions,\r\n) => StateChangeSubscription<TState, THandler>\r\n\r\n/**\r\n * This is part of the simplux internal API and should not be accessed\r\n * except by simplux extensions.\r\n *\r\n * @internal\r\n */\r\nexport interface _SimpluxModuleInternals<TState> {\r\n  /**\r\n   * The unique name of the module.\r\n   */\r\n  readonly name: string\r\n\r\n  /**\r\n   * The mock state value to return instead of the real state when the\r\n   * module's state is accessed.\r\n   */\r\n  mockStateValue?: TState\r\n\r\n  /**\r\n   * Track all module mutations to be able to detect duplicates etc.\r\n   */\r\n  readonly mutations: MutationDefinitions<TState>\r\n\r\n  /**\r\n   * Mock functions that should be called instead of real mutations.\r\n   */\r\n  mutationMocks?: { [name: string]: (...args: any[]) => TState }\r\n\r\n  /**\r\n   * Track generated selector IDs for the module.\r\n   */\r\n  lastSelectorId?: number\r\n\r\n  /**\r\n   * Mock functions that should be called instead of real selectors.\r\n   */\r\n  selectorMocks?: { [selectorId: number]: (...args: any[]) => any }\r\n\r\n  /**\r\n   * A proxy to the Redux store's dispatch function. This is part of the\r\n   * simplux internal API and should not be accessed except by simplux\r\n   * extensions.\r\n   */\r\n  readonly dispatch: (action: AnyAction) => void\r\n\r\n  /**\r\n   * A proxy to the Redux store's dispatch function. This is part of the\r\n   * simplux internal API and should not be accessed except by simplux\r\n   * extensions.\r\n   */\r\n  readonly getReducer: () => Reducer\r\n\r\n  /**\r\n   * Get the current module state.\r\n   *\r\n   * @returns the module state\r\n   */\r\n  readonly getState: () => Immutable<TState>\r\n}\r\n\r\n/**\r\n * Interface for efficiently identifying simplux module objects at compile time.\r\n *\r\n * @public\r\n */\r\nexport interface SimpluxModuleMarker<TState> {\r\n  /**\r\n   * A symbol that allows efficient compile-time and run-time identification\r\n   * of simplux module 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_MODULE]: TState\r\n}\r\n\r\n/**\r\n * A simplux module that contains some state and allows controlled modification\r\n * and observation of the state.\r\n *\r\n * @public\r\n */\r\nexport interface SimpluxModule<TState> extends SimpluxModuleMarker<TState> {\r\n  /**\r\n   * A selector for getting the current module state.\r\n   *\r\n   * @returns the module state\r\n   */\r\n  readonly state: SimpluxSelector<TState, [], Immutable<TState>>\r\n\r\n  /**\r\n   * Replace the whole module state.\r\n   *\r\n   * @param state - the state to set for the module\r\n   */\r\n  readonly setState: (state: Immutable<TState>) => void\r\n\r\n  /**\r\n   * Register a handler to be called whenever the module's state\r\n   * changes. The handler will be called immediately with the module's\r\n   * current state when subscribing (can be changed by providing custom\r\n   * options).\r\n   *\r\n   * @param handler - the function to call whenever the module's state changes\r\n   *\r\n   * @returns a subscription that can be unsubscribed from to remove the handler\r\n   */\r\n  readonly subscribeToStateChanges: SubscribeToStateChanges<TState>\r\n\r\n  /**\r\n   * Internal state that is used by simplux. This is part of the simplux\r\n   * internal API and should not be accessed except by simplux extensions.\r\n   *\r\n   * @internal\r\n   */\r\n  readonly $simplux: _SimpluxModuleInternals<TState>\r\n}\r\n\r\n/**\r\n * This is part of the simplux internal API and should not be accessed\r\n * except by simplux extensions.\r\n *\r\n * @internal\r\n */\r\nexport function createModule<TState>(\r\n  store: _SimpluxStore,\r\n  config: SimpluxModuleConfig<TState>,\r\n): SimpluxModule<TState> {\r\n  if (process.env.NODE_ENV !== 'production') {\r\n    if (!config.name) {\r\n      throw new Error('you must provide a module name!')\r\n    }\r\n  }\r\n\r\n  const { getState, dispatch, subscribe, setReducer } = store\r\n\r\n  const internals: _SimpluxModuleInternals<TState> = {\r\n    name: config.name,\r\n    mutations: {},\r\n    dispatch,\r\n    getReducer: () => simpluxStore.getReducer(config.name),\r\n    getState: getModuleState,\r\n  }\r\n\r\n  function getModuleState(): Immutable<TState> {\r\n    return internals.mockStateValue || getState()[config.name]\r\n  }\r\n\r\n  function setModuleState(state: Immutable<TState>) {\r\n    dispatch({\r\n      type: `@simplux/${config.name}/setState`,\r\n      state,\r\n    })\r\n  }\r\n\r\n  let unsubscribeFromStore: (() => void) | undefined\r\n  let latestModuleState = config.initialState as Immutable<TState>\r\n  const handlers: StateChangeHandler<TState>[] = []\r\n\r\n  type Required<T> = { [prop in keyof T]-?: T[prop] }\r\n\r\n  const defaultStateChangeHandlerOptions: Required<StateChangeHandlerOptions> = {\r\n    shouldSkipInitialInvocation: false,\r\n  }\r\n\r\n  const subscribeToStateChanges: SubscribeToStateChanges<TState> = (\r\n    handler,\r\n    options = {},\r\n  ) => {\r\n    const fullOptions = {\r\n      ...defaultStateChangeHandlerOptions,\r\n      ...options,\r\n    }\r\n\r\n    handlers.push(handler)\r\n\r\n    if (handlers.length === 1) {\r\n      latestModuleState = getModuleState()\r\n      unsubscribeFromStore = subscribe(() => {\r\n        const moduleState = getModuleState()\r\n\r\n        if (moduleState !== latestModuleState) {\r\n          const previousModuleState = latestModuleState\r\n          latestModuleState = moduleState\r\n\r\n          for (const handler of handlers) {\r\n            handler(moduleState, previousModuleState)\r\n          }\r\n        }\r\n      })\r\n    }\r\n\r\n    if (!fullOptions.shouldSkipInitialInvocation) {\r\n      handler(latestModuleState, latestModuleState)\r\n    }\r\n\r\n    const unsubscribe = () => {\r\n      const idx = handlers.indexOf(handler)\r\n      if (idx >= 0) {\r\n        handlers.splice(idx, 1)\r\n      }\r\n\r\n      if (handlers.length === 0 && unsubscribeFromStore) {\r\n        unsubscribeFromStore()\r\n        unsubscribeFromStore = undefined\r\n      }\r\n    }\r\n\r\n    return {\r\n      unsubscribe,\r\n      handler: (handler as unknown) as ResolvedStateChangeHandler<\r\n        TState,\r\n        typeof handler\r\n      >,\r\n    }\r\n  }\r\n\r\n  const moduleReducer = createModuleReducer(\r\n    config.name,\r\n    config.initialState,\r\n    internals.mutations,\r\n  )\r\n\r\n  const moduleReducerWithImmerSupport = createImmerReducer(moduleReducer)\r\n\r\n  setReducer(config.name, moduleReducerWithImmerSupport)\r\n\r\n  type ShallowMutable<T> = { -readonly [prop in keyof T]: T[prop] }\r\n\r\n  const result: ShallowMutable<SimpluxModule<TState>> = {\r\n    state: undefined!,\r\n    setState: setModuleState,\r\n    subscribeToStateChanges,\r\n    $simplux: internals,\r\n    [SIMPLUX_MODULE]: '' as any,\r\n  }\r\n\r\n  const selectors = createSelectors(result, {\r\n    state: (s) => s,\r\n  })\r\n\r\n  result.state = selectors.state\r\n\r\n  return result\r\n}\r\n\r\n/**\r\n * Checks if an object is a simplux module.\r\n *\r\n * @param object - the object to check\r\n *\r\n * @returns true if the object is a simplux module\r\n *\r\n * @internal\r\n */\r\nexport function _isSimpluxModule<TState, TOther>(\r\n  object: SimpluxModuleMarker<TState> | TOther,\r\n): object is SimpluxModule<TState> {\r\n  return (object as any)?.[SIMPLUX_MODULE] === ''\r\n}\r\n"],"names":[],"mappings":"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"}