@aerogel/core 0.0.0-next.b85327579d32f21c6a9fa21142f0165cdd320d7e → 0.0.0-next.d34923f3b144e8f6720e6a9cdadb2cd4fb4ab289

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 (114) 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 +1568 -252
  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 +13 -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 +25 -5
  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 +25 -15
  20. package/src/components/forms/AGCheckbox.vue +7 -1
  21. package/src/components/forms/AGForm.vue +9 -10
  22. package/src/components/forms/AGInput.vue +10 -6
  23. package/src/components/forms/AGSelect.story.vue +46 -0
  24. package/src/components/forms/AGSelect.vue +60 -0
  25. package/src/components/forms/index.ts +5 -6
  26. package/src/components/headless/forms/AGHeadlessButton.vue +18 -12
  27. package/src/components/headless/forms/AGHeadlessInput.ts +29 -4
  28. package/src/components/headless/forms/AGHeadlessInput.vue +17 -7
  29. package/src/components/headless/forms/AGHeadlessInputDescription.vue +28 -0
  30. package/src/components/headless/forms/AGHeadlessInputInput.vue +43 -5
  31. package/src/components/headless/forms/AGHeadlessInputLabel.vue +8 -2
  32. package/src/components/headless/forms/AGHeadlessInputTextArea.vue +42 -0
  33. package/src/components/headless/forms/AGHeadlessSelect.ts +42 -0
  34. package/src/components/headless/forms/AGHeadlessSelect.vue +77 -0
  35. package/src/components/headless/forms/AGHeadlessSelectButton.vue +24 -0
  36. package/src/components/headless/forms/AGHeadlessSelectError.vue +26 -0
  37. package/src/components/headless/forms/AGHeadlessSelectLabel.vue +24 -0
  38. package/src/components/headless/forms/AGHeadlessSelectOption.ts +4 -0
  39. package/src/components/headless/forms/AGHeadlessSelectOption.vue +39 -0
  40. package/src/components/headless/forms/AGHeadlessSelectOptions.ts +3 -0
  41. package/src/components/headless/forms/composition.ts +10 -0
  42. package/src/components/headless/forms/index.ts +12 -1
  43. package/src/components/headless/index.ts +1 -0
  44. package/src/components/headless/modals/AGHeadlessModal.ts +27 -0
  45. package/src/components/headless/modals/AGHeadlessModal.vue +3 -5
  46. package/src/components/headless/modals/index.ts +4 -6
  47. package/src/components/headless/snackbars/AGHeadlessSnackbar.vue +10 -0
  48. package/src/components/headless/snackbars/index.ts +40 -0
  49. package/src/components/index.ts +4 -1
  50. package/src/components/interfaces.ts +9 -0
  51. package/src/components/lib/AGErrorMessage.vue +16 -0
  52. package/src/components/lib/AGLink.vue +9 -0
  53. package/src/components/{basic → lib}/AGMarkdown.vue +13 -7
  54. package/src/components/lib/AGMeasured.vue +15 -0
  55. package/src/components/lib/AGStartupCrash.vue +31 -0
  56. package/src/components/lib/index.ts +5 -0
  57. package/src/components/modals/AGAlertModal.ts +15 -0
  58. package/src/components/modals/AGAlertModal.vue +4 -16
  59. package/src/components/modals/AGConfirmModal.ts +33 -0
  60. package/src/components/modals/AGConfirmModal.vue +9 -13
  61. package/src/components/modals/AGErrorReportModal.ts +46 -0
  62. package/src/components/modals/AGErrorReportModal.vue +54 -0
  63. package/src/components/modals/AGErrorReportModalButtons.vue +111 -0
  64. package/src/components/modals/AGErrorReportModalTitle.vue +25 -0
  65. package/src/components/modals/AGLoadingModal.ts +23 -0
  66. package/src/components/modals/AGLoadingModal.vue +4 -8
  67. package/src/components/modals/AGModal.ts +1 -1
  68. package/src/components/modals/AGModal.vue +16 -13
  69. package/src/components/modals/AGModalTitle.vue +9 -0
  70. package/src/components/modals/AGPromptModal.ts +36 -0
  71. package/src/components/modals/AGPromptModal.vue +34 -0
  72. package/src/components/modals/index.ts +16 -7
  73. package/src/components/snackbars/AGSnackbar.vue +36 -0
  74. package/src/components/snackbars/index.ts +3 -0
  75. package/src/components/utils.ts +10 -0
  76. package/src/directives/index.ts +20 -3
  77. package/src/directives/measure.ts +21 -0
  78. package/src/errors/Errors.ts +65 -12
  79. package/src/errors/index.ts +26 -1
  80. package/src/errors/utils.ts +19 -0
  81. package/src/forms/Form.ts +57 -9
  82. package/src/forms/index.ts +1 -0
  83. package/src/forms/utils.ts +15 -0
  84. package/src/jobs/Job.ts +5 -0
  85. package/src/jobs/index.ts +7 -0
  86. package/src/lang/Lang.ts +15 -23
  87. package/src/main.histoire.ts +1 -0
  88. package/src/main.ts +3 -2
  89. package/src/plugins/Plugin.ts +1 -0
  90. package/src/plugins/index.ts +19 -0
  91. package/src/services/App.state.ts +22 -4
  92. package/src/services/App.ts +38 -5
  93. package/src/services/Cache.ts +43 -0
  94. package/src/services/Events.test.ts +39 -0
  95. package/src/services/Events.ts +100 -30
  96. package/src/services/Service.ts +55 -16
  97. package/src/services/index.ts +11 -5
  98. package/src/services/store.ts +8 -5
  99. package/src/testing/index.ts +25 -0
  100. package/src/ui/UI.state.ts +10 -1
  101. package/src/ui/UI.ts +168 -26
  102. package/src/ui/index.ts +12 -3
  103. package/src/utils/composition/events.ts +1 -0
  104. package/src/utils/index.ts +1 -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/index.ts +0 -3
  114. package/src/globals.ts +0 -6
