@nakedobjects/gemini
Version:
Single Page Application client for a Naked Objects application.
143 lines • 24 kB
JavaScript
import { Component, Input, ViewChild } from '@angular/core';
import { BehaviorSubject } from 'rxjs';
import { accept, dropOn, focus, paste, safeUnsubscribe } from '../helpers-components';
import * as i0 from "@angular/core";
import * as i1 from "@nakedobjects/view-models";
import * as i2 from "@angular/common";
import * as i3 from "@angular/forms";
import * as i4 from "@angular/cdk/drag-drop";
import * as i5 from "../clear.directive";
export class AutoCompleteComponent {
dragAndDrop;
constructor(dragAndDrop) {
this.dragAndDrop = dragAndDrop;
}
fieldViewModel;
bSubject;
sub;
currentIndex = -1;
inputField;
canDrop = false;
dragOver = false;
set model(m) {
this.fieldViewModel = m;
this.dragAndDrop.setDropZoneId(this.modelPaneId);
}
get model() {
return this.fieldViewModel;
}
form;
get modelPaneId() {
return this.model.paneArgId;
}
get modelId() {
return this.model.id;
}
get control() {
return this.form.controls[this.model.id];
}
get choices() {
return this.model.choices;
}
get accept() {
// eslint-disable-next-line @typescript-eslint/no-this-alias
const _this = this;
return (cdkDrag, _cdkDropList) => {
return accept(_this.model, _this, cdkDrag.data);
};
}
drop(event) {
const cdkDrag = event.item;
if (event.isPointerOverContainer) {
dropOn(cdkDrag.data, this.model, this);
}
this.canDrop = false;
this.dragOver = false;
}
exit() {
this.canDrop = false;
this.dragOver = false;
}
enter() {
this.dragOver = true;
}
classes() {
return {
[this.model.color]: true,
'candrop': this.canDrop,
'dnd-drag-over': this.dragOver,
};
}
get description() {
return this.model.description;
}
paste(event) {
paste(event, this.model, this, () => this.dragAndDrop.getCopyViewModel(), () => this.dragAndDrop.setCopyViewModel(null));
}
clear() {
this.model.clear();
this.control.reset();
}
select(item) {
this.model.choices = [];
this.model.selectedChoice = item;
this.control.reset(item);
}
choiceName = (choice) => choice.name;
get subject() {
if (!this.bSubject) {
const initialValue = this.control.value;
this.bSubject = new BehaviorSubject(initialValue);
this.sub = this.control.valueChanges.subscribe((data) => {
this.bSubject.next(data);
this.currentIndex = -1;
});
}
return this.bSubject;
}
isSelected(i) {
return { 'selected': i === this.currentIndex };
}
onArrowUp() {
this.currentIndex--;
this.currentIndex = this.currentIndex < -1 ? -1 : this.currentIndex;
return false;
}
onArrowDown() {
this.currentIndex++;
const maxIndex = this.choices.length - 1;
this.currentIndex = this.currentIndex > maxIndex ? maxIndex : this.currentIndex;
return false;
}
selectCurrent() {
const maxIndex = this.choices.length - 1;
if (this.currentIndex >= 0 && this.currentIndex <= maxIndex) {
this.select(this.choices[this.currentIndex]);
return false;
}
return true;
}
ngOnDestroy() {
this.dragAndDrop.clearDropZoneId(this.modelPaneId);
safeUnsubscribe(this.sub);
}
focus() {
return focus(this.inputField);
}
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.9", ngImport: i0, type: AutoCompleteComponent, deps: [{ token: i1.DragAndDropService }], target: i0.ɵɵFactoryTarget.Component });
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.9", type: AutoCompleteComponent, selector: "nof-auto-complete", inputs: { model: "model", form: "form" }, viewQueries: [{ propertyName: "inputField", first: true, predicate: ["focus"], descendants: true }], ngImport: i0, template: "<div [formGroup]=\"form\">\n <input\n #focus\n [id]=\"modelPaneId\"\n class=\"value droppable\"\n cdkDropList\n (cdkDropListDropped)=\"drop($event)\"\n (cdkDropListExited)=\"exit()\"\n (cdkDropListEntered)=\"enter()\"\n [cdkDropListEnterPredicate]=\"accept\"\n [ngClass]=\"classes()\"\n placeholder=\"{{description}}\"\n type=\"text\"\n [formControlName]=\"modelId\"\n [nofClear]=\"subject\"\n (clear)=\"clear()\"\n (keydown.arrowup)=\"onArrowUp()\"\n (keydown.arrowdown)=\"onArrowDown()\"\n (keyup.enter)=\"selectCurrent()\"/>\n <div class=\"suggestions\" *ngIf=\"choices.length > 0\">\n <ul >\n <li [ngClass]=\"isSelected(i)\" *ngFor=\"let choice of choices; let i = index\">\n <a (click)=\"select(choice)\">{{choiceName(choice)}}</a>\n </li>\n </ul>\n </div>\n</div>", styles: [".suggestions ul{list-style-type:none;width:var(--field-value-width);border-style:solid;border-width:1px;border-color:#000;margin:0;font-family:var(--font-text-family);font-size:var(--font-size-1);padding-left:0;font-weight:var(--font-weight-3)}.validation{font-family:var(--font-text-family);font-size:var(--font-size-1)}input{width:var(--field-value-width);padding-left:var(--space-3);border-style:solid;border-width:1px;border-color:gray}li a{padding-left:var(--space-3)}li:hover,.selected{background-color:var(--header-background-color);background-color:gray}.ng-clearable{background-image:var(--clearable-image);background-repeat:no-repeat;background-position:right -10px;background-size:8px}.ng-clearable[class*=link-color]{background-image:var(--clearable-link-image)}.ng-clearable.ng-x{background-position:right 2px center}.ng-clearable.ng-onX{cursor:pointer}input::-ms-clear{display:none}.value.droppable{border:solid;border-width:1px;width:var(--field-value-width);height:var(--field-value-height)}.value.droppable{border-color:#000}.value.droppable.dnd-drag-over.candrop{border-color:var(--approval-color);cursor:pointer}.value.droppable.dnd-drag-over{border-color:var(--warning-color)}\n"], dependencies: [{ kind: "directive", type: i2.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i3.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i3.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i3.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i3.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i3.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "directive", type: i4.CdkDropList, selector: "[cdkDropList], cdk-drop-list", inputs: ["cdkDropListConnectedTo", "cdkDropListData", "cdkDropListOrientation", "id", "cdkDropListLockAxis", "cdkDropListDisabled", "cdkDropListSortingDisabled", "cdkDropListEnterPredicate", "cdkDropListSortPredicate", "cdkDropListAutoScrollDisabled", "cdkDropListAutoScrollStep", "cdkDropListElementContainer"], outputs: ["cdkDropListDropped", "cdkDropListEntered", "cdkDropListExited", "cdkDropListSorted"], exportAs: ["cdkDropList"] }, { kind: "directive", type: i5.ClearDirective, selector: "[nofClear]", inputs: ["nofClear"], outputs: ["clear"] }] });
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.9", ngImport: i0, type: AutoCompleteComponent, decorators: [{
type: Component,
args: [{ selector: 'nof-auto-complete', template: "<div [formGroup]=\"form\">\n <input\n #focus\n [id]=\"modelPaneId\"\n class=\"value droppable\"\n cdkDropList\n (cdkDropListDropped)=\"drop($event)\"\n (cdkDropListExited)=\"exit()\"\n (cdkDropListEntered)=\"enter()\"\n [cdkDropListEnterPredicate]=\"accept\"\n [ngClass]=\"classes()\"\n placeholder=\"{{description}}\"\n type=\"text\"\n [formControlName]=\"modelId\"\n [nofClear]=\"subject\"\n (clear)=\"clear()\"\n (keydown.arrowup)=\"onArrowUp()\"\n (keydown.arrowdown)=\"onArrowDown()\"\n (keyup.enter)=\"selectCurrent()\"/>\n <div class=\"suggestions\" *ngIf=\"choices.length > 0\">\n <ul >\n <li [ngClass]=\"isSelected(i)\" *ngFor=\"let choice of choices; let i = index\">\n <a (click)=\"select(choice)\">{{choiceName(choice)}}</a>\n </li>\n </ul>\n </div>\n</div>", styles: [".suggestions ul{list-style-type:none;width:var(--field-value-width);border-style:solid;border-width:1px;border-color:#000;margin:0;font-family:var(--font-text-family);font-size:var(--font-size-1);padding-left:0;font-weight:var(--font-weight-3)}.validation{font-family:var(--font-text-family);font-size:var(--font-size-1)}input{width:var(--field-value-width);padding-left:var(--space-3);border-style:solid;border-width:1px;border-color:gray}li a{padding-left:var(--space-3)}li:hover,.selected{background-color:var(--header-background-color);background-color:gray}.ng-clearable{background-image:var(--clearable-image);background-repeat:no-repeat;background-position:right -10px;background-size:8px}.ng-clearable[class*=link-color]{background-image:var(--clearable-link-image)}.ng-clearable.ng-x{background-position:right 2px center}.ng-clearable.ng-onX{cursor:pointer}input::-ms-clear{display:none}.value.droppable{border:solid;border-width:1px;width:var(--field-value-width);height:var(--field-value-height)}.value.droppable{border-color:#000}.value.droppable.dnd-drag-over.candrop{border-color:var(--approval-color);cursor:pointer}.value.droppable.dnd-drag-over{border-color:var(--warning-color)}\n"] }]
}], ctorParameters: () => [{ type: i1.DragAndDropService }], propDecorators: { inputField: [{
type: ViewChild,
args: ['focus', { static: false }]
}], model: [{
type: Input,
args: [{ required: true }]
}], form: [{
type: Input,
args: [{ required: true }]
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"auto-complete.component.js","sourceRoot":"","sources":["../../../../gemini/src/auto-complete/auto-complete.component.ts","../../../../gemini/src/auto-complete/auto-complete.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CdkDrag, CdkDragDrop, CdkDropList } from '@angular/cdk/drag-drop';\nimport { Component, ElementRef, Input, OnDestroy, ViewChild } from '@angular/core';\nimport { AbstractControl, FormGroup } from '@angular/forms';\nimport { ChoiceViewModel, DragAndDropService, FieldViewModel, IDraggableViewModel } from '@nakedobjects/view-models';\nimport { Dictionary } from 'lodash';\nimport { BehaviorSubject, SubscriptionLike as ISubscription } from 'rxjs';\nimport { accept, dropOn, focus, paste, safeUnsubscribe } from '../helpers-components';\n\n@Component({\n    selector: 'nof-auto-complete',\n    templateUrl: 'auto-complete.component.html',\n    styleUrls: ['auto-complete.component.css']\n})\nexport class AutoCompleteComponent implements OnDestroy {\n\n    constructor(\n        private readonly dragAndDrop: DragAndDropService\n    ) { }\n\n    private fieldViewModel!: FieldViewModel;\n    private bSubject?: BehaviorSubject<unknown>;\n    private sub?: ISubscription;\n    private currentIndex = -1;\n\n    @ViewChild('focus', {static: false})\n    inputField?: ElementRef;\n    \n    canDrop = false;\n    dragOver = false;\n\n    @Input({required : true})\n    set model(m: FieldViewModel) {\n        this.fieldViewModel = m;\n        this.dragAndDrop.setDropZoneId(this.modelPaneId);\n    }\n\n    get model() {\n        return this.fieldViewModel;\n    }\n\n    @Input({required : true})\n    form!: FormGroup;\n\n\n    get modelPaneId() {\n        return this.model.paneArgId;\n    }\n\n    get modelId() {\n        return this.model.id;\n    }\n\n    get control(): AbstractControl {\n        return this.form.controls[this.model.id];\n    }\n\n    get choices(): ChoiceViewModel[] {\n        return this.model.choices;\n    }\n\n    get accept() {\n        // eslint-disable-next-line @typescript-eslint/no-this-alias\n        const _this = this;\n        return (cdkDrag: CdkDrag<IDraggableViewModel>, _cdkDropList: CdkDropList) => {\n            return accept(_this.model, _this, cdkDrag.data);\n        };\n    }\n\n    drop(event: CdkDragDrop<CdkDrag<IDraggableViewModel>>) {\n        const cdkDrag: CdkDrag<IDraggableViewModel> = event.item;\n        if (event.isPointerOverContainer) {\n            dropOn(cdkDrag.data, this.model, this);\n        }\n        this.canDrop = false;\n        this.dragOver = false;\n    }\n\n    exit() {\n        this.canDrop = false;\n        this.dragOver = false;\n    }\n\n    enter() {\n        this.dragOver = true;\n    }\n\n    classes(): Dictionary<boolean | null> {\n        return {\n            [this.model.color]: true,\n            'candrop': this.canDrop,\n            'dnd-drag-over': this.dragOver,\n        };\n    }\n\n    get description() {\n        return this.model.description;\n    }\n\n    paste(event: KeyboardEvent) {\n        paste(event, this.model, this, () => this.dragAndDrop.getCopyViewModel(), () => this.dragAndDrop.setCopyViewModel(null));\n    }\n\n    clear() {\n        this.model.clear();\n        this.control.reset();\n    }\n\n    select(item: ChoiceViewModel) {\n        this.model.choices = [];\n        this.model.selectedChoice = item;\n        this.control.reset(item);\n    }\n\n    choiceName = (choice: ChoiceViewModel) => choice.name;\n\n    get subject() {\n        if (!this.bSubject) {\n            const initialValue = this.control.value;\n            this.bSubject = new BehaviorSubject(initialValue);\n\n            this.sub = this.control.valueChanges.subscribe((data) => {\n                this.bSubject!.next(data);\n                this.currentIndex = -1;\n            });\n        }\n\n        return this.bSubject;\n    }\n\n    isSelected(i: number) {\n        return { 'selected': i === this.currentIndex };\n    }\n\n    onArrowUp() {\n        this.currentIndex--;\n        this.currentIndex = this.currentIndex < -1 ? -1 : this.currentIndex;\n        return false;\n    }\n\n    onArrowDown() {\n        this.currentIndex++;\n        const maxIndex = this.choices.length - 1;\n        this.currentIndex = this.currentIndex > maxIndex ? maxIndex : this.currentIndex;\n        return false;\n    }\n\n    selectCurrent() {\n        const maxIndex = this.choices.length - 1;\n        if (this.currentIndex >= 0 && this.currentIndex <= maxIndex) {\n            this.select(this.choices[this.currentIndex]);\n            return false;\n        }\n        return true;\n    }\n\n    ngOnDestroy(): void {\n        this.dragAndDrop.clearDropZoneId(this.modelPaneId);\n        safeUnsubscribe(this.sub);\n    }\n\n    focus() {\n        return focus(this.inputField);\n    }\n}\n","<div [formGroup]=\"form\">\n    <input\n        #focus\n        [id]=\"modelPaneId\"\n        class=\"value droppable\"\n        cdkDropList\n        (cdkDropListDropped)=\"drop($event)\"\n        (cdkDropListExited)=\"exit()\"\n        (cdkDropListEntered)=\"enter()\"\n        [cdkDropListEnterPredicate]=\"accept\"\n        [ngClass]=\"classes()\"\n        placeholder=\"{{description}}\"\n        type=\"text\"\n        [formControlName]=\"modelId\"\n        [nofClear]=\"subject\"\n        (clear)=\"clear()\"\n        (keydown.arrowup)=\"onArrowUp()\"\n        (keydown.arrowdown)=\"onArrowDown()\"\n        (keyup.enter)=\"selectCurrent()\"/>\n    <div class=\"suggestions\" *ngIf=\"choices.length > 0\">\n        <ul >\n            <li [ngClass]=\"isSelected(i)\" *ngFor=\"let choice of choices; let i = index\">\n                <a (click)=\"select(choice)\">{{choiceName(choice)}}</a>\n            </li>\n        </ul>\n    </div>\n</div>"]}