@dotglitch/ngx-common 1.1.17 → 1.1.19

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 (52) hide show
  1. package/components/command-palette/breadcrumb/breadcrumb.component.d.ts +16 -0
  2. package/components/command-palette/command-palette.component.d.ts +21 -3
  3. package/components/command-palette/shortcut/shortcut.component.d.ts +1 -1
  4. package/components/menu/menu.component.d.ts +1 -1
  5. package/directives/image-cache.directive.d.ts +46 -0
  6. package/directives/menu.directive.d.ts +1 -1
  7. package/esm2022/components/command-palette/breadcrumb/breadcrumb.component.mjs +30 -0
  8. package/esm2022/components/command-palette/command-palette.component.mjs +191 -54
  9. package/esm2022/components/command-palette/shortcut/shortcut.component.mjs +5 -5
  10. package/esm2022/components/dynamic-html/dynamic-html.component.mjs +3 -3
  11. package/esm2022/components/dynamic-html/dynamic-html.module.mjs +4 -4
  12. package/esm2022/components/dynamic-html/dynamic-html.service.mjs +3 -3
  13. package/esm2022/components/filemanager/file-grid/file-grid.component.mjs +3 -3
  14. package/esm2022/components/filemanager/filemanager.component.mjs +3 -3
  15. package/esm2022/components/filemanager/folder-rename/folder-rename.component.mjs +3 -3
  16. package/esm2022/components/filemanager/toolbar/breadcrumb/breadcrumb.component.mjs +3 -3
  17. package/esm2022/components/filemanager/toolbar/icon-button/icon-button.component.mjs +3 -3
  18. package/esm2022/components/filemanager/toolbar/toolbar.component.mjs +3 -3
  19. package/esm2022/components/filemanager/tree-view/tree-view.component.mjs +3 -3
  20. package/esm2022/components/lazy-loader/lazy-loader.component.mjs +3 -3
  21. package/esm2022/components/lazy-loader/lazy-loader.module.mjs +4 -4
  22. package/esm2022/components/lazy-loader/lazy-loader.service.mjs +3 -3
  23. package/esm2022/components/menu/menu.component.mjs +11 -8
  24. package/esm2022/components/react-magic-wrapper/react-magic-wrapper.component.mjs +3 -3
  25. package/esm2022/components/tabulator/tabulator.component.mjs +3 -3
  26. package/esm2022/components/tooltip/tooltip.component.mjs +3 -3
  27. package/esm2022/components/vscode/vscode.component.mjs +3 -3
  28. package/esm2022/directives/image-cache.directive.mjs +143 -0
  29. package/esm2022/directives/menu.directive.mjs +27 -29
  30. package/esm2022/directives/tooltip.directive.mjs +24 -28
  31. package/esm2022/pipes/html-bypass.pipe.mjs +3 -3
  32. package/esm2022/pipes/resource-bypass.pipe.mjs +3 -3
  33. package/esm2022/pipes/script-bypass.pipe.mjs +3 -3
  34. package/esm2022/pipes/style-bypass.pipe.mjs +3 -3
  35. package/esm2022/pipes/url-bypass.pipe.mjs +3 -3
  36. package/esm2022/public-api.mjs +2 -1
  37. package/esm2022/services/command-palette.service.mjs +8 -5
  38. package/esm2022/services/dependency.service.mjs +3 -3
  39. package/esm2022/services/dialog.service.mjs +3 -3
  40. package/esm2022/services/fetch.service.mjs +3 -3
  41. package/esm2022/services/file.service.mjs +3 -3
  42. package/esm2022/services/keyboard.service.mjs +3 -3
  43. package/esm2022/services/navigation.service.mjs +3 -3
  44. package/esm2022/services/theme.service.mjs +3 -3
  45. package/fesm2022/{dotglitch-ngx-common-folder-rename.component-0RqscdRS.mjs → dotglitch-ngx-common-folder-rename.component-VI_uH5-6.mjs} +8 -7
  46. package/fesm2022/dotglitch-ngx-common-folder-rename.component-VI_uH5-6.mjs.map +1 -0
  47. package/fesm2022/dotglitch-ngx-common.mjs +515 -215
  48. package/fesm2022/dotglitch-ngx-common.mjs.map +1 -1
  49. package/package.json +5 -4
  50. package/public-api.d.ts +1 -0
  51. package/services/command-palette.service.d.ts +20 -1
  52. package/fesm2022/dotglitch-ngx-common-folder-rename.component-0RqscdRS.mjs.map +0 -1
@@ -0,0 +1,16 @@
1
+ import { CommandPaletteComponent } from '../command-palette.component';
2
+ import { CommandAction } from '../../../services/command-palette.service';
3
+ import * as i0 from "@angular/core";
4
+ export declare class BreadcrumbComponent {
5
+ private commandPalette;
6
+ breadcrumbs: {
7
+ action: CommandAction;
8
+ commands: CommandAction[];
9
+ destroying: boolean;
10
+ selectedIndex: number;
11
+ }[];
12
+ constructor(commandPalette: CommandPaletteComponent);
13
+ selectBreadcrumb(crumb: any): void;
14
+ static ɵfac: i0.ɵɵFactoryDeclaration<BreadcrumbComponent, never>;
15
+ static ɵcmp: i0.ɵɵComponentDeclaration<BreadcrumbComponent, "ngx-commandpalette-breadcrumb", never, { "breadcrumbs": { "alias": "breadcrumbs"; "required": false; }; }, {}, never, never, true, never>;
16
+ }
@@ -1,20 +1,38 @@
1
- import { OnInit } from '@angular/core';
1
+ import { ChangeDetectorRef, ElementRef, OnInit } from '@angular/core';
2
2
  import { CommandAction, CommandPaletteService } from '../../services/command-palette.service';
3
3
  import { MatDialogRef } from '@angular/material/dialog';
4
+ import { NgScrollbar } from 'ngx-scrollbar';
4
5
  import * as i0 from "@angular/core";
5
6
  export declare class CommandPaletteComponent implements OnInit {
6
7
  private readonly commandPalette;
7
8
  private readonly dialog;
9
+ private readonly elementRef;
10
+ private readonly changeDetector;
11
+ scrollbar: NgScrollbar;
12
+ textInput: ElementRef<HTMLInputElement>;
8
13
  contextElement: HTMLElement;
14
+ get el(): HTMLElement;
15
+ readonly MAT_ICON_REGEX: RegExp;
9
16
  queryString: string;
10
17
  activeIndex: number;
18
+ readonly rowHeight = 29;
19
+ readonly padding = 6;
11
20
  commands: CommandAction[];
12
21
  filteredCommands: CommandAction[];
13
- constructor(commandPalette: CommandPaletteService, dialog: MatDialogRef<any>, data: any);
22
+ breadcrumbs: {
23
+ action: CommandAction;
24
+ commands: CommandAction[];
25
+ destroying: boolean;
26
+ selectedIndex: number;
27
+ }[];
28
+ constructor(commandPalette: CommandPaletteService, dialog: MatDialogRef<any>, elementRef: ElementRef, changeDetector: ChangeDetectorRef, data: any);
14
29
  ngOnInit(): void;
15
30
  onKeyDown(evt: KeyboardEvent): void;
31
+ private filterResults;
32
+ setCommandList(commands: CommandAction[]): void;
33
+ private focusRow;
16
34
  executeCommand(command: CommandAction): void;
17
35
  onBlur(): void;
18
36
  static ɵfac: i0.ɵɵFactoryDeclaration<CommandPaletteComponent, never>;
19
- static ɵcmp: i0.ɵɵComponentDeclaration<CommandPaletteComponent, "app-command-palette", never, { "contextElement": { "alias": "contextElement"; "required": false; }; }, {}, never, never, true, never>;
37
+ static ɵcmp: i0.ɵɵComponentDeclaration<CommandPaletteComponent, "ngx-command-palette", never, { "contextElement": { "alias": "contextElement"; "required": false; }; }, {}, never, never, true, never>;
20
38
  }
@@ -5,5 +5,5 @@ export declare class ShortcutComponent {
5
5
  keys: string[];
6
6
  ngOnChanges(): void;
7
7
  static ɵfac: i0.ɵɵFactoryDeclaration<ShortcutComponent, never>;
8
- static ɵcmp: i0.ɵɵComponentDeclaration<ShortcutComponent, "app-shortcut", never, { "shortcut": { "alias": "shortcut"; "required": false; }; }, {}, never, never, true, never>;
8
+ static ɵcmp: i0.ɵɵComponentDeclaration<ShortcutComponent, "ngx-commandpalette-shortcut", never, { "shortcut": { "alias": "shortcut"; "required": false; }; }, {}, never, never, true, never>;
9
9
  }
@@ -50,7 +50,7 @@ export declare class MenuComponent implements OnInit {
50
50
  * @param evt
51
51
  * @returns
52
52
  */
53
- onMenuItemClick(item: MenuItem, row: HTMLTableRowElement): Promise<any>;
53
+ onMenuItemClick(item: MenuItem, row: HTMLTableRowElement, keepOpen?: boolean): Promise<any>;
54
54
  /**
55
55
  *
56
56
  * @param label
@@ -0,0 +1,46 @@
1
+ import { ElementRef, InjectionToken } from '@angular/core';
2
+ import * as i0 from "@angular/core";
3
+ export type NgxImageCacheConfig = {
4
+ /**
5
+ * Image to use as a placeholder while loading the main image
6
+ * Recommended to use inlined SVG or a base64 encoded image
7
+ */
8
+ loadingPlaceholder?: string;
9
+ /**
10
+ * Image to use as a placeholder where images fail to load
11
+ * Recommended to use inlined SVG or a base64 encoded image
12
+ */
13
+ brokenPlaceholder?: string;
14
+ };
15
+ export declare const NGX_IMAGE_CACHE_CONFIG: InjectionToken<NgxImageCacheConfig>;
16
+ export type NgxImageCacheConfiguration = {
17
+ /**
18
+ * Max age to cache an image in milliseconds.
19
+ * If set to `0` or a negative number, images will never expire.
20
+ */
21
+ maxAge: number;
22
+ /**
23
+ * Set to `false` to disable memory caching for the image
24
+ * If both `cacheInMemory` and `cacheInIndexedDB` are false,
25
+ * no caching will happen. (You'll still get the loader)
26
+ */
27
+ cacheInMemory: boolean;
28
+ /**
29
+ * Set to `false` to disable indexedDB caching for the image
30
+ * If both `cacheInMemory` and `cacheInIndexedDB` are false,
31
+ * no caching will happen. (You'll still get the loader)
32
+ */
33
+ cacheInIndexedDB: boolean;
34
+ };
35
+ export declare class NgxImageCacheDirective {
36
+ private readonly element;
37
+ private readonly cacheConfig;
38
+ url: string;
39
+ configuration: NgxImageCacheConfiguration;
40
+ private get el();
41
+ constructor(element: ElementRef, cacheConfig: NgxImageCacheConfig);
42
+ ngOnChanges(): void;
43
+ getCachedImage(): Promise<void>;
44
+ static ɵfac: i0.ɵɵFactoryDeclaration<NgxImageCacheDirective, [null, { optional: true; }]>;
45
+ static ɵdir: i0.ɵɵDirectiveDeclaration<NgxImageCacheDirective, "img[ngx-cache]", never, { "url": { "alias": "source"; "required": false; }; "configuration": { "alias": "ngx-cache-config"; "required": false; }; }, {}, never, never, true, never>;
46
+ }
@@ -24,7 +24,7 @@ export declare class MenuDirective {
24
24
  config: MenuOptions;
25
25
  constructor(dialog: MatDialog, viewContainer: ViewContainerRef);
26
26
  ngAfterViewInit(): void;
27
- openMenu(evt: PointerEvent, items?: MenuItem[]): Promise<[value: any]>;
27
+ openMenu(evt: PointerEvent, items?: MenuItem[], keepOpen?: boolean): Promise<[value: any]>;
28
28
  static ɵfac: i0.ɵɵFactoryDeclaration<MenuDirective, never>;
29
29
  static ɵdir: i0.ɵɵDirectiveDeclaration<MenuDirective, "[ngx-contextmenu],[ngx-menu]", never, { "data": { "alias": "ngx-menu-context"; "required": false; }; "ctxMenuItems": { "alias": "ngx-contextmenu"; "required": false; }; "menuItems": { "alias": "ngx-menu"; "required": false; }; "config": { "alias": "ngx-menu-config"; "required": false; }; }, {}, never, never, true, never>;
30
30
  }