@@ -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,5 +1,6 @@
1
1
  import { MagicObject, PromisedValue, Storage, isEmpty, objectDeepClone, objectOnly } from '@noeldemartin/utils';
2
2
  import type { Constructor } from '@noeldemartin/utils';
3
+ import type { MaybeRef } from 'vue';
3
4
  import type { Store } from 'pinia';
4
5
 
5
6
  import ServiceBootError from '@/errors/ServiceBootError';
@@ -8,9 +9,12 @@ import { defineServiceStore } from '@/services/store';
8
9
  export type ServiceState = Record<string, any>; // eslint-disable-line @typescript-eslint/no-explicit-any
9
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];
17
+ [K in keyof TComputedState]: (state: UnrefServiceState<TState>) => TComputedState[K];
14
18
  } & ThisType<{
15
19
  readonly [K in keyof TComputedState]: TComputedState[K];
16
20
  }>;
@@ -20,12 +24,14 @@ export function defineServiceState<
20
24
  ComputedState extends ServiceState = {}
21
25
  >(options: {
22
26
  name: string;
23
- initialState: State;
27
+ initialState: State | (() => State);
24
28
  persist?: (keyof State)[];
25
29
  computed?: ComputedStateDefinition<State, ComputedState>;
26
30
  serialize?: (state: Partial<State>) => Partial<State>;
27
- }): Constructor<State> & Constructor<ComputedState> & Constructor<Service<State, ComputedState, Partial<State>>> {
28
- return class extends Service<State, ComputedState> {
31
+ }): Constructor<UnrefServiceState<State>> &
32
+ Constructor<ComputedState> &
33
+ Constructor<Service<UnrefServiceState<State>, ComputedState, Partial<UnrefServiceState<State>>>> {
34
+ return class extends Service<UnrefServiceState<State>, ComputedState> {
29
35
 
30
36
  public static persist = (options.persist as string[]) ?? [];
31
37
 
@@ -37,21 +43,41 @@ export function defineServiceState<
37
43
  return options.name ?? null;
38
44
  }
39
45
 
40
- protected getInitialState(): State {
41
- 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>);
42
68
  }
