Framework7+Vue.js Spotify播放器 - 实例详解(3)

参考:Framework7+Vue.js Spotify播放器 - 实例详解(1)
Framework7+Vue.js Spotify播放器 - 实例详解(2)
回顾一下:

  1. 用Template模板初始化
  2. 添加 Font Awesome Icon 图标库
  3. Framework7 View和Page概念
  4. 更新main视图
  5. 自定义样式文件CSS
  6. 初始化App应用,关联slider和显示的数值
  7. 调用Spotify API,处理返回数据
  8. 新的List页面(搜索结果)

9. media页面(歌曲详情)

当我们点击搜索结果里某一首歌曲,需要打开歌曲详情页面,并且能够播放

新建media.vue:

  • 添加组件props: ['mediaId']。动态路由,以歌曲Id为索引,会自动添加到props里,这样详情页面就知道需要展示哪首歌曲了
  • 组件mounted时,引入store.searchTracks
  • 数据访问:searchTracks[mediaId]
  • 本页面不需要工具栏,设置page.no-toolbar
  • <audio>是HTML5自带的音频播放元素,后面在Android设备上播放,需要替换成Media.plugin
# \spotifyApp\src\assets\vue\Media.vue
<template>
  <f7-page name="media" no-toolbar id="media">
    <f7-navbar>
      <f7-nav-left back-link="Back" sliding></f7-nav-left>
      <f7-nav-center sliding>Media Detail</f7-nav-center>
      <f7-nav-right>
        <f7-link icon="icon-bars" open-panel=""></f7-link>
      </f7-nav-right>
    </f7-navbar>
    <f7-block-title>Track Info</f7-block-title>
    <f7-list>
      <f7-list-item :media="'![]( + searchTracks[mediaId].album.images[1].url +)'" :title="searchTracks[mediaId].name">
      </f7-list-item>
      <f7-list-item>
        <f7-label>Type</f7-label>
        <f7-label>{{searchTracks[mediaId].type}}</f7-label>
      </f7-list-item>
      <f7-list-item>
        <f7-label>Artist</f7-label>
        <f7-label>{{searchTracks[mediaId].artists[0].name}}</f7-label>
      </f7-list-item>
    </f7-list>
    <f7-block-title>Album Info</f7-block-title>
    <f7-list form>
      <f7-list-item title="">
        <f7-label>Album</f7-label>
        <f7-label>{{searchTracks[mediaId].album.name}}</f7-label>
      </f7-list-item>
      <f7-list-item>
        <f7-label>Popularity Rank</f7-label>
        <f7-input type="range" min="0" max="100" step="1" v-model="sliderVal" disabled/>
        <f7-input type="text" disabled value="90" class="rangeVal" />
      </f7-list-item>
    </f7-list>
    <f7-block-title>Preview</f7-block-title>
    <f7-block inset>
      <f7-block inner>
        <div id="previewAudio" align="center">
          <audio controls preload :src="searchTracks[mediaId].preview_url"></audio>
        </div>
      </f7-block>
    </f7-block>
    <f7-list>
      <f7-list-item class="bottom">
        <a href="#" class="item-media link favorite"><i class="icon fa fa-star fa-2x"></i> </a>
        <div class="item-title">
          <a :href="searchTracks[mediaId].external_urls.spotify" class="link external">Open in Spotify</a>
        </div>
        <a href="#" class="share item-media link"><i class="icon fa fa-external-link fa-2x"></i></a>
      </f7-list-item>
    </f7-list>

  </f7-page>
</template>
<script>
import store from '../../store.js'
export default {
  name: 'list',
  props: ['mediaId'], //  <!-- $route.params are also passed as component props: -->
  data() {
    return {
      searchTracks: [],
    }
  },
  mounted() {
    this.searchTracks = store.searchTracks;
  }
}
</script>

Media.vue添加到路由:

# \spotifyApp\src\routes.js
export default [
  {
    path: '/list/',
    component: require('./assets/vue/List.vue')
  },
  {
    path: '/media/:mediaId/',
    component: require('./assets/vue/Media.vue')
  }
]

List.vue里,以路由名+Index访问就行,实际运行时就像/media/0/、/media/1/、/media/2/

# \spotifyApp\src\assets\vue\List.vue
<f7-list-item swipeout v-for="(item, index) in searchTracks" :key="item.id" 
:link="'/media/'+index+'/'" :media="'![]( + item.album.images[2].url +)'" :title="item.name" :subtitle="item.artists[0].name" :text="item.album.name"> 

为了页面美观,main.css里作了一些小调整:

# \spotifyApp\src\assets\css\main.css
.statusbar-overlay {
    background: #000000;
}
i.icon.icon-spotify {
    width: 29px;
    height: 29px;
    background-image: url("../images/spotify.png");
    margin-right: 3px;
}
/* Item View bottom action buttons */
li.bottom {
    position: fixed;
    bottom: 0px;
    width: 100%;
}
/* Add space before actual range val */
.rangeVal {
    padding-left: 5px;
}
.list-block {
    margin: 10px 0 !important;
}
/* Format content headers */
.content-block-title {
    margin: 18px 15px 10px;
}
/* Format the long track titles */
.list-block .item-title {
    white-space: inherit;
    position: relative;
    overflow: hidden;
    text-overflow: ellipsis;
    max-width: 100%;
}
/* Formats the Popularity Rank title to display slider better */
.list-block .item-title.label {
    width: 50%;
}

img.detail {
  max-width: 100%;
}

.list-block .item-media {
  flex-shrink: 1;
}

到此,你的页面应该像这样:

Paste_Image.png

