[Ionic 2从入门到精通] 5.6 自定义的管道和所有相片的飞页

本节课中,我们将制作自定义管道用来更友好的展示照片的拍摄日期,我们也将完成应用的关键部分之一也就是slideshow。虽然他没有拍照那么复杂,但是他算是整个应用的灵魂。

创建一个自定义管道

我们在本书的基础部分已经涵盖了管道是什么,这里还是跟你简单温习一下:管道实际上允许我们在数据进行展示之前对他进行修改。
我们此处的目标是创建一个标签显示此照片上几天之前拍摄的,这样,“3 天前”,“10 天前”。目前而言,我们已经在照片的数据上存储了拍摄日期,不幸的是,他看起是酱紫的:

Sun Mar 06 2016 00:40:02 GMT+1000 (AEST)

也就是天底下最不友好的日期显示方式。所以,这里是@Pipe最好的使用情景:我们创建一个管道用来接收丑陋的日期格式,转换到“X 天前”这样的格式,然后返回之。
我们先看看如何创建这个管道,然后在慢慢讨论。
> 修改 src/pipes/days-agp.ts 为如下:

import { Injectable, Pipe } from '@angular/core';

@Pipe({
    name: 'daysAgo'
})
@Injectable()
export class DaysAgo {
    transform(value, args?) {
        let now = new Date();
        let oneDay = 24 * 60 * 60 * 1000;
        let diffDays = Math.round(Math.abs((value.getTime() - now.getTime())/(oneDay)));

        return diffDays;
    }
}

@Pipe装饰器里面我们提佛那个一个名字daysAgo,意思就是在模板中可以直接通过使用这个名字作为关键字来使用这个管道。使用管道的时候,永远都需要传入数据,然后数据都会被传入transform函数也就是上面的value。我们会把我们的照片对象的Date对象传入到这个管道,这就是我们这里要做的。同时注意可以通过管道传入额外的参数,也就是我们为什么用args?,因为这不是强制要求传入的参数。
首先我们来一点数学魔法得出照片上几天之前拍摄的,(Stack Overflow也有这个,我的解决方式可能比他丑点儿),然后返回他。返回的值就是将被渲染到模板上去的内容。
如果你还记得之前的课的话,我们已经在app.module.ts中设置好这个管道了。
> 修改home.html中的photo项为如下:

<ion-item>
<img [src]="photo.image" />
<ion-badge item-right light>{{photo.date | daysAgo}} days ago</ion-badge>
</ion-item>

注意了(译者敲了下黑板),我们加了这个句:

{{photo.date | daysAgo}}

这样写会把photo.date传入到daysAgo管道,然后将daysAgo返回的东西展示最这里。最终结果是一个带有类似“5 天前”的徽章。

为所有照片创建一个Slideshow

我们已经创建好了Slideshow的模板,所以现在我们需要制作一些逻辑来循环和展示所有照片,同时我们也要添加一个打开slideshow的途径(重启途径也要)。
我们还是先从playSlideshow函数开始,这样我们可以真正的打开页面。
> 修改 src/pages/home/home.ts的 playSlideshow 函数为如下:

playSlideshow(): void {
    if(this.photos.length > 1){
        let modal = this.modalCtrl.create(SlideshowPage, {photos:
        this.photos});
        modal.present();
    } else {
        let alert = this.simpleAlert.createAlert('Oops!', 'You need at least two photos before you can play a slideshow.');
        alert.present();
    }
}

跟创建警告提示框类似,我们给用户撞见了一个模态框页面。首先,我们创建了一个Modal,然后通过NavController呈现到用户面前。在这里,我们使用已经创建好的SlideshowPage来创建模态框,也将所有的照片数据传入进去。这样我们可以在Slideshow页面上获取到这些数据。
注意,我们只有在用户有超过1张照片的时候才会触发他(因为没有或者只有1张照片的slideshow就不是真的slideshow了,对吧?),否则会显示警告提示框。
现在,我们来定义Slideshow页的类定义。这个类挺小的,所以先全部贴出来,然后分段讲解。
> 修改 src/pages/slideshow/slideshow.ts 为如下:

import { NavParams, ViewController } from 'ionic-angular';
import { Component, ElementRef, ViewChild } from '@angular/core';

@Component({
    selector: 'page-slideshow',
    templateUrl: 'slideshow.html'
})
export class SlideshowPage {
    @ViewChild('imagePlayer') imagePlayer: ElementRef;
    imagePlayerInterval: any;
    photos: any;

    constructor(public navParams: NavParams, public viewCtrl: ViewController) {
        this.photos = this.navParams.get('photos');
    }

    ionViewDidEnter(){
        this.playPhotos();
    }

    closeModal(){
        this.viewCtrl.dismiss();
    }

    playPhotos(){
        let imagePlayer = this.imagePlayer.nativeElement;
        let i = 0;
        //Clear any interval already set
        clearInterval(this.imagePlayerInterval);
        //Restart
        this.imagePlayerInterval = setInterval(() => {
            if(i < this.photos.length){
                imagePlayer.src = this.photos[i].image;
                i++;
            }
            else {
                clearInterval(this.imagePlayerInterval);
            }
        }, 500);
    }
}

这里有些东西可能你不大熟悉。我们导入了NavParams服务可以用他拿到在创建的时候传过来的数据。你可以看到我们在构造器里面通过this.navParams.get('photos');获取数据。这里另一个奇怪的东西是ViewController。我们需要用到这个来隐藏模态框(我们这里定义了一个closeModal函数提供给模板里面的按钮使用)。
这里最重要的函数是playPhotos,这就是我们循环所有照片然后在页面上相应改变对于的图片元素的地方。首先,我们通过之前设置好的@ViewChild引用得到图片元素(之前模板里面通过 #本地变量 的方式定义的)的引用,我们清理了当前正在运行的所有定时器(防止用户在slideshow还在播放的时候突然重启slideshow)。然后我们开始循环this.photos里的照片。如果还剩下照片可以展示,图片的src属性将会使用下一张照片的数据更新,如果没有的话,就会清理计时器。上面代码中,定时器的间隔是500毫秒,也就是0.5秒,我们可以根据个人喜好通过调整这个数值来调整slideshow的快和慢。(甚至给用户提供选择来控制)
playPhotos函数上ionViewDidEnter函数自动触发的,ionViewDisEnter函数上在进入视图的时候自动执行的,这个函数在我们每次进入视图的时候都会执行。

总结

这是另一节比较小的课,但是这里讲的东西都蛮酷的。至此,应用的大部分主体功能都完成了,我们只要加上一些边边角角就可以了。当然,我们还是得梅花他,但是下节课我们要学习的是整合本地通知和社交分享功能。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,992评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,212评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,535评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,197评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,310评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,383评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,409评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,191评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,621评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,910评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,084评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,763评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,403评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,083评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,318评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,946评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,967评论 2 351

推荐阅读更多精彩内容