feat(picker): add ios/md/wp picker styles

This commit is contained in:
Adam Bradley
2016-04-26 16:03:44 -05:00
parent 066ab712c0
commit aa9a667a3f
8 changed files with 932 additions and 153 deletions

View File

@ -5,45 +5,59 @@
// -------------------------------------------------- // --------------------------------------------------
$picker-ios-height: 260px !default; $picker-ios-height: 260px !default;
$picker-ios-background-color: #cfd5da !default; $picker-ios-border-color: $list-ios-border-color !default;
$picker-ios-background-color: $list-ios-background-color !default;
$picker-ios-toolbar-height: 44px !default; $picker-ios-toolbar-height: 44px !default;
$picker-ios-toolbar-background-color: #f7f7f8 !default; $picker-ios-toolbar-background-color: $picker-ios-background-color !default;
$picker-ios-button-height: $picker-ios-toolbar-height !default; $picker-ios-button-height: $picker-ios-toolbar-height !default;
$picker-ios-button-text-color: color($colors-ios, primary) !default; $picker-ios-button-text-color: $link-ios-color !default;
$picker-ios-button-background-color: transparent !default; $picker-ios-button-background-color: transparent !default;
$picker-ios-option-offset-y: 90px !default; $picker-ios-column-padding: 0 12px !default;
$picker-ios-option-font-size: 18px !default;
$picker-ios-option-line-height: 24px !default; $picker-ios-option-padding: 0 10px !default;
$picker-ios-option-text-color: $list-ios-text-color !default;
$picker-ios-option-font-size: 22px !default;
$picker-ios-option-height: 42px !default;
$picker-ios-option-offset-y: (($picker-ios-height - $picker-ios-toolbar-height) / 2) - ($picker-ios-option-height / 2) - 10 !default;
$picker-highlight-opacity: .8 !default;
.picker-wrapper { .picker-wrapper {
height: $picker-ios-height; height: $picker-ios-height;
border-top: 1px solid #929499; border-top: 1px solid $picker-ios-border-color;
background: $picker-ios-background-color; background: $picker-ios-background-color;
} }
.hairlines .picker-wrapper {
border-width: $hairlines-width;
}
.picker-toolbar { .picker-toolbar {
display: flex; display: flex;
height: $picker-ios-toolbar-height; height: $picker-ios-toolbar-height;
border-bottom: 1px solid $picker-ios-border-color;
background: $picker-ios-toolbar-background-color; background: $picker-ios-toolbar-background-color;
} }
.hairlines .picker-wrapper,
.hairlines .picker-toolbar {
border-width: $hairlines-width;
}
.picker-toolbar-button { .picker-toolbar-button {
flex: 1; flex: 1;
text-align: right; text-align: right;
} }
.picker-toolbar-cancel { .picker-toolbar-cancel {
font-weight: normal;
text-align: left; text-align: left;
} }
@ -57,23 +71,85 @@ $picker-ios-option-line-height: 24px !default;
background: $picker-ios-button-background-color; background: $picker-ios-button-background-color;
} }
.picker-offset { .picker-columns {
transform: translateY($picker-ios-option-offset-y); height: $picker-ios-height - $picker-ios-toolbar-height;
perspective: 1800px;
} }
.picker-column { .picker-col {
padding: 0 10px; padding: $picker-ios-column-padding;
transform-style: preserve-3d;
} }
.picker-prefix, .picker-prefix,
.picker-suffix, .picker-suffix,
.picker-options { .picker-opts {
padding: 0 8px; top: $picker-ios-option-offset-y;
font-size: $picker-ios-option-font-size; font-size: $picker-ios-option-font-size;
line-height: $picker-ios-option-line-height; line-height: $picker-ios-option-height;
color: $picker-ios-option-text-color;
transform-style: preserve-3d;
pointer-events: none;
} }
.picker-prefix, .picker-opt {
.picker-suffix { margin: 0;
padding: 0 padding: $picker-ios-option-padding;
width: calc(100% - 24px);
font-size: $picker-ios-option-font-size;
line-height: $picker-ios-option-height;
background: transparent;
transform-origin: center center;
transform-style: preserve-3d;
transition-timing-function: ease-out;
backface-visibility: hidden;
pointer-events: auto;
}
.picker-above-highlight {
position: absolute;
top: 0;
left: 0;
z-index: 10;
display: block;
width: 100%;
height: $picker-ios-option-offset-y + 4px;
border-bottom: 1px solid $picker-ios-border-color;
background: linear-gradient(to bottom,
rgba($picker-ios-background-color, 1) 20%,
rgba($picker-ios-background-color, .7) 100%);
transform: translate3d(0, 0, 90px);
}
.picker-below-highlight {
position: absolute;
top: $picker-ios-option-offset-y + $picker-ios-option-height - 4;
left: 0;
z-index: 11;
display: block;
width: 100%;
height: $picker-ios-option-offset-y + $picker-ios-option-height;
border-top: 1px solid $picker-ios-border-color;
background: linear-gradient(to top,
rgba($picker-ios-background-color, 1) 30%,
rgba($picker-ios-background-color, .7) 100%);
transform: translate3d(0, 0, 90px);
} }

