ack-angular
Version:
Extra special directives, components, providers and pipes to aide in tackling everyday interface development needs in Angular2
195 lines • 21.8 kB
JavaScript
import { array } from "../pipes.class";
import { ContentChildren, TemplateRef, Component, Input, Output, EventEmitter } from "@angular/core";
import { TemplateReader } from "../TemplateReader.class";
import { string as ackOptions } from "./templates/ack-options.pug";
import * as i0 from "@angular/core";
import * as i1 from "@angular/common";
import * as i2 from "../pipes";
export class AckOptions {
constructor(ElementRef) {
this.ElementRef = ElementRef;
this.array = [];
this.stylize = true;
this.TemplateReader = new TemplateReader({
lastTemplateName: "templateRef",
types: {
option: "templateRef",
selected: "selected"
}
});
this.modelChange = new EventEmitter();
}
ngAfterViewInit() {
Promise.resolve().then(() => {
if (this.inputTemplateRefs) {
this.TemplateReader.readTemplates(this.inputTemplateRefs);
}
if (this.templateRefs) {
this.TemplateReader.readTemplates(this.templateRefs);
}
});
}
selectItem(item) {
const value = this.getArrayItemValue(item);
const isArrayMode = this.multiple || this.modelAsArray;
if (isArrayMode) {
const modelIndex = this.modelIndex(item);
if (modelIndex >= 0) {
this.model.splice(modelIndex, 1);
}
else {
this.model.push(this.getArrayItemModel(item));
}
if (this.max) {
while (this.model.length > this.max) {
this.model.shift();
}
}
}
else {
if (this.toggleable && this.model == value) {
delete this.model;
}
else {
this.model = this.getArrayItemModel(item);
}
}
this.emitChange();
}
emitChange() {
this.modelChange.emit(this.model);
const form = getParentByTagName(this.ElementRef.nativeElement, 'form');
if (form)
this.fireFormEvents(form);
}
fireFormEvents(form) {
let event = document.createEvent("HTMLEvents");
event.initEvent("input", true, true);
form.dispatchEvent(event);
event = document.createEvent("HTMLEvents");
event.initEvent("change", true, true);
form.dispatchEvent(event);
}
getArrayItemModel(item) {
if (this.arrayToModelKey != null) {
if (this.arrayToModelKey == '') {
return item;
}
const split = this.arrayToModelKey.split('.');
var scope = item;
while (split.length) {
if (scope == null)
return null;
let key = split.shift();
scope = scope[key];
}
return scope;
}
return this.getArrayItemValue(item);
}
getArrayItemValue(item) {
if (!this.arrayKey)
return item;
let items = this.arrayKey.split('.');
var scope = item;
while (items.length) {
if (scope == null)
return null;
let firstItem = items.shift();
scope = scope[firstItem];
}
return scope;
}
getModelValueToArrayItem(modelValue) {
if (!this.modelKey)
return modelValue;
let items = this.modelKey.split('.');
var scope = modelValue;
while (items.length) {
if (scope == null)
return null;
let firstItem = items.shift();
scope = scope[firstItem];
}
return scope;
}
modelIndex(item) {
this.model = array(this.model);
for (let i = this.model.length - 1; i >= 0; --i) {
let value = this.getArrayItemValue(item);
let modelValue = this.getModelValueToArrayItem(this.model[i]);
if (value == modelValue)
return i;
}
return -1;
}
isItemSelected(item) {
return this.modelIndex(item) >= 0;
}
getItemClass(item) {
const selected = this.isItemSelected(item);
let string = '';
if (this.stylize) {
string += 'cursor-pointer pad-h pad-v-sm border-grey-6x border-bottom ';
}
if (this.stylize && selected) {
string += 'bg-warning ';
}
if (this.stylize && !selected) {
string += 'hover-bg-grey-5x ';
}
return string;
}
}
AckOptions.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.2.4", ngImport: i0, type: AckOptions, deps: [{ token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Component });
AckOptions.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.2.4", type: AckOptions, selector: "ack-options", inputs: { array: "array", stylize: "stylize", multiple: "multiple", modelAsArray: "modelAsArray", max: "max", toggleable: "toggleable", inputTemplateRefs: "inputTemplateRefs", model: "model", arrayKey: "arrayKey", modelKey: "modelKey", arrayToModelKey: "arrayToModelKey" }, outputs: { modelChange: "modelChange" }, queries: [{ propertyName: "templateRefs", predicate: TemplateRef }], ngImport: i0, template: "<div [ngClass]=\"{'border-grey-6x border-top':stylize}\"></div><div *ngFor=\"let item of array|array\" (click)=\"selectItem(item)\" [ngClass]=\"getItemClass(item)\"><ng-template *ngIf=\"TemplateReader.templates.selected && isItemSelected(item)\" [ngTemplateOutlet]=\"TemplateReader.templates.selected\" [ngTemplateOutletContext]=\"{item:item}\"></ng-template><ng-template *ngIf=\"TemplateReader.templates.templateRef && (!TemplateReader.templates.selected || !isItemSelected(item))\" [ngTemplateOutlet]=\"TemplateReader.templates.templateRef\" [ngTemplateOutletContext]=\"{item:item, selected:isItemSelected(item)}\"></ng-template><span *ngIf=\"!TemplateReader.templates.templateRef\">{{ item }}</span></div>", isInline: true, dependencies: [{ kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "pipe", type: i2.ForceArray, name: "array" }] });
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.4", ngImport: i0, type: AckOptions, decorators: [{
type: Component,
args: [{
selector: "ack-options",
template: ackOptions
//,exportAs:"AckOptions"
}]
}], ctorParameters: function () { return [{ type: i0.ElementRef }]; }, propDecorators: { array: [{
type: Input
}], stylize: [{
type: Input
}], multiple: [{
type: Input
}], modelAsArray: [{
type: Input
}], max: [{
type: Input
}], toggleable: [{
type: Input
}], templateRefs: [{
type: ContentChildren,
args: [TemplateRef]
}], inputTemplateRefs: [{
type: Input
}], model: [{
type: Input
}], modelChange: [{
type: Output
}], arrayKey: [{
type: Input
}], modelKey: [{
type: Input
}], arrayToModelKey: [{
type: Input
}] } });
export function getParentByTagName(node, tagname) {
let parent;
if (node === null || tagname === '')
return;
parent = node.parentNode;
tagname = tagname.toUpperCase();
while (parent && parent.tagName !== "HTML") {
if (parent.tagName === tagname) {
return parent;
}
parent = parent.parentNode;
}
return;
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"AckOptions.component.js","sourceRoot":"","sources":["../../../src/components/AckOptions.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { array } from \"../pipes.class\"\nimport {\n  ElementRef,\n  ContentChildren,\n  TemplateRef,\n  Component,\n  Input,\n  Output,\n  EventEmitter\n} from \"@angular/core\"\nimport { TemplateReader } from \"../TemplateReader.class\"\nimport { string as ackOptions } from \"./templates/ack-options.pug\"\n\n@Component({\n  selector:\"ack-options\",\n  template:ackOptions\n  //,exportAs:\"AckOptions\"\n}) export class AckOptions{\n  @Input() array = []\n  @Input() stylize:boolean = true\n  @Input() multiple!: boolean\n  @Input() modelAsArray!: boolean\n  @Input() max!: number\n  @Input() toggleable!: boolean\n\n  TemplateReader:TemplateReader = new TemplateReader({\n    lastTemplateName:\"templateRef\",\n    types:{\n      option:\"templateRef\",\n      selected:\"selected\"\n    }\n  })\n\n  @ContentChildren(TemplateRef) templateRefs:any//TemplateRef<any>[]\n  @Input() inputTemplateRefs:any//TemplateRef<any>[]\n\n  @Input() model:any\n  @Output() modelChange:EventEmitter<any> = new EventEmitter()\n\n  @Input() arrayKey!: string\n  @Input() modelKey!: string\n  @Input() arrayToModelKey!: string\n\n  constructor(public ElementRef:ElementRef){}\n\n  ngAfterViewInit(){\n    Promise.resolve().then(()=>{\n      if( this.inputTemplateRefs ){\n        this.TemplateReader.readTemplates(this.inputTemplateRefs)\n      }\n      if( this.templateRefs ){\n        this.TemplateReader.readTemplates(this.templateRefs)\n      }\n    })\n  }\n\n  selectItem(item: any) {\n    const value = this.getArrayItemValue(item)\n    const isArrayMode = this.multiple || this.modelAsArray\n\n    if( isArrayMode ){\n      const modelIndex = this.modelIndex( item )\n      if(modelIndex>=0){\n        this.model.splice(modelIndex, 1)\n      }else{\n        this.model.push( this.getArrayItemModel(item) )\n      }\n\n      if(this.max){\n        while(this.model.length > this.max){\n          this.model.shift()\n        }\n      }\n    }else{\n      if(this.toggleable && this.model==value){\n        delete this.model\n      }else{\n        this.model = this.getArrayItemModel(item)\n      }\n    }\n\n    this.emitChange()\n  }\n\n  emitChange(){\n    this.modelChange.emit( this.model )\n\n    const form = getParentByTagName(this.ElementRef.nativeElement,'form')\n    if(form)this.fireFormEvents(form)\n  }\n\n  fireFormEvents(form: any) {\n    let event = document.createEvent(\"HTMLEvents\");\n    event.initEvent(\"input\", true, true);\n    form.dispatchEvent(event)\n\n    event = document.createEvent(\"HTMLEvents\");\n    event.initEvent(\"change\", true, true);\n    form.dispatchEvent(event)\n  }\n\n  getArrayItemModel(item: any) {\n    if(this.arrayToModelKey!=null){\n      if(this.arrayToModelKey==''){\n        return item\n      }\n\n      const split = this.arrayToModelKey.split('.')\n      var scope = item\n      while(split.length){\n        if(scope==null)return null\n        let key = <string>split.shift()\n        scope = scope[ key ]\n      }\n      return scope\n    }\n    return this.getArrayItemValue( item )\n  }\n\n  getArrayItemValue(item: any){\n    if(!this.arrayKey)return item\n\n    let items = this.arrayKey.split('.')\n    var scope = item\n    while(items.length){\n      if( scope==null )return null\n      let firstItem = <string>items.shift()\n      scope = scope[ firstItem ]\n    }\n\n    return scope\n  }\n\n  getModelValueToArrayItem( modelValue:any ):any{\n    if(!this.modelKey)return modelValue\n\n    let items:string[] = this.modelKey.split('.')\n    var scope = modelValue\n    while( items.length ){\n      if( scope==null )return null\n      let firstItem = <string>items.shift()\n      scope = scope[ firstItem ]\n    }\n\n    return scope\n  }\n\n  modelIndex(item: any) {\n    this.model = array(this.model)\n    for(let i=this.model.length-1; i >= 0; --i){\n      let value = this.getArrayItemValue( item )\n      let modelValue = this.getModelValueToArrayItem( this.model[i] )\n      if( value == modelValue )return i\n    }\n    return -1\n  }\n\n  isItemSelected(item: any) {\n    return this.modelIndex(item)>=0\n  }\n\n  getItemClass(item: any): string{\n    const selected = this.isItemSelected(item)\n    let string = ''\n\n    if(this.stylize){\n      string += 'cursor-pointer pad-h pad-v-sm border-grey-6x border-bottom '\n    }\n\n    if(this.stylize && selected){\n      string += 'bg-warning '\n    }\n\n    if(this.stylize && !selected){\n      string += 'hover-bg-grey-5x '\n    }\n\n    return string\n\n  }\n}\n\nexport function getParentByTagName(node: any, tagname: any) {\n  let parent;\n  if (node === null || tagname === '') return;\n  parent  = node.parentNode;\n  tagname = tagname.toUpperCase();\n\n  while (parent && parent.tagName !== \"HTML\") {\n    if (parent.tagName === tagname) {\n      return parent;\n    }\n    parent = parent.parentNode;\n  }\n\n  return;\n}\n"]}