@aerogel/core 0.0.0-next.980a397d575dcb5ff8c5a0bff769d09f938ea03c → 0.0.0-next.b18f4e0acd39431045c2f444c711303890143193

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 (55) 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 +471 -96
  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 +13 -3
  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/forms/AGCheckbox.vue +7 -1
  13. package/src/components/forms/AGInput.vue +8 -6
  14. package/src/components/forms/AGSelect.story.vue +21 -3
  15. package/src/components/forms/AGSelect.vue +10 -3
  16. package/src/components/headless/forms/AGHeadlessInput.ts +5 -10
  17. package/src/components/headless/forms/AGHeadlessSelect.ts +20 -22
  18. package/src/components/headless/forms/AGHeadlessSelect.vue +23 -22
  19. package/src/components/headless/forms/AGHeadlessSelectOption.vue +6 -6
  20. package/src/components/headless/modals/AGHeadlessModal.ts +19 -1
  21. package/src/components/headless/modals/AGHeadlessModal.vue +3 -5
  22. package/src/components/headless/snackbars/index.ts +23 -8
  23. package/src/components/lib/AGMeasured.vue +15 -0
  24. package/src/components/lib/index.ts +1 -0
  25. package/src/components/modals/AGAlertModal.ts +15 -0
  26. package/src/components/modals/AGAlertModal.vue +3 -14
  27. package/src/components/modals/AGConfirmModal.ts +17 -0
  28. package/src/components/modals/AGConfirmModal.vue +6 -10
  29. package/src/components/modals/AGErrorReportModal.ts +27 -1
  30. package/src/components/modals/AGErrorReportModal.vue +7 -15
  31. package/src/components/modals/AGErrorReportModalButtons.vue +4 -2
  32. package/src/components/modals/AGLoadingModal.ts +14 -0
  33. package/src/components/modals/AGLoadingModal.vue +3 -7
  34. package/src/components/modals/AGModal.vue +14 -12
  35. package/src/components/modals/AGPromptModal.ts +30 -0
  36. package/src/components/modals/AGPromptModal.vue +34 -0
  37. package/src/components/modals/index.ts +11 -19
  38. package/src/components/snackbars/AGSnackbar.vue +2 -8
  39. package/src/components/utils.ts +10 -0
  40. package/src/directives/index.ts +3 -1
  41. package/src/directives/measure.ts +12 -0
  42. package/src/errors/Errors.ts +11 -6
  43. package/src/errors/index.ts +1 -1
  44. package/src/forms/Form.ts +1 -0
  45. package/src/services/App.state.ts +0 -1
  46. package/src/services/App.ts +20 -2
  47. package/src/services/Service.ts +14 -8
  48. package/src/services/index.ts +1 -1
  49. package/src/ui/UI.ts +92 -11
  50. package/src/ui/index.ts +8 -3
  51. package/src/utils/composition/events.ts +1 -0
  52. package/src/utils/index.ts +1 -0
  53. package/src/utils/tailwindcss.test.ts +26 -0
  54. package/src/utils/tailwindcss.ts +7 -0
  55. package/src/utils/vue.ts +10 -1
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,26 @@ 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
+ }
47
+
32
48
  export interface ShowSnackbarOptions {
33
49
  component?: Component;
34
50
  color?: SnackbarColor;
@@ -47,33 +63,98 @@ export class UIService extends Service {
47
63
  public alert(message: string): void;
48
64
  public alert(title: string, message: string): void;
49
65
  public alert(messageOrTitle: string, message?: string): void {
50
- const options = typeof message === 'string' ? { title: messageOrTitle, message } : { message: messageOrTitle };
66
+ const getProperties = (): AGAlertModalProps => {
67
+ if (typeof message !== 'string') {
68
+ return { message: messageOrTitle };
69
+ }
70
+
71
+ return {
72
+ title: messageOrTitle,
73
+ message,
74
+ };
75
+ };
51
76
 
52
- this.openModal(this.requireComponent(UIComponents.AlertModal), options);
77
+ this.openModal(this.requireComponent(UIComponents.AlertModal), getProperties());
53
78
  }
54
79
 
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>>(
80
+ public async confirm(message: string, options?: ConfirmOptions): Promise<boolean>;
81
+ public async confirm(title: string, message: string, options?: ConfirmOptions): Promise<boolean>;
82
+ public async confirm(
83
+ messageOrTitle: string,
84
+ messageOrOptions?: string | ConfirmOptions,
85
+ options?: ConfirmOptions,
86
+ ): Promise<boolean> {
87
+ const getProperties = (): AGConfirmModalProps => {
88
+ if (typeof messageOrOptions !== 'string') {
89
+ return {
90
+ message: messageOrTitle,
91
+ ...(messageOrOptions ?? {}),
92
+ };
93
+ }
94
+
95
+ return {
96
+ title: messageOrTitle,
97
+ message: messageOrOptions,
98
+ ...(options ?? {}),
99
+ };
100
+ };
101
+
102
+ const modal = await this.openModal<ModalComponent<AGConfirmModalProps, boolean>>(
60
103
  this.requireComponent(UIComponents.ConfirmModal),
61
- options,
104
+ getProperties(),
62
105
  );
63
106
  const result = await modal.beforeClose;
64
107
 
65
108
  return result ?? false;
66
109
  }
67
110
 
111
+ public async prompt(message: string, options?: PromptOptions): Promise<string | null>;
112
+ public async prompt(title: string, message: string, options?: PromptOptions): Promise<string | null>;
113
+ public async prompt(
114
+ messageOrTitle: string,
115
+ messageOrOptions?: string | PromptOptions,
116
+ options?: PromptOptions,
117
+ ): Promise<string | null> {
118
+ const getProperties = (): AGPromptModalProps => {
119
+ if (typeof messageOrOptions !== 'string') {
120
+ return {
121
+ message: messageOrTitle,
122
+ ...(messageOrOptions ?? {}),
123
+ };
124
+ }
125
+
126
+ return {
127
+ title: messageOrTitle,
128
+ message: messageOrOptions,
129
+ ...(options ?? {}),
130
+ };
131
+ };
132
+
133
+ const modal = await this.openModal<ModalComponent<AGPromptModalProps, string | null>>(
134
+ this.requireComponent(UIComponents.PromptModal),
135
+ getProperties(),
136
+ );
137
+ const result = await modal.beforeClose;
138
+
139
+ return result ?? null;
140
+ }
141
+
68
142
  public async loading<T>(operation: Promise<T>): Promise<T>;
69
143
  public async loading<T>(message: string, operation: Promise<T>): Promise<T>;
70
144
  public async loading<T>(messageOrOperation: string | Promise<T>, operation?: Promise<T>): Promise<T> {
71
- operation = typeof messageOrOperation === 'string' ? (operation as Promise<T>) : messageOrOperation;
145
+ const getProperties = (): AGLoadingModalProps => {
146
+ if (typeof messageOrOperation !== 'string') {
147
+ return {};
148
+ }
72
149
 
73
- const message = typeof messageOrOperation === 'string' ? messageOrOperation : undefined;
74
- const modal = await this.openModal(this.requireComponent(UIComponents.LoadingModal), { message });
150
+ return { message: messageOrOperation };
151
+ };
152
+
153
+ const modal = await this.openModal(this.requireComponent(UIComponents.LoadingModal), getProperties());
75
154
 
76
155
  try {
156
+ operation = typeof messageOrOperation === 'string' ? (operation as Promise<T>) : messageOrOperation;
157
+
77
158
  const [result] = await Promise.all([operation, after({ seconds: 1 })]);
78
159
 
79
160
  return result;
package/src/ui/index.ts CHANGED
@@ -8,13 +8,16 @@ import AGAlertModal from '../components/modals/AGAlertModal.vue';
8
8
  import AGConfirmModal from '../components/modals/AGConfirmModal.vue';
9
9
  import AGErrorReportModal from '../components/modals/AGErrorReportModal.vue';
10
10
  import AGLoadingModal from '../components/modals/AGLoadingModal.vue';
11
+ import AGPromptModal from '../components/modals/AGPromptModal.vue';
11
12
  import AGSnackbar from '../components/snackbars/AGSnackbar.vue';
13
+ import AGStartupCrash from '../components/lib/AGStartupCrash.vue';
12
14
  import type { UIComponent } from './UI';
13
15
 
14
- export { UI, UIComponents, UIComponent };
15
-
16
16
  const services = { $ui: UI };
17
17
 
18
+ export * from './UI';
19
+ export { default as UI } from './UI';
20
+
18
21
  export type UIServices = typeof services;
19
22
 
20
23
  export default definePlugin({
@@ -24,7 +27,9 @@ export default definePlugin({
24
27
  [UIComponents.ConfirmModal]: AGConfirmModal,
25
28
  [UIComponents.ErrorReportModal]: AGErrorReportModal,
26
29
  [UIComponents.LoadingModal]: AGLoadingModal,
30
+ [UIComponents.PromptModal]: AGPromptModal,
27
31
  [UIComponents.Snackbar]: AGSnackbar,
32
+ [UIComponents.StartupCrash]: AGStartupCrash,
28
33
  };
29
34
 
30
35
  Object.entries({
@@ -37,7 +42,7 @@ export default definePlugin({
37
42
  });
38
43
 
39
44
  declare module '@/bootstrap/options' {
40
- interface AerogelOptions {
45
+ export interface AerogelOptions {
41
46
  components?: Partial<Record<UIComponent, Component>>;
42
47
  }
43
48
  }
@@ -14,6 +14,7 @@ export function useEvent<Event extends EventWithPayload>(
14
14
  event: Event,
15
15
  listener: EventListener<EventsPayload[Event]>
16
16
  ): void;
17
+ export function useEvent<Payload>(event: string, listener: (payload: Payload) => unknown): void;
17
18
  export function useEvent<Event extends string>(event: UnknownEvent<Event>, listener: EventListener): void;
18
19
 
19
20
  export function useEvent(event: string, listener: EventListener): void {
@@ -1,4 +1,5 @@
1
1
  export * from './composition/events';
2
2
  export * from './composition/forms';
3
3
  export * from './composition/hooks';
4
+ export * from './tailwindcss';
4
5
  export * from './vue';
@@ -0,0 +1,26 @@
1
+ import { describe, expect, it } from 'vitest';
2
+
3
+ import { removeInteractiveClasses } from './tailwindcss';
4
+
5
+ describe('TailwindCSS utils', () => {
6
+
7
+ it('Removes interactive classes', () => {
8
+ const cases: [string, string][] = [
9
+ ['text-red hover:text-green', 'text-red'],
10
+ ['text-red hover:text-green text-lg', 'text-red text-lg'],
11
+ [
12
+ `
13
+ text-red text-lg
14
+ focus:text-yellow
15
+ hover:focus:text-black
16
+ `,
17
+ 'text-red text-lg',
18
+ ],
19
+ ];
20
+
21
+ cases.forEach(([original, expected]) => {
22
+ expect(removeInteractiveClasses(original)).toEqual(expected);
23
+ });
24
+ });
25
+
26
+ });
@@ -0,0 +1,7 @@
1
+ export function removeInteractiveClasses(classes: string): string {
2
+ return classes
3
+ .split(/\s+/)
4
+ .filter((className) => !/^(hover|focus|focus-visible):/.test(className))
5
+ .join(' ')
6
+ .trim();
7
+ }
package/src/utils/vue.ts CHANGED
@@ -1,5 +1,5 @@
1
1
  import { fail } from '@noeldemartin/utils';
2
- import { inject, reactive, ref } from 'vue';
2
+ import { computed, inject, reactive, ref, watch } from 'vue';
3
3
  import type { Directive, InjectionKey, PropType, Ref, UnwrapNestedRefs } from 'vue';
4
4
 
5
5
  type BaseProp<T> = {
@@ -30,6 +30,15 @@ export function componentRef<T>(): Ref<UnwrapNestedRefs<T> | undefined> {
30
30
  return ref<UnwrapNestedRefs<T>>();
31
31
  }
32
32
 
33
+ export function computedAsync<T>(getter: () => Promise<T>): Ref<T | undefined> {
34
+ const result = ref<T>();
35
+ const asyncValue = computed(getter);
36
+
37
+ watch(asyncValue, async () => (result.value = await asyncValue.value), { immediate: true });
38
+
39
+ return result;
40
+ }
41
+
33
42
  export function defineDirective(directive: Directive): Directive {
34
43
  return directive;
35
44
  }