View File

@ -3,3 +3,159 @@
// Material Design Picker // Material Design Picker
// -------------------------------------------------- // --------------------------------------------------
$picker-md-height: 260px !default;
$picker-md-border-color: $list-md-border-color !default;
$picker-md-background-color: $list-md-background-color !default;
$picker-md-toolbar-height: 44px !default;
$picker-md-toolbar-background-color: $picker-md-background-color !default;
$picker-md-button-height: $picker-md-toolbar-height !default;
$picker-md-button-text-color: $link-md-color !default;
$picker-md-button-background-color: transparent !default;
$picker-md-column-padding: 0 12px !default;
$picker-md-option-padding: 0 10px !default;
$picker-md-option-text-color: $list-md-text-color !default;
$picker-md-option-font-size: 18px !default;
$picker-md-option-height: 42px !default;
$picker-md-option-offset-y: (($picker-md-height - $picker-md-toolbar-height) / 2) - ($picker-md-option-height / 2) - 10 !default;
$picker-md-option-selected-font-size: 24px !default;
$picker-md-option-selected-color: $link-md-color !default;
$picker-highlight-opacity: .8 !default;
.picker-wrapper {
height: $picker-md-height;
border-top: 1px solid $picker-md-border-color;
background: $picker-md-background-color;
}
.picker-toolbar {
display: flex;
justify-content: flex-end;
height: $picker-md-toolbar-height;
background: $picker-md-toolbar-background-color;
}
.hairlines .picker-wrapper,
.hairlines .picker-toolbar {
border-width: $hairlines-width;
}
.picker-button,
.picker-button.activated {
margin: 0;
height: $picker-md-button-height;
color: $picker-md-button-text-color;
background: $picker-md-button-background-color;
box-shadow: none;
}
.picker-columns {
height: $picker-md-height - $picker-md-toolbar-height;
perspective: 1800px;
}
.picker-col {
padding: $picker-md-column-padding;
transform-style: preserve-3d;
}
.picker-prefix,
.picker-suffix,
.picker-opts {
top: $picker-md-option-offset-y;
font-size: $picker-md-option-font-size;
line-height: $picker-md-option-height;
color: $picker-md-option-text-color;
transform-style: preserve-3d;
pointer-events: none;
}
.picker-opts ion-button-effect {
display: none;
}
.picker-opt {
margin: 0;
padding: $picker-md-option-padding;
width: calc(100% - 24px);
font-size: $picker-md-option-font-size;
line-height: $picker-md-option-height;
background: transparent;
transition-timing-function: ease-out;
backface-visibility: hidden;
pointer-events: auto;
}
.picker-opt .button-inner {
transition: 200ms;
}
.picker-prefix,
.picker-suffix,
.picker-opt-selected {
font-size: $picker-md-option-selected-font-size;
color: $picker-md-option-selected-color;
}
.picker-above-highlight {
position: absolute;
top: 0;
left: 0;
z-index: 10;
width: 100%;
height: $picker-md-option-offset-y + 4px;
border-bottom: 1px solid $picker-md-border-color;
background: linear-gradient(to bottom,
rgba($picker-md-background-color, 1) 20%,
rgba($picker-md-background-color, .7) 100%);
transform: translate3d(0, 0, 90px);
}
.picker-below-highlight {
position: absolute;
top: $picker-md-option-offset-y + $picker-md-option-height - 4;
left: 0;
z-index: 11;
width: 100%;
height: $picker-md-option-offset-y + $picker-md-option-height;
border-top: 1px solid $picker-md-border-color;
background: linear-gradient(to top,
rgba($picker-md-background-color, 1) 30%,
rgba($picker-md-background-color, .7) 100%);
transform: translate3d(0, 0, 90px);
}

View File

@ -29,9 +29,11 @@ ion-picker-cmp {
left: 0; left: 0;
z-index: $z-index-overlay-wrapper; z-index: $z-index-overlay-wrapper;
display: flex; display: flex;
flex-direction: column;
overflow: hidden; overflow: hidden;
flex-direction: column;
margin: auto; margin: auto;
width: $picker-width; width: $picker-width;
@ -41,32 +43,79 @@ ion-picker-cmp {
} }
.picker-columns { .picker-columns {
position: relative;
display: flex; display: flex;
flex: 1;
overflow: hidden; overflow: hidden;
justify-content: center;
} }
.picker-offset { .picker-col {
position: relative;
display: flex; display: flex;
max-height: 100%;
} }
.picker-column { .picker-opts {
flex: 1; position: relative;
width: 100%;
min-width: 50px;
max-width: 100%;
} }
.picker-prefix { .picker-prefix {
position: relative;
flex: 1; flex: 1;
min-width: 50px;
min-width: 45%;
text-align: right; text-align: right;
white-space: nowrap;
} }
.picker-suffix { .picker-suffix {
position: relative;
flex: 1; flex: 1;
min-width: 50px;
min-width: 45%;
text-align: left; text-align: left;
white-space: nowrap;
} }
.picker-option { .picker-opt {
flex: 1; position: absolute;
top: 0;
left: 0;
overflow: hidden; overflow: hidden;
white-space: nowrap;
flex: 1;
width: 100%;
text-align: center;
text-overflow: ellipsis; text-overflow: ellipsis;
white-space: nowrap;
}
.picker-opts-left .button-inner {
justify-content: flex-start;
}
.picker-opts-right .button-inner {
justify-content: flex-end;
}
.picker-above-highlight,
.picker-below-highlight {
display: none;
pointer-events: none;
} }

