@aerogel/core 0.0.0-next.c8f032a868370824898e171969aec1bb6827688e → 0.0.0-next.d547095ca85c86c1e41f5c7fa170d0ba856c3f4d

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (115) hide show
  1. package/dist/aerogel-core.cjs.js +1 -1
  2. package/dist/aerogel-core.cjs.js.map +1 -1
  3. package/dist/aerogel-core.d.ts +1511 -236
  4. package/dist/aerogel-core.esm.js +1 -1
  5. package/dist/aerogel-core.esm.js.map +1 -1
  6. package/histoire.config.ts +7 -0
  7. package/noeldemartin.config.js +4 -1
  8. package/package.json +14 -4
  9. package/postcss.config.js +6 -0
  10. package/src/assets/histoire.css +3 -0
  11. package/src/bootstrap/bootstrap.test.ts +4 -3
  12. package/src/bootstrap/index.ts +27 -4
  13. package/src/bootstrap/options.ts +3 -0
  14. package/src/components/AGAppLayout.vue +7 -2
  15. package/src/components/AGAppModals.vue +15 -0
  16. package/src/components/AGAppOverlays.vue +10 -8
  17. package/src/components/AGAppSnackbars.vue +13 -0
  18. package/src/components/constants.ts +8 -0
  19. package/src/components/forms/AGButton.vue +33 -10
  20. package/src/components/forms/AGCheckbox.vue +41 -0
  21. package/src/components/forms/AGInput.vue +15 -9
  22. package/src/components/forms/AGSelect.story.vue +46 -0
  23. package/src/components/forms/AGSelect.vue +60 -0
  24. package/src/components/forms/index.ts +5 -5
  25. package/src/components/headless/forms/AGHeadlessButton.vue +12 -12
  26. package/src/components/headless/forms/AGHeadlessInput.ts +23 -3
  27. package/src/components/headless/forms/AGHeadlessInput.vue +11 -8
  28. package/src/components/headless/forms/AGHeadlessInputError.vue +1 -1
  29. package/src/components/headless/forms/AGHeadlessInputInput.vue +17 -3
  30. package/src/components/headless/forms/AGHeadlessInputLabel.vue +8 -2
  31. package/src/components/headless/forms/AGHeadlessSelect.ts +42 -0
  32. package/src/components/headless/forms/AGHeadlessSelect.vue +77 -0
  33. package/src/components/headless/forms/AGHeadlessSelectButton.vue +24 -0
  34. package/src/components/headless/forms/AGHeadlessSelectError.vue +26 -0
  35. package/src/components/headless/forms/AGHeadlessSelectLabel.vue +24 -0
  36. package/src/components/headless/forms/AGHeadlessSelectOption.ts +4 -0
  37. package/src/components/headless/forms/AGHeadlessSelectOption.vue +39 -0
  38. package/src/components/headless/forms/AGHeadlessSelectOptions.ts +3 -0
  39. package/src/components/headless/forms/index.ts +9 -1
  40. package/src/components/headless/index.ts +1 -0
  41. package/src/components/headless/modals/AGHeadlessModal.ts +27 -0
  42. package/src/components/headless/modals/AGHeadlessModal.vue +3 -5
  43. package/src/components/headless/modals/AGHeadlessModalPanel.vue +5 -1
  44. package/src/components/headless/modals/index.ts +4 -6
  45. package/src/components/headless/snackbars/AGHeadlessSnackbar.vue +10 -0
  46. package/src/components/headless/snackbars/index.ts +40 -0
  47. package/src/components/index.ts +3 -1
  48. package/src/components/lib/AGErrorMessage.vue +16 -0
  49. package/src/components/lib/AGLink.vue +9 -0
  50. package/src/components/lib/AGMarkdown.vue +36 -0
  51. package/src/components/lib/AGMeasured.vue +15 -0
  52. package/src/components/lib/AGStartupCrash.vue +31 -0
  53. package/src/components/lib/index.ts +5 -0
  54. package/src/components/modals/AGAlertModal.ts +15 -0
  55. package/src/components/modals/AGAlertModal.vue +4 -16
  56. package/src/components/modals/AGConfirmModal.ts +27 -0
  57. package/src/components/modals/AGConfirmModal.vue +8 -12
  58. package/src/components/modals/AGErrorReportModal.ts +46 -0
  59. package/src/components/modals/AGErrorReportModal.vue +54 -0
  60. package/src/components/modals/AGErrorReportModalButtons.vue +111 -0
  61. package/src/components/modals/AGErrorReportModalTitle.vue +25 -0
  62. package/src/components/modals/AGLoadingModal.ts +23 -0
  63. package/src/components/modals/AGLoadingModal.vue +15 -0
  64. package/src/components/modals/AGModal.ts +1 -1
  65. package/src/components/modals/AGModal.vue +26 -5
  66. package/src/components/modals/AGModalTitle.vue +9 -0
  67. package/src/components/modals/AGPromptModal.ts +30 -0
  68. package/src/components/modals/AGPromptModal.vue +34 -0
  69. package/src/components/modals/index.ts +16 -6
  70. package/src/components/snackbars/AGSnackbar.vue +36 -0
  71. package/src/components/snackbars/index.ts +3 -0
  72. package/src/components/utils.ts +10 -0
  73. package/src/directives/index.ts +20 -3
  74. package/src/directives/measure.ts +21 -0
  75. package/src/errors/Errors.state.ts +31 -0
  76. package/src/errors/Errors.ts +185 -0
  77. package/src/errors/index.ts +46 -0
  78. package/src/errors/utils.ts +19 -0
  79. package/src/forms/Form.test.ts +21 -0
  80. package/src/forms/Form.ts +34 -15
  81. package/src/forms/utils.ts +17 -0
  82. package/src/jobs/Job.ts +5 -0
  83. package/src/jobs/index.ts +7 -0
  84. package/src/lang/Lang.ts +11 -15
  85. package/src/lang/index.ts +3 -5
  86. package/src/lang/utils.ts +4 -0
  87. package/src/main.histoire.ts +1 -0
  88. package/src/main.ts +4 -2
  89. package/src/plugins/Plugin.ts +1 -0
  90. package/src/plugins/index.ts +19 -0
  91. package/src/services/App.state.ts +9 -2
  92. package/src/services/App.ts +43 -3
  93. package/src/services/Events.test.ts +39 -0
  94. package/src/services/Events.ts +100 -30
  95. package/src/services/Service.ts +174 -53
  96. package/src/services/index.ts +22 -4
  97. package/src/services/store.ts +30 -0
  98. package/src/testing/index.ts +25 -0
  99. package/src/ui/UI.state.ts +11 -1
  100. package/src/ui/UI.ts +169 -20
  101. package/src/ui/index.ts +15 -4
  102. package/src/utils/composition/events.ts +1 -0
  103. package/src/utils/composition/forms.ts +11 -0
  104. package/src/utils/index.ts +2 -0
  105. package/src/utils/markdown.ts +11 -2
  106. package/src/utils/tailwindcss.test.ts +26 -0
  107. package/src/utils/tailwindcss.ts +7 -0
  108. package/src/utils/vue.ts +15 -4
  109. package/tailwind.config.js +4 -0
  110. package/tsconfig.json +1 -0
  111. package/vite.config.ts +2 -1
  112. package/.eslintrc.js +0 -3
  113. package/src/components/basic/AGMarkdown.vue +0 -35
  114. package/src/components/basic/index.ts +0 -3
  115. package/src/globals.ts +0 -6
