import {Component, ElementRef, Renderer, HostListener, ViewEncapsulation} from '@angular/core'; import {Animation} from '../../animations/animation'; import {Config} from '../../config/config'; import {isPresent} from '../../util/util'; import {Key} from '../../util/key'; import {NavParams} from '../nav/nav-params'; import {Transition, TransitionOptions} from '../../transitions/transition'; import {ViewController} from '../nav/view-controller'; /** * @private */ @Component({ selector: 'ion-alert', template: `

`, host: { 'role': 'dialog', '[attr.aria-labelledby]': 'hdrId', '[attr.aria-describedby]': 'descId' }, encapsulation: ViewEncapsulation.None, }) export class AlertCmp { private activeId: string; private descId: string; private d: { cssClass?: string; message?: string; subTitle?: string; buttons?: any[]; inputs?: any[]; enableBackdropDismiss?: boolean; }; private enabled: boolean; private hdrId: string; private id: number; private inputType: string; private lastClick: number; private msgId: string; private subHdrId: string; constructor( private _viewCtrl: ViewController, private _elementRef: ElementRef, private _config: Config, params: NavParams, renderer: Renderer ) { this.d = params.data; if (this.d.cssClass) { this.d.cssClass.split(' ').forEach(cssClass => { renderer.setElementClass(_elementRef.nativeElement, cssClass, true); }); } this.id = (++alertIds); this.descId = ''; this.hdrId = 'alert-hdr-' + this.id; this.subHdrId = 'alert-subhdr-' + this.id; this.msgId = 'alert-msg-' + this.id; this.activeId = ''; this.lastClick = 0; if (this.d.message) { this.descId = this.msgId; } else if (this.d.subTitle) { this.descId = this.subHdrId; } if (!this.d.message) { this.d.message = ''; } } ionViewLoaded() { // normalize the data let data = this.d; data.buttons = data.buttons.map(button => { if (typeof button === 'string') { return { text: button }; } return button; }); data.inputs = data.inputs.map((input, index) => { return { type: input.type || 'text', name: isPresent(input.name) ? input.name : index, placeholder: isPresent(input.placeholder) ? input.placeholder : '', value: isPresent(input.value) ? input.value : '', label: input.label, checked: !!input.checked, disabled: !!input.disabled, id: 'alert-input-' + this.id + '-' + index }; }); // An alert can be created with several different inputs. Radios, // checkboxes and inputs are all accepted, but they cannot be mixed. let inputTypes: any[] = []; data.inputs.forEach(input => { if (inputTypes.indexOf(input.type) < 0) { inputTypes.push(input.type); } }); if (inputTypes.length > 1 && (inputTypes.indexOf('checkbox') > -1 || inputTypes.indexOf('radio') > -1)) { console.warn('Alert cannot mix input types: ' + (inputTypes.join('/')) + '. Please see alert docs for more info.'); } this.inputType = inputTypes.length ? inputTypes[0] : null; let checkedInput = this.d.inputs.find(input => input.checked); if (checkedInput) { this.activeId = checkedInput.id; } } @HostListener('body:keyup', ['$event']) private _keyUp(ev: KeyboardEvent) { if (this.enabled && this._viewCtrl.isLast()) { if (ev.keyCode === Key.ENTER) { if (this.lastClick + 1000 < Date.now()) { // do not fire this click if there recently was already a click // this can happen when the button has focus and used the enter // key to click the button. However, both the click handler and // this keyup event will fire, so only allow one of them to go. console.debug('alert, enter button'); let button = this.d.buttons[this.d.buttons.length - 1]; this.btnClick(button); } } else if (ev.keyCode === Key.ESCAPE) { console.debug('alert, escape button'); this.bdClick(); } } } ionViewDidEnter() { let activeElement: any = document.activeElement; if (document.activeElement) { activeElement.blur(); } let focusableEle = this._elementRef.nativeElement.querySelector('input,button'); if (focusableEle) { focusableEle.focus(); } this.enabled = true; } btnClick(button: any, dismissDelay?: number) { if (!this.enabled) { return; } // keep the time of the most recent button click this.lastClick = Date.now(); let shouldDismiss = true; if (button.handler) { // a handler has been provided, execute it // pass the handler the values from the inputs if (button.handler(this.getValues()) === false) { // if the return value of the handler is false then do not dismiss shouldDismiss = false; } } if (shouldDismiss) { setTimeout(() => { this.dismiss(button.role); }, dismissDelay || this._config.get('pageTransitionDelay')); } } rbClick(checkedInput: any) { if (this.enabled) { this.d.inputs.forEach(input => { input.checked = (checkedInput === input); }); this.activeId = checkedInput.id; } } cbClick(checkedInput: any) { if (this.enabled) { checkedInput.checked = !checkedInput.checked; } } bdClick() { if (this.enabled && this.d.enableBackdropDismiss) { let cancelBtn = this.d.buttons.find(b => b.role === 'cancel'); if (cancelBtn) { this.btnClick(cancelBtn, 1); } else { this.dismiss('backdrop'); } } } dismiss(role: any): Promise { return this._viewCtrl.dismiss(this.getValues(), role); } getValues() { if (this.inputType === 'radio') { // this is an alert with radio buttons (single value select) // return the one value which is checked, otherwise undefined let checkedInput = this.d.inputs.find(i => i.checked); return checkedInput ? checkedInput.value : undefined; } if (this.inputType === 'checkbox') { // this is an alert with checkboxes (multiple value select) // return an array of all the checked values return this.d.inputs.filter(i => i.checked).map(i => i.value); } // this is an alert with text inputs // return an object of all the values with the input name as the key let values: {[k: string]: string} = {}; this.d.inputs.forEach(i => { values[i.name] = i.value; }); return values; } } /** * Animations for alerts */ class AlertPopIn extends Transition { constructor(enteringView: ViewController, leavingView: ViewController, opts: TransitionOptions) { super(enteringView, leavingView, opts); let ele = enteringView.pageRef().nativeElement; let backdrop = new Animation(ele.querySelector('ion-backdrop')); let wrapper = new Animation(ele.querySelector('.alert-wrapper')); wrapper.fromTo('opacity', 0.01, 1).fromTo('scale', 1.1, 1); backdrop.fromTo('opacity', 0.01, 0.3); this .easing('ease-in-out') .duration(200) .add(backdrop) .add(wrapper); } } Transition.register('alert-pop-in', AlertPopIn); class AlertPopOut extends Transition { constructor(enteringView: ViewController, leavingView: ViewController, opts: TransitionOptions) { super(enteringView, leavingView, opts); let ele = leavingView.pageRef().nativeElement; let backdrop = new Animation(ele.querySelector('ion-backdrop')); let wrapper = new Animation(ele.querySelector('.alert-wrapper')); wrapper.fromTo('opacity', 0.99, 0).fromTo('scale', 1, 0.9); backdrop.fromTo('opacity', 0.3, 0); this .easing('ease-in-out') .duration(200) .add(backdrop) .add(wrapper); } } Transition.register('alert-pop-out', AlertPopOut); class AlertMdPopIn extends Transition { constructor(enteringView: ViewController, leavingView: ViewController, opts: TransitionOptions) { super(enteringView, leavingView, opts); let ele = enteringView.pageRef().nativeElement; let backdrop = new Animation(ele.querySelector('ion-backdrop')); let wrapper = new Animation(ele.querySelector('.alert-wrapper')); wrapper.fromTo('opacity', 0.01, 1).fromTo('scale', 1.1, 1); backdrop.fromTo('opacity', 0.01, 0.5); this .easing('ease-in-out') .duration(200) .add(backdrop) .add(wrapper); } } Transition.register('alert-md-pop-in', AlertMdPopIn); class AlertMdPopOut extends Transition { constructor(enteringView: ViewController, leavingView: ViewController, opts: TransitionOptions) { super(enteringView, leavingView, opts); let ele = leavingView.pageRef().nativeElement; let backdrop = new Animation(ele.querySelector('ion-backdrop')); let wrapper = new Animation(ele.querySelector('.alert-wrapper')); wrapper.fromTo('opacity', 0.99, 0).fromTo('scale', 1, 0.9); backdrop.fromTo('opacity', 0.5, 0); this .easing('ease-in-out') .duration(200) .add(backdrop) .add(wrapper); } } Transition.register('alert-md-pop-out', AlertMdPopOut); class AlertWpPopIn extends Transition { constructor(enteringView: ViewController, leavingView: ViewController, opts: TransitionOptions) { super(enteringView, leavingView, opts); let ele = enteringView.pageRef().nativeElement; let backdrop = new Animation(ele.querySelector('ion-backdrop')); let wrapper = new Animation(ele.querySelector('.alert-wrapper')); wrapper.fromTo('opacity', 0.01, 1).fromTo('scale', 1.3, 1); backdrop.fromTo('opacity', 0.01, 0.5); this .easing('cubic-bezier(0,0 0.05,1)') .duration(200) .add(backdrop) .add(wrapper); } } Transition.register('alert-wp-pop-in', AlertWpPopIn); class AlertWpPopOut extends Transition { constructor(enteringView: ViewController, leavingView: ViewController, opts: TransitionOptions) { super(enteringView, leavingView, opts); let ele = leavingView.pageRef().nativeElement; let backdrop = new Animation(ele.querySelector('ion-backdrop')); let wrapper = new Animation(ele.querySelector('.alert-wrapper')); wrapper.fromTo('opacity', 0.99, 0).fromTo('scale', 1, 1.3); backdrop.fromTo('opacity', 0.5, 0); this .easing('ease-out') .duration(150) .add(backdrop) .add(wrapper); } } Transition.register('alert-wp-pop-out', AlertWpPopOut); let alertIds = -1;