View File

@ -1,38 +1,24 @@
import {Component, ElementRef, Input, ViewChild, Renderer, HostListener, ChangeDetectionStrategy, ViewEncapsulation} from 'angular2/core'; import {Component, ElementRef, Input, ViewChild, Renderer, HostListener, ViewEncapsulation} from 'angular2/core';
import {NgClass, NgIf, NgFor} from 'angular2/common';
import {Animation} from '../../animations/animation'; import {Animation} from '../../animations/animation';
import {Transition, TransitionOptions} from '../../transitions/transition'; import {Transition, TransitionOptions} from '../../transitions/transition';
import {Config} from '../../config/config'; import {Config} from '../../config/config';
import {isPresent} from '../../util/util'; import {isPresent, isString, isNumber} from '../../util/util';
import {NavParams} from '../nav/nav-params'; import {NavParams} from '../nav/nav-params';
import {ViewController} from '../nav/view-controller'; import {ViewController} from '../nav/view-controller';
import {raf, CSS, pointerCoord} from '../../util/dom'; import {nativeRaf, cancelRaf, CSS, pointerCoord} from '../../util/dom';
/** /**
* @name Picker * @name Picker
* @description * @description
* *
* @usage
* ```ts
* constructor(private nav: NavController) {}
*
* presentSelector() {
* let picker = Picker.create({
*
* });
* this.nav.present(picker);
* }
*
* ```
*
*/ */
export class Picker extends ViewController { export class Picker extends ViewController {
constructor(opts: PickerOptions = {}) { constructor(opts: PickerOptions = {}) {
opts.columns = opts.columns || []; opts.columns = opts.columns || [];
opts.buttons = opts.buttons || [];
opts.enableBackdropDismiss = isPresent(opts.enableBackdropDismiss) ? !!opts.enableBackdropDismiss : true; opts.enableBackdropDismiss = isPresent(opts.enableBackdropDismiss) ? !!opts.enableBackdropDismiss : true;
super(PickerDisplayCmp, opts); super(PickerDisplayCmp, opts);
@ -54,6 +40,20 @@ export class Picker extends ViewController {
return this._nav && this._nav.config.get(key); return this._nav && this._nav.config.get(key);
} }
/**
* @param {any} button Picker toolbar button
*/
addButton(button: any) {
this.data.buttons.push(button);
}
/**
* @param {any} button Picker toolbar button
*/
addColumn(column: PickerColumn) {
this.data.columns.push(column);
}
/** /**
* @param {string} cssClass CSS class name to add to the picker's outer wrapper. * @param {string} cssClass CSS class name to add to the picker's outer wrapper.
*/ */
@ -68,95 +68,140 @@ export class Picker extends ViewController {
} }
/** /**
* @private * @private
*/ */
@Component({ @Component({
selector: '.picker-column', selector: '.picker-col',
template: template:
'<div class="picker-offset">' + '<div *ngIf="col.prefix" class="picker-prefix" [style.width]="col.prefixWidth">{{col.prefix}}</div>' +
'<div *ngIf="col.prefix" class="picker-prefix">{{col.prefix}}</div>' + '<div class="picker-opts" #colEle [style.width]="col.optionsWidth">' +
'<div class="picker-options" #colEle>' + '<button *ngFor="#o of col.options; #i=index" (click)="optClick($event, i)" type="button" category="picker-opt">' +
'<div *ngFor="#o of col.options" class="picker-option">' +
'{{o.text}}' + '{{o.text}}' +
'</button>' +
'</div>' + '</div>' +
'</div>' + '<div *ngIf="col.suffix" class="picker-suffix" [style.width]="col.suffixWidth">{{col.suffix}}</div>',
'<div *ngIf="col.suffix" class="picker-suffix">{{col.suffix}}</div>' +
'</div>',
host: { host: {
'[style.flex]': 'col.flex', '[style.min-width]': 'col.columnWidth',
'[class.picker-opts-left]': 'col.align=="left"',
'[class.picker-opts-right]': 'col.align=="right"',
'(touchstart)': 'pointerStart($event)', '(touchstart)': 'pointerStart($event)',
'(touchmove)': 'pointerMove($event)', '(touchmove)': 'pointerMove($event)',
'(touchend)': 'pointerEnd($event)', '(touchend)': 'pointerEnd($event)',
'(mousedown)': 'pointerStart($event)', '(mousedown)': 'pointerStart($event)',
'(mousemove)': 'pointerMove($event)', '(mousemove)': 'pointerMove($event)',
'(mouseup)': 'pointerEnd($event)', '(body:mouseup)': 'pointerEnd($event)',
'(body:mouseout)': 'mouseOut($event)',
} }
}) })
class PickerColumnCmp { class PickerColumnCmp {
@ViewChild('colEle') colEle: ElementRef; @ViewChild('colEle') colEle: ElementRef;
@Input() col: PickerColumn; @Input() col: PickerColumn;
y: number; y: number = 0;
colHeight: number; colHeight: number;
optHeight: number; optHeight: number;
velocity: number; velocity: number;
pos: number[] = []; pos: number[] = [];
scrollingDown: boolean;
msPrv: number = 0; msPrv: number = 0;
startY: number = null; startY: number = null;
rafId: number;
bounceFrom: number;
maxY: number;
rotateFactor: number;
constructor(config: Config) {
this.rotateFactor = config.getNumber('pickerRotateFactor', 0);
}
ngAfterViewInit() { ngAfterViewInit() {
// get the scrollable element within the column
let colEle: HTMLElement = this.colEle.nativeElement; let colEle: HTMLElement = this.colEle.nativeElement;
this.colHeight = colEle.clientHeight; this.colHeight = colEle.clientHeight;
// get the height of one option
this.optHeight = (colEle.firstElementChild ? colEle.firstElementChild.clientHeight : 0); this.optHeight = (colEle.firstElementChild ? colEle.firstElementChild.clientHeight : 0);
this.setY(0, true); // set the scroll position for the selected option
let selectedIndex = this.col.options.indexOf(this.col.selected);
this.setSelected(selectedIndex, 0);
} }
pointerStart(ev) { pointerStart(ev) {
console.debug('picker, pointerStart', ev.type, this.startY);
if (this.isPrevented(ev)) { if (this.isPrevented(ev)) {
// do not both with mouse events if a touch event already fired
return; return;
} }
// cancel any previous raf's that haven't fired yet
cancelRaf(this.rafId);
// remember where the pointer started from`
this.startY = pointerCoord(ev).y; this.startY = pointerCoord(ev).y;
// reset everything
this.velocity = 0; this.velocity = 0;
this.pos.length = 0; this.pos.length = 0;
this.pos.push(this.startY, Date.now()); this.pos.push(this.startY, Date.now());
this.maxY = (this.optHeight * (this.col.options.length - 1)) * -1;
console.debug('picker, pointerStart', ev.type, this.startY);
} }
pointerMove(ev) { pointerMove(ev) {
ev.preventDefault();
ev.stopPropagation();
if (this.startY !== null) { if (this.startY !== null) {
if (this.isPrevented(ev)) { if (this.isPrevented(ev)) {
return; return;
} }
let currentY = pointerCoord(ev).y; var currentY = pointerCoord(ev).y;
console.debug('picker, pointerMove', ev.type, currentY);
this.pos.push(currentY, Date.now()); this.pos.push(currentY, Date.now());
this.setY(this.startY + currentY, false);
// update the scroll position relative to pointer start position
var y = this.y + (currentY - this.startY);
if (y > 0) {
// scrolling up higher than scroll area
y = Math.pow(y, 0.8);
this.bounceFrom = y;
} else if (y < this.maxY) {
// scrolling down below scroll area
y = y + Math.pow(this.maxY - y, 0.9);
this.bounceFrom = y;
} else {
this.bounceFrom = 0;
}
this.update(y, 0, false);
} }
} }
pointerEnd(ev) { pointerEnd(ev) {
if (this.startY !== null) {
if (this.isPrevented(ev)) { if (this.isPrevented(ev)) {
return; return;
} }
this.velocity = 0;
if (this.bounceFrom > 0) {
// bounce back up
this.update(0, 100, true);
} else if (this.bounceFrom < 0) {
// bounce back down
this.update(this.maxY, 100, true);
} else if (this.startY !== null) {
var endY = pointerCoord(ev).y; var endY = pointerCoord(ev).y;
console.debug('picker, pointerEnd', ev.type, endY); console.debug('picker, pointerEnd', ev.type, endY);
this.pos.push(endY, Date.now()); this.pos.push(endY, Date.now());
this.velocity = 0;
this.scrollingDown = (endY < this.startY);
var endPos = (this.pos.length - 1); var endPos = (this.pos.length - 1);
var startPos = endPos; var startPos = endPos;
@ -176,45 +221,135 @@ class PickerColumnCmp {
this.velocity = ((movedTop / timeOffset) * FRAME_MS); this.velocity = ((movedTop / timeOffset) * FRAME_MS);
} }
this.setY(this.startY + endY, true); if (Math.abs(endY - this.startY) > 3) {
ev.preventDefault();
ev.stopPropagation();
this.decelerate(); var y = this.y + (endY - this.startY);
this.update(y, 0, true);
}
}
this.startY = null; this.startY = null;
this.decelerate();
}
mouseOut(ev) {
if (ev.target.classList.contains('picker-col')) {
this.pointerEnd(ev);
} }
} }
decelerate() { decelerate() {
var self = this; var y = 0;
cancelRaf(this.rafId);
if (self.velocity) { if (isNaN(this.y) || !this.optHeight) {
self.velocity *= DECELERATION_FRICTION; // fallback in case numbers get outta wack
console.log(`decelerate velocity ${self.velocity}`); this.update(y, 0, true);
var y = self.y + self.velocity; } else if (Math.abs(this.velocity) > 0) {
self.setY(y, true); // still decelerating
this.velocity *= DECELERATION_FRICTION;
raf(self.decelerate.bind(self)); // do not let it go slower than a velocity of 1
this.velocity = (this.velocity > 0 ? Math.max(this.velocity, 1) : Math.min(this.velocity, -1));
} else if (self.y % this.optHeight !== 0) { y = Math.round(this.y - this.velocity);
self.y = self.y + (this.scrollingDown ? -1 : 1); if (y > 0) {
// whoops, it's trying to scroll up farther than the options we have!
y = 0;
this.velocity = 0;
console.log(`lock in ${self.y}`); } else if (y < this.maxY) {
// gahh, it's trying to scroll down farther than we can!
y = this.maxY;
this.velocity = 0;
}
self.setY(self.y, true); console.log(`decelerate y: ${y}, velocity: ${this.velocity}, optHeight: ${this.optHeight}`);
raf(self.decelerate.bind(self)); this.update(y, 0, true);
if (y % this.optHeight !== 0 || Math.abs(this.velocity) > 1) {
// isn't locked in yet, keep decelerating until it is
this.rafId = nativeRaf(this.decelerate.bind(this));
}
} else if (this.y % this.optHeight !== 0) {
// needs to still get locked into a position so options line up
var currentPos = Math.abs(this.y % this.optHeight);
// create a velocity in the direction it needs to scroll
this.velocity = (currentPos > (this.optHeight / 2) ? 1 : -1);
this.decelerate();
} }
} }
setY(yOffset: number, saveY: boolean) { optClick(ev, index: number) {
let y = yOffset + this.y; if (!this.velocity) {
ev.preventDefault();
ev.stopPropagation();
console.log(`y: ${y}, yOffset: ${yOffset}, colHeight: ${this.colHeight}, optHeight: ${this.optHeight}`); this.setSelected(index, 150);
}
}
let colEleStyle = this.colEle.nativeElement.style; setSelected(selectedIndex: number, duration: number) {
colEleStyle[CSS.transform] = `translate3d(0px,${y}px,0px)`; // if there is a selected index, then figure out it's y position
// if there isn't a selected index, then just use the top y position
let y = (selectedIndex > -1) ? ((selectedIndex * this.optHeight) * -1) : 0;
cancelRaf(this.rafId);
this.velocity = 0;
// so what y position we're at
this.update(y, duration, true);
}
update(y: number, duration: number, saveY: boolean) {
// ensure we've got a good round number :)
y = Math.round(y);
let selectedIndex = Math.abs(Math.round(y / this.optHeight));
this.col.selected = this.col.options[selectedIndex];
let colEle: HTMLElement = this.colEle.nativeElement;
let optElements: any = colEle.querySelectorAll('.picker-opt');
for (var i = 0; i < optElements.length; i++) {
var optEle: HTMLElement = optElements[i];
var optTop = (i * this.optHeight);
var optOffset = (optTop + y);
var rotateX = (optOffset * this.rotateFactor);
var translateX = 0;
var translateY = 0;
var translateZ = 0;
if (this.rotateFactor !== 0) {
translateX = 10;
translateZ = 90;
if (rotateX > 90 || rotateX < -90) {
translateX = -9999;
rotateX = 0;
}
} else {
translateY = optOffset;
}
optEle.style[CSS.transform] = `rotateX(${rotateX}deg) translate3d(${translateX}px,${translateY}px,${translateZ}px)`;
optEle.style[CSS.transitionDuration] = (duration > 0 ? duration + 'ms' : '');
optEle.classList[i === selectedIndex ? 'add' : 'remove']('picker-opt-selected');
}
if (saveY) { if (saveY) {
this.y = y; this.y = y;
@ -223,9 +358,12 @@ class PickerColumnCmp {
isPrevented(ev) { isPrevented(ev) {
if (ev.type.indexOf('touch') > -1) { if (ev.type.indexOf('touch') > -1) {
// this is a touch event, so prevent mouse events for a while
this.msPrv = Date.now() + 2000; this.msPrv = Date.now() + 2000;
} else if (this.msPrv > Date.now() && ev.type.indexOf('mouse') > -1) { } else if (this.msPrv > Date.now() && ev.type.indexOf('mouse') > -1) {
// this is a mouse event, and a touch event already happend recently
// prevent the calling method from continuing
ev.preventDefault(); ev.preventDefault();
ev.stopPropagation(); ev.stopPropagation();
return true; return true;
@ -245,21 +383,21 @@ class PickerColumnCmp {
'<div class="picker-wrapper">' + '<div class="picker-wrapper">' +
'<div class="picker-toolbar">' + '<div class="picker-toolbar">' +
'<div *ngFor="#b of d.buttons" class="picker-toolbar-button" [ngClass]="b.cssRole">' + '<div *ngFor="#b of d.buttons" class="picker-toolbar-button" [ngClass]="b.cssRole">' +
'<button (click)="btnClick(b)" [ngClass]="b.cssClass" class="picker-button">' + '<button (click)="btnClick(b)" [ngClass]="b.cssClass" class="picker-button" clear>' +
'{{b.text}}' + '{{b.text}}' +
'<ion-button-effect></ion-button-effect>' +
'</button>' + '</button>' +
'</div>' + '</div>' +
'</div>' + '</div>' +
'<div class="picker-columns">' + '<div class="picker-columns">' +
'<div *ngFor="#c of d.columns" [col]="c" class="picker-column"></div>' + '<div class="picker-above-highlight"></div>' +
'<div *ngFor="#c of d.columns" [col]="c" class="picker-col"></div>' +
'<div class="picker-below-highlight"></div>' +
'</div>' + '</div>' +
'</div>', '</div>',
host: { host: {
'role': 'dialog' 'role': 'dialog'
}, },
directives: [NgClass, NgIf, NgFor, PickerColumnCmp], directives: [PickerColumnCmp],
changeDetection: ChangeDetectionStrategy.OnPush,
encapsulation: ViewEncapsulation.None, encapsulation: ViewEncapsulation.None,
}) })
class PickerDisplayCmp { class PickerDisplayCmp {
@ -293,7 +431,7 @@ class PickerDisplayCmp {
let data = this.d; let data = this.d;
data.buttons = data.buttons.map(button => { data.buttons = data.buttons.map(button => {
if (typeof button === 'string') { if (isString(button)) {
return { text: button }; return { text: button };
} }
if (button.role) { if (button.role) {
@ -302,10 +440,34 @@ class PickerDisplayCmp {
return button; return button;
}); });
// clean up dat data
data.columns = data.columns.map(column => { data.columns = data.columns.map(column => {
if (!column.flex) { if (!isPresent(column.columnWidth)) {
column.flex = 1; column.columnWidth = (100 / data.columns.length) + '%';
} }
if (!isPresent(column.options)) {
column.options = [];
}
column.options = column.options.map(inputOpt => {
let opt: PickerColumnOption = {
text: '',
value: ''
};
if (isPresent(inputOpt)) {
if (isString(inputOpt) || isNumber(inputOpt)) {
opt.text = inputOpt;
opt.value = inputOpt;
} else {
opt.text = isPresent(inputOpt.text) ? inputOpt.text : inputOpt.value;
opt.value = isPresent(inputOpt.value) ? inputOpt.value : inputOpt.text;
}
}
return opt;
});
return column; return column;
}); });
} }
@ -384,7 +546,7 @@ class PickerDisplayCmp {
// return an object of all the values with the input name as the key // return an object of all the values with the input name as the key
let values = {}; let values = {};
this.d.columns.forEach(col => { this.d.columns.forEach(col => {
values[col.name] = col.value; values[col.name] = col.selected ? col.selected.value : null;
}); });
return values; return values;
} }
@ -404,19 +566,20 @@ export interface PickerOptions {
export interface PickerColumn { export interface PickerColumn {
name?: string; name?: string;
value?: string; selected?: PickerColumnOption;
prefix?: string; prefix?: string;
suffix?: string; suffix?: string;
options: PickerColumnOption[]; options: PickerColumnOption[];
flex?: number;
cssClass?: string; cssClass?: string;
columnWidth?: string;
prefixWidth?: string;
suffixWidth?: string;
optionsWidth?: string;
} }
export interface PickerColumnOption { export interface PickerColumnOption {
value?: string; value?: any;
text?: string; text?: any;
checked?: boolean;
id?: string;
} }
@ -458,7 +621,5 @@ Transition.register('picker-slide-out', PickerSlideOut);
let pickerIds = -1; let pickerIds = -1;
const MIN_VELOCITY_START_DECELERATION = 4;
const MIN_VELOCITY_CONTINUE_DECELERATION = 0.12;
const DECELERATION_FRICTION = 0.97; const DECELERATION_FRICTION = 0.97;
const FRAME_MS = (1000 / 60); const FRAME_MS = (1000 / 60);

View File

@ -3,3 +3,171 @@
// Windows Picker // Windows Picker
// -------------------------------------------------- // --------------------------------------------------
$picker-wp-height: 260px !default;
$picker-wp-border-color: $list-wp-border-color !default;
$picker-wp-background-color: $list-wp-background-color !default;
$picker-wp-toolbar-height: 44px !default;
$picker-wp-toolbar-background-color: $picker-wp-background-color !default;
$picker-wp-button-height: $picker-wp-toolbar-height !default;
$picker-wp-button-text-color: $link-wp-color !default;
$picker-wp-button-background-color: transparent !default;
$picker-wp-column-padding: 0 12px !default;
$picker-wp-option-padding: 0 10px !default;
$picker-wp-option-text-color: $list-wp-text-color !default;
$picker-wp-option-font-size: 18px !default;
$picker-wp-option-height: 42px !default;
$picker-wp-option-offset-y: (($picker-wp-height - $picker-wp-toolbar-height) / 2) - ($picker-wp-option-height / 2) - 10 !default;
$picker-wp-option-selected-font-size: 24px !default;
$picker-wp-option-selected-color: $link-wp-color !default;
$picker-highlight-opacity: .8 !default;
.picker-wrapper {
height: $picker-wp-height;
border-top: 1px solid $picker-wp-border-color;
background: $picker-wp-background-color;
}
.picker-toolbar {
display: flex;
justify-content: flex-end;
height: $picker-wp-toolbar-height;
background: $picker-wp-toolbar-background-color;
}
.hairlines .picker-wrapper,
.hairlines .picker-toolbar {
border-width: $hairlines-width;
}
.picker-toolbar-button {
flex: 1;
text-align: right;
}
.picker-toolbar-cancel {
font-weight: normal;
text-align: left;
}
.picker-button,
.picker-button.activated {
margin: 0;
height: $picker-wp-button-height;
color: $picker-wp-button-text-color;
background: $picker-wp-button-background-color;
box-shadow: none;
}
.picker-columns {
height: $picker-wp-height - $picker-wp-toolbar-height;
perspective: 1800px;
}
.picker-col {
padding: $picker-wp-column-padding;
transform-style: preserve-3d;
}
.picker-prefix,
.picker-suffix,
.picker-opts {
top: $picker-wp-option-offset-y;
font-size: $picker-wp-option-font-size;
line-height: $picker-wp-option-height;
color: $picker-wp-option-text-color;
transform-style: preserve-3d;
pointer-events: none;
}
.picker-opts ion-button-effect {
display: none;
}
.picker-opt {
margin: 0;
padding: $picker-wp-option-padding;
width: calc(100% - 24px);
font-size: $picker-wp-option-font-size;
line-height: $picker-wp-option-height;
background: transparent;
transition-timing-function: ease-out;
backface-visibility: hidden;
pointer-events: auto;
}
.picker-opt .button-inner {
transition: 200ms;
}
.picker-prefix,
.picker-suffix,
.picker-opt-selected {
font-size: $picker-wp-option-selected-font-size;
color: $picker-wp-option-selected-color;
}
.picker-above-highlight {
position: absolute;
top: 0;
left: 0;
z-index: 10;
width: 100%;
height: $picker-wp-option-offset-y + 4px;
border-bottom: 1px solid $picker-wp-border-color;
background: linear-gradient(to bottom,
rgba($picker-wp-background-color, 1) 20%,
rgba($picker-wp-background-color, .7) 100%);
transform: translate3d(0, 0, 90px);
}
.picker-below-highlight {
position: absolute;
top: $picker-wp-option-offset-y + $picker-wp-option-height - 4;
left: 0;
z-index: 11;
width: 100%;
height: $picker-wp-option-offset-y + $picker-wp-option-height;
border-top: 1px solid $picker-wp-border-color;
background: linear-gradient(to top,
rgba($picker-wp-background-color, 1) 30%,
rgba($picker-wp-background-color, .7) 100%);
transform: translate3d(0, 0, 90px);
}

View File

@ -1,64 +1,208 @@
import {ViewEncapsulation} from 'angular2/core';
import {App, Page, Picker, NavController} from 'ionic-angular'; import {App, Page, Picker, NavController} from 'ionic-angular';
@Page({ @Page({
templateUrl: 'main.html' templateUrl: 'main.html',
encapsulation: ViewEncapsulation.None,
}) })
class E2EPage { class E2EPage {
smoothie: string;
timer: string;
constructor(private nav: NavController) { constructor(private nav: NavController) {}
setTimeout(() => {
this.presentPicker()
}, 250);
}
presentPicker() { twoColumns() {
let picker = Picker.create({ let picker = Picker.create({
buttons: [ buttons: [
{ {
text: 'Cancel', text: 'Cancel',
role: 'cancel' role: 'cancel'
}, },
'Done' {
text: 'Done',
handler: (data) => {
this.smoothie = `${data.flavor1} ${data.flavor2}`;
}
}
], ],
columns: [ columns: [
{ {
prefix: 'prefix', name: 'flavor1',
suffix: 'suffix', align: 'right',
options: [ options: [
{ text: 'Jan' }, { text: 'Mango' },
{ text: 'Feb' }, { text: 'Banana' },
{ text: 'Mar' }, { text: 'Cherry' },
{ text: 'Apr' }, { text: 'Strawberry' },
{ text: 'May' }, { text: 'Raspberry' },
{ text: 'Jun' }, { text: 'Blueberry' },
{ text: 'Jul' }, { text: 'Peach' },
{ text: 'Aug' }, { text: 'Coconut' },
{ text: 'Sep' }, { text: 'Pineapple' },
{ text: 'Oct' }, { text: 'Honeydew' },
{ text: 'Nov' }, { text: 'Watermelon' },
{ text: 'Dec' }, { text: 'Grape' },
{ text: 'Avocado' },
{ text: 'Kiwi' },
{ text: 'Orange' },
{ text: 'Papaya' },
] ]
}, },
// { {
// prefix: 'prefix', name: 'flavor2',
// suffix: 'suffix', align: 'left',
// options: [ options: [
// { text: 'Jan' }, { text: 'Banana' },
// { text: 'Feb' }, { text: 'Orange' },
// { text: 'Mar' }, { text: 'Grape' },
// { text: 'Apr' }, { text: 'Watermelon' },
// { text: 'May' }, { text: 'Strawberry' },
// { text: 'Jun' }, { text: 'Papaya' },
// { text: 'Jul' }, { text: 'Kiwi' },
// { text: 'Aug' }, { text: 'Cherry' },
// { text: 'Sep' }, { text: 'Raspberry' },
// { text: 'Oct' }, { text: 'Mango' },
// { text: 'Nov' }, { text: 'Pineapple' },
// { text: 'Dec' }, { text: 'Peach' },
// ] { text: 'Avocado' },
// }, { text: 'Honeydew' },
{ text: 'Blueberry' },
{ text: 'Coconut' },
] ]
},
]
});
this.nav.present(picker);
}
prefixLabel() {
let picker = Picker.create({
buttons: [
{
text: 'Nerp',
role: 'cancel'
},
{
text: 'Woot!',
handler: (data) => {
this.smoothie = `${data.flavor1}`;
}
}
],
columns: [
{
name: 'flavor1',
align: 'left',
prefix: 'Flavor',
options: [
{ text: 'Mango' },
{ text: 'Banana' },
{ text: 'Cherry' },
{ text: 'Strawberry' },
{ text: 'Raspberry' },
{ text: 'Blueberry' },
{ text: 'Peach' },
{ text: 'Coconut' },
{ text: 'Pineapple' },
{ text: 'Honeydew' },
{ text: 'Watermelon' },
{ text: 'Grape' },
{ text: 'Avocado' },
{ text: 'Kiwi' },
{ text: 'Orange' },
{ text: 'Papaya' },
]
}
]
});
this.nav.present(picker);
}
suffixLabel() {
let picker = Picker.create({
buttons: [
{
text: 'No',
role: 'cancel'
},
{
text: 'Si',
handler: (data) => {
this.smoothie = `${data.flavor1}`;
}
}
],
columns: [
{
name: 'flavor1',
align: 'right',
suffix: 'flavor',
options: [
{ text: 'Mango' },
{ text: 'Banana' },
{ text: 'Cherry' },
{ text: 'Strawberry' },
{ text: 'Raspberry' },
{ text: 'Blueberry' },
{ text: 'Peach' },
{ text: 'Coconut' },
{ text: 'Pineapple' },
{ text: 'Honeydew' },
{ text: 'Watermelon' },
{ text: 'Grape' },
{ text: 'Avocado' },
{ text: 'Kiwi' },
{ text: 'Orange' },
{ text: 'Papaya' },
]
}
]
});
this.nav.present(picker);
}
columnSizes() {
let picker = Picker.create();
picker.addButton({
text: 'Cancel',
role: 'cancel'
});
picker.addButton({
text: 'Set Timer',
handler: (data) => {
this.timer = `${data.hour}:${data.min}`;
}
});
picker.addColumn({
name: 'hour',
suffix: 'hour',
columnWidth: '30%',
optionsWidth: '50px',
options: Array.apply(null, {length: 23}).map(Number.call, Number)
});
var minuteOptions = [];
for (var i = 0; i < 60; i++) {
minuteOptions.push({
text: i,
value: ('0' + i).slice(-2)
});
}
picker.addColumn({
name: 'min',
suffix: 'min',
columnWidth: '40%',
optionsWidth: '80px',
options: minuteOptions
}); });
this.nav.present(picker); this.nav.present(picker);

View File

@ -4,6 +4,30 @@
<ion-content padding> <ion-content padding>
<button block class="e2ePresentPicker" (click)="presentPicker()">Picker</button> <button block class="e2eTwoColumns" (click)="twoColumns()">
2 Columns
</button>
<button block class="e2ePrefixLabel" (click)="prefixLabel()">
Prefix Label
</button>
<button block class="e2eSuffixLabel" (click)="suffixLabel()">
Suffix Label
</button>
<button block class="e2eColumnSizesLabel" (click)="columnSizes()">
Columns with sizes
</button>
<p aria-hidden="true" padding>
<code>Smoothie: {{ smoothie }}</code><br>
<code>Timer: {{ timer }}</code><br>
</p>
</ion-content> </ion-content>
<style>
</style>

View File

@ -33,6 +33,7 @@ Config.setModeConfig('ios', {
pickerEnter: 'picker-slide-in', pickerEnter: 'picker-slide-in',
pickerLeave: 'picker-slide-out', pickerLeave: 'picker-slide-out',
pickerRotateFactor: -0.46,
spinner: 'ios', spinner: 'ios',