@@ -1,7 +1,47 @@
1
- import { facade } from '@noeldemartin/utils';
1
+ import { PromisedValue, facade, forever, updateLocationQueryParameters } from '@noeldemartin/utils';
2
+
3
+ import Events from '@/services/Events';
4
+ import type { Plugin } from '@/plugins';
2
5
 
3
6
  import Service from './App.state';
4
7
 
5
- export class AppService extends Service {}
8
+ export class AppService extends Service {
9
+
10
+ public readonly ready = new PromisedValue<void>();
11
+ public readonly mounted = new PromisedValue<void>();
12
+
13
+ public isReady(): boolean {
14
+ return this.ready.isResolved();
15
+ }
16
+
17
+ public isMounted(): boolean {
18
+ return this.mounted.isResolved();
19
+ }
20
+
21
+ public async whenReady<T>(callback: () => T): Promise<T> {
22
+ const result = await this.ready.then(callback);
23
+
24
+ return result;
25
+ }
26
+
27
+ public async reload(queryParameters?: Record<string, string | undefined>): Promise<void> {
28
+ queryParameters && updateLocationQueryParameters(queryParameters);
29
+
30
+ location.reload();
31
+
32
+ // Stall until the reload happens
33
+ await forever();
34
+ }
35
+
36
+ public plugin<T extends Plugin = Plugin>(name: string): T | null {
37
+ return (this.plugins[name] as T) ?? null;
38
+ }
39
+
40
+ protected async boot(): Promise<void> {
41
+ Events.once('application-ready', () => this.ready.resolve());
42
+ Events.once('application-mounted', () => this.mounted.resolve());
43
+ }
44
+
45
+ }
6
46
 
