@aerogel/core 0.0.0-next.980a397d575dcb5ff8c5a0bff769d09f938ea03c → 0.0.0-next.9f9564ab9f8da05f60d7868db361edbc5601ee39

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 (78) 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 +689 -140
  4. package/dist/aerogel-core.esm.js +1 -1
  5. package/dist/aerogel-core.esm.js.map +1 -1
  6. package/package.json +2 -2
  7. package/src/bootstrap/bootstrap.test.ts +3 -3
  8. package/src/bootstrap/index.ts +16 -5
  9. package/src/bootstrap/options.ts +3 -0
  10. package/src/components/AGAppLayout.vue +3 -2
  11. package/src/components/AGAppOverlays.vue +5 -1
  12. package/src/components/AGAppSnackbars.vue +1 -1
  13. package/src/components/forms/AGCheckbox.vue +7 -1
  14. package/src/components/forms/AGForm.vue +9 -10
  15. package/src/components/forms/AGInput.vue +10 -6
  16. package/src/components/forms/AGSelect.story.vue +21 -3
  17. package/src/components/forms/AGSelect.vue +10 -3
  18. package/src/components/headless/forms/AGHeadlessButton.vue +17 -12
  19. package/src/components/headless/forms/AGHeadlessInput.ts +12 -13
  20. package/src/components/headless/forms/AGHeadlessInput.vue +3 -3
  21. package/src/components/headless/forms/AGHeadlessInputDescription.vue +28 -0
  22. package/src/components/headless/forms/AGHeadlessInputInput.vue +40 -4
  23. package/src/components/headless/forms/AGHeadlessInputTextArea.vue +40 -0
  24. package/src/components/headless/forms/AGHeadlessSelect.ts +20 -22
  25. package/src/components/headless/forms/AGHeadlessSelect.vue +23 -22
  26. package/src/components/headless/forms/AGHeadlessSelectOption.vue +6 -6
  27. package/src/components/headless/forms/composition.ts +10 -0
  28. package/src/components/headless/forms/index.ts +3 -0
  29. package/src/components/headless/modals/AGHeadlessModal.ts +19 -1
  30. package/src/components/headless/modals/AGHeadlessModal.vue +3 -5
  31. package/src/components/headless/snackbars/index.ts +23 -8
  32. package/src/components/lib/AGErrorMessage.vue +2 -2
  33. package/src/components/lib/AGMarkdown.vue +9 -4
  34. package/src/components/lib/AGMeasured.vue +15 -0
  35. package/src/components/lib/index.ts +1 -0
  36. package/src/components/modals/AGAlertModal.ts +15 -0
  37. package/src/components/modals/AGAlertModal.vue +3 -14
  38. package/src/components/modals/AGConfirmModal.ts +17 -0
  39. package/src/components/modals/AGConfirmModal.vue +6 -10
  40. package/src/components/modals/AGErrorReportModal.ts +27 -1
  41. package/src/components/modals/AGErrorReportModal.vue +7 -15
  42. package/src/components/modals/AGErrorReportModalButtons.vue +4 -2
  43. package/src/components/modals/AGLoadingModal.ts +14 -0
  44. package/src/components/modals/AGLoadingModal.vue +3 -7
  45. package/src/components/modals/AGModal.vue +14 -12
  46. package/src/components/modals/AGPromptModal.ts +30 -0
  47. package/src/components/modals/AGPromptModal.vue +34 -0
  48. package/src/components/modals/index.ts +11 -19
  49. package/src/components/snackbars/AGSnackbar.vue +2 -8
  50. package/src/components/utils.ts +10 -0
  51. package/src/directives/index.ts +5 -1
  52. package/src/directives/measure.ts +21 -0
  53. package/src/errors/Errors.ts +26 -24
  54. package/src/errors/index.ts +2 -11
  55. package/src/errors/utils.ts +19 -0
  56. package/src/forms/Form.ts +43 -3
  57. package/src/forms/index.ts +1 -0
  58. package/src/forms/utils.ts +15 -0
  59. package/src/jobs/Job.ts +5 -0
  60. package/src/jobs/index.ts +7 -0
  61. package/src/lang/Lang.ts +14 -22
  62. package/src/main.ts +3 -0
  63. package/src/services/App.state.ts +1 -2
  64. package/src/services/App.ts +21 -3
  65. package/src/services/Cache.ts +43 -0
  66. package/src/services/Events.test.ts +39 -0
  67. package/src/services/Events.ts +100 -30
  68. package/src/services/Service.ts +42 -12
  69. package/src/services/index.ts +5 -2
  70. package/src/services/store.ts +8 -5
  71. package/src/testing/index.ts +25 -0
  72. package/src/ui/UI.ts +101 -15
  73. package/src/ui/index.ts +8 -3
  74. package/src/utils/composition/events.ts +1 -0
  75. package/src/utils/index.ts +1 -0
  76. package/src/utils/tailwindcss.test.ts +26 -0
  77. package/src/utils/tailwindcss.ts +7 -0
  78. package/src/utils/vue.ts +10 -1
