@angular-mdl/core
Version:
Angular components, directives and styles based on material design lite https://getmdl.io.
268 lines • 34.5 kB
JavaScript
import { Component, Injectable, Input, Renderer2, ViewChild, ViewEncapsulation, } from "@angular/core";
import { MdlError } from "../common/mdl-error";
import * as i0 from "@angular/core";
import * as i1 from "@angular/common";
const BOTTOM_LEFT = "bottom-left";
const BOTTOM_RIGHT = "bottom-right";
const TOP_LEFT = "top-left";
const TOP_RIGHT = "top-right";
const UNALIGNED = "unaligned";
// Total duration of the menu animation.
const TRANSITION_DURATION_SECONDS = 0.3;
// The fraction of the total duration we want to use for menu item animations.
const TRANSITION_DURATION_FRACTION = 0.8;
// How long the menu stays open after choosing an option (so the user can see
// the ripple).
const CLOSE_TIMEOUT = 175;
const CSS_ALIGN_MAP = {};
CSS_ALIGN_MAP[BOTTOM_LEFT] = "mdl-menu--bottom-left";
CSS_ALIGN_MAP[BOTTOM_RIGHT] = "mdl-menu--bottom-right";
CSS_ALIGN_MAP[TOP_LEFT] = "mdl-menu--top-left";
CSS_ALIGN_MAP[TOP_RIGHT] = "mdl-menu--top-right";
CSS_ALIGN_MAP[UNALIGNED] = "mdl-menu--unaligned";
export class MdlMenuError extends MdlError {
}
export class MdlMenuRegisty {
constructor() {
this.menuComponents = [];
}
add(menuComponent) {
this.menuComponents.push(menuComponent);
}
remove(menuComponent) {
const fromIndex = this.menuComponents.indexOf(menuComponent);
this.menuComponents.splice(fromIndex, 1);
}
hideAllExcept(menuComponent) {
this.menuComponents.forEach((component) => {
if (component !== menuComponent) {
component.hide();
}
});
}
}
MdlMenuRegisty.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: MdlMenuRegisty, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
MdlMenuRegisty.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: MdlMenuRegisty, providedIn: "root" });
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: MdlMenuRegisty, decorators: [{
type: Injectable,
args: [{
providedIn: "root",
}]
}] });
export class MdlMenuComponent {
constructor(renderer, menuRegistry) {
this.renderer = renderer;
this.menuRegistry = menuRegistry;
this.cssPosition = "mdl-menu--bottom-left";
this.isVisible = false;
this.menuRegistry.add(this);
}
ngOnInit() {
this.cssPosition = CSS_ALIGN_MAP[this.position ?? BOTTOM_LEFT];
}
ngAfterViewInit() {
this.container = this.containerChild?.nativeElement;
this.menuElement = this.menuElementChild?.nativeElement;
this.outline = this.outlineChild?.nativeElement;
// Add a click listener to the document, to close the menu.
const callback = () => {
if (this.isVisible) {
this.hide();
}
return true;
};
this.renderer.listen("window", "click", callback);
this.renderer.listen("window", "touchstart", callback);
}
toggle(event, mdlButton) {
if (!mdlButton) {
throw new MdlMenuError(`MdlButtonComponent is required`);
}
if (this.isVisible) {
this.hide();
}
else {
this.show(event, mdlButton);
}
}
hideOnItemClicked() {
// Wait some time before closing menu, so the user can see the ripple.
setTimeout(() => {
this.hide();
}, CLOSE_TIMEOUT);
}
hide() {
// Remove all transition delays; menu items fade out concurrently.
document.querySelectorAll("mdl-menu-item").forEach((el) => {
el.style.removeProperty("transition-delay");
});
// this.menuItemComponents.toArray().forEach(mi => {
// mi.element.style.removeProperty('transition-delay');
// });
// Measure the inner element.
const rect = this.menuElement?.getBoundingClientRect();
const height = rect?.height;
const width = rect?.width;
// Turn on animation, and apply the final clip. Also make invisible.
// This triggers the transitions.
this.renderer.addClass(this.menuElement, "is-animating");
this.applyClip(height, width);
this.renderer.removeClass(this.container, "is-visible");
// Clean up after the animation is complete.
this.addAnimationEndListener();
this.isVisible = false;
}
show(event, mdlButton) {
this.menuRegistry.hideAllExcept(this);
event.stopPropagation();
const forElement = mdlButton.element;
const rect = forElement.getBoundingClientRect();
const forRect = forElement?.parentElement?.getBoundingClientRect();
if (!this.container || !forRect) {
return;
}
if (this.position === UNALIGNED) {
// Do not position the menu automatically. Requires the developer to
// manually specify position.
}
else if (this.position === BOTTOM_RIGHT) {
// Position below the "for" element, aligned to its right.
this.container.style.right = forRect.right - rect.right + "px";
this.container.style.top =
forElement.offsetTop + forElement.offsetHeight + "px";
}
else if (this.position === TOP_LEFT) {
// Position above the "for" element, aligned to its left.
this.container.style.left = forElement.offsetLeft + "px";
this.container.style.bottom = forRect.bottom - rect.top + "px";
}
else if (this.position === TOP_RIGHT) {
// Position above the "for" element, aligned to its right.
this.container.style.right = forRect.right - rect.right + "px";
this.container.style.bottom = forRect.bottom - rect.top + "px";
}
else {
// Default: position below the "for" element, aligned to its left.
this.container.style.left = forElement.offsetLeft + "px";
this.container.style.top =
forElement.offsetTop + forElement.offsetHeight + "px";
}
// Measure the inner element.
const height = this.menuElement?.getBoundingClientRect().height ?? 0;
const width = this.menuElement?.getBoundingClientRect().width ?? 0;
this.container.style.width = width + "px";
this.container.style.height = height + "px";
if (this.outline) {
this.outline.style.width = width + "px";
this.outline.style.height = height + "px";
}
const transitionDuration = TRANSITION_DURATION_SECONDS * TRANSITION_DURATION_FRACTION;
document.querySelectorAll("mdl-menu-item").forEach((el) => {
const mi = el;
let itemDelay;
if (this.position === TOP_LEFT || this.position === TOP_RIGHT) {
itemDelay =
((height - mi.offsetTop - mi.offsetHeight) / height) *
transitionDuration +
"s";
}
else {
itemDelay = (mi.offsetTop / height) * transitionDuration + "s";
}
mi.style.transitionDelay = itemDelay;
});
// Apply the initial clip to the text before we start animating.
this.applyClip(height, width);
this.renderer.addClass(this.container, "is-visible");
if (this.menuElement) {
this.menuElement.style.clip =
"rect(0 " + width + "px " + height + "px 0)";
this.renderer.addClass(this.menuElement, "is-animating");
}
this.addAnimationEndListener();
this.isVisible = true;
}
ngOnDestroy() {
this.menuRegistry.remove(this);
}
addAnimationEndListener() {
this.renderer.listen(this.menuElement, "transitionend", () => {
this.renderer.removeClass(this.menuElement, "is-animating");
return true;
});
}
applyClip(height, width) {
if (!this.menuElement) {
return;
}
if (this.position === UNALIGNED) {
// Do not clip.
this.menuElement.style.clip = "";
}
else if (this.position === BOTTOM_RIGHT) {
// Clip to the top right corner of the menu.
this.menuElement.style.clip =
"rect(0 " + width + "px " + "0 " + width + "px)";
}
else if (this.position === TOP_LEFT) {
// Clip to the bottom left corner of the menu.
this.menuElement.style.clip =
"rect(" + height + "px 0 " + height + "px 0)";
}
else if (this.position === TOP_RIGHT) {
// Clip to the bottom right corner of the menu.
this.menuElement.style.clip =
"rect(" +
height +
"px " +
width +
"px " +
height +
"px " +
width +
"px)";
}
else {
// Default: do not clip (same as clipping to the top left corner).
this.menuElement.style.clip = "";
}
}
}
MdlMenuComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: MdlMenuComponent, deps: [{ token: i0.Renderer2 }, { token: MdlMenuRegisty }], target: i0.ɵɵFactoryTarget.Component });
MdlMenuComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.0.4", type: MdlMenuComponent, selector: "mdl-menu", inputs: { position: ["mdl-menu-position", "position"] }, viewQueries: [{ propertyName: "containerChild", first: true, predicate: ["container"], descendants: true, static: true }, { propertyName: "menuElementChild", first: true, predicate: ["menuElement"], descendants: true, static: true }, { propertyName: "outlineChild", first: true, predicate: ["outline"], descendants: true, static: true }], exportAs: ["mdlMenu"], ngImport: i0, template: `
<div #container class="mdl-menu__container is-upgraded">
<div #outline class="mdl-menu__outline" [ngClass]="cssPosition"></div>
<div class="mdl-menu" #menuElement>
<ng-content></ng-content>
</div>
</div>
`, isInline: true, dependencies: [{ kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }], encapsulation: i0.ViewEncapsulation.None });
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: MdlMenuComponent, decorators: [{
type: Component,
args: [{
selector: "mdl-menu",
exportAs: "mdlMenu",
template: `
<div #container class="mdl-menu__container is-upgraded">
<div #outline class="mdl-menu__outline" [ngClass]="cssPosition"></div>
<div class="mdl-menu" #menuElement>
<ng-content></ng-content>
</div>
</div>
`,
encapsulation: ViewEncapsulation.None,
}]
}], ctorParameters: function () { return [{ type: i0.Renderer2 }, { type: MdlMenuRegisty }]; }, propDecorators: { position: [{
type: Input,
args: ["mdl-menu-position"]
}], containerChild: [{
type: ViewChild,
args: ["container", { static: true }]
}], menuElementChild: [{
type: ViewChild,
args: ["menuElement", { static: true }]
}], outlineChild: [{
type: ViewChild,
args: ["outline", { static: true }]
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"mdl-menu.component.js","sourceRoot":"","sources":["../../../../../../projects/core/src/lib/menu/mdl-menu.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewInit,\n  Component,\n  ElementRef,\n  Injectable,\n  Input,\n  OnDestroy,\n  OnInit,\n  Renderer2,\n  ViewChild,\n  ViewEncapsulation,\n} from \"@angular/core\";\nimport { MdlError } from \"../common/mdl-error\";\nimport { MdlButtonComponent } from \"../button/mdl-button.component\";\n\nconst BOTTOM_LEFT = \"bottom-left\";\nconst BOTTOM_RIGHT = \"bottom-right\";\nconst TOP_LEFT = \"top-left\";\nconst TOP_RIGHT = \"top-right\";\nconst UNALIGNED = \"unaligned\";\n\n// Total duration of the menu animation.\nconst TRANSITION_DURATION_SECONDS = 0.3;\n// The fraction of the total duration we want to use for menu item animations.\nconst TRANSITION_DURATION_FRACTION = 0.8;\n// How long the menu stays open after choosing an option (so the user can see\n// the ripple).\nconst CLOSE_TIMEOUT = 175;\n\nconst CSS_ALIGN_MAP: { [key: string]: string } = {};\nCSS_ALIGN_MAP[BOTTOM_LEFT] = \"mdl-menu--bottom-left\";\nCSS_ALIGN_MAP[BOTTOM_RIGHT] = \"mdl-menu--bottom-right\";\nCSS_ALIGN_MAP[TOP_LEFT] = \"mdl-menu--top-left\";\nCSS_ALIGN_MAP[TOP_RIGHT] = \"mdl-menu--top-right\";\nCSS_ALIGN_MAP[UNALIGNED] = \"mdl-menu--unaligned\";\n\nexport class MdlMenuError extends MdlError {}\n\n@Injectable({\n  providedIn: \"root\",\n})\nexport class MdlMenuRegisty {\n  menuComponents: MdlMenuComponent[] = [];\n\n  add(menuComponent: MdlMenuComponent): void {\n    this.menuComponents.push(menuComponent);\n  }\n\n  remove(menuComponent: MdlMenuComponent): void {\n    const fromIndex = this.menuComponents.indexOf(menuComponent);\n    this.menuComponents.splice(fromIndex, 1);\n  }\n\n  hideAllExcept(menuComponent: MdlMenuComponent): void {\n    this.menuComponents.forEach((component) => {\n      if (component !== menuComponent) {\n        component.hide();\n      }\n    });\n  }\n}\n\n@Component({\n  selector: \"mdl-menu\",\n  exportAs: \"mdlMenu\",\n  template: `\n    <div #container class=\"mdl-menu__container is-upgraded\">\n      <div #outline class=\"mdl-menu__outline\" [ngClass]=\"cssPosition\"></div>\n      <div class=\"mdl-menu\" #menuElement>\n        <ng-content></ng-content>\n      </div>\n    </div>\n  `,\n  encapsulation: ViewEncapsulation.None,\n})\nexport class MdlMenuComponent implements OnInit, AfterViewInit, OnDestroy {\n  // eslint-disable-next-line\n  @Input(\"mdl-menu-position\")\n  position: string | undefined;\n\n  @ViewChild(\"container\", { static: true })\n  containerChild: ElementRef | undefined;\n  @ViewChild(\"menuElement\", { static: true })\n  menuElementChild: ElementRef | undefined;\n  @ViewChild(\"outline\", { static: true })\n  outlineChild: ElementRef | undefined;\n\n  public cssPosition = \"mdl-menu--bottom-left\";\n  private container: HTMLElement | undefined;\n  private menuElement: HTMLElement | undefined;\n  private outline: HTMLElement | undefined;\n  private isVisible = false;\n\n  constructor(\n    private renderer: Renderer2,\n    private menuRegistry: MdlMenuRegisty\n  ) {\n    this.menuRegistry.add(this);\n  }\n\n  ngOnInit(): void {\n    this.cssPosition = CSS_ALIGN_MAP[this.position ?? BOTTOM_LEFT];\n  }\n\n  ngAfterViewInit(): void {\n    this.container = this.containerChild?.nativeElement;\n    this.menuElement = this.menuElementChild?.nativeElement;\n    this.outline = this.outlineChild?.nativeElement;\n\n    // Add a click listener to the document, to close the menu.\n    const callback = () => {\n      if (this.isVisible) {\n        this.hide();\n      }\n      return true;\n    };\n    this.renderer.listen(\"window\", \"click\", callback);\n    this.renderer.listen(\"window\", \"touchstart\", callback);\n  }\n\n  toggle(event: Event, mdlButton: MdlButtonComponent): void {\n    if (!mdlButton) {\n      throw new MdlMenuError(`MdlButtonComponent is required`);\n    }\n    if (this.isVisible) {\n      this.hide();\n    } else {\n      this.show(event, mdlButton);\n    }\n  }\n\n  hideOnItemClicked(): void {\n    // Wait some time before closing menu, so the user can see the ripple.\n    setTimeout(() => {\n      this.hide();\n    }, CLOSE_TIMEOUT);\n  }\n\n  hide(): void {\n    // Remove all transition delays; menu items fade out concurrently.\n    document.querySelectorAll(\"mdl-menu-item\").forEach((el) => {\n      (el as HTMLElement).style.removeProperty(\"transition-delay\");\n    });\n    // this.menuItemComponents.toArray().forEach(mi => {\n    //   mi.element.style.removeProperty('transition-delay');\n    // });\n\n    // Measure the inner element.\n    const rect = this.menuElement?.getBoundingClientRect();\n    const height = rect?.height;\n    const width = rect?.width;\n\n    // Turn on animation, and apply the final clip. Also make invisible.\n    // This triggers the transitions.\n    this.renderer.addClass(this.menuElement, \"is-animating\");\n    this.applyClip(height, width);\n    this.renderer.removeClass(this.container, \"is-visible\");\n\n    // Clean up after the animation is complete.\n    this.addAnimationEndListener();\n\n    this.isVisible = false;\n  }\n\n  show(event: Event, mdlButton: MdlButtonComponent): void {\n    this.menuRegistry.hideAllExcept(this);\n\n    event.stopPropagation();\n\n    const forElement = mdlButton.element;\n    const rect = forElement.getBoundingClientRect();\n    const forRect = forElement?.parentElement?.getBoundingClientRect();\n\n    if (!this.container || !forRect) {\n      return;\n    }\n    if (this.position === UNALIGNED) {\n      // Do not position the menu automatically. Requires the developer to\n      // manually specify position.\n    } else if (this.position === BOTTOM_RIGHT) {\n      // Position below the \"for\" element, aligned to its right.\n      this.container.style.right = forRect.right - rect.right + \"px\";\n      this.container.style.top =\n        forElement.offsetTop + forElement.offsetHeight + \"px\";\n    } else if (this.position === TOP_LEFT) {\n      // Position above the \"for\" element, aligned to its left.\n      this.container.style.left = forElement.offsetLeft + \"px\";\n      this.container.style.bottom = forRect.bottom - rect.top + \"px\";\n    } else if (this.position === TOP_RIGHT) {\n      // Position above the \"for\" element, aligned to its right.\n      this.container.style.right = forRect.right - rect.right + \"px\";\n      this.container.style.bottom = forRect.bottom - rect.top + \"px\";\n    } else {\n      // Default: position below the \"for\" element, aligned to its left.\n      this.container.style.left = forElement.offsetLeft + \"px\";\n      this.container.style.top =\n        forElement.offsetTop + forElement.offsetHeight + \"px\";\n    }\n\n    // Measure the inner element.\n    const height = this.menuElement?.getBoundingClientRect().height ?? 0;\n    const width = this.menuElement?.getBoundingClientRect().width ?? 0;\n\n    this.container.style.width = width + \"px\";\n    this.container.style.height = height + \"px\";\n    if (this.outline) {\n      this.outline.style.width = width + \"px\";\n      this.outline.style.height = height + \"px\";\n    }\n\n    const transitionDuration =\n      TRANSITION_DURATION_SECONDS * TRANSITION_DURATION_FRACTION;\n    document.querySelectorAll(\"mdl-menu-item\").forEach((el) => {\n      const mi = el as HTMLElement;\n      let itemDelay;\n      if (this.position === TOP_LEFT || this.position === TOP_RIGHT) {\n        itemDelay =\n          ((height - mi.offsetTop - mi.offsetHeight) / height) *\n            transitionDuration +\n          \"s\";\n      } else {\n        itemDelay = (mi.offsetTop / height) * transitionDuration + \"s\";\n      }\n      mi.style.transitionDelay = itemDelay;\n    });\n\n    // Apply the initial clip to the text before we start animating.\n    this.applyClip(height, width);\n\n    this.renderer.addClass(this.container, \"is-visible\");\n    if (this.menuElement) {\n      this.menuElement.style.clip =\n        \"rect(0 \" + width + \"px \" + height + \"px 0)\";\n      this.renderer.addClass(this.menuElement, \"is-animating\");\n    }\n\n    this.addAnimationEndListener();\n\n    this.isVisible = true;\n  }\n\n  ngOnDestroy(): void {\n    this.menuRegistry.remove(this);\n  }\n\n  private addAnimationEndListener() {\n    this.renderer.listen(this.menuElement, \"transitionend\", () => {\n      this.renderer.removeClass(this.menuElement, \"is-animating\");\n      return true;\n    });\n  }\n\n  private applyClip(height: number | undefined, width: number | undefined) {\n    if (!this.menuElement) {\n      return;\n    }\n    if (this.position === UNALIGNED) {\n      // Do not clip.\n      this.menuElement.style.clip = \"\";\n    } else if (this.position === BOTTOM_RIGHT) {\n      // Clip to the top right corner of the menu.\n      this.menuElement.style.clip =\n        \"rect(0 \" + width + \"px \" + \"0 \" + width + \"px)\";\n    } else if (this.position === TOP_LEFT) {\n      // Clip to the bottom left corner of the menu.\n      this.menuElement.style.clip =\n        \"rect(\" + height + \"px 0 \" + height + \"px 0)\";\n    } else if (this.position === TOP_RIGHT) {\n      // Clip to the bottom right corner of the menu.\n      this.menuElement.style.clip =\n        \"rect(\" +\n        height +\n        \"px \" +\n        width +\n        \"px \" +\n        height +\n        \"px \" +\n        width +\n        \"px)\";\n    } else {\n      // Default: do not clip (same as clipping to the top left corner).\n      this.menuElement.style.clip = \"\";\n    }\n  }\n}\n"]}