@@ -0,0 +1,30 @@
1
+ import { Component, Input } from '@angular/core';
2
+ import * as i0 from "@angular/core";
3
+ import * as i1 from "../command-palette.component";
4
+ export class BreadcrumbComponent {
5
+ constructor(commandPalette) {
6
+ this.commandPalette = commandPalette;
7
+ this.breadcrumbs = [];
8
+ }
9
+ selectBreadcrumb(crumb) {
10
+ const index = this.breadcrumbs.indexOf(crumb);
11
+ if (index == -1)
12
+ throw new Error("Something terrible happened.");
13
+ const layer = this.breadcrumbs.at(-1);
14
+ layer.destroying = true;
15
+ setTimeout(() => {
16
+ this.commandPalette.setCommandList(this.breadcrumbs.at(-2).commands);
17
+ this.commandPalette.breadcrumbs.pop();
18
+ this.commandPalette.activeIndex = layer.selectedIndex;
19
+ }, 190);
20
+ }
21
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.1.2", ngImport: i0, type: BreadcrumbComponent, deps: [{ token: i1.CommandPaletteComponent }], target: i0.ɵɵFactoryTarget.Component }); }
22
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.1.2", type: BreadcrumbComponent, isStandalone: true, selector: "ngx-commandpalette-breadcrumb", inputs: { breadcrumbs: "breadcrumbs" }, ngImport: i0, template: "@for (crumb of breadcrumbs; track crumb; let i = $index) {\n <div class=\"crumb\">\n <div\n class=\"crumb_inner\"\n [class.destroy]=\"crumb.destroying\"\n [class.root]=\"i == 0\"\n (click)=\"selectBreadcrumb(crumb)\"\n >\n {{crumb.action.label}}\n </div>\n </div>\n}\n", styles: [":host{display:flex}.crumb{overflow:hidden;height:26px;padding:0 16px;margin:0 -16px}.crumb .crumb_inner{--background-color: #3498db;color:#fff;position:relative;background:var(--background-color);padding:2px 5px 0 6px;margin-right:12px;text-align:center;white-space:pre;font-size:14px;-webkit-user-select:none;user-select:none;cursor:pointer;transition:transform var(--transition),opacity var(--transition),width var(--transition);animation:crumbIn var(--transition) 1}.crumb .crumb_inner.destroy{animation:crumbOut var(--transition) 1}.crumb .crumb_inner:before,.crumb .crumb_inner:after{content:\"\";position:absolute;top:0;border:0 solid var(--background-color);border-width:14px 8px;width:0;height:0}.crumb .crumb_inner:before{left:-14px;border-left-color:#0000}.crumb .crumb_inner:after{left:100%;border-color:#0000;border-left-color:var(--background-color)}.crumb .crumb_inner:hover{--background-color: #1abc9c}.crumb .crumb_inner:active{--background-color: #16a085}.crumb:nth-child(2n) .crumb_inner{--background-color: #2980b9}.crumb:first-child .crumb_inner{padding-left:6px;border-radius:4px 0 0 4px}.crumb:first-child .crumb_inner:before{border:none}.crumb:last-child:not(:only-child) .crumb_inner{padding-right:6px;border-radius:0 4px 4px 0}.crumb:last-child:not(:only-child) .crumb_inner:after{border:none}@keyframes crumbIn{0%{transform:translate(-10px);opacity:0}to{transform:translate(0);opacity:1}}@keyframes crumbOut{to{transform:translate(-10px);opacity:0}0%{transform:translate(0);opacity:1}}mat-form-field input{transition:width .2s ease,opacity .15s 0ms cubic-bezier(.4,0,.2,1)!important}\n"] }); }
23
+ }
24
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.1.2", ngImport: i0, type: BreadcrumbComponent, decorators: [{
25
+ type: Component,
26
+ args: [{ selector: 'ngx-commandpalette-breadcrumb', standalone: true, template: "@for (crumb of breadcrumbs; track crumb; let i = $index) {\n <div class=\"crumb\">\n <div\n class=\"crumb_inner\"\n [class.destroy]=\"crumb.destroying\"\n [class.root]=\"i == 0\"\n (click)=\"selectBreadcrumb(crumb)\"\n >\n {{crumb.action.label}}\n </div>\n </div>\n}\n", styles: [":host{display:flex}.crumb{overflow:hidden;height:26px;padding:0 16px;margin:0 -16px}.crumb .crumb_inner{--background-color: #3498db;color:#fff;position:relative;background:var(--background-color);padding:2px 5px 0 6px;margin-right:12px;text-align:center;white-space:pre;font-size:14px;-webkit-user-select:none;user-select:none;cursor:pointer;transition:transform var(--transition),opacity var(--transition),width var(--transition);animation:crumbIn var(--transition) 1}.crumb .crumb_inner.destroy{animation:crumbOut var(--transition) 1}.crumb .crumb_inner:before,.crumb .crumb_inner:after{content:\"\";position:absolute;top:0;border:0 solid var(--background-color);border-width:14px 8px;width:0;height:0}.crumb .crumb_inner:before{left:-14px;border-left-color:#0000}.crumb .crumb_inner:after{left:100%;border-color:#0000;border-left-color:var(--background-color)}.crumb .crumb_inner:hover{--background-color: #1abc9c}.crumb .crumb_inner:active{--background-color: #16a085}.crumb:nth-child(2n) .crumb_inner{--background-color: #2980b9}.crumb:first-child .crumb_inner{padding-left:6px;border-radius:4px 0 0 4px}.crumb:first-child .crumb_inner:before{border:none}.crumb:last-child:not(:only-child) .crumb_inner{padding-right:6px;border-radius:0 4px 4px 0}.crumb:last-child:not(:only-child) .crumb_inner:after{border:none}@keyframes crumbIn{0%{transform:translate(-10px);opacity:0}to{transform:translate(0);opacity:1}}@keyframes crumbOut{to{transform:translate(-10px);opacity:0}0%{transform:translate(0);opacity:1}}mat-form-field input{transition:width .2s ease,opacity .15s 0ms cubic-bezier(.4,0,.2,1)!important}\n"] }]
27
+ }], ctorParameters: () => [{ type: i1.CommandPaletteComponent }], propDecorators: { breadcrumbs: [{
28
+ type: Input
29
+ }] } });
30
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYnJlYWRjcnVtYi5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9jb21tb24vc3JjL2NvbXBvbmVudHMvY29tbWFuZC1wYWxldHRlL2JyZWFkY3J1bWIvYnJlYWRjcnVtYi5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9jb21tb24vc3JjL2NvbXBvbmVudHMvY29tbWFuZC1wYWxldHRlL2JyZWFkY3J1bWIvYnJlYWRjcnVtYi5jb21wb25lbnQuaHRtbCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsU0FBUyxFQUFFLEtBQUssRUFBRSxNQUFNLGVBQWUsQ0FBQzs7O0FBVWpELE1BQU0sT0FBTyxtQkFBbUI7SUFTNUIsWUFDWSxjQUF1QztRQUF2QyxtQkFBYyxHQUFkLGNBQWMsQ0FBeUI7UUFSMUMsZ0JBQVcsR0FLZCxFQUFFLENBQUM7SUFNVCxDQUFDO0lBRUQsZ0JBQWdCLENBQUMsS0FBSztRQUNsQixNQUFNLEtBQUssR0FBRyxJQUFJLENBQUMsV0FBVyxDQUFDLE9BQU8sQ0FBQyxLQUFLLENBQUMsQ0FBQztRQUM5QyxJQUFJLEtBQUssSUFBSSxDQUFDLENBQUM7WUFDWCxNQUFNLElBQUksS0FBSyxDQUFDLDhCQUE4QixDQUFDLENBQUM7UUFFcEQsTUFBTSxLQUFLLEdBQUcsSUFBSSxDQUFDLFdBQVcsQ0FBQyxFQUFFLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUN0QyxLQUFLLENBQUMsVUFBVSxHQUFHLElBQUksQ0FBQztRQUN4QixVQUFVLENBQUMsR0FBRyxFQUFFO1lBQ1osSUFBSSxDQUFDLGNBQWMsQ0FBQyxjQUFjLENBQUMsSUFBSSxDQUFDLFdBQVcsQ0FBQyxFQUFFLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxRQUFRLENBQUMsQ0FBQztZQUNyRSxJQUFJLENBQUMsY0FBYyxDQUFDLFdBQVcsQ0FBQyxHQUFHLEVBQUUsQ0FBQztZQUN0QyxJQUFJLENBQUMsY0FBYyxDQUFDLFdBQVcsR0FBRyxLQUFLLENBQUMsYUFBYSxDQUFDO1FBQzFELENBQUMsRUFBRSxHQUFHLENBQUMsQ0FBQTtJQUNYLENBQUM7OEdBM0JRLG1CQUFtQjtrR0FBbkIsbUJBQW1CLGlJQ1ZoQyxvV0FZQTs7MkZERmEsbUJBQW1CO2tCQU4vQixTQUFTOytCQUNJLCtCQUErQixjQUc3QixJQUFJOzRGQUlQLFdBQVc7c0JBQW5CLEtBQUsiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb21wb25lbnQsIElucHV0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBDb21tYW5kUGFsZXR0ZUNvbXBvbmVudCB9IGZyb20gJy4uL2NvbW1hbmQtcGFsZXR0ZS5jb21wb25lbnQnO1xuaW1wb3J0IHsgQ29tbWFuZEFjdGlvbiB9IGZyb20gJy4uLy4uLy4uL3NlcnZpY2VzL2NvbW1hbmQtcGFsZXR0ZS5zZXJ2aWNlJztcblxuQENvbXBvbmVudCh7XG4gICAgc2VsZWN0b3I6ICduZ3gtY29tbWFuZHBhbGV0dGUtYnJlYWRjcnVtYicsXG4gICAgdGVtcGxhdGVVcmw6ICcuL2JyZWFkY3J1bWIuY29tcG9uZW50Lmh0bWwnLFxuICAgIHN0eWxlVXJsczogWycuL2JyZWFkY3J1bWIuY29tcG9uZW50LnNjc3MnXSxcbiAgICBzdGFuZGFsb25lOiB0cnVlXG59KVxuZXhwb3J0IGNsYXNzIEJyZWFkY3J1bWJDb21wb25lbnQge1xuXG4gICAgQElucHV0KCkgYnJlYWRjcnVtYnM6IHtcbiAgICAgICAgYWN0aW9uOiBDb21tYW5kQWN0aW9uLFxuICAgICAgICBjb21tYW5kczogQ29tbWFuZEFjdGlvbltdLFxuICAgICAgICBkZXN0cm95aW5nOiBib29sZWFuLFxuICAgICAgICBzZWxlY3RlZEluZGV4OiBudW1iZXJcbiAgICB9W10gPSBbXTtcblxuICAgIGNvbnN0cnVjdG9yKFxuICAgICAgICBwcml2YXRlIGNvbW1hbmRQYWxldHRlOiBDb21tYW5kUGFsZXR0ZUNvbXBvbmVudFxuICAgICkge1xuXG4gICAgfVxuXG4gICAgc2VsZWN0QnJlYWRjcnVtYihjcnVtYikge1xuICAgICAgICBjb25zdCBpbmRleCA9IHRoaXMuYnJlYWRjcnVtYnMuaW5kZXhPZihjcnVtYik7XG4gICAgICAgIGlmIChpbmRleCA9PSAtMSlcbiAgICAgICAgICAgIHRocm93IG5ldyBFcnJvcihcIlNvbWV0aGluZyB0ZXJyaWJsZSBoYXBwZW5lZC5cIik7XG5cbiAgICAgICAgY29uc3QgbGF5ZXIgPSB0aGlzLmJyZWFkY3J1bWJzLmF0KC0xKTtcbiAgICAgICAgbGF5ZXIuZGVzdHJveWluZyA9IHRydWU7XG4gICAgICAgIHNldFRpbWVvdXQoKCkgPT4ge1xuICAgICAgICAgICAgdGhpcy5jb21tYW5kUGFsZXR0ZS5zZXRDb21tYW5kTGlzdCh0aGlzLmJyZWFkY3J1bWJzLmF0KC0yKS5jb21tYW5kcyk7XG4gICAgICAgICAgICB0aGlzLmNvbW1hbmRQYWxldHRlLmJyZWFkY3J1bWJzLnBvcCgpO1xuICAgICAgICAgICAgdGhpcy5jb21tYW5kUGFsZXR0ZS5hY3RpdmVJbmRleCA9IGxheWVyLnNlbGVjdGVkSW5kZXg7XG4gICAgICAgIH0sIDE5MClcbiAgICB9XG59XG4iLCJAZm9yIChjcnVtYiBvZiBicmVhZGNydW1iczsgdHJhY2sgY3J1bWI7IGxldCBpID0gJGluZGV4KSB7XG4gICAgPGRpdiBjbGFzcz1cImNydW1iXCI+XG4gICAgICAgIDxkaXZcbiAgICAgICAgICAgIGNsYXNzPVwiY3J1bWJfaW5uZXJcIlxuICAgICAgICAgICAgW2NsYXNzLmRlc3Ryb3ldPVwiY3J1bWIuZGVzdHJveWluZ1wiXG4gICAgICAgICAgICBbY2xhc3Mucm9vdF09XCJpID09IDBcIlxuICAgICAgICAgICAgKGNsaWNrKT1cInNlbGVjdEJyZWFkY3J1bWIoY3J1bWIpXCJcbiAgICAgICAgPlxuICAgICAgICAgICAgICAgIHt7Y3J1bWIuYWN0aW9uLmxhYmVsfX1cbiAgICAgICAgPC9kaXY+XG4gICAgPC9kaXY+XG59XG4iXX0=
@@ -1,51 +1,105 @@
1
- import { Component, HostListener, Inject, Input } from '@angular/core';
1
+ import { Component, HostListener, Inject, Input, ViewChild, isDevMode } from '@angular/core';
2
2
  import { MatInputModule } from '@angular/material/input';