@@ -1,4 +1,4 @@
1
- import { facade, forever, updateLocationQueryParameters } from '@noeldemartin/utils';
1
+ import { PromisedValue, facade, forever, updateLocationQueryParameters } from '@noeldemartin/utils';
2
2
 
3
3
  import Events from '@/services/Events';
4
4
  import type { Plugin } from '@/plugins';
@@ -7,6 +7,23 @@ import Service from './App.state';
7
7
 
8
8
  export class AppService extends Service {
9
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
+
10
27
  public async reload(queryParameters?: Record<string, string | undefined>): Promise<void> {
11
28
  queryParameters && updateLocationQueryParameters(queryParameters);
12
29
 
@@ -21,9 +38,10 @@ export class AppService extends Service {
21
38
  }
22
39
 
23
40
  protected async boot(): Promise<void> {
24
- Events.once('application-mounted', () => this.setState({ isMounted: true }));
41
+ Events.once('application-ready', () => this.ready.resolve());
42
+ Events.once('application-mounted', () => this.mounted.resolve());
25
43
  }
26
44
 
27
45
  }
28
46
 
29
- export default facade(new AppService());
47
+ export default facade(AppService);
@@ -0,0 +1,43 @@
1
+ import { PromisedValue, facade, tap } from '@noeldemartin/utils';
2
+
3
+ import Service from '@/services/Service';
4
+
5
+ export class CacheService extends Service {
6
+
7
+ private cache?: PromisedValue<Cache> = undefined;
8
+
9
+ public async get(url: string): Promise<Response | null> {
10
+ const cache = await this.open();
11
+ const response = await cache.match(url);
12
+
13
+ return response ?? null;
14
+ }
15
+
16
+ public async store(url: string, response: Response): Promise<void> {
17
+ const cache = await this.open();
18
+
19
+ await cache.put(url, response);
20
+ }
21
+
22
+ public async replace(url: string, response: Response): Promise<void> {
23
+ const cache = await this.open();
24
+ const keys = await cache.keys(url);
25
+
26
+ if (keys.length === 0) {
27
+ return;
28
+ }
29
+
30
+ await cache.put(url, response);
31
+ }
32
+
33
+ protected async open(): Promise<Cache> {
34
+ return (this.cache =
35
+ this.cache ??
36
+ tap(new PromisedValue<Cache>(), (cache) => {
37
+ caches.open('app').then((instance) => cache.resolve(instance));
38
+ }));
39
+ }
40
+
41
+ }
42
+
43
+ export default facade(CacheService);
@@ -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,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();
@@ -104,6 +130,10 @@ export default class Service<
104
130
  return this._booted;
105
131
  }
106
132
 