7
- export default facade(new AppService());
47
+ export default facade(AppService);
@@ -0,0 +1,39 @@
1
+ import { beforeEach, describe, expect, it } from 'vitest';
2
+
3
+ import Events, { EventListenerPriorities } from './Events';
4
+
5
+ describe('Events', () => {
6
+
7
+ beforeEach(() => void Events.reset());
8
+
9
+ it('registers listeners', async () => {
10
+ // Arrange
11
+ let counter = 0;
12
+
13
+ Events.on('trigger', () => counter++);
14
+
15
+ // Act
16
+ await Events.emit('trigger');
17
+ await Events.emit('trigger');
18
+ await Events.emit('trigger');
19
+
20
+ // Assert
21
+ expect(counter).toEqual(3);
22
+ });
23
+
24
+ it('triggers listeners by priority', async () => {
25
+ // Arrange
26
+ const storage: string[] = [];
27
+
28
+ Events.on('trigger', () => storage.push('second'));
29
+ Events.on('trigger', { priority: EventListenerPriorities.Low }, () => storage.push('third'));
30
+ Events.on('trigger', { priority: EventListenerPriorities.High }, () => storage.push('first'));
31
+
32
+ // Act
33
+ await Events.emit('trigger');
34
+
35
+ // Assert
36
+ expect(storage).toEqual(['first', 'second', 'third']);
37
+ });
38
+
39
+ });
@@ -1,9 +1,13 @@
1
- import { arr, facade, tap } from '@noeldemartin/utils';
2
- import type { FluentArray } from '@noeldemartin/utils';
1
+ import { arrayRemove, facade, fail, tap } from '@noeldemartin/utils';
3
2
 
4
3
  import Service from '@/services/Service';
5
4
 
6
5
  export interface EventsPayload {}
6
+ export interface EventListenerOptions {
7
+ priority: number;
8
+ }
9
+ export type AerogelGlobalEvents = Partial<{ [Event in EventWithoutPayload]: () => unknown }> &
10
+ Partial<{ [Event in EventWithPayload]: EventListener<EventsPayload[Event]> }>;
7
11
 
8
12
  export type EventListener<T = unknown> = (payload: T) => unknown;
9
13
  export type UnknownEvent<T> = T extends keyof EventsPayload ? never : T;
@@ -16,70 +20,136 @@ export type EventWithPayload = {
16
20
  [K in keyof EventsPayload]: EventsPayload[K] extends void ? never : K;
17
21
  }[keyof EventsPayload];
18
22
 