3
3
  import { MAT_DIALOG_DATA } from '@angular/material/dialog';
4
4
  import { ScrollingModule } from '@angular/cdk/scrolling';
5
- import { NgScrollbarModule } from 'ngx-scrollbar';
5
+ import { NgScrollbar, NgScrollbarModule } from 'ngx-scrollbar';
6
6
  import { ShortcutComponent } from './shortcut/shortcut.component';
7
+ import { MatIconModule } from '@angular/material/icon';
8
+ import { BreadcrumbComponent } from './breadcrumb/breadcrumb.component';
7
9
  import * as i0 from "@angular/core";
8
10
  import * as i1 from "../../services/command-palette.service";
9
11
  import * as i2 from "@angular/material/dialog";
10
- import * as i3 from "@angular/material/input";
11
- import * as i4 from "@angular/material/form-field";
12
- import * as i5 from "@angular/cdk/scrolling";
13
- import * as i6 from "ngx-scrollbar";
12
+ import * as i3 from "@angular/material/icon";
13
+ import * as i4 from "@angular/material/input";
14
+ import * as i5 from "@angular/material/form-field";
15
+ import * as i6 from "@angular/cdk/scrolling";
16
+ import * as i7 from "ngx-scrollbar";
14
17
  export class CommandPaletteComponent {
15
- constructor(commandPalette, dialog, data) {
18
+ get el() { return this.elementRef.nativeElement; }
19
+ constructor(commandPalette, dialog, elementRef, changeDetector, data) {
16
20
  this.commandPalette = commandPalette;
17
21
  this.dialog = dialog;
22
+ this.elementRef = elementRef;
23
+ this.changeDetector = changeDetector;
24
+ this.MAT_ICON_REGEX = /[:\/\.]/;
18
25
  this.queryString = "";
19
26
  this.activeIndex = 0;
27
+ this.rowHeight = 29;
28
+ this.padding = 6;
20
29
  this.commands = [];
21
30
  this.filteredCommands = [];
31
+ this.breadcrumbs = [];
22
32
  this.contextElement = this.contextElement ?? data.contextElement;
23
33
  }
24
34
  ngOnInit() {
25
- this.commands = this.commandPalette.getRegisteredCommands(this.contextElement)
26
- .filter(c => c.visibleInList != false);
27
- this.filteredCommands = this.commands;
28
- // Reset the filter labels
29
- this.commands.forEach(command => command['_renderedLabel'] = '');
35
+ this.commands = this.commandPalette.getRegisteredCommands(this.contextElement);
36
+ this.setCommandList(this.commands);
37
+ // TODO: custom name
38
+ this.breadcrumbs.push({
39
+ action: { label: "/" },
40
+ commands: this.commands,
41
+ destroying: false,
42
+ selectedIndex: 0
43
+ });
30
44
  }
31
45
  onKeyDown(evt) {
32
- if (evt.key == "Enter") {
33
- // Fire the first command
34
- if (this.filteredCommands.length > 0)
35
- this.executeCommand(this.filteredCommands[this.activeIndex]);
36
- evt.stopPropagation();
37
- return;
38
- }
39
- if (evt.key == "ArrowUp") {
40
- // Fire the first command
41
- this.activeIndex = Math.max(this.activeIndex - 1, 0);
42
- evt.stopPropagation();
43
- return;
44
- }
45
- if (evt.key == "ArrowDown") {
46
- this.activeIndex = Math.min(this.filteredCommands.length - 1, this.activeIndex + 1);
47
- evt.stopPropagation();
48
- return;
46
+ switch (evt.key) {
47
+ case "Enter": {
48
+ // Fire the first command
49
+ if (this.filteredCommands.length > 0)
50
+ this.executeCommand(this.filteredCommands[this.activeIndex]);
51
+ evt.stopPropagation();
52
+ return;
53
+ }
54
+ case "ArrowUp": {
55
+ this.activeIndex = Math.max(this.activeIndex - 1, 0);
56
+ evt.stopPropagation();
57
+ this.focusRow();
58
+ return;
59
+ }
60
+ case "ArrowDown": {
61
+ this.activeIndex = Math.min(this.filteredCommands.length - 1, this.activeIndex + 1);
62
+ evt.stopPropagation();
63
+ this.focusRow();
64
+ return;
65
+ }
66
+ case "PageUp": {
67
+ // Fire the first command
68
+ this.activeIndex = Math.max(this.activeIndex - 12, 0);
69
+ evt.stopPropagation();
70
+ this.focusRow();
71
+ return;
72
+ }
73
+ case "PageDown": {
74
+ this.activeIndex = Math.min(this.filteredCommands.length - 1, this.activeIndex + 12);
75
+ evt.stopPropagation();
76
+ this.focusRow();
77
+ return;
78
+ }
79
+ case "Escape": {
80
+ this.dialog.close();
81
+ evt.stopPropagation();
82
+ return;
83
+ }
84
+ case "Backspace": {
85
+ // If we have no characters and we're hitting backspace, go back
86
+ // to the previous menu in the breadcrumb
87
+ if (this.queryString.length == 0 && this.breadcrumbs.length > 1) {
88
+ const layer = this.breadcrumbs.at(-1);
89
+ layer.destroying = true;
90
+ setTimeout(() => {
91
+ this.setCommandList(this.breadcrumbs.at(-2).commands);
92
+ this.breadcrumbs.pop();
93
+ this.activeIndex = layer.selectedIndex;
94
+ }, 190);
95
+ return;
96
+ }
97
+ else {
98
+ break;
99
+ }
100
+ }
101
+ case "Delete": {
102
+ }
49
103
  }
50
104
  this.activeIndex = 0;
51
105
  this.commands.forEach(c => c['_renderedLabel'] = '');
@@ -53,17 +107,23 @@ export class CommandPaletteComponent {
53
107
  // value so that it's updated
54
108
  setTimeout(() => {
55
109
  this.queryString = evt.target.value;
56
- // Whitespace doesn't count.
57
- if (this.queryString.trim().length == 0) {
58
- this.filteredCommands = this.commands;
59
- return;
60
- }
61
- const queryChars = this.queryString
62
- .toLowerCase()
63
- .split('');
64
- const matchedCommands = [];
65
- for (const command of this.commands) {
66
- const { label } = command;
110
+ this.filterResults();
111
+ });
112
+ }
113
+ async filterResults() {
114
+ // Whitespace doesn't count.
115
+ if (this.queryString.trim().length == 0) {
116
+ this.filteredCommands = this.commands;
117
+ return;
118
+ }
119
+ const queryChars = this.queryString
120
+ .toLowerCase()
121
+ .split('');
122
+ const matchedCommands = [];
123
+ for (const command of this.commands) {
124
+ const { label } = command;
125
+ // Check the label
126
+ if (command.label) {
67
127
  const commandChars = label
68
128
  .toLowerCase()
69
129
  .split('');
@@ -87,34 +147,111 @@ export class CommandPaletteComponent {
87
147
  matchedCommands.push(command);
88
148
  }
89
149
  }
90
- this.filteredCommands = matchedCommands;
91
- });
150
+ // Check the hint
151
+ if (command.hint) {
152
+ const commandChars = label
153
+ .toLowerCase()
154
+ .split('');
155
+ let renderedHint = '';
156
+ let lastIndex = 0;
157
+ let isMatch = true;
158
+ for (const char of queryChars) {
159
+ const index = commandChars.indexOf(char, lastIndex);
160
+ if (index == -1) {
161
+ isMatch = false;
162
+ break;
163
+ }
164
+ else {
165
+ renderedHint += label.slice(lastIndex, index) + `<b>${label.slice(index, index + 1)}</b>`;
166
+ lastIndex = index + 1;
167
+ }
168
+ }
169
+ renderedHint += label.slice(lastIndex);
170
+ if (isMatch) {
171
+ command['_renderedHint'] = renderedHint;
172
+ matchedCommands.push(command);
173
+ }
174
+ }
175
+ }
176
+ this.filteredCommands = matchedCommands;
177
+ }
178
+ setCommandList(commands) {
179
+ this.commands = commands
180
+ .filter(c => c.visibleInList != false);
181
+ this.filteredCommands = this.commands;
182
+ this.queryString = '';
183
+ this.activeIndex = 0;
184
+ // Reset the filter labels
185
+ this.commands.forEach(command => command['_renderedLabel'] = '');
186
+ }
187
+ focusRow() {
188
+ const top = this.activeIndex * this.rowHeight;
189
+ const height = this.rowHeight;
190
+ const viewTop = this.scrollbar?.viewport?.scrollTop;
191
+ const viewHeight = this.scrollbar?.viewport?.clientHeight;
192
+ const viewBottom = viewTop + viewHeight;
193
+ if (top < viewTop) {
194
+ this.scrollbar.viewport.nativeElement.scrollTo({ top: top + this.padding });
195
+ }
196
+ else if ((top + height) > viewBottom) {
197
+ this.scrollbar.viewport.nativeElement.scrollTo({ top: ((top + this.rowHeight) - viewHeight) + this.padding });
198
+ }
199
+ // Immediately check for changes to update template
200
+ this.changeDetector.detectChanges();
92
201
  }
93
202
  executeCommand(command) {
94
- this.commandPalette.invokeAction(command);
95
- this.dialog.close();
203
+ // Open a sub menu of items
204
+ if (Array.isArray(command.subMenu)) {
205
+ this.breadcrumbs.push({
206
+ action: command,
207
+ commands: command.subMenu,
208
+ selectedIndex: this.activeIndex,
209
+ destroying: false
210
+ });
211
+ this.setCommandList(command.subMenu);
212
+ }
213
+ // Directly invoke the action and kill the dialog
214
+ else {
215
+ this.commandPalette.invokeAction(command);
216
+ this.dialog.close();
217
+ }
96
218
  }
97
219
  onBlur() {
98
- // this.dialog.close();
220
+ if (!isDevMode()) {
221
+ this.dialog.close();
222
+ }
99
223
  }
100
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.1.1", ngImport: i0, type: CommandPaletteComponent, deps: [{ token: i1.CommandPaletteService }, { token: i2.MatDialogRef }, { token: MAT_DIALOG_DATA }], target: i0.ɵɵFactoryTarget.Component }); }
101
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.1.1", type: CommandPaletteComponent, isStandalone: true, selector: "app-command-palette", inputs: { contextElement: "contextElement" }, host: { listeners: { "window:blur": "onBlur()" } }, ngImport: i0, template: "<mat-form-field style=\"width: 100%; padding: 8px 8px 0 8px\">\n <input\n matInput\n type=\"text\"\n [value]=\"queryString\"\n (keydown)=\"onKeyDown($event)\"\n >\n</mat-form-field>\n\n@if (filteredCommands.length == 0) {\n <div class=\"commands\">\n <div class=\"command selected\">\n <div class=\"label\">No matching results.</div>\n </div>\n </div>\n}\n@else {\n <div\n class=\"commands\"\n [style.flex]=\"(filteredCommands.length * 28) + 'px'\"\n >\n <ng-scrollbar>\n <cdk-virtual-scroll-viewport itemSize=\"28\" scrollViewport>\n <div\n *cdkVirtualFor=\"let command of filteredCommands; let index = index\"\n class=\"command\"\n [class.selected]=\"index==activeIndex\"\n (click)=\"executeCommand(command)\"\n >\n <div\n class=\"label\"\n [innerHTML]=\"command['_renderedLabel'] || command.label || command.description\"\n ></div>\n <div>\n @for (shortcut of (command.shortcutKey?.find ? command.shortcutKey : [command.shortcutKey]); track shortcut) {\n @if (shortcut) {\n <app-shortcut [shortcut]=\"shortcut\"></app-shortcut>\n }\n }\n </div>\n </div>\n </cdk-virtual-scroll-viewport>\n </ng-scrollbar>\n </div>\n}\n\n", styles: [":host{display:flex;flex-direction:column;width:860px;max-height:460px;border:1px solid #484848;border-radius:5px;background-color:#222;-webkit-user-select:none;user-select:none}.commands{max-height:100%;overflow:hidden}.commands .command{display:flex;height:28px;padding-left:16px;padding-right:32px;align-items:center;border-radius:3px;justify-content:space-between}.commands .command:hover{background-color:#2a2d2e}.commands .command.selected{background-color:#04395e}.commands .command .label ::ng-deep b{color:#2196f3}ng-scrollbar.ng-scrollbar{--scrollbar-padding: 0px;--scrollbar-size: 14px;--scrollbar-border-radius: 0;--scrollbar-thumb-color: #0000;--scrollbar-thumb-transition: height ease-out .15s, width ease-out .15s, background-color ease 1.2s}ng-scrollbar.ng-scrollbar:hover{--scrollbar-thumb-color: #444}cdk-virtual-scroll-viewport{height:100%}:host ::ng-deep .mdc-text-field--no-label:not(.mdc-text-field--outlined):not(.mdc-text-field--textarea) .mat-mdc-form-field-infix{padding:4px 0}:host ::ng-deep .mat-mdc-form-field-infix{min-height:32px}:host ::ng-deep .mat-mdc-text-field-wrapper{margin-bottom:8px}:host ::ng-deep .mat-mdc-form-field-subscript-wrapper{display:none}::ng-deep .dark :host{border:1px solid #e5e5e5;background-color:#f8f8f8}::ng-deep .dark :host .commands .command:hover{background-color:#f2f2f2}::ng-deep .dark :host .commands .command.selected{background-color:#e8e8e8}\n"], dependencies: [{ kind: "ngmodule", type: MatInputModule }, { kind: "directive", type: i3.MatInput, selector: "input[matInput], textarea[matInput], select[matNativeControl], input[matNativeControl], textarea[matNativeControl]", inputs: ["disabled", "id", "placeholder", "name", "required", "type", "errorStateMatcher", "aria-describedby", "value", "readonly"], exportAs: ["matInput"] }, { kind: "component", type: i4.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "ngmodule", type: ScrollingModule }, { kind: "directive", type: i5.CdkFixedSizeVirtualScroll, selector: "cdk-virtual-scroll-viewport[itemSize]", inputs: ["itemSize", "minBufferPx", "maxBufferPx"] }, { kind: "directive", type: i5.CdkVirtualForOf, selector: "[cdkVirtualFor][cdkVirtualForOf]", inputs: ["cdkVirtualForOf", "cdkVirtualForTrackBy", "cdkVirtualForTemplate", "cdkVirtualForTemplateCacheSize"] }, { kind: "component", type: i5.CdkVirtualScrollViewport, selector: "cdk-virtual-scroll-viewport", inputs: ["orientation", "appendOnly"], outputs: ["scrolledIndexChange"] }, { kind: "ngmodule", type: NgScrollbarModule }, { kind: "component", type: i6.NgScrollbar, selector: "ng-scrollbar", inputs: ["disabled", "sensorDisabled", "pointerEventsDisabled", "viewportPropagateMouseMove", "autoHeightDisabled", "autoWidthDisabled", "viewClass", "trackClass", "thumbClass", "minThumbSize", "trackClickScrollDuration", "pointerEventsMethod", "track", "visibility", "appearance", "position", "sensorDebounce", "scrollAuditTime"], outputs: ["updated"], exportAs: ["ngScrollbar"] }, { kind: "directive", type: i6.ScrollViewport, selector: "[scrollViewport]" }, { kind: "component", type: ShortcutComponent, selector: "app-shortcut", inputs: ["shortcut"] }] }); }
224
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.1.2", ngImport: i0, type: CommandPaletteComponent, deps: [{ token: i1.CommandPaletteService }, { token: i2.MatDialogRef }, { token: i0.ElementRef }, { token: i0.ChangeDetectorRef }, { token: MAT_DIALOG_DATA }], target: i0.ɵɵFactoryTarget.Component }); }
225
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.1.2", type: CommandPaletteComponent, isStandalone: true, selector: "ngx-command-palette", inputs: { contextElement: "contextElement" }, host: { listeners: { "click": "textInput.nativeElement.focus()", "window:blur": "onBlur()", "window:resize": "onBlur()" } }, viewQueries: [{ propertyName: "scrollbar", first: true, predicate: NgScrollbar, descendants: true }, { propertyName: "textInput", first: true, predicate: ["textinput"], descendants: true }], ngImport: i0, template: "<mat-form-field style=\"width: 100%; padding: 8px 8px 0 8px\">\n <ngx-commandpalette-breadcrumb\n [breadcrumbs]=\"breadcrumbs\"\n />\n <input\n #textinput\n matInput\n type=\"text\"\n [value]=\"queryString\"\n (keydown)=\"onKeyDown($event)\"\n >\n</mat-form-field>\n\n@if (filteredCommands.length == 0) {\n <div class=\"commands\">\n <div class=\"command selected\">\n <div class=\"label\">No matching results.</div>\n </div>\n </div>\n}\n@else {\n <div\n class=\"commands\"\n [style.flex]=\"(filteredCommands.length * rowHeight) + 'px'\"\n [class.shadow]=\"scrollbar?.viewport?.scrollTop > 2\"\n >\n <ng-scrollbar #scrollbar>\n <cdk-virtual-scroll-viewport [itemSize]=\"rowHeight\" scrollViewport [minBufferPx]=\"150\">\n <div\n *cdkVirtualFor=\"let command of filteredCommands; let index = index\"\n class=\"command\"\n [style.height]=\"rowHeight + 'px'\"\n [class.selected]=\"index==activeIndex\"\n [class.has-icon]=\"command.icon\"\n [attr.index]=\"index\"\n (click)=\"executeCommand(command)\"\n >\n @if (command.icon) {\n <div class=\"icon\">\n @if (!MAT_ICON_REGEX.test(command.icon)) {\n <mat-icon [fontIcon]=\"command.icon\"></mat-icon>\n }\n @else {\n <img [src]=\"command.icon\"/>\n }\n </div>\n }\n\n <div\n class=\"label\"\n [innerHTML]=\"command['_renderedLabel'] || command.label || command.description\"\n ></div>\n\n @if (command['_renderedHint'] || command.hint) {\n <div\n class=\"hint\"\n [innerHTML]=\"command['_renderedHint'] || command.hint\"\n ></div>\n }\n <div style=\"flex: 1\"></div>\n\n <div>\n @for (shortcut of (command.shortcutKey?.find ? command.shortcutKey : [command.shortcutKey]); track shortcut) {\n @if (shortcut) {\n <ngx-commandpalette-shortcut [shortcut]=\"shortcut\"/>\n }\n }\n </div>\n </div>\n </cdk-virtual-scroll-viewport>\n </ng-scrollbar>\n </div>\n}\n\n", styles: [":host{display:flex;flex-direction:column;width:860px;max-height:460px;border:1px solid #484848;border-radius:6px;background-color:#222;-webkit-user-select:none;user-select:none;overflow:hidden;--text-color: #ccc;--transition: .25s ease}.commands{max-height:100%;overflow:hidden;position:relative;padding:0 6px}.commands .command{display:flex;padding-left:16px;padding-right:32px;align-items:center;border-radius:3px;justify-content:space-between;color:var(--text-color);font-size:15.5px;background-color:#0000}.commands .command:hover{background-color:#2a2d2e}.commands .command.selected{background-color:#04395e}.commands .command.has-icon{padding-left:8px}.commands .command .label ::ng-deep b{color:#2196f3}.commands .command .icon{width:38px;height:100%;display:flex;align-items:center;justify-content:center}.commands .command img{max-height:100%;padding:4px}.commands .command .hint{margin-left:12px;opacity:.75}.commands.shadow:after{box-shadow:#000 0 6px 6px -6px inset}.commands:after{content:\"\";position:absolute;top:0;left:0;width:100%;height:6px;box-shadow:#0000 0 6px 6px -6px inset;transition:box-shadow .3s ease}ng-scrollbar.ng-scrollbar{--scrollbar-padding: 0px;--scrollbar-size: 14px;--scrollbar-border-radius: 0;--scrollbar-thumb-color: #4440;--scrollbar-thumb-transition: height ease-out .15s, width ease-out .15s, background-color ease 1.2s;animation:fadeScrollbar 1.2s ease}ng-scrollbar.ng-scrollbar:hover{--scrollbar-thumb-color: #444f}@keyframes fadeScrollbar{0%{--scrollbar-thumb-color: #444f}to{--scrollbar-thumb-color: #4440}}cdk-virtual-scroll-viewport{height:100%;padding-bottom:6px}:host ::ng-deep .mdc-text-field--no-label:not(.mdc-text-field--outlined):not(.mdc-text-field--textarea) .mat-mdc-form-field-infix{padding:2px 0 4px}:host ::ng-deep .mat-mdc-form-field-infix{min-height:32px;display:flex}:host ::ng-deep .mat-mdc-text-field-wrapper{margin-bottom:8px;padding:0 12px}:host ::ng-deep .mat-mdc-form-field-subscript-wrapper{display:none}::ng-deep .light app-command-palette{border:1px solid #e5e5e5;background-color:#f8f8f8;--text-color: #262626}::ng-deep .light app-command-palette .commands .command:hover{background-color:#f2f2f2}::ng-deep .light app-command-palette .commands .command.selected{background-color:#e8e8e8}\n"], dependencies: [{ kind: "ngmodule", type: MatIconModule }, { kind: "component", type: i3.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "ngmodule", type: MatInputModule }, { kind: "directive", type: i4.MatInput, selector: "input[matInput], textarea[matInput], select[matNativeControl], input[matNativeControl], textarea[matNativeControl]", inputs: ["disabled", "id", "placeholder", "name", "required", "type", "errorStateMatcher", "aria-describedby", "value", "readonly"], exportAs: ["matInput"] }, { kind: "component", type: i5.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "ngmodule", type: ScrollingModule }, { kind: "directive", type: i6.CdkFixedSizeVirtualScroll, selector: "cdk-virtual-scroll-viewport[itemSize]", inputs: ["itemSize", "minBufferPx", "maxBufferPx"] }, { kind: "directive", type: i6.CdkVirtualForOf, selector: "[cdkVirtualFor][cdkVirtualForOf]", inputs: ["cdkVirtualForOf", "cdkVirtualForTrackBy", "cdkVirtualForTemplate", "cdkVirtualForTemplateCacheSize"] }, { kind: "component", type: i6.CdkVirtualScrollViewport, selector: "cdk-virtual-scroll-viewport", inputs: ["orientation", "appendOnly"], outputs: ["scrolledIndexChange"] }, { kind: "ngmodule", type: NgScrollbarModule }, { kind: "component", type: i7.NgScrollbar, selector: "ng-scrollbar", inputs: ["disabled", "sensorDisabled", "pointerEventsDisabled", "viewportPropagateMouseMove", "autoHeightDisabled", "autoWidthDisabled", "viewClass", "trackClass", "thumbClass", "minThumbSize", "trackClickScrollDuration", "pointerEventsMethod", "track", "visibility", "appearance", "position", "sensorDebounce", "scrollAuditTime"], outputs: ["updated"], exportAs: ["ngScrollbar"] }, { kind: "directive", type: i7.ScrollViewport, selector: "[scrollViewport]" }, { kind: "component", type: ShortcutComponent, selector: "ngx-commandpalette-shortcut", inputs: ["shortcut"] }, { kind: "component", type: BreadcrumbComponent, selector: "ngx-commandpalette-breadcrumb", inputs: ["breadcrumbs"] }] }); }
102
226
  }