133
+ public hasPersistedState(): boolean {
134
+ return Storage.has(this._name);
135
+ }
136
+
107
137
  public hasState<P extends keyof State>(property: P): boolean {
108
138
  if (!this._store) {
109
139
  return false;
@@ -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,
@@ -50,7 +53,7 @@ export default definePlugin({
50
53
  });
51
54
 
52
55
  declare module '@/bootstrap/options' {
53
- interface AerogelOptions {
56
+ export interface AerogelOptions {
54
57
  services?: Record<string, Service>;
55
58
  }
56
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
+ }
package/src/ui/UI.ts CHANGED
@@ -5,6 +5,7 @@ import type { ObjectValues } from '@noeldemartin/utils';
5
5
 
6
6
  import Events from '@/services/Events';
7
7
  import type { SnackbarAction, SnackbarColor } from '@/components/headless/snackbars';
8
+ import type { AGAlertModalProps, AGConfirmModalProps, AGLoadingModalProps, AGPromptModalProps } from '@/components';
8
9
 
9
10
  import Service from './UI.state';
10
11
  import type { Modal, ModalComponent, Snackbar } from './UI.state';
@@ -24,11 +25,27 @@ export const UIComponents = {
24
25
  ConfirmModal: 'confirm-modal',
25
26
  ErrorReportModal: 'error-report-modal',
26
27
  LoadingModal: 'loading-modal',
28
+ PromptModal: 'prompt-modal',
27
29
  Snackbar: 'snackbar',
30
+ StartupCrash: 'startup-crash',
28
31
  } as const;
29
32
 
30
33
  export type UIComponent = ObjectValues<typeof UIComponents>;
31
34
 
35
+ export interface ConfirmOptions {
36
+ acceptText?: string;
37
+ cancelText?: string;
38
+ }
39
+
40
+ export interface PromptOptions {
41
+ label?: string;
42
+ defaultValue?: string;
43
+ placeholder?: string;
44
+ acceptText?: string;
45
+ cancelText?: string;
46
+ trim?: boolean;
47
+ }
48
+
32
49
  export interface ShowSnackbarOptions {
33
50
  component?: Component;
34
51
  color?: SnackbarColor;
@@ -47,33 +64,100 @@ export class UIService extends Service {
47
64
  public alert(message: string): void;
48
65
  public alert(title: string, message: string): void;
49
66
  public alert(messageOrTitle: string, message?: string): void {
50
- const options = typeof message === 'string' ? { title: messageOrTitle, message } : { message: messageOrTitle };
67
+ const getProperties = (): AGAlertModalProps => {
68
+ if (typeof message !== 'string') {
69
+ return { message: messageOrTitle };
70
+ }
71
+
72
+ return {
73
+ title: messageOrTitle,
74
+ message,
75
+ };
76
+ };
51
77
 
52
- this.openModal(this.requireComponent(UIComponents.AlertModal), options);
78
+ this.openModal(this.requireComponent(UIComponents.AlertModal), getProperties());
53
79
  }
54
80
 
55
- public async confirm(message: string): Promise<boolean>;
56
- public async confirm(title: string, message: string): Promise<boolean>;
57
- public async confirm(messageOrTitle: string, message?: string): Promise<boolean> {
58
- const options = typeof message === 'string' ? { title: messageOrTitle, message } : { message: messageOrTitle };
59
- const modal = await this.openModal<ModalComponent<{ message: string }, boolean>>(
81
+ public async confirm(message: string, options?: ConfirmOptions): Promise<boolean>;
82
+ public async confirm(title: string, message: string, options?: ConfirmOptions): Promise<boolean>;
83
+ public async confirm(
84
+ messageOrTitle: string,
85
+ messageOrOptions?: string | ConfirmOptions,
86
+ options?: ConfirmOptions,
87
+ ): Promise<boolean> {
88
+ const getProperties = (): AGConfirmModalProps => {
89
+ if (typeof messageOrOptions !== 'string') {
90
+ return {
91
+ message: messageOrTitle,
92
+ ...(messageOrOptions ?? {}),
93
+ };
94
+ }
95
+
96
+ return {
97
+ title: messageOrTitle,
98
+ message: messageOrOptions,
99
+ ...(options ?? {}),
100
+ };
101
+ };
102
+
103
+ const modal = await this.openModal<ModalComponent<AGConfirmModalProps, boolean>>(
60
104
  this.requireComponent(UIComponents.ConfirmModal),
61
- options,
105
+ getProperties(),
62
106
  );
63
107
  const result = await modal.beforeClose;
64
108
 
65
109
  return result ?? false;
66
110
  }
67
111
 
112
+ public async prompt(message: string, options?: PromptOptions): Promise<string | null>;
113
+ public async prompt(title: string, message: string, options?: PromptOptions): Promise<string | null>;
114
+ public async prompt(
115
+ messageOrTitle: string,
116
+ messageOrOptions?: string | PromptOptions,
117
+ options?: PromptOptions,
118
+ ): Promise<string | null> {
119
+ const trim = options?.trim ?? true;
120
+ const getProperties = (): AGPromptModalProps => {
121
+ if (typeof messageOrOptions !== 'string') {
122
+ return {
123
+ message: messageOrTitle,
124
+ ...(messageOrOptions ?? {}),
125
+ };
126
+ }
127
+
128
+ return {
129
+ title: messageOrTitle,
130
+ message: messageOrOptions,
131
+ ...(options ?? {}),
132
+ };
133
+ };
134
+
135
+ const modal = await this.openModal<ModalComponent<AGPromptModalProps, string | null>>(
136
+ this.requireComponent(UIComponents.PromptModal),
137
+ getProperties(),
138
+ );
139
+ const rawResult = await modal.beforeClose;
140
+ const result = trim ? rawResult?.trim() : rawResult;
141
+
142
+ return result ?? null;
143
+ }
144
+
68
145
  public async loading<T>(operation: Promise<T>): Promise<T>;
69
146
  public async loading<T>(message: string, operation: Promise<T>): Promise<T>;
70
147
  public async loading<T>(messageOrOperation: string | Promise<T>, operation?: Promise<T>): Promise<T> {
71
- operation = typeof messageOrOperation === 'string' ? (operation as Promise<T>) : messageOrOperation;
148
+ const getProperties = (): AGLoadingModalProps => {
149
+ if (typeof messageOrOperation !== 'string') {
150
+ return {};
151
+ }
72
152
 
73
- const message = typeof messageOrOperation === 'string' ? messageOrOperation : undefined;
74
- const modal = await this.openModal(this.requireComponent(UIComponents.LoadingModal), { message });
153
+ return { message: messageOrOperation };
154
+ };
155
+
156
+ const modal = await this.openModal(this.requireComponent(UIComponents.LoadingModal), getProperties());
75
157
 
76
158
  try {
159
+ operation = typeof messageOrOperation === 'string' ? (operation as Promise<T>) : messageOrOperation;
160
+
77
161
  const [result] = await Promise.all([operation, after({ seconds: 1 })]);
78
162
 
79
163
  return result;
@@ -86,7 +170,7 @@ export class UIService extends Service {
86
170
  const snackbar: Snackbar = {
87
171
  id: uuid(),
88
172
  properties: { message, ...options },
89
- component: options.component ?? markRaw(this.requireComponent(UIComponents.Snackbar)),
173
+ component: markRaw(options.component ?? this.requireComponent(UIComponents.Snackbar)),
90
174
  };
91
175
 
92
176
  this.setState('snackbars', this.snackbars.concat(snackbar));
@@ -189,14 +273,16 @@ export class UIService extends Service {
189
273
 
190
274
  }
191
275
 
192
- export default facade(new UIService());
276
+ export default facade(UIService);
193
277
 
194
278
  declare module '@/services/Events' {
195
279
  export interface EventsPayload {
196
- 'modal-will-close': { modal: Modal; result?: unknown };
197
- 'modal-closed': { modal: Modal; result?: unknown };
198
280
  'close-modal': { id: string; result?: unknown };
199
281
  'hide-modal': { id: string };
282
+ 'hide-overlays-backdrop': void;
283
+ 'modal-closed': { modal: Modal; result?: unknown };
284
+ 'modal-will-close': { modal: Modal; result?: unknown };
200
285
  'show-modal': { id: string };
286
+ 'show-overlays-backdrop': void;
201
287
  }
202
288
  }