23
+ export const EventListenerPriorities = {
24
+ Low: -256,
25
+ Default: 0,
26
+ High: 256,
27
+ } as const;
28
+
19
29
  export class EventsService extends Service {
20
30
 
21
- private listeners: Record<string, FluentArray<EventListener>> = {};
31
+ private listeners: Record<string, { priorities: number[]; handlers: Record<number, EventListener[]> }> = {};
32
+
33
+ protected async boot(): Promise<void> {
34
+ Object.entries(globalThis.__aerogelEvents__ ?? {}).forEach(([event, listener]) =>
35
+ this.on(event as string, listener as EventListener));
36
+ }
22
37
 
23
38
  public emit<Event extends EventWithoutPayload>(event: Event): Promise<void>;
24
39
  public emit<Event extends EventWithPayload>(event: Event, payload: EventsPayload[Event]): Promise<void>;
25
40
  public emit<Event extends string>(event: UnknownEvent<Event>, payload?: unknown): Promise<void>;
26
41
  public async emit(event: string, payload?: unknown): Promise<void> {
27
- const listeners = [...(this.listeners[event] ?? [])];
42
+ const listeners = this.listeners[event] ?? { priorities: [], handlers: {} };
28
43
 
29
- await Promise.all(listeners.map((listener) => listener(payload)) ?? []);
44
+ for (const priority of listeners.priorities) {
45
+ await Promise.all(listeners.handlers[priority]?.map((listener) => listener(payload)) ?? []);
46
+ }
30
47
  }
31
48
 
49
+ /* eslint-disable max-len */
32
50
  public on<Event extends EventWithoutPayload>(event: Event, listener: () => unknown): () => void;
33
- public on<Event extends EventWithPayload>(
34
- event: Event,
35
- listener: EventListener<EventsPayload[Event]>
36
- ): () => void | void;
37
-
51
+ public on<Event extends EventWithoutPayload>(event: Event, options: Partial<EventListenerOptions>, listener: () => unknown): () => void; // prettier-ignore
52
+ public on<Event extends EventWithPayload>(event: Event, listener: EventListener<EventsPayload[Event]>): () => void | void; // prettier-ignore
53
+ public on<Event extends EventWithPayload>(event: Event, options: Partial<EventListenerOptions>, listener: EventListener<EventsPayload[Event]>): () => void | void; // prettier-ignore
38
54
  public on<Event extends string>(event: UnknownEvent<Event>, listener: EventListener): () => void;
39
- public on(event: string, listener: EventListener): () => void {
40
- (this.listeners[event] ??= arr<EventListener>([])).push(listener);
55
+ public on<Event extends string>(event: UnknownEvent<Event>, options: Partial<EventListenerOptions>, listener: EventListener): () => void; // prettier-ignore
56
+ /* eslint-enable max-len */
41
57
 
42
- return () => this.off(event, listener);
58
+ public on(
59
+ event: string,
60
+ optionsOrListener: Partial<EventListenerOptions> | EventListener,
61
+ listener?: EventListener,
62
+ ): () => void {
63
+ const options = typeof optionsOrListener === 'function' ? {} : optionsOrListener;
64
+ const handler = typeof optionsOrListener === 'function' ? optionsOrListener : (listener as EventListener);
65
+
66
+ this.registerListener(event, options, handler);
67
+
68
+ return () => this.off(event, handler);
43
69
  }
44
70
 
71
+ /* eslint-disable max-len */
45
72
  public once<Event extends EventWithoutPayload>(event: Event, listener: () => unknown): () => void;
46
- public once<Event extends EventWithPayload>(
47
- event: Event,
48
- listener: EventListener<EventsPayload[Event]>
49
- ): () => void | void;
50
-
73
+ public once<Event extends EventWithoutPayload>(event: Event, options: Partial<EventListenerOptions>, listener: () => unknown): () => void; // prettier-ignore
74
+ public once<Event extends EventWithPayload>(event: Event, listener: EventListener<EventsPayload[Event]>): () => void | void; // prettier-ignore
75
+ public once<Event extends EventWithPayload>(event: Event, options: Partial<EventListenerOptions>, listener: EventListener<EventsPayload[Event]>): () => void | void; // prettier-ignore
51
76
  public once<Event extends string>(event: UnknownEvent<Event>, listener: EventListener): () => void;
52
- public once(event: string, listener: EventListener): () => void {
77
+ public once<Event extends string>(event: UnknownEvent<Event>, options: Partial<EventListenerOptions>, listener: EventListener): () => void; // prettier-ignore
78
+ /* eslint-enable max-len */
79
+
80
+ public once(
81
+ event: string,
82
+ optionsOrListener: Partial<EventListenerOptions> | EventListener,
83
+ listener?: EventListener,
84
+ ): () => void {
53
85
  let onceListener: EventListener | null = null;
86
+ const options = typeof optionsOrListener === 'function' ? {} : optionsOrListener;
87
+ const handler = typeof optionsOrListener === 'function' ? optionsOrListener : (listener as EventListener);
54
88
 
55
89
  return tap(
56
90
  () => onceListener && this.off(event, onceListener),
57
91
  (off) => {
58
- (this.listeners[event] ??= arr<EventListener>([])).push(
59
- (onceListener = (...args) => {
60
- off();
92
+ onceListener = (...args) => {
93
+ off();
61
94
 
62
- return listener(...args);
63
- }),
64
- );
95
+ return handler(...args);
96
+ };
97
+
98
+ this.registerListener(event, options, handler);
65
99
  },
66
100
  );
67
101
  }
68
102
 
69
103
  public off(event: string, listener: EventListener): void {
70
- const eventListeners = this.listeners[event];
104
+ const listeners = this.listeners[event];
71
105
 
72
- if (!eventListeners) {
106
+ if (!listeners) {
73
107
  return;
74
108
  }
75
109
 
76
- eventListeners.remove(listener);
110
+ const priorities = [...listeners.priorities];
111
+
112
+ for (const priority of priorities) {
113
+ arrayRemove(listeners.handlers[priority] ?? [], listener);
77
114
 
78
- if (eventListeners.isEmpty()) {
115
+ if (listeners.handlers[priority]?.length === 0) {
116
+ delete listeners.handlers[priority];
117
+ arrayRemove(listeners.priorities, priority);
118
+ }
119
+ }
120
+
121
+ if (listeners.priorities.length === 0) {
79
122
  delete this.listeners[event];
80
123
  }
81
124
  }
82
125
 
126
+ protected registerListener(event: string, options: Partial<EventListenerOptions>, handler: EventListener): void {
127
+ const priority = options.priority ?? 0;
128
+
129
+ if (!(event in this.listeners)) {
130
+ this.listeners[event] = { priorities: [], handlers: {} };
131
+ }
132
+
133
+ const priorities =
134
+ this.listeners[event]?.priorities ?? fail<number[]>(`priorities missing for event '${event}'`);
135
+ const handlers =
136
+ this.listeners[event]?.handlers ??
137
+ fail<Record<number, EventListener[]>>(`handlers missing for event '${event}'`);
138
+
139
+ if (!priorities.includes(priority)) {
140
+ priorities.push(priority);
141
+ priorities.sort((a, b) => b - a);
142
+ handlers[priority] = [];
143
+ }
144
+
145
+ handlers[priority]?.push(handler);
146
+ }
147
+
83
148
  }
84
149
 
85
- export default facade(new EventsService());
150
+ export default facade(EventsService);
151
+
152
+ declare global {
153
+ // eslint-disable-next-line no-var
154
+ var __aerogelEvents__: AerogelGlobalEvents | undefined;
155
+ }
@@ -1,75 +1,126 @@
1
- import { computed, reactive } from 'vue';
2
- import { MagicObject, PromisedValue } from '@noeldemartin/utils';
3
- import type { ComputedRef } from 'vue';
1
+ import { MagicObject, PromisedValue, Storage, isEmpty, objectDeepClone, objectOnly } from '@noeldemartin/utils';
4
2
  import type { Constructor } from '@noeldemartin/utils';
3
+ import type { MaybeRef } from 'vue';
4
+ import type { Store } from 'pinia';
5
5
 
6
6
  import ServiceBootError from '@/errors/ServiceBootError';
7
+ import { defineServiceStore } from '@/services/store';
7
8
 
8
9
  export type ServiceState = Record<string, any>; // eslint-disable-line @typescript-eslint/no-explicit-any
9
- export type DefaultServiceState = {};
10
+ export type DefaultServiceState = any; // eslint-disable-line @typescript-eslint/no-explicit-any
10
11
  export type ServiceConstructor<T extends Service = Service> = Constructor<T> & typeof Service;
12
+ export type UnrefServiceState<State extends ServiceState> = {
13
+ [K in keyof State]: State[K] extends MaybeRef<infer T> ? T : State[K];
14
+ };
11
15
 
12
16
  export type ComputedStateDefinition<TState extends ServiceState, TComputedState extends ServiceState> = {
13
- [K in keyof TComputedState]: (state: TState) => TComputedState[K];
14
- };
17
+ [K in keyof TComputedState]: (state: UnrefServiceState<TState>) => TComputedState[K];
18
+ } & ThisType<{
19
+ readonly [K in keyof TComputedState]: TComputedState[K];
20
+ }>;
15
21
 
16
22
  export function defineServiceState<
17
23
  State extends ServiceState = ServiceState,
18
24
  ComputedState extends ServiceState = {}
19
25
  >(options: {
20
- initialState: State;
26
+ name: string;
27
+ initialState: State | (() => State);
28
+ persist?: (keyof State)[];
21
29
  computed?: ComputedStateDefinition<State, ComputedState>;
22
- }): Constructor<State> & Constructor<ComputedState> & ServiceConstructor {
23
- return class extends Service<State, ComputedState> {
30
+ serialize?: (state: Partial<State>) => Partial<State>;
31
+ }): Constructor<UnrefServiceState<State>> &
32
+ Constructor<ComputedState> &
33
+ Constructor<Service<UnrefServiceState<State>, ComputedState, Partial<UnrefServiceState<State>>>> {
34
+ return class extends Service<UnrefServiceState<State>, ComputedState> {
35
+
36
+ public static persist = (options.persist as string[]) ?? [];
37
+
38
+ protected usesStore(): boolean {
39
+ return true;
40
+ }
41
+
42
+ protected getName(): string | null {
43
+ return options.name ?? null;
44
+ }
24
45
 
25
- protected getInitialState(): State {
26
- return options.initialState;
46
+ protected getInitialState(): UnrefServiceState<State> {
47
+ if (typeof options.initialState === 'function') {
48
+ return options.initialState();
49
+ }
50
+
51
+ return Object.entries(options.initialState).reduce((state, [key, value]) => {
52
+ try {
53
+ value = structuredClone(value);
54
+ } catch (error) {
55
+ // eslint-disable-next-line no-console
56
+ console.warn(
57
+ `Could not clone '${key}' state from ${this.getName()} service, ` +
58
+ 'this may cause problems if you\'re using multiple instances of the service ' +
59
+ '(for example, in unit tests).\n' +
60
+ 'To fix this problem, declare your initialState as a function instead.',
61
+ );
62
+ }
63
+
64
+ state[key as keyof State] = value;
65
+
66
+ return state;
67
+ }, {} as UnrefServiceState<State>);
27
68
  }
28
69
 
29
- protected getComputedStateDefinition(): ComputedStateDefinition<State, ComputedState> {
30
- return options.computed ?? ({} as ComputedStateDefinition<State, ComputedState>);
70
+ protected getComputedStateDefinition(): ComputedStateDefinition<UnrefServiceState<State>, ComputedState> {
71
+ return (options.computed ?? {}) as ComputedStateDefinition<UnrefServiceState<State>, ComputedState>;
31
72
  }
32
-
33
- } as unknown as Constructor<State> & Constructor<ComputedState> & ServiceConstructor;
73
+
74
+ protected serializePersistedState(state: Partial<State>): Partial<State> {
75
+ return options.serialize?.(state) ?? state;
76
+ }
77
+
78
+ } as unknown as Constructor<UnrefServiceState<State>> &
79
+ Constructor<ComputedState> &
80
+ Constructor<Service<UnrefServiceState<State>, ComputedState, Partial<UnrefServiceState<State>>>>;
34
81
  }
35
82
 
36
83
  export default class Service<
37
84
  State extends ServiceState = DefaultServiceState,
38
- ComputedState extends ServiceState = {}
85
+ ComputedState extends ServiceState = {},
86
+ ServiceStorage extends Partial<State> = Partial<State>
39
87
  > extends MagicObject {
40
88
 
41
- protected _namespace: string;
89
+ public static persist: string[] = [];
90
+
91
+ protected _name: string;
42
92
  private _booted: PromisedValue<void>;
43
- private _state: State;
44
- private _computedState: Record<keyof ComputedState, ComputedRef>;
93
+ private _computedStateKeys: Set<keyof State>;
94
+ private _store: Store | false;
45
95
 
46
96
  constructor() {
47
97
  super();
48
98
 
49
- this._namespace = new.target.name;
50
- this._booted = new PromisedValue();
51
- this._state = reactive(this.getInitialState());
52
- this._computedState = Object.entries(this.getComputedStateDefinition()).reduce(
53
- (computedState, [name, method]) => {
54
- computedState[name as keyof ComputedState] = computed(() => method(this._state));
99
+ const getters = this.getComputedStateDefinition();
55
100
 
56
- return computedState;
57
- },
58
- {} as Record<keyof ComputedState, ComputedRef>,
59
- );
101
+ this._name = this.getName() ?? new.target.name;
102
+ this._booted = new PromisedValue();
103
+ this._computedStateKeys = new Set(Object.keys(getters));
104
+ this._store =
105
+ this.usesStore() &&
106
+ defineServiceStore(this._name, {
107
+ state: () => this.getInitialState(),
108
+
109
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
110
+ getters: getters as any,
111
+ });
60
112
  }
61
113
 
62
114
  public get booted(): PromisedValue<void> {
63
115
  return this._booted;
64
116
  }
65
117
 
66
- public launch(namespace?: string): Promise<void> {
67
- const handleError = (error: unknown) => this._booted.reject(new ServiceBootError(this._namespace, error));
68
-
69
- this._namespace = namespace ?? this._namespace;
118
+ public launch(): Promise<void> {
119
+ const handleError = (error: unknown) => this._booted.reject(new ServiceBootError(this._name, error));
70
120
 
71
121
  try {
72
- this.boot()
122
+ this.frameworkBoot()
123
+ .then(() => this.boot())
73
124
  .then(() => this._booted.resolve())
74
125
  .catch(handleError);
75
126
  } catch (error) {
@@ -79,15 +130,52 @@ export default class Service<
79
130
  return this._booted;
80
131
  }
81
132
 
133
+ public hasPersistedState(): boolean {
134
+ return Storage.has(this._name);
135
+ }
136
+
137
+ public hasState<P extends keyof State>(property: P): boolean {
138
+ if (!this._store) {
139
+ return false;
140
+ }
141
+
142
+ return property in this._store.$state || this._computedStateKeys.has(property);
143
+ }
144
+
145
+ public getState(): State;
146
+ public getState<P extends keyof State>(property: P): State[P];
147
+ public getState<P extends keyof State>(property?: P): State | State[P] {
148
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
149
+ const store = this._store as any;
150
+
151
+ if (property) {
152
+ return store ? store[property] : undefined;
153
+ }
154
+
155
+ return store ? store : {};
156
+ }
157
+
158
+ public setState<P extends keyof State>(property: P, value: State[P]): void;
159
+ public setState(state: Partial<State>): void;
160
+ public setState<P extends keyof State>(stateOrProperty: P | Partial<State>, value?: State[P]): void {
161
+ if (!this._store) {
162
+ return;
163
+ }
164
+
165
+ const state = (
166
+ typeof stateOrProperty === 'string' ? { [stateOrProperty]: value } : stateOrProperty
167
+ ) as Partial<State>;
168
+
169
+ Object.assign(this._store.$state, state);
170
+
171
+ this.onStateUpdated(state);
172
+ }
173
+
82
174
  protected __get(property: string): unknown {
83
175
  if (this.hasState(property)) {
84
176
  return this.getState(property);
85
177
  }
86
178
 
87
- if (this.hasComputedState(property)) {
88
- return this.getComputedState(property);
89
- }
90
-
91
179
  return super.__get(property);
92
180
  }
93
181
 
@@ -95,26 +183,33 @@ export default class Service<
95
183
  this.setState({ [property]: value } as Partial<State>);
96
184
  }
97
185
 
98
- protected hasState<P extends keyof State>(property: P): boolean {
99
- return property in this._state;
100
- }
186
+ protected onStateUpdated(state: Partial<State>): void {
187
+ // TODO fix this.static()
188
+ const persist = (this.constructor as unknown as { persist: string[] }).persist;
189
+ const persisted = objectOnly(state, persist);
101
190
 
102
- protected hasComputedState<P extends keyof State>(property: P): boolean {
103
- return property in this._computedState;
104
- }
191
+ if (isEmpty(persisted)) {
192
+ return;
193
+ }
194
+
195
+ const storage = Storage.get<ServiceStorage>(this._name);
196
+
197
+ if (!storage) {
198
+ return;
199
+ }
105
200
 
106
- protected getState(): State;
107
- protected getState<P extends keyof State>(property: P): State[P];
108
- protected getState<P extends keyof State>(property?: P): State | State[P] {
109
- return property ? this._state[property] : this._state;
201
+ Storage.set(this._name, {
202
+ ...storage,
203
+ ...this.serializePersistedState(objectDeepClone(persisted) as Partial<State>),
204
+ });
110
205
  }
111
206
 
112
- protected getComputedState<P extends keyof ComputedState>(property: P): ComputedState[P] {
113
- return this._computedState[property]?.value;
207
+ protected usesStore(): boolean {
208
+ return false;
114
209
  }
115
210
 
116
- protected setState(state: Partial<State>): void {
117
- Object.assign(this._state, state);
211
+ protected getName(): string | null {
212
+ return null;
118
213
  }
119
214
 
120
215
  protected getInitialState(): State {
@@ -125,8 +220,34 @@ export default class Service<
125
220
  return {} as ComputedStateDefinition<State, ComputedState>;
126
221
  }
127
222
 
223
+ protected serializePersistedState(state: Partial<State>): Partial<State> {
224
+ return state;
225
+ }
226
+
227
+ protected async frameworkBoot(): Promise<void> {
228
+ this.initializePersistedState();
229
+ }
230
+
128
231
  protected async boot(): Promise<void> {
129
- //
232
+ // Placeholder for overrides, don't place any functionality here.
233
+ }
234
+
235
+ protected initializePersistedState(): void {
236
+ // TODO fix this.static()
237
+ const persist = (this.constructor as unknown as { persist: string[] }).persist;
238
+
239
+ if (!this.usesStore() || isEmpty(persist)) {
240
+ return;
241
+ }
242
+
243
+ if (Storage.has(this._name)) {
244
+ const persisted = Storage.require<ServiceStorage>(this._name);
245
+ this.setState(persisted);
246
+
247
+ return;
248
+ }
249
+
250
+ Storage.set(this._name, objectOnly(this.getState(), persist));
130
251
  }
131
252
 
132
253
  }
@@ -5,10 +5,12 @@ import { definePlugin } from '@/plugins';
5
5
  import App from './App';
6
6
  import Events from './Events';
7
7
  import Service from './Service';
8
+ import { getPiniaStore } from './store';
8
9
 
9
10
  export * from './App';
10
11
  export * from './Events';
11
12
  export * from './Service';
13
+ export * from './store';
12
14
 
13
15
  export { App, Events, Service };
14
16
 
@@ -24,20 +26,36 @@ export interface Services extends DefaultServices {}
24
26
  export async function bootServices(app: VueApp, services: Record<string, Service>): Promise<void> {
25
27
  await Promise.all(
26
28
  Object.entries(services).map(async ([name, service]) => {
27
- // eslint-disable-next-line no-console
28
- await service.launch(name.slice(1)).catch((error) => console.error(error));
29
+ await service
30
+ .launch()
31
+ .catch((error) => app.config.errorHandler?.(error, null, `Failed launching ${name}.`));
29
32
  }),
30
33
  );
31
34
 
32
35
  Object.assign(app.config.globalProperties, services);
36
+
37
+ App.development && Object.assign(window, services);
33
38
  }
34
39
 
35
40
  export default definePlugin({
36
- async install(app) {
37
- await bootServices(app, defaultServices);
41
+ async install(app, options) {
42
+ const services = {
43
+ ...defaultServices,
44
+ ...options.services,
45
+ };
46
+
47
+ app.use(getPiniaStore());
48
+
49
+ await bootServices(app, services);
38
50
  },
39
51
  });
40
52
 
53
+ declare module '@/bootstrap/options' {
54
+ export interface AerogelOptions {
55
+ services?: Record<string, Service>;
56
+ }
57
+ }
58
+
41
59
  declare module '@vue/runtime-core' {
42
60
  interface ComponentCustomProperties extends Services {}
43
61
  }
@@ -0,0 +1,30 @@
1
+ import { tap } from '@noeldemartin/utils';
2
+ import { createPinia, defineStore, setActivePinia } from 'pinia';
3
+ import type { DefineStoreOptions, Pinia, StateTree, Store, _GettersTree } from 'pinia';
4
+
5
+ let _store: Pinia | null = null;
6
+
7
+ function initializePiniaStore(): Pinia {
8
+ return _store ?? resetPiniaStore();
9
+ }
10
+
11
+ export function resetPiniaStore(): Pinia {
12
+ return tap(createPinia(), (store) => {
13
+ _store = store;
14
+
15
+ setActivePinia(store);
16
+ });
17
+ }
18
+
19
+ export function getPiniaStore(): Pinia {
20
+ return _store ?? initializePiniaStore();
21
+ }
22
+
23
+ export function defineServiceStore<Id extends string, S extends StateTree = {}, G extends _GettersTree<S> = {}, A = {}>(
24
+ name: Id,
25
+ options: Omit<DefineStoreOptions<Id, S, G, A>, 'id'>,
26
+ ): Store<Id, S, G, A> {
27
+ initializePiniaStore();
28
+
29
+ return defineStore(name, options)();
30
+ }