103
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.1.1", ngImport: i0, type: CommandPaletteComponent, decorators: [{
227
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.1.2", ngImport: i0, type: CommandPaletteComponent, decorators: [{
104
228
  type: Component,
105
- args: [{ selector: 'app-command-palette', imports: [
229
+ args: [{ selector: 'ngx-command-palette', host: {
230
+ "(click)": "textInput.nativeElement.focus()"
231
+ }, imports: [
232
+ MatIconModule,
106
233
  MatInputModule,
107
234
  ScrollingModule,
108
235
  NgScrollbarModule,
109
- ShortcutComponent
110
- ], standalone: true, template: "<mat-form-field style=\"width: 100%; padding: 8px 8px 0 8px\">\n <input\n matInput\n type=\"text\"\n [value]=\"queryString\"\n (keydown)=\"onKeyDown($event)\"\n >\n</mat-form-field>\n\n@if (filteredCommands.length == 0) {\n <div class=\"commands\">\n <div class=\"command selected\">\n <div class=\"label\">No matching results.</div>\n </div>\n </div>\n}\n@else {\n <div\n class=\"commands\"\n [style.flex]=\"(filteredCommands.length * 28) + 'px'\"\n >\n <ng-scrollbar>\n <cdk-virtual-scroll-viewport itemSize=\"28\" scrollViewport>\n <div\n *cdkVirtualFor=\"let command of filteredCommands; let index = index\"\n class=\"command\"\n [class.selected]=\"index==activeIndex\"\n (click)=\"executeCommand(command)\"\n >\n <div\n class=\"label\"\n [innerHTML]=\"command['_renderedLabel'] || command.label || command.description\"\n ></div>\n <div>\n @for (shortcut of (command.shortcutKey?.find ? command.shortcutKey : [command.shortcutKey]); track shortcut) {\n @if (shortcut) {\n <app-shortcut [shortcut]=\"shortcut\"></app-shortcut>\n }\n }\n </div>\n </div>\n </cdk-virtual-scroll-viewport>\n </ng-scrollbar>\n </div>\n}\n\n", styles: [":host{display:flex;flex-direction:column;width:860px;max-height:460px;border:1px solid #484848;border-radius:5px;background-color:#222;-webkit-user-select:none;user-select:none}.commands{max-height:100%;overflow:hidden}.commands .command{display:flex;height:28px;padding-left:16px;padding-right:32px;align-items:center;border-radius:3px;justify-content:space-between}.commands .command:hover{background-color:#2a2d2e}.commands .command.selected{background-color:#04395e}.commands .command .label ::ng-deep b{color:#2196f3}ng-scrollbar.ng-scrollbar{--scrollbar-padding: 0px;--scrollbar-size: 14px;--scrollbar-border-radius: 0;--scrollbar-thumb-color: #0000;--scrollbar-thumb-transition: height ease-out .15s, width ease-out .15s, background-color ease 1.2s}ng-scrollbar.ng-scrollbar:hover{--scrollbar-thumb-color: #444}cdk-virtual-scroll-viewport{height:100%}:host ::ng-deep .mdc-text-field--no-label:not(.mdc-text-field--outlined):not(.mdc-text-field--textarea) .mat-mdc-form-field-infix{padding:4px 0}:host ::ng-deep .mat-mdc-form-field-infix{min-height:32px}:host ::ng-deep .mat-mdc-text-field-wrapper{margin-bottom:8px}:host ::ng-deep .mat-mdc-form-field-subscript-wrapper{display:none}::ng-deep .dark :host{border:1px solid #e5e5e5;background-color:#f8f8f8}::ng-deep .dark :host .commands .command:hover{background-color:#f2f2f2}::ng-deep .dark :host .commands .command.selected{background-color:#e8e8e8}\n"] }]
111
- }], ctorParameters: () => [{ type: i1.CommandPaletteService }, { type: i2.MatDialogRef }, { type: undefined, decorators: [{
236
+ ShortcutComponent,
237
+ BreadcrumbComponent
238
+ ], standalone: true, template: "<mat-form-field style=\"width: 100%; padding: 8px 8px 0 8px\">\n <ngx-commandpalette-breadcrumb\n [breadcrumbs]=\"breadcrumbs\"\n />\n <input\n #textinput\n matInput\n type=\"text\"\n [value]=\"queryString\"\n (keydown)=\"onKeyDown($event)\"\n >\n</mat-form-field>\n\n@if (filteredCommands.length == 0) {\n <div class=\"commands\">\n <div class=\"command selected\">\n <div class=\"label\">No matching results.</div>\n </div>\n </div>\n}\n@else {\n <div\n class=\"commands\"\n [style.flex]=\"(filteredCommands.length * rowHeight) + 'px'\"\n [class.shadow]=\"scrollbar?.viewport?.scrollTop > 2\"\n >\n <ng-scrollbar #scrollbar>\n <cdk-virtual-scroll-viewport [itemSize]=\"rowHeight\" scrollViewport [minBufferPx]=\"150\">\n <div\n *cdkVirtualFor=\"let command of filteredCommands; let index = index\"\n class=\"command\"\n [style.height]=\"rowHeight + 'px'\"\n [class.selected]=\"index==activeIndex\"\n [class.has-icon]=\"command.icon\"\n [attr.index]=\"index\"\n (click)=\"executeCommand(command)\"\n >\n @if (command.icon) {\n <div class=\"icon\">\n @if (!MAT_ICON_REGEX.test(command.icon)) {\n <mat-icon [fontIcon]=\"command.icon\"></mat-icon>\n }\n @else {\n <img [src]=\"command.icon\"/>\n }\n </div>\n }\n\n <div\n class=\"label\"\n [innerHTML]=\"command['_renderedLabel'] || command.label || command.description\"\n ></div>\n\n @if (command['_renderedHint'] || command.hint) {\n <div\n class=\"hint\"\n [innerHTML]=\"command['_renderedHint'] || command.hint\"\n ></div>\n }\n <div style=\"flex: 1\"></div>\n\n <div>\n @for (shortcut of (command.shortcutKey?.find ? command.shortcutKey : [command.shortcutKey]); track shortcut) {\n @if (shortcut) {\n <ngx-commandpalette-shortcut [shortcut]=\"shortcut\"/>\n }\n }\n </div>\n </div>\n </cdk-virtual-scroll-viewport>\n </ng-scrollbar>\n </div>\n}\n\n", styles: [":host{display:flex;flex-direction:column;width:860px;max-height:460px;border:1px solid #484848;border-radius:6px;background-color:#222;-webkit-user-select:none;user-select:none;overflow:hidden;--text-color: #ccc;--transition: .25s ease}.commands{max-height:100%;overflow:hidden;position:relative;padding:0 6px}.commands .command{display:flex;padding-left:16px;padding-right:32px;align-items:center;border-radius:3px;justify-content:space-between;color:var(--text-color);font-size:15.5px;background-color:#0000}.commands .command:hover{background-color:#2a2d2e}.commands .command.selected{background-color:#04395e}.commands .command.has-icon{padding-left:8px}.commands .command .label ::ng-deep b{color:#2196f3}.commands .command .icon{width:38px;height:100%;display:flex;align-items:center;justify-content:center}.commands .command img{max-height:100%;padding:4px}.commands .command .hint{margin-left:12px;opacity:.75}.commands.shadow:after{box-shadow:#000 0 6px 6px -6px inset}.commands:after{content:\"\";position:absolute;top:0;left:0;width:100%;height:6px;box-shadow:#0000 0 6px 6px -6px inset;transition:box-shadow .3s ease}ng-scrollbar.ng-scrollbar{--scrollbar-padding: 0px;--scrollbar-size: 14px;--scrollbar-border-radius: 0;--scrollbar-thumb-color: #4440;--scrollbar-thumb-transition: height ease-out .15s, width ease-out .15s, background-color ease 1.2s;animation:fadeScrollbar 1.2s ease}ng-scrollbar.ng-scrollbar:hover{--scrollbar-thumb-color: #444f}@keyframes fadeScrollbar{0%{--scrollbar-thumb-color: #444f}to{--scrollbar-thumb-color: #4440}}cdk-virtual-scroll-viewport{height:100%;padding-bottom:6px}:host ::ng-deep .mdc-text-field--no-label:not(.mdc-text-field--outlined):not(.mdc-text-field--textarea) .mat-mdc-form-field-infix{padding:2px 0 4px}:host ::ng-deep .mat-mdc-form-field-infix{min-height:32px;display:flex}:host ::ng-deep .mat-mdc-text-field-wrapper{margin-bottom:8px;padding:0 12px}:host ::ng-deep .mat-mdc-form-field-subscript-wrapper{display:none}::ng-deep .light app-command-palette{border:1px solid #e5e5e5;background-color:#f8f8f8;--text-color: #262626}::ng-deep .light app-command-palette .commands .command:hover{background-color:#f2f2f2}::ng-deep .light app-command-palette .commands .command.selected{background-color:#e8e8e8}\n"] }]
239
+ }], ctorParameters: () => [{ type: i1.CommandPaletteService }, { type: i2.MatDialogRef }, { type: i0.ElementRef }, { type: i0.ChangeDetectorRef }, { type: undefined, decorators: [{
112
240
  type: Inject,
113
241
  args: [MAT_DIALOG_DATA]
114
- }] }], propDecorators: { contextElement: [{
242
+ }] }], propDecorators: { scrollbar: [{
243
+ type: ViewChild,
244
+ args: [NgScrollbar]
245
+ }], textInput: [{
246
+ type: ViewChild,
247
+ args: ['textinput']
248
+ }], contextElement: [{
115
249
  type: Input
116
250
  }], onBlur: [{
117
251
  type: HostListener,
118
252
  args: ["window:blur"]
253
+ }, {
254
+ type: HostListener,
255
+ args: ["window:resize"]
119
256
  }] } });