43
69
 
44
- protected getComputedStateDefinition(): ComputedStateDefinition<State, ComputedState> {
45
- return options.computed ?? ({} as ComputedStateDefinition<State, ComputedState>);
70
+ protected getComputedStateDefinition(): ComputedStateDefinition<UnrefServiceState<State>, ComputedState> {
71
+ return (options.computed ?? {}) as ComputedStateDefinition<UnrefServiceState<State>, ComputedState>;
46
72
  }
47
73
 
48
74
  protected serializePersistedState(state: Partial<State>): Partial<State> {
49
75
  return options.serialize?.(state) ?? state;
50
76
  }
51
-
52
- } as unknown as Constructor<State> &
77
+
78
+ } as unknown as Constructor<UnrefServiceState<State>> &
53
79
  Constructor<ComputedState> &
54
- Constructor<Service<State, ComputedState, Partial<State>>>;
80
+ Constructor<Service<UnrefServiceState<State>, ComputedState, Partial<UnrefServiceState<State>>>>;
55
81
  }
56
82
 
57
83
  export default class Service<
@@ -65,7 +91,7 @@ export default class Service<
65
91
  protected _name: string;
66
92
  private _booted: PromisedValue<void>;
67
93
  private _computedStateKeys: Set<keyof State>;
68
- private _store?: Store | false;
94
+ private _store: Store | false;
69
95
 