10. 更多Mobile App元素:侧边栏Sidebar和元素左滑右滑Swipeout

手机应用,都会用到这种侧边栏Sidebar:

Paste_Image.png

很简单,在main.vue里添加side panel:

  • f7-panel:定义在f7-views之外,因为是全局共享的,子页面也需要打开Sidebar
  • f7-panel left reveal:定义在左侧出现,reveal是不覆盖原页面内容。如果需要覆盖效果,使用cover就行
  • 打开sidebar:F7已经默认写好了,使用<f7-link icon="icon-bars" open-panel="left"></f7-link>就行,简单吧?
  • 关闭sidebar:点击侧边栏外面,就自动关闭了。当然,你也可以添加按钮,手动触发关闭事件
  • @click,methods: {} 对于选项,添加点击事件处理就行。
  • TODO:点击About,打开“about.vue”页面,你来试一试吧!
# \spotifyApp\src\main.vue
<template>
  <!-- App -->
  <div id="app">
    <!-- Statusbar -->
    <f7-statusbar></f7-statusbar>
    <!-- Left Panel -->
    <f7-panel left reveal layout="dark">
      <f7-view id="left-panel-view" navbar-through :dynamic-navbar="true">
        <f7-navbar title="Left Panel"></f7-navbar>
        <f7-pages>
          <f7-page>
            <f7-list>
              <f7-list-item media="<i class='icon fa fa-cog'></i>" @click="onSetting" title="Settings">
              </f7-list-item>
              <f7-list-item media="<i class='icon fa fa-star'></i>" @click="onFavorite" title="Favorites">
              </f7-list-item>
              <f7-list-item media="<i class='icon fa fa-info-circle'></i>" @click="onAbout" title="About">
              </f7-list-item>
            </f7-list>
          </f7-page>
        </f7-pages>
      </f7-view>
    </f7-panel>
    <f7-views>
      <!-- Main view-->
      <f7-view main :dynamicNavbar="true">
        <!-- Top Navigation Bar-->
        <f7-navbar>
          <f7-nav-left>
            <f7-link icon="icon-bars" open-panel="left"></f7-link>
          </f7-nav-left>
          <f7-nav-center>Spotify Browser</f7-nav-center>
        </f7-navbar>
        <!-- Pages -->
        <f7-pages navbar-through toolbar-through>
  。。。

iPhone手机,大家有过体验,比如短信左滑,就会出现“删除”的快捷按键,很方便。前面我们已经实现了左滑、右滑效果,现在来添加具体的Swipeout功能:


Paste_Image.png

修改List.vue,监听@click事件

  • @click="addFavorite(index)":带入index参数,知道添加哪首歌到收藏夹
  • favoriteList:添加到store.js。这样,共享收藏夹数据给其它组件,比如sidebar
# \spotifyApp\src\store.js
const store = {
  searchTracks: [],
  favoriteList: [],
};
export default store;
  • preview(index):调用了Phonegap Media插件,需要手动安装:$ phonegap plugin add cordova-plugin-media --save。左滑后,点击Preview,会预览播放7秒钟
  • share(index):调用手机原生的分享功能,后续会实现

Swipeout在浏览器上体验跟手机上略有不同,建议通过Phoegap Developer手机App来测试。参考://www.greatytc.com/p/c24086ac57ba

# \spotifyApp\src\assets\vue\List.vue
...
    <f7-list class="searchbar-found" media-list id="mediaList">
      <f7-list-item swipeout v-for="(item, index) in searchTracks" :key="item.id" :link="'/media/'+index+'/'" :media="'![]( + item.album.images[2].url +)'" :title="item.name" :subtitle="'<span class=singerPrex>by </span>'+item.artists[0].name" :text="item.album.name">
        <f7-swipeout-actions right>
          <f7-swipeout-button class="bg-orange">
            <a class="bg-orange favorite" @click="addFavorite(index)"><i class="icon fa fa-star fa-2x"></i></a>
          </f7-swipeout-button>
          <f7-swipeout-button class="bg-blue">
            <a href="#" class="bg-blue share"  @click="share(index)"><i class="icon fa fa-share fa-2x"></i></a>
          </f7-swipeout-button>
        </f7-swipeout-actions>
        <f7-swipeout-actions left>
          <f7-swipeout-button class="bg-green">
            <a href="#" class="bg-green preview" @click="preview(index)"><i class="icon fa fa-play fa-2x"></i></a>
          </f7-swipeout-button>
        </f7-swipeout-actions>
      </f7-list-item>
    </f7-list>
...
<script>
import store from '../../store.js'
export default {
  name: 'list',
  data() {
    return {
      searchTracks: [],
      favoriteList: [],
    }
  },
  mounted() {
    window.Dom7(document).on('deviceready', () => {
      console.log("List Page is ready!");
      this.searchTracks = store.searchTracks;
      this.favoriteList = store.favoriteList;
    });
  },
  methods: {
    addFavorite(index) {
      store.favoriteList.push(this.searchTracks[index]);
      window.f7.alert(this.searchTracks[index].name + ' added to favorites!');
    },
    preview(index) {
      var item = this.searchTracks[index];
      window.f7.alert("Previewing " + item.name);
      var media = new Media(item.preview_url, function() {
        console.log("Media Success");
      }, function(error) {
        console.log("Media fail " + error)
      }, null);
      media.play();
      setTimeout(function() {
          media.stop()
        }, 7000) //preview for 7 seconds
    },
    share(index) {
        window.f7.alert(this.searchTracks[index].name + ' shared to...');
    }
  }
}
</script>

11. 让你的App更有原生体验

--> 请看详解(4)

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

推荐阅读更多精彩内容