mirror of
https://github.com/ionic-team/ionic-framework.git
synced 2025-08-20 12:29:55 +08:00
705 lines
15 KiB
TypeScript
705 lines
15 KiB
TypeScript
import {Directive, Component, ElementRef, Host, EventEmitter, Input, Output} from 'angular2/core';
|
|
import {NgClass} from 'angular2/common';
|
|
|
|
import {Ion} from '../ion';
|
|
import {Animation} from '../../animations/animation';
|
|
import {Gesture} from '../../gestures/gesture';
|
|
import {DragGesture} from '../../gestures/drag-gesture';
|
|
import {dom} from '../../util';
|
|
import {CSS} from '../../util/dom';
|
|
import {debounce, isTrueProperty, defaults} from '../../util/util';
|
|
|
|
import {Swiper} from './swiper-widget';
|
|
import {Scroll} from '../scroll/scroll';
|
|
|
|
|
|
/**
|
|
* @name Slides
|
|
* @description
|
|
* Slides is a slide box implementation based on Swiper.js
|
|
*
|
|
* @usage
|
|
* ```ts
|
|
* @Page({
|
|
* template: `
|
|
* <ion-slides pager (change)="onSlideChanged($event)" (move)="onSlideMove($event)">
|
|
* <ion-slide>
|
|
* <h3>Thank you for choosing the Awesome App!</h3>
|
|
* <p>
|
|
* The number one app for everything awesome.
|
|
* </p>
|
|
* </ion-slide>
|
|
* <ion-slide>
|
|
* <h3>Using Awesome</h3>
|
|
* <div id="list">
|
|
* <h5>Just three steps:</h5>
|
|
* <ol>
|
|
* <li>Be awesome</li>
|
|
* <li>Stay awesome</li>
|
|
* <li>There is no step 3</li>
|
|
* </ol>
|
|
* </div>
|
|
* </ion-slide>
|
|
* <ion-slide>
|
|
* <h3>Any questions?</h3>
|
|
* </ion-slide>
|
|
* </ion-slides>
|
|
* `
|
|
*})
|
|
*
|
|
*```
|
|
* @demo /docs/v2/demos/slides/
|
|
* @see {@link /docs/v2/components#slides Slides Component Docs}
|
|
*
|
|
* Swiper.js:
|
|
* The most modern mobile touch slider and framework with hardware accelerated transitions
|
|
*
|
|
* http://www.idangero.us/swiper/
|
|
*
|
|
* Copyright 2015, Vladimir Kharlampidi
|
|
* The iDangero.us
|
|
* http://www.idangero.us/
|
|
*
|
|
* Licensed under MIT
|
|
*/
|
|
@Component({
|
|
selector: 'ion-slides',
|
|
template:
|
|
'<div class="swiper-container">' +
|
|
'<div class="swiper-wrapper">' +
|
|
'<ng-content></ng-content>' +
|
|
'</div>' +
|
|
'<div [class.hide]="!showPager" class="swiper-pagination"></div>' +
|
|
'</div>',
|
|
directives: [NgClass]
|
|
})
|
|
export class Slides extends Ion {
|
|
|
|
/**
|
|
* @private
|
|
*/
|
|
public rapidUpdate: Function;
|
|
|
|
/**
|
|
* @private
|
|
*/
|
|
private showPager: boolean;
|
|
|
|
/**
|
|
* @private
|
|
*/
|
|
private slider: Swiper;
|
|
|
|
/**
|
|
* @private
|
|
*/
|
|
private maxScale: number;
|
|
|
|
/**
|
|
* @private
|
|
*/
|
|
private zoomElement: HTMLElement;
|
|
|
|
/**
|
|
* @private
|
|
*/
|
|
private zoomGesture: Gesture;
|
|
|
|
/**
|
|
* @private
|
|
*/
|
|
private scale: number;
|
|
|
|
/**
|
|
* @private
|
|
*/
|
|
private zoomLastPosX: number;
|
|
|
|
/**
|
|
* @private
|
|
*/
|
|
private zoomLastPosY: number;
|
|
|
|
/**
|
|
* @private
|
|
*/
|
|
private viewportWidth: number;
|
|
|
|
/**
|
|
* @private
|
|
*/
|
|
private viewportHeight: number;
|
|
|
|
/**
|
|
* @private
|
|
*/
|
|
private enableZoom: boolean;
|
|
|
|
/**
|
|
* @private
|
|
*/
|
|
private touch: {
|
|
x: number,
|
|
y: number,
|
|
startX: number,
|
|
startY: number,
|
|
deltaX: number,
|
|
deltaY: number,
|
|
lastX: number,
|
|
lastY: number,
|
|
target: HTMLElement,
|
|
zoomable: HTMLElement,
|
|
zoomableWidth: number,
|
|
zoomableHeight: number
|
|
};
|
|
|
|
/**
|
|
* @input {boolean} Whether the slide should show the pager or not
|
|
*/
|
|
@Input() pager: any;
|
|
|
|
/**
|
|
* @input {any} Any slider options you want to configure, see swiper parameters: http://www.idangero.us/swiper/api/
|
|
*/
|
|
@Input() options: any;
|
|
|
|
/**
|
|
* @input {number} Whether or not the slider can zoom in or out
|
|
*/
|
|
@Input() zoom: any;
|
|
|
|
/**
|
|
* @input {number} how long it should take to zoom a slide
|
|
*/
|
|
@Input() zoomDuration: any;
|
|
|
|
/**
|
|
* @input {number} the max scale an slide can be zoomed
|
|
*/
|
|
@Input() zoomMax: any;
|
|
|
|
/**
|
|
* @output {any} expression to evaluate when a slide has been changed
|
|
*/
|
|
@Output() change: EventEmitter<any> = new EventEmitter();
|
|
|
|
/**
|
|
* @output {any} expression to evaluate when a slide change starts
|
|
*/
|
|
@Output() slideChangeStart: EventEmitter<any> = new EventEmitter();
|
|
|
|
/**
|
|
* @output {any} expression to evaluate when a slide moves
|
|
*/
|
|
@Output() move: EventEmitter<any> = new EventEmitter();
|
|
|
|
/**
|
|
* @private
|
|
* @param {ElementRef} elementRef TODO
|
|
*/
|
|
constructor(elementRef: ElementRef) {
|
|
super(elementRef);
|
|
this.rapidUpdate = debounce(() => {
|
|
this.update();
|
|
}, 10);
|
|
}
|
|
|
|
/**
|
|
* @private
|
|
*/
|
|
ngOnInit() {
|
|
if (!this.options) {
|
|
this.options = {};
|
|
}
|
|
|
|
this.showPager = isTrueProperty(this.pager);
|
|
|
|
var options = defaults({
|
|
pagination: '.swiper-pagination',
|
|
}, this.options);
|
|
|
|
options.onTap = (swiper, e) => {
|
|
this.onTap(swiper, e);
|
|
return this.options.onTap && this.options.onTap(swiper, e);
|
|
};
|
|
options.onClick = (swiper, e) => {
|
|
this.onClick(swiper, e);
|
|
return this.options.onClick && this.options.onClick(swiper, e);
|
|
};
|
|
options.onDoubleTap = (swiper, e) => {
|
|
this.onDoubleTap(swiper, e);
|
|
return this.options.onDoubleTap && this.options.onDoubleTap(swiper, e);
|
|
};
|
|
options.onTransitionStart = (swiper, e) => {
|
|
this.onTransitionStart(swiper, e);
|
|
return this.options.onTransitionStart && this.options.onTransitionStart(swiper, e);
|
|
};
|
|
options.onTransitionEnd = (swiper, e) => {
|
|
this.onTransitionEnd(swiper, e);
|
|
return this.options.onTransitionEnd && this.options.onTransitionEnd(swiper, e);
|
|
};
|
|
options.onSlideChangeStart = (swiper) => {
|
|
this.slideChangeStart.emit(swiper);
|
|
return this.options.onSlideChangeStart && this.options.onSlideChangeStart(swiper);
|
|
};
|
|
options.onSlideChangeEnd = (swiper) => {
|
|
this.change.emit(swiper);
|
|
return this.options.onSlideChangeEnd && this.options.onSlideChangeEnd(swiper);
|
|
};
|
|
options.onLazyImageLoad = (swiper, slide, img) => {
|
|
return this.options.onLazyImageLoad && this.options.onLazyImageLoad(swiper, slide, img);
|
|
};
|
|
options.onLazyImageReady = (swiper, slide, img) => {
|
|
return this.options.onLazyImageReady && this.options.onLazyImageReady(swiper, slide, img);
|
|
};
|
|
options.onSliderMove = (swiper, e) => {
|
|
this.move.emit(swiper);
|
|
return this.options.onSliderMove && this.options.onSliderMove(swiper, e);
|
|
};
|
|
|
|
setTimeout(() => {
|
|
var swiper = new Swiper(this.getNativeElement().children[0], options);
|
|
this.slider = swiper;
|
|
});
|
|
|
|
/*
|
|
* TODO: Finish this
|
|
if (util.isTrueProperty(this.zoom)) {
|
|
this.enableZoom = true;
|
|
setTimeout(() => {
|
|
this.initZoom();
|
|
})
|
|
}
|
|
*/
|
|
|
|
}
|
|
|
|
/**
|
|
* @private
|
|
*/
|
|
onTap(swiper, e) {
|
|
}
|
|
/**
|
|
* @private
|
|
*/
|
|
onClick(swiper, e) {
|
|
}
|
|
/**
|
|
* @private
|
|
*/
|
|
onDoubleTap(swiper, e) {
|
|
|
|
this.toggleZoom(swiper, e);
|
|
}
|
|
/**
|
|
* @private
|
|
*/
|
|
onLazyImageLoad(swiper, slide, img) {
|
|
}
|
|
/**
|
|
* @private
|
|
*/
|
|
onLazyImageReady(swiper, slide, img) {
|
|
}
|
|
|
|
/*
|
|
nextButton(swiper, e) {
|
|
}
|
|
prevButton() {
|
|
}
|
|
indexButton() {
|
|
}
|
|
*/
|
|
|
|
/**
|
|
* @private
|
|
*/
|
|
initZoom() {
|
|
this.zoomDuration = this.zoomDuration || 230;
|
|
this.maxScale = this.zoomMax || 3;
|
|
|
|
this.zoomElement = this.getNativeElement().children[0].children[0];
|
|
|
|
this.zoomElement && this.zoomElement.classList.add('ion-scroll-zoom');
|
|
|
|
this.zoomGesture = new Gesture(this.zoomElement);
|
|
this.zoomGesture.listen();
|
|
|
|
this.scale = 1;
|
|
|
|
this.zoomLastPosX = 0;
|
|
this.zoomLastPosY = 0;
|
|
|
|
|
|
let last_scale, startX, startY, posX = 0, posY = 0, zoomRect;
|
|
|
|
this.viewportWidth = this.getNativeElement().offsetWidth;
|
|
this.viewportHeight = this.getNativeElement().offsetHeight;
|
|
|
|
this.zoomElement.addEventListener('touchstart', (e) => {
|
|
this.onTouchStart(e);
|
|
});
|
|
|
|
this.zoomElement.addEventListener('touchmove', (e) => {
|
|
this.onTouchMove(e);
|
|
});
|
|
|
|
this.zoomElement.addEventListener('touchend', (e) => {
|
|
this.onTouchEnd(e);
|
|
});
|
|
|
|
this.zoomGesture.on('pinchstart', (e) => {
|
|
last_scale = this.scale;
|
|
console.debug('Last scale', e.scale);
|
|
});
|
|
|
|
this.zoomGesture.on('pinch', (e) => {
|
|
this.scale = Math.max(1, Math.min(last_scale * e.scale, 10));
|
|
console.debug('Scaling', this.scale);
|
|
this.zoomElement.style[CSS.transform] = 'scale(' + this.scale + ')';
|
|
|
|
zoomRect = this.zoomElement.getBoundingClientRect();
|
|
});
|
|
|
|
this.zoomGesture.on('pinchend', () => {
|
|
// last_scale = Math.max(1, Math.min(last_scale * e.scale, 10));
|
|
if (this.scale > this.maxScale) {
|
|
let za = new Animation(this.zoomElement)
|
|
.duration(this.zoomDuration)
|
|
.easing('linear')
|
|
.from('scale', this.scale)
|
|
.to('scale', this.maxScale);
|
|
za.play();
|
|
|
|
this.scale = this.maxScale;
|
|
}
|
|
});
|
|
}
|
|
|
|
/**
|
|
* @private
|
|
*/
|
|
resetZoom() {
|
|
|
|
if (this.zoomElement) {
|
|
|
|
this.zoomElement.parentElement.style[CSS.transform] = '';
|
|
this.zoomElement.style[CSS.transform] = 'scale(1)';
|
|
}
|
|
|
|
this.scale = 1;
|
|
this.zoomLastPosX = 0;
|
|
this.zoomLastPosY = 0;
|
|
}
|
|
|
|
/**
|
|
* @private
|
|
*/
|
|
toggleZoom(swiper, e) {
|
|
console.debug('Try toggle zoom');
|
|
if (!this.enableZoom) { return; }
|
|
|
|
console.debug('Toggling zoom', e);
|
|
|
|
/*
|
|
let x = e.pointers[0].clientX;
|
|
let y = e.pointers[0].clientY;
|
|
|
|
let mx = this.viewportWidth / 2;
|
|
let my = this.viewportHeight / 2;
|
|
|
|
let tx, ty;
|
|
|
|
if (x > mx) {
|
|
// Greater than half
|
|
tx = -x;
|
|
} else {
|
|
// Less than or equal to half
|
|
tx = (this.viewportWidth - x);
|
|
}
|
|
if (y > my) {
|
|
ty = -y;
|
|
} else {
|
|
ty = y-my;
|
|
}
|
|
|
|
console.debug(y);
|
|
*/
|
|
|
|
let zi = new Animation(this.touch.target.children[0])
|
|
.duration(this.zoomDuration)
|
|
.easing('linear');
|
|
|
|
let zw = new Animation(this.touch.target.children[0])
|
|
.duration(this.zoomDuration)
|
|
.easing('linear');
|
|
|
|
let za = new Animation();
|
|
za.add(zi);
|
|
|
|
if (this.scale > 1) {
|
|
// zoom out
|
|
|
|
// zw.fromTo('translateX', posX + 'px', '0px');
|
|
// zw.fromTo('translateY', posY + 'px', '0px');
|
|
|
|
zi.from('scale', this.scale);
|
|
zi.to('scale', 1);
|
|
za.play();
|
|
|
|
// posX = 0;
|
|
// posY = 0;
|
|
|
|
this.scale = 1;
|
|
} else {
|
|
// zoom in
|
|
|
|
// zw.fromTo('translateX', posX + 'px', tx + 'px');
|
|
// zw.fromTo('translateY', posY + 'px', ty + 'px');
|
|
|
|
zi.from('scale', this.scale);
|
|
zi.to('scale', this.maxScale);
|
|
za.play();
|
|
|
|
// posX = tx;
|
|
// posY = ty;
|
|
|
|
this.scale = this.maxScale;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @private
|
|
*/
|
|
onTransitionStart(swiper, e) {
|
|
}
|
|
/**
|
|
* @private
|
|
*/
|
|
onTransitionEnd(swiper, e) {
|
|
}
|
|
|
|
/**
|
|
* @private
|
|
*/
|
|
onTouchStart(e) {
|
|
console.debug('Touch start', e);
|
|
|
|
// TODO: Support mice as well
|
|
|
|
let target = ((dom.closest(e.target, '.slide').children[0] as HTMLElement).children[0] as HTMLElement);
|
|
|
|
this.touch = {
|
|
x: null,
|
|
y: null,
|
|
startX: e.touches[0].clientX,
|
|
startY: e.touches[0].clientY,
|
|
deltaX: 0,
|
|
deltaY: 0,
|
|
lastX: 0,
|
|
lastY: 0,
|
|
target: target.parentElement,
|
|
zoomable: target,
|
|
zoomableWidth: target.offsetWidth,
|
|
zoomableHeight: target.offsetHeight
|
|
};
|
|
console.debug('Target', this.touch.target);
|
|
|
|
// TODO: android prevent default
|
|
}
|
|
|
|
/**
|
|
* @private
|
|
*/
|
|
onTouchMove(e) {
|
|
this.touch.deltaX = e.touches[0].clientX - this.touch.startX;
|
|
this.touch.deltaY = e.touches[0].clientY - this.touch.startY;
|
|
|
|
// TODO: Make sure we need to transform (image is bigger than viewport)
|
|
|
|
let zoomableScaledWidth = this.touch.zoomableWidth * this.scale;
|
|
let zoomableScaledHeight = this.touch.zoomableHeight * this.scale;
|
|
|
|
let x1 = Math.min((this.viewportWidth / 2) - zoomableScaledWidth / 2, 0);
|
|
let x2 = -x1;
|
|
let y1 = Math.min((this.viewportHeight / 2) - zoomableScaledHeight / 2, 0);
|
|
let y2 = -y1;
|
|
|
|
console.debug('BOUNDS', x1, x2, y1, y2);
|
|
|
|
if (this.scale <= 1) {
|
|
return;
|
|
}
|
|
|
|
console.debug('PAN', e);
|
|
|
|
// move image
|
|
this.touch.x = this.touch.deltaX + this.touch.lastX;
|
|
this.touch.y = this.touch.deltaY + this.touch.lastY;
|
|
|
|
if (this.touch.x < x1) {
|
|
console.debug('OUT ON LEFT');
|
|
}
|
|
if (this.touch.x > x2 ) {
|
|
console.debug('OUT ON RIGHT');
|
|
}
|
|
|
|
if (this.touch.x > this.viewportWidth) {
|
|
// too far on the left side, let the event bubble up (to enable slider on edges, for example)
|
|
} else if (-this.touch.x > this.viewportWidth) {
|
|
// too far on the right side, let the event bubble up (to enable slider on edges, for example)
|
|
} else {
|
|
console.debug('TRANSFORM', this.touch.x, this.touch.y, this.touch.target);
|
|
// this.touch.target.style[CSS.transform] = 'translateX(' + this.touch.x + 'px) translateY(' + this.touch.y + 'px)';
|
|
this.touch.target.style[CSS.transform] = 'translateX(' + this.touch.x + 'px) translateY(' + this.touch.y + 'px)';
|
|
e.preventDefault();
|
|
e.stopPropagation();
|
|
return false;
|
|
}
|
|
|
|
}
|
|
|
|
/**
|
|
* @private
|
|
*/
|
|
onTouchEnd(e) {
|
|
console.debug('PANEND', e);
|
|
|
|
if (this.scale > 1) {
|
|
|
|
if (Math.abs(this.touch.x) > this.viewportWidth) {
|
|
// TODO what is posX?
|
|
var posX = posX > 0 ? this.viewportWidth - 1 : -(this.viewportWidth - 1);
|
|
console.debug('Setting on posx', this.touch.x);
|
|
}
|
|
|
|
/*
|
|
if (posY > this.viewportHeight/2) {
|
|
let z = new Animation(this.zoomElement.parentElement);
|
|
z.fromTo('translateY', posY + 'px', Math.min(this.viewportHeight/2 + 30, posY));
|
|
z.play();
|
|
} else {
|
|
let z = new Animation(this.zoomElement.parentElement);
|
|
z.fromTo('translateY', posY + 'px', Math.max(this.viewportHeight/2 - 30, posY));
|
|
z.play();
|
|
}
|
|
*/
|
|
|
|
this.touch.lastX = this.touch.x;
|
|
this.touch.lastY = this.touch.y;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @private
|
|
* Update the underlying slider implementation. Call this if you've added or removed
|
|
* child slides.
|
|
*/
|
|
update() {
|
|
setTimeout(() => {
|
|
this.slider.update();
|
|
|
|
// Don't allow pager to show with > 10 slides
|
|
if (this.slider.slides.length > 10) {
|
|
this.showPager = false;
|
|
}
|
|
});
|
|
}
|
|
|
|
/**
|
|
* @private
|
|
*/
|
|
next() {
|
|
this.slider.slideNext();
|
|
}
|
|
|
|
/**
|
|
* @private
|
|
*/
|
|
prev() {
|
|
this.slider.slidePrev();
|
|
}
|
|
|
|
/**
|
|
* @private
|
|
*/
|
|
getIndex(): number {
|
|
return this.slider.activeIndex;
|
|
}
|
|
|
|
/**
|
|
* @private
|
|
*/
|
|
getNumSlides(): number {
|
|
return this.slider.slides.length;
|
|
}
|
|
|
|
/**
|
|
* @private
|
|
*/
|
|
isAtEnd(): boolean {
|
|
return this.slider.isEnd;
|
|
}
|
|
|
|
/**
|
|
* @private
|
|
*/
|
|
isAtBeginning(): boolean {
|
|
return this.slider.isBeginning;
|
|
}
|
|
|
|
/**
|
|
* @private
|
|
*/
|
|
getSliderWidget() {
|
|
return this.slider;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @name Slide
|
|
* @description
|
|
* `ion-slide` is a child component of `ion-slides` and is where all your individule slide content will be rendered too.
|
|
*
|
|
* @demo /docs/v2/demos/slides/
|
|
* @see {@link /docs/v2/api/components/slides/Slides/ Slides API Docs}
|
|
*/
|
|
@Component({
|
|
selector: 'ion-slide',
|
|
template: '<div class="slide-zoom"><ng-content></ng-content></div>'
|
|
})
|
|
export class Slide {
|
|
|
|
/**
|
|
* @private
|
|
*/
|
|
private ele: HTMLElement;
|
|
|
|
|
|
/**
|
|
* @private
|
|
*/
|
|
@Input() zoom;
|
|
|
|
constructor(
|
|
elementRef: ElementRef,
|
|
@Host() slides: Slides
|
|
) {
|
|
this.ele = elementRef.nativeElement;
|
|
this.ele.classList.add('swiper-slide');
|
|
|
|
slides.rapidUpdate();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @private
|
|
*/
|
|
@Directive({
|
|
selector: 'slide-lazy',
|
|
host: {
|
|
'class': 'swiper-lazy'
|
|
}
|
|
})
|
|
export class SlideLazy {}
|