70
96
  constructor() {
71
97
  super();
@@ -93,7 +119,8 @@ export default class Service<
93
119
  const handleError = (error: unknown) => this._booted.reject(new ServiceBootError(this._name, error));
94
120
 
95
121
  try {
96
- this.boot()
122
+ this.frameworkBoot()
123
+ .then(() => this.boot())
97
124
  .then(() => this._booted.resolve())
98
125
  .catch(handleError);
99
126
  } catch (error) {
@@ -103,6 +130,10 @@ export default class Service<
103
130
  return this._booted;
104
131
  }
105
132
 
133
+ public hasPersistedState(): boolean {
134
+ return Storage.has(this._name);
135
+ }
136
+
106
137
  public hasState<P extends keyof State>(property: P): boolean {
107
138
  if (!this._store) {
108
139
  return false;
@@ -161,7 +192,11 @@ export default class Service<
161
192
  return;
162
193
  }
163
194
 
164
- const storage = Storage.require<ServiceStorage>(this._name);
195
+ const storage = Storage.get<ServiceStorage>(this._name);
196
+
197
+ if (!storage) {
198
+ return;
199
+ }
165
200
 
166
201
  Storage.set(this._name, {
167
202
  ...storage,
@@ -189,11 +224,15 @@ export default class Service<
189
224
  return state;
190
225
  }
191
226
 
227
+ protected async frameworkBoot(): Promise<void> {
228
+ this.initializePersistedState();
229
+ }
230
+
192
231
  protected async boot(): Promise<void> {
193
- this.restorePersistedState();
232
+ // Placeholder for overrides, don't place any functionality here.
194
233
  }
195
234
 
196
- protected restorePersistedState(): void {
235
+ protected initializePersistedState(): void {
197
236
  // TODO fix this.static()
198
237
  const persist = (this.constructor as unknown as { persist: string[] }).persist;
199
238
 
@@ -3,15 +3,18 @@ import type { App as VueApp } from 'vue';
3
3
  import { definePlugin } from '@/plugins';
4
4
 
5
5
  import App from './App';
6
+ import Cache from './Cache';
6
7
  import Events from './Events';
7
8
  import Service from './Service';
8
9
  import { getPiniaStore } from './store';
9
10
 
10
11
  export * from './App';
12
+ export * from './Cache';
11
13
  export * from './Events';
12
14
  export * from './Service';
15
+ export * from './store';
13
16
 
14
- export { App, Events, Service };
17
+ export { App, Cache, Events, Service };
15
18
 
16
19
  const defaultServices = {
17
20
  $app: App,
@@ -24,13 +27,16 @@ export interface Services extends DefaultServices {}
24
27
 
25
28
  export async function bootServices(app: VueApp, services: Record<string, Service>): Promise<void> {
26
29
  await Promise.all(
27
- Object.entries(services).map(async ([_, service]) => {
28
- // eslint-disable-next-line no-console
29
- await service.launch().catch((error) => console.error(error));
30
+ Object.entries(services).map(async ([name, service]) => {
31
+ await service
32
+ .launch()
33
+ .catch((error) => app.config.errorHandler?.(error, null, `Failed launching ${name}.`));
30
34
  }),
31
35
  );
32
36
 
33
37
  Object.assign(app.config.globalProperties, services);
38
+
39
+ App.development && Object.assign(window, services);
34
40
  }
35
41
 
36
42
  export default definePlugin({
@@ -47,7 +53,7 @@ export default definePlugin({
47
53
  });
48
54
 
49
55
  declare module '@/bootstrap/options' {
50
- interface AerogelOptions {
56
+ export interface AerogelOptions {
51
57
  services?: Record<string, Service>;
52
58
  }
53
59
  }
@@ -1,16 +1,19 @@
1
+ import { tap } from '@noeldemartin/utils';
1
2
  import { createPinia, defineStore, setActivePinia } from 'pinia';
2
3
  import type { DefineStoreOptions, Pinia, StateTree, Store, _GettersTree } from 'pinia';
3
4
 
4
5
  let _store: Pinia | null = null;
5
6
 
6
7
  function initializePiniaStore(): Pinia {
7
- if (!_store) {
8
- _store = createPinia();
8
+ return _store ?? resetPiniaStore();
9
+ }
9
10
 
10
- setActivePinia(_store);
11
- }
11
+ export function resetPiniaStore(): Pinia {
12
+ return tap(createPinia(), (store) => {
13
+ _store = store;
12
14
 
13
- return _store;
15
+ setActivePinia(store);
16
+ });
14
17
  }
15
18
 
16
19
  export function getPiniaStore(): Pinia {
@@ -0,0 +1,25 @@
1
+ import type { GetClosureArgs } from '@noeldemartin/utils';
2
+
3
+ import Events from '@/services/Events';
4
+ import { definePlugin } from '@/plugins';
5
+
6
+ export interface AerogelTestingRuntime {
7
+ on: (typeof Events)['on'];
8
+ }
9
+
10
+ export default definePlugin({
11
+ async install() {
12
+ if (import.meta.env.MODE !== 'testing') {
13
+ return;
14
+ }
15
+
16
+ globalThis.testingRuntime = {
17
+ on: ((...args: GetClosureArgs<(typeof Events)['on']>) => Events.on(...args)) as (typeof Events)['on'],
18
+ };
19
+ },
20
+ });
21
+
22
+ declare global {
23
+ // eslint-disable-next-line no-var
24
+ var testingRuntime: AerogelTestingRuntime | undefined;
25
+ }
@@ -17,7 +17,16 @@ export interface ModalComponent<
17
17
  Result = unknown
18
18
  > {}
19
19
 
20
+ export interface Snackbar {
21
+ id: string;
22
+ component: Component;
23
+ properties: Record<string, unknown>;
24
+ }
25
+
20
26
  export default defineServiceState({
21
27
  name: 'ui',
22
- initialState: { modals: [] as Modal[] },
28
+ initialState: {
29
+ modals: [] as Modal[],
30
+ snackbars: [] as Snackbar[],
31
+ },
23
32
  });