120
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"command-palette.component.js","sourceRoot":"","sources":["../../../../../packages/common/src/components/command-palette/command-palette.component.ts","../../../../../packages/common/src/components/command-palette/command-palette.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, HostListener, Inject, Input, OnInit } from '@angular/core';\nimport { CommandAction, CommandPaletteService } from '../../services/command-palette.service';\nimport { MatInputModule } from '@angular/material/input';\nimport { MAT_DIALOG_DATA, MatDialogRef } from '@angular/material/dialog';\nimport { ScrollingModule } from '@angular/cdk/scrolling';\nimport { NgScrollbarModule } from 'ngx-scrollbar';\nimport { ShortcutComponent } from './shortcut/shortcut.component';\n\n@Component({\n    selector: 'app-command-palette',\n    templateUrl: './command-palette.component.html',\n    styleUrls: ['./command-palette.component.scss'],\n    imports: [\n        MatInputModule,\n        ScrollingModule,\n        NgScrollbarModule,\n        ShortcutComponent\n    ],\n    standalone: true\n})\nexport class CommandPaletteComponent implements OnInit {\n\n    @Input() contextElement: HTMLElement;\n\n    queryString = \"\";\n    activeIndex = 0;\n\n    commands: CommandAction[] = [];\n    filteredCommands: CommandAction[] = [];\n\n    constructor(\n        private readonly commandPalette: CommandPaletteService,\n        private readonly dialog: MatDialogRef<any>,\n        @Inject(MAT_DIALOG_DATA) data: any\n    ) {\n        this.contextElement = this.contextElement ?? data.contextElement;\n    }\n\n    ngOnInit() {\n        this.commands = this.commandPalette.getRegisteredCommands(this.contextElement)\n            .filter(c => c.visibleInList != false);\n        this.filteredCommands = this.commands;\n\n        // Reset the filter labels\n        this.commands.forEach(command => command['_renderedLabel'] = '');\n    }\n\n    onKeyDown(evt: KeyboardEvent) {\n        if (evt.key == \"Enter\") {\n            // Fire the first command\n            if (this.filteredCommands.length > 0)\n                this.executeCommand(this.filteredCommands[this.activeIndex]);\n\n            evt.stopPropagation();\n            return;\n        }\n\n        if (evt.key == \"ArrowUp\") {\n            // Fire the first command\n            this.activeIndex = Math.max(this.activeIndex-1, 0);\n\n            evt.stopPropagation();\n            return;\n        }\n\n        if (evt.key == \"ArrowDown\") {\n            this.activeIndex = Math.min(this.filteredCommands.length-1, this.activeIndex+1);\n\n            evt.stopPropagation();\n            return;\n        }\n\n        this.activeIndex = 0;\n        this.commands.forEach(c => c['_renderedLabel'] = '');\n\n        // Check in the next tick to get the input's\n        // value so that it's updated\n        setTimeout(() => {\n            this.queryString = (evt.target as HTMLInputElement).value;\n\n            // Whitespace doesn't count.\n            if (this.queryString.trim().length == 0) {\n                this.filteredCommands = this.commands;\n                return;\n            }\n\n            const queryChars = this.queryString\n                .toLowerCase()\n                .split('');\n\n            const matchedCommands: CommandAction[] = [];\n\n            for (const command of this.commands) {\n                const { label } = command;\n\n                const commandChars = label\n                    .toLowerCase()\n                    .split('');\n\n                let renderedLabel = '';\n                let lastIndex = 0;\n                let isMatch = true;\n\n                for (const char of queryChars) {\n                    const index = commandChars.indexOf(char, lastIndex);\n\n                    if (index == -1) {\n                        isMatch = false;\n                        break;\n                    }\n                    else {\n                        renderedLabel += label.slice(lastIndex, index) + `<b>${label.slice(index, index+1) }</b>`;\n                        lastIndex = index+1;\n                    }\n                }\n\n                renderedLabel += label.slice(lastIndex);\n\n                if (isMatch) {\n                    command['_renderedLabel'] = renderedLabel;\n                    matchedCommands.push(command);\n                }\n            }\n\n            this.filteredCommands = matchedCommands;\n        })\n    }\n\n    executeCommand(command: CommandAction) {\n        this.commandPalette.invokeAction(command);\n        this.dialog.close();\n    }\n\n    @HostListener(\"window:blur\")\n    onBlur() {\n        // this.dialog.close();\n    }\n}\n","<mat-form-field style=\"width: 100%; padding: 8px 8px 0 8px\">\n    <input\n        matInput\n        type=\"text\"\n        [value]=\"queryString\"\n        (keydown)=\"onKeyDown($event)\"\n    >\n</mat-form-field>\n\n@if (filteredCommands.length == 0) {\n    <div class=\"commands\">\n        <div class=\"command selected\">\n            <div class=\"label\">No matching results.</div>\n        </div>\n    </div>\n}\n@else {\n    <div\n        class=\"commands\"\n        [style.flex]=\"(filteredCommands.length * 28) + 'px'\"\n    >\n        <ng-scrollbar>\n            <cdk-virtual-scroll-viewport itemSize=\"28\" scrollViewport>\n                <div\n                    *cdkVirtualFor=\"let command of filteredCommands; let index = index\"\n                    class=\"command\"\n                    [class.selected]=\"index==activeIndex\"\n                    (click)=\"executeCommand(command)\"\n                >\n                    <div\n                        class=\"label\"\n                        [innerHTML]=\"command['_renderedLabel'] || command.label || command.description\"\n                    ></div>\n                    <div>\n                        @for (shortcut of (command.shortcutKey?.find ? command.shortcutKey : [command.shortcutKey]); track shortcut) {\n                            @if (shortcut) {\n                                <app-shortcut [shortcut]=\"shortcut\"></app-shortcut>\n                            }\n                        }\n                    </div>\n                </div>\n            </cdk-virtual-scroll-viewport>\n        </ng-scrollbar>\n    </div>\n}\n\n"]}
257
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"command-palette.component.js","sourceRoot":"","sources":["../../../../../packages/common/src/components/command-palette/command-palette.component.ts","../../../../../packages/common/src/components/command-palette/command-palette.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAqB,SAAS,EAAc,YAAY,EAAE,MAAM,EAAE,KAAK,EAAU,SAAS,EAAE,SAAS,EAAE,MAAM,eAAe,CAAC;AAEpI,OAAO,EAAE,cAAc,EAAE,MAAM,yBAAyB,CAAC;AACzD,OAAO,EAAE,eAAe,EAAgB,MAAM,0BAA0B,CAAC;AACzE,OAAO,EAAE,eAAe,EAAE,MAAM,wBAAwB,CAAC;AACzD,OAAO,EAAE,WAAW,EAAE,iBAAiB,EAAE,MAAM,eAAe,CAAC;AAC/D,OAAO,EAAE,iBAAiB,EAAE,MAAM,+BAA+B,CAAC;AAClE,OAAO,EAAE,aAAa,EAAE,MAAM,wBAAwB,CAAC;AACvD,OAAO,EAAE,mBAAmB,EAAE,MAAM,mCAAmC,CAAC;;;;;;;;;AAmBxE,MAAM,OAAO,uBAAuB;IAOhC,IAAI,EAAE,KAAK,OAAO,IAAI,CAAC,UAAU,CAAC,aAA4B,CAAC,CAAC,CAAC;IAmBjE,YACqB,cAAqC,EACrC,MAAyB,EACzB,UAAsB,EACtB,cAAiC,EACzB,IAAS;QAJjB,mBAAc,GAAd,cAAc,CAAuB;QACrC,WAAM,GAAN,MAAM,CAAmB;QACzB,eAAU,GAAV,UAAU,CAAY;QACtB,mBAAc,GAAd,cAAc,CAAmB;QArB7C,mBAAc,GAAG,SAAS,CAAC;QAEpC,gBAAW,GAAG,EAAE,CAAC;QACjB,gBAAW,GAAG,CAAC,CAAC;QACP,cAAS,GAAG,EAAE,CAAC;QACf,YAAO,GAAG,CAAC,CAAC;QAErB,aAAQ,GAAoB,EAAE,CAAC;QAC/B,qBAAgB,GAAoB,EAAE,CAAC;QAEvC,gBAAW,GAKL,EAAE,CAAC;QASL,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,cAAc,IAAI,IAAI,CAAC,cAAc,CAAC;IACrE,CAAC;IAED,QAAQ;QACJ,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,cAAc,CAAC,qBAAqB,CAAC,IAAI,CAAC,cAAc,CAAC,CAAA;QAC9E,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;QAEnC,oBAAoB;QACpB,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC;YAClB,MAAM,EAAE,EAAE,KAAK,EAAE,GAAG,EAAE;YACtB,QAAQ,EAAE,IAAI,CAAC,QAAQ;YACvB,UAAU,EAAE,KAAK;YACjB,aAAa,EAAE,CAAC;SACnB,CAAC,CAAC;IACP,CAAC;IAED,SAAS,CAAC,GAAkB;QACxB,QAAO,GAAG,CAAC,GAAG,EAAE,CAAC;YACb,KAAK,OAAO,CAAC,CAAC,CAAC;gBACX,yBAAyB;gBACzB,IAAI,IAAI,CAAC,gBAAgB,CAAC,MAAM,GAAG,CAAC;oBAChC,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC,CAAC;gBAEjE,GAAG,CAAC,eAAe,EAAE,CAAC;gBACtB,OAAO;YACX,CAAC;YACD,KAAK,SAAS,CAAC,CAAC,CAAC;gBACb,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,WAAW,GAAC,CAAC,EAAE,CAAC,CAAC,CAAC;gBAEnD,GAAG,CAAC,eAAe,EAAE,CAAC;gBACtB,IAAI,CAAC,QAAQ,EAAE,CAAC;gBAChB,OAAO;YACX,CAAC;YACD,KAAK,WAAW,CAAC,CAAC,CAAC;gBACf,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,gBAAgB,CAAC,MAAM,GAAC,CAAC,EAAE,IAAI,CAAC,WAAW,GAAC,CAAC,CAAC,CAAC;gBAEhF,GAAG,CAAC,eAAe,EAAE,CAAC;gBACtB,IAAI,CAAC,QAAQ,EAAE,CAAC;gBAChB,OAAO;YACX,CAAC;YACD,KAAK,QAAQ,CAAC,CAAC,CAAC;gBACZ,yBAAyB;gBACzB,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,WAAW,GAAC,EAAE,EAAE,CAAC,CAAC,CAAC;gBAEpD,GAAG,CAAC,eAAe,EAAE,CAAC;gBACtB,IAAI,CAAC,QAAQ,EAAE,CAAC;gBAChB,OAAO;YACX,CAAC;YACD,KAAK,UAAU,CAAC,CAAC,CAAC;gBACd,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,gBAAgB,CAAC,MAAM,GAAC,CAAC,EAAE,IAAI,CAAC,WAAW,GAAC,EAAE,CAAC,CAAC;gBAEjF,GAAG,CAAC,eAAe,EAAE,CAAC;gBACtB,IAAI,CAAC,QAAQ,EAAE,CAAC;gBAChB,OAAO;YACX,CAAC;YACD,KAAK,QAAQ,CAAC,CAAC,CAAC;gBACZ,IAAI,CAAC,MAAM,CAAC,KAAK,EAAE,CAAC;gBACpB,GAAG,CAAC,eAAe,EAAE,CAAC;gBACtB,OAAO;YACX,CAAC;YACD,KAAK,WAAW,CAAC,CAAC,CAAC;gBACf,gEAAgE;gBAChE,yCAAyC;gBACzC,IAAI,IAAI,CAAC,WAAW,CAAC,MAAM,IAAI,CAAC,IAAI,IAAI,CAAC,WAAW,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;oBAC9D,MAAM,KAAK,GAAG,IAAI,CAAC,WAAW,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC;oBACtC,KAAK,CAAC,UAAU,GAAG,IAAI,CAAC;oBACxB,UAAU,CAAC,GAAG,EAAE;wBACZ,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,WAAW,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC;wBACtD,IAAI,CAAC,WAAW,CAAC,GAAG,EAAE,CAAC;wBACvB,IAAI,CAAC,WAAW,GAAG,KAAK,CAAC,aAAa,CAAC;oBAC3C,CAAC,EAAE,GAAG,CAAC,CAAA;oBACP,OAAO;gBACX,CAAC;qBACI,CAAC;oBACF,MAAM;gBACV,CAAC;YACL,CAAC;YACD,KAAK,QAAQ,CAAC,CAAC,CAAC;YAEhB,CAAC;QACL,CAAC;QAED,IAAI,CAAC,WAAW,GAAG,CAAC,CAAC;QACrB,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,gBAAgB,CAAC,GAAG,EAAE,CAAC,CAAC;QAErD,4CAA4C;QAC5C,6BAA6B;QAC7B,UAAU,CAAC,GAAG,EAAE;YACZ,IAAI,CAAC,WAAW,GAAI,GAAG,CAAC,MAA2B,CAAC,KAAK,CAAC;YAC1D,IAAI,CAAC,aAAa,EAAE,CAAC;QACzB,CAAC,CAAC,CAAA;IACN,CAAC;IAEO,KAAK,CAAC,aAAa;QACvB,4BAA4B;QAC5B,IAAI,IAAI,CAAC,WAAW,CAAC,IAAI,EAAE,CAAC,MAAM,IAAI,CAAC,EAAE,CAAC;YACtC,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC,QAAQ,CAAC;YACtC,OAAO;QACX,CAAC;QAED,MAAM,UAAU,GAAG,IAAI,CAAC,WAAW;aAC9B,WAAW,EAAE;aACb,KAAK,CAAC,EAAE,CAAC,CAAC;QAEf,MAAM,eAAe,GAAoB,EAAE,CAAC;QAE5C,KAAK,MAAM,OAAO,IAAI,IAAI,CAAC,QAAQ,EAAE,CAAC;YAClC,MAAM,EAAE,KAAK,EAAE,GAAG,OAAO,CAAC;YAE1B,kBAAkB;YAClB,IAAI,OAAO,CAAC,KAAK,EAAE,CAAC;gBAChB,MAAM,YAAY,GAAG,KAAK;qBACrB,WAAW,EAAE;qBACb,KAAK,CAAC,EAAE,CAAC,CAAC;gBAEf,IAAI,aAAa,GAAG,EAAE,CAAC;gBACvB,IAAI,SAAS,GAAG,CAAC,CAAC;gBAClB,IAAI,OAAO,GAAG,IAAI,CAAC;gBAEnB,KAAK,MAAM,IAAI,IAAI,UAAU,EAAE,CAAC;oBAC5B,MAAM,KAAK,GAAG,YAAY,CAAC,OAAO,CAAC,IAAI,EAAE,SAAS,CAAC,CAAC;oBAEpD,IAAI,KAAK,IAAI,CAAC,CAAC,EAAE,CAAC;wBACd,OAAO,GAAG,KAAK,CAAC;wBAChB,MAAM;oBACV,CAAC;yBACI,CAAC;wBACF,aAAa,IAAI,KAAK,CAAC,KAAK,CAAC,SAAS,EAAE,KAAK,CAAC,GAAG,MAAM,KAAK,CAAC,KAAK,CAAC,KAAK,EAAE,KAAK,GAAG,CAAC,CAAC,MAAM,CAAC;wBAC3F,SAAS,GAAG,KAAK,GAAG,CAAC,CAAC;oBAC1B,CAAC;gBACL,CAAC;gBAED,aAAa,IAAI,KAAK,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;gBAExC,IAAI,OAAO,EAAE,CAAC;oBACV,OAAO,CAAC,gBAAgB,CAAC,GAAG,aAAa,CAAC;oBAC1C,eAAe,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;gBAClC,CAAC;YACL,CAAC;YAED,iBAAiB;YACjB,IAAI,OAAO,CAAC,IAAI,EAAE,CAAC;gBACf,MAAM,YAAY,GAAG,KAAK;qBACrB,WAAW,EAAE;qBACb,KAAK,CAAC,EAAE,CAAC,CAAC;gBAEf,IAAI,YAAY,GAAG,EAAE,CAAC;gBACtB,IAAI,SAAS,GAAG,CAAC,CAAC;gBAClB,IAAI,OAAO,GAAG,IAAI,CAAC;gBAEnB,KAAK,MAAM,IAAI,IAAI,UAAU,EAAE,CAAC;oBAC5B,MAAM,KAAK,GAAG,YAAY,CAAC,OAAO,CAAC,IAAI,EAAE,SAAS,CAAC,CAAC;oBAEpD,IAAI,KAAK,IAAI,CAAC,CAAC,EAAE,CAAC;wBACd,OAAO,GAAG,KAAK,CAAC;wBAChB,MAAM;oBACV,CAAC;yBACI,CAAC;wBACF,YAAY,IAAI,KAAK,CAAC,KAAK,CAAC,SAAS,EAAE,KAAK,CAAC,GAAG,MAAM,KAAK,CAAC,KAAK,CAAC,KAAK,EAAE,KAAK,GAAG,CAAC,CAAC,MAAM,CAAC;wBAC1F,SAAS,GAAG,KAAK,GAAG,CAAC,CAAC;oBAC1B,CAAC;gBACL,CAAC;gBAED,YAAY,IAAI,KAAK,CAAC,KAAK,CAAC,SAAS,CAAC,CAAC;gBAEvC,IAAI,OAAO,EAAE,CAAC;oBACV,OAAO,CAAC,eAAe,CAAC,GAAG,YAAY,CAAC;oBACxC,eAAe,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;gBAClC,CAAC;YACL,CAAC;QACL,CAAC;QAED,IAAI,CAAC,gBAAgB,GAAG,eAAe,CAAC;IAC5C,CAAC;IAEM,cAAc,CAAC,QAAyB;QAC3C,IAAI,CAAC,QAAQ,GAAG,QAAQ;aACnB,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,aAAa,IAAI,KAAK,CAAC,CAAC;QAC3C,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC,QAAQ,CAAC;QACtC,IAAI,CAAC,WAAW,GAAG,EAAE,CAAC;QACtB,IAAI,CAAC,WAAW,GAAG,CAAC,CAAC;QAErB,0BAA0B;QAC1B,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC,OAAO,CAAC,gBAAgB,CAAC,GAAG,EAAE,CAAC,CAAC;IACrE,CAAC;IAEO,QAAQ;QACZ,MAAM,GAAG,GAAG,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,SAAS,CAAC;QAC9C,MAAM,MAAM,GAAG,IAAI,CAAC,SAAS,CAAC;QAE9B,MAAM,OAAO,GAAG,IAAI,CAAC,SAAS,EAAE,QAAQ,EAAE,SAAS,CAAC;QACpD,MAAM,UAAU,GAAG,IAAI,CAAC,SAAS,EAAE,QAAQ,EAAE,YAAY,CAAC;QAC1D,MAAM,UAAU,GAAG,OAAO,GAAG,UAAU,CAAC;QAExC,IAAI,GAAG,GAAG,OAAO,EAAE,CAAC;YAChB,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,aAAa,CAAC,QAAQ,CAAC,EAAE,GAAG,EAAE,GAAG,GAAG,IAAI,CAAC,OAAO,EAAE,CAAC,CAAA;QAC/E,CAAC;aACI,IAAI,CAAC,GAAG,GAAG,MAAM,CAAC,GAAG,UAAU,EAAE,CAAC;YACnC,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,aAAa,CAAC,QAAQ,CAAC,EAAE,GAAG,EAAE,CAAC,CAAC,GAAG,GAAG,IAAI,CAAC,SAAS,CAAC,GAAG,UAAU,CAAC,GAAG,IAAI,CAAC,OAAO,EAAE,CAAC,CAAA;QACjH,CAAC;QAED,mDAAmD;QACnD,IAAI,CAAC,cAAc,CAAC,aAAa,EAAE,CAAC;IACxC,CAAC;IAED,cAAc,CAAC,OAAsB;QACjC,2BAA2B;QAC3B,IAAI,KAAK,CAAC,OAAO,CAAC,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC;YACjC,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC;gBAClB,MAAM,EAAE,OAAO;gBACf,QAAQ,EAAE,OAAO,CAAC,OAAO;gBACzB,aAAa,EAAE,IAAI,CAAC,WAAW;gBAC/B,UAAU,EAAE,KAAK;aACpB,CAAC,CAAC;YAEH,IAAI,CAAC,cAAc,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC;QACzC,CAAC;QACD,iDAAiD;aAC5C,CAAC;YACF,IAAI,CAAC,cAAc,CAAC,YAAY,CAAC,OAAO,CAAC,CAAC;YAC1C,IAAI,CAAC,MAAM,CAAC,KAAK,EAAE,CAAC;QACxB,CAAC;IACL,CAAC;IAID,MAAM;QACF,IAAI,CAAC,SAAS,EAAE,EAAE,CAAC;YACf,IAAI,CAAC,MAAM,CAAC,KAAK,EAAE,CAAC;QACxB,CAAC;IACL,CAAC;8GAvQQ,uBAAuB,8IA+BpB,eAAe;kGA/BlB,uBAAuB,qSAErB,WAAW,yIC7B1B,8uFA0EA,ixEDxDQ,aAAa,mLACb,cAAc,qjBACd,eAAe,2jBACf,iBAAiB,ijBACjB,iBAAiB,8FACjB,mBAAmB;;2FAId,uBAAuB;kBAjBnC,SAAS;+BACI,qBAAqB,QAGzB;wBACF,SAAS,EAAE,iCAAiC;qBAC/C,WACQ;wBACL,aAAa;wBACb,cAAc;wBACd,eAAe;wBACf,iBAAiB;wBACjB,iBAAiB;wBACjB,mBAAmB;qBACtB,cACW,IAAI;;0BAiCX,MAAM;2BAAC,eAAe;yCA7BH,SAAS;sBAAhC,SAAS;uBAAC,WAAW;gBACE,SAAS;sBAAhC,SAAS;uBAAC,WAAW;gBAEb,cAAc;sBAAtB,KAAK;gBA8PN,MAAM;sBAFL,YAAY;uBAAC,aAAa;;sBAC1B,YAAY;uBAAC,eAAe","sourcesContent":["import { ChangeDetectorRef, Component, ElementRef, HostListener, Inject, Input, OnInit, ViewChild, isDevMode } from '@angular/core';\nimport { CommandAction, CommandPaletteService } from '../../services/command-palette.service';\nimport { MatInputModule } from '@angular/material/input';\nimport { MAT_DIALOG_DATA, MatDialogRef } from '@angular/material/dialog';\nimport { ScrollingModule } from '@angular/cdk/scrolling';\nimport { NgScrollbar, NgScrollbarModule } from 'ngx-scrollbar';\nimport { ShortcutComponent } from './shortcut/shortcut.component';\nimport { MatIconModule } from '@angular/material/icon';\nimport { BreadcrumbComponent } from './breadcrumb/breadcrumb.component';\n\n@Component({\n    selector: 'ngx-command-palette',\n    templateUrl: './command-palette.component.html',\n    styleUrls: ['./command-palette.component.scss'],\n    host: {\n        \"(click)\": \"textInput.nativeElement.focus()\"\n    },\n    imports: [\n        MatIconModule,\n        MatInputModule,\n        ScrollingModule,\n        NgScrollbarModule,\n        ShortcutComponent,\n        BreadcrumbComponent\n    ],\n    standalone: true\n})\nexport class CommandPaletteComponent implements OnInit {\n\n    @ViewChild(NgScrollbar) scrollbar: NgScrollbar;\n    @ViewChild('textinput') textInput: ElementRef<HTMLInputElement>;\n\n    @Input() contextElement: HTMLElement;\n\n    get el() { return this.elementRef.nativeElement as HTMLElement; }\n\n    readonly MAT_ICON_REGEX = /[:\\/\\.]/;\n\n    queryString = \"\";\n    activeIndex = 0;\n    readonly rowHeight = 29;\n    readonly padding = 6;\n\n    commands: CommandAction[] = [];\n    filteredCommands: CommandAction[] = [];\n\n    breadcrumbs: {\n        action: CommandAction,\n        commands: CommandAction[],\n        destroying: boolean,\n        selectedIndex: number\n    }[] = [];\n\n    constructor(\n        private readonly commandPalette: CommandPaletteService,\n        private readonly dialog: MatDialogRef<any>,\n        private readonly elementRef: ElementRef,\n        private readonly changeDetector: ChangeDetectorRef,\n        @Inject(MAT_DIALOG_DATA) data: any\n    ) {\n        this.contextElement = this.contextElement ?? data.contextElement;\n    }\n\n    ngOnInit() {\n        this.commands = this.commandPalette.getRegisteredCommands(this.contextElement)\n        this.setCommandList(this.commands);\n\n        // TODO: custom name\n        this.breadcrumbs.push({\n            action: { label: \"/\" },\n            commands: this.commands,\n            destroying: false,\n            selectedIndex: 0\n        });\n    }\n\n    onKeyDown(evt: KeyboardEvent) {\n        switch(evt.key) {\n            case \"Enter\": {\n                // Fire the first command\n                if (this.filteredCommands.length > 0)\n                    this.executeCommand(this.filteredCommands[this.activeIndex]);\n\n                evt.stopPropagation();\n                return;\n            }\n            case \"ArrowUp\": {\n                this.activeIndex = Math.max(this.activeIndex-1, 0);\n\n                evt.stopPropagation();\n                this.focusRow();\n                return;\n            }\n            case \"ArrowDown\": {\n                this.activeIndex = Math.min(this.filteredCommands.length-1, this.activeIndex+1);\n\n                evt.stopPropagation();\n                this.focusRow();\n                return;\n            }\n            case \"PageUp\": {\n                // Fire the first command\n                this.activeIndex = Math.max(this.activeIndex-12, 0);\n\n                evt.stopPropagation();\n                this.focusRow();\n                return;\n            }\n            case \"PageDown\": {\n                this.activeIndex = Math.min(this.filteredCommands.length-1, this.activeIndex+12);\n\n                evt.stopPropagation();\n                this.focusRow();\n                return;\n            }\n            case \"Escape\": {\n                this.dialog.close();\n                evt.stopPropagation();\n                return;\n            }\n            case \"Backspace\": {\n                // If we have no characters and we're hitting backspace, go back\n                // to the previous menu in the breadcrumb\n                if (this.queryString.length == 0 && this.breadcrumbs.length > 1) {\n                    const layer = this.breadcrumbs.at(-1);\n                    layer.destroying = true;\n                    setTimeout(() => {\n                        this.setCommandList(this.breadcrumbs.at(-2).commands);\n                        this.breadcrumbs.pop();\n                        this.activeIndex = layer.selectedIndex;\n                    }, 190)\n                    return;\n                }\n                else {\n                    break;\n                }\n            }\n            case \"Delete\": {\n\n            }\n        }\n\n        this.activeIndex = 0;\n        this.commands.forEach(c => c['_renderedLabel'] = '');\n\n        // Check in the next tick to get the input's\n        // value so that it's updated\n        setTimeout(() => {\n            this.queryString = (evt.target as HTMLInputElement).value;\n            this.filterResults();\n        })\n    }\n\n    private async filterResults() {\n        // Whitespace doesn't count.\n        if (this.queryString.trim().length == 0) {\n            this.filteredCommands = this.commands;\n            return;\n        }\n\n        const queryChars = this.queryString\n            .toLowerCase()\n            .split('');\n\n        const matchedCommands: CommandAction[] = [];\n\n        for (const command of this.commands) {\n            const { label } = command;\n\n            // Check the label\n            if (command.label) {\n                const commandChars = label\n                    .toLowerCase()\n                    .split('');\n\n                let renderedLabel = '';\n                let lastIndex = 0;\n                let isMatch = true;\n\n                for (const char of queryChars) {\n                    const index = commandChars.indexOf(char, lastIndex);\n\n                    if (index == -1) {\n                        isMatch = false;\n                        break;\n                    }\n                    else {\n                        renderedLabel += label.slice(lastIndex, index) + `<b>${label.slice(index, index + 1)}</b>`;\n                        lastIndex = index + 1;\n                    }\n                }\n\n                renderedLabel += label.slice(lastIndex);\n\n                if (isMatch) {\n                    command['_renderedLabel'] = renderedLabel;\n                    matchedCommands.push(command);\n                }\n            }\n\n            // Check the hint\n            if (command.hint) {\n                const commandChars = label\n                    .toLowerCase()\n                    .split('');\n\n                let renderedHint = '';\n                let lastIndex = 0;\n                let isMatch = true;\n\n                for (const char of queryChars) {\n                    const index = commandChars.indexOf(char, lastIndex);\n\n                    if (index == -1) {\n                        isMatch = false;\n                        break;\n                    }\n                    else {\n                        renderedHint += label.slice(lastIndex, index) + `<b>${label.slice(index, index + 1)}</b>`;\n                        lastIndex = index + 1;\n                    }\n                }\n\n                renderedHint += label.slice(lastIndex);\n\n                if (isMatch) {\n                    command['_renderedHint'] = renderedHint;\n                    matchedCommands.push(command);\n                }\n            }\n        }\n\n        this.filteredCommands = matchedCommands;\n    }\n\n    public setCommandList(commands: CommandAction[]) {\n        this.commands = commands\n            .filter(c => c.visibleInList != false);\n        this.filteredCommands = this.commands;\n        this.queryString = '';\n        this.activeIndex = 0;\n\n        // Reset the filter labels\n        this.commands.forEach(command => command['_renderedLabel'] = '');\n    }\n\n    private focusRow() {\n        const top = this.activeIndex * this.rowHeight;\n        const height = this.rowHeight;\n\n        const viewTop = this.scrollbar?.viewport?.scrollTop;\n        const viewHeight = this.scrollbar?.viewport?.clientHeight;\n        const viewBottom = viewTop + viewHeight;\n\n        if (top < viewTop) {\n            this.scrollbar.viewport.nativeElement.scrollTo({ top: top + this.padding })\n        }\n        else if ((top + height) > viewBottom) {\n            this.scrollbar.viewport.nativeElement.scrollTo({ top: ((top + this.rowHeight) - viewHeight) + this.padding })\n        }\n\n        // Immediately check for changes to update template\n        this.changeDetector.detectChanges();\n    }\n\n    executeCommand(command: CommandAction) {\n        // Open a sub menu of items\n        if (Array.isArray(command.subMenu)) {\n            this.breadcrumbs.push({\n                action: command,\n                commands: command.subMenu,\n                selectedIndex: this.activeIndex,\n                destroying: false\n            });\n\n            this.setCommandList(command.subMenu);\n        }\n        // Directly invoke the action and kill the dialog\n        else {\n            this.commandPalette.invokeAction(command);\n            this.dialog.close();\n        }\n    }\n\n    @HostListener(\"window:blur\")\n    @HostListener(\"window:resize\")\n    onBlur() {\n        if (!isDevMode()) {\n            this.dialog.close();\n        }\n    }\n}\n","<mat-form-field style=\"width: 100%; padding: 8px 8px 0 8px\">\n    <ngx-commandpalette-breadcrumb\n        [breadcrumbs]=\"breadcrumbs\"\n    />\n    <input\n        #textinput\n        matInput\n        type=\"text\"\n        [value]=\"queryString\"\n        (keydown)=\"onKeyDown($event)\"\n    >\n</mat-form-field>\n\n@if (filteredCommands.length == 0) {\n    <div class=\"commands\">\n        <div class=\"command selected\">\n            <div class=\"label\">No matching results.</div>\n        </div>\n    </div>\n}\n@else {\n    <div\n        class=\"commands\"\n        [style.flex]=\"(filteredCommands.length * rowHeight) + 'px'\"\n        [class.shadow]=\"scrollbar?.viewport?.scrollTop > 2\"\n    >\n        <ng-scrollbar #scrollbar>\n            <cdk-virtual-scroll-viewport [itemSize]=\"rowHeight\" scrollViewport [minBufferPx]=\"150\">\n                <div\n                    *cdkVirtualFor=\"let command of filteredCommands; let index = index\"\n                    class=\"command\"\n                    [style.height]=\"rowHeight + 'px'\"\n                    [class.selected]=\"index==activeIndex\"\n                    [class.has-icon]=\"command.icon\"\n                    [attr.index]=\"index\"\n                    (click)=\"executeCommand(command)\"\n                >\n                    @if (command.icon) {\n                        <div class=\"icon\">\n                            @if (!MAT_ICON_REGEX.test(command.icon)) {\n                                <mat-icon [fontIcon]=\"command.icon\"></mat-icon>\n                            }\n                            @else {\n                                <img [src]=\"command.icon\"/>\n                            }\n                        </div>\n                    }\n\n                    <div\n                        class=\"label\"\n                        [innerHTML]=\"command['_renderedLabel'] || command.label || command.description\"\n                    ></div>\n\n                    @if (command['_renderedHint'] || command.hint) {\n                        <div\n                            class=\"hint\"\n                            [innerHTML]=\"command['_renderedHint'] || command.hint\"\n                        ></div>\n                    }\n                    <div style=\"flex: 1\"></div>\n\n                    <div>\n                        @for (shortcut of (command.shortcutKey?.find ? command.shortcutKey : [command.shortcutKey]); track shortcut) {\n                            @if (shortcut) {\n                                <ngx-commandpalette-shortcut [shortcut]=\"shortcut\"/>\n                            }\n                        }\n                    </div>\n                </div>\n            </cdk-virtual-scroll-viewport>\n        </ng-scrollbar>\n    </div>\n}\n\n"]}