@dotglitch/ngx-common 1.1.18 → 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.
- package/components/command-palette/breadcrumb/breadcrumb.component.d.ts +16 -0
- package/components/command-palette/command-palette.component.d.ts +21 -3
- package/components/command-palette/shortcut/shortcut.component.d.ts +1 -1
- package/directives/image-cache.directive.d.ts +46 -0
- package/esm2022/components/command-palette/breadcrumb/breadcrumb.component.mjs +30 -0
- package/esm2022/components/command-palette/command-palette.component.mjs +191 -54
- package/esm2022/components/command-palette/shortcut/shortcut.component.mjs +5 -5
- package/esm2022/components/dynamic-html/dynamic-html.component.mjs +3 -3
- package/esm2022/components/dynamic-html/dynamic-html.module.mjs +4 -4
- package/esm2022/components/dynamic-html/dynamic-html.service.mjs +3 -3
- package/esm2022/components/filemanager/file-grid/file-grid.component.mjs +3 -3
- package/esm2022/components/filemanager/filemanager.component.mjs +3 -3
- package/esm2022/components/filemanager/folder-rename/folder-rename.component.mjs +3 -3
- package/esm2022/components/filemanager/toolbar/breadcrumb/breadcrumb.component.mjs +3 -3
- package/esm2022/components/filemanager/toolbar/icon-button/icon-button.component.mjs +3 -3
- package/esm2022/components/filemanager/toolbar/toolbar.component.mjs +3 -3
- package/esm2022/components/filemanager/tree-view/tree-view.component.mjs +3 -3
- package/esm2022/components/lazy-loader/lazy-loader.component.mjs +3 -3
- package/esm2022/components/lazy-loader/lazy-loader.module.mjs +4 -4
- package/esm2022/components/lazy-loader/lazy-loader.service.mjs +3 -3
- package/esm2022/components/menu/menu.component.mjs +3 -3
- package/esm2022/components/react-magic-wrapper/react-magic-wrapper.component.mjs +3 -3
- package/esm2022/components/tabulator/tabulator.component.mjs +3 -3
- package/esm2022/components/tooltip/tooltip.component.mjs +3 -3
- package/esm2022/components/vscode/vscode.component.mjs +3 -3
- package/esm2022/directives/image-cache.directive.mjs +143 -0
- package/esm2022/directives/menu.directive.mjs +3 -3
- package/esm2022/directives/tooltip.directive.mjs +3 -3
- package/esm2022/pipes/html-bypass.pipe.mjs +3 -3
- package/esm2022/pipes/resource-bypass.pipe.mjs +3 -3
- package/esm2022/pipes/script-bypass.pipe.mjs +3 -3
- package/esm2022/pipes/style-bypass.pipe.mjs +3 -3
- package/esm2022/pipes/url-bypass.pipe.mjs +3 -3
- package/esm2022/public-api.mjs +2 -1
- package/esm2022/services/command-palette.service.mjs +8 -5
- package/esm2022/services/dependency.service.mjs +3 -3
- package/esm2022/services/dialog.service.mjs +3 -3
- package/esm2022/services/fetch.service.mjs +3 -3
- package/esm2022/services/file.service.mjs +3 -3
- package/esm2022/services/keyboard.service.mjs +3 -3
- package/esm2022/services/navigation.service.mjs +3 -3
- package/esm2022/services/theme.service.mjs +3 -3
- package/fesm2022/{dotglitch-ngx-common-folder-rename.component-0RqscdRS.mjs → dotglitch-ngx-common-folder-rename.component-VI_uH5-6.mjs} +8 -7
- package/fesm2022/dotglitch-ngx-common-folder-rename.component-VI_uH5-6.mjs.map +1 -0
- package/fesm2022/dotglitch-ngx-common.mjs +467 -162
- package/fesm2022/dotglitch-ngx-common.mjs.map +1 -1
- package/package.json +5 -4
- package/public-api.d.ts +1 -0
- package/services/command-palette.service.d.ts +20 -1
- 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
|
-
|
|
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, "
|
|
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, "
|
|
8
|
+
static ɵcmp: i0.ɵɵComponentDeclaration<ShortcutComponent, "ngx-commandpalette-shortcut", never, { "shortcut": { "alias": "shortcut"; "required": false; }; }, {}, never, never, true, never>;
|
|
9
9
|
}
|
|
@@ -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
|
+
}
|
|
@@ -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/
|
|
11
|
-
import * as i4 from "@angular/material/
|
|
12
|
-
import * as i5 from "@angular/
|
|
13
|
-
import * as i6 from "
|
|
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
|
-
|
|
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
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
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
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
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
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
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
|
-
|
|
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
|
-
|
|
95
|
-
|
|
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
|
-
|
|
220
|
+
if (!isDevMode()) {
|
|
221
|
+
this.dialog.close();
|
|
222
|
+
}
|
|
99
223
|
}
|
|
100
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.1.
|
|
101
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.1.
|
|
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.
|
|
227
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.1.2", ngImport: i0, type: CommandPaletteComponent, decorators: [{
|
|
104
228
|
type: Component,
|
|
105
|
-
args: [{ selector: '
|
|
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
|
-
|
|
111
|
-
|
|
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: {
|
|
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":"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","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"]}
|