🎶 基于 React 实现的仿 iOS 客户端网易云音乐。
在线地址:戳我(PC 浏览器需切换到移动端模式)
移动端体验:
- React 16.3
- TypeScript
- Mobx + Redux
- react-redux
- react-router-v4
- Scss
目前只实现了上面四个页面,但是总体的结构已经形成了,其他页面的添加只是时间上的问题 (其实是懒),暂时没有实现,下面是目前已实现的功能的细节:
像首页的 banner 或者推荐歌单等,都是不会被共享的局部状态,使用 Mobx 来进行请求的发起和状态的管理。
播放器的状态是一个全局状态,包括当前的播放列表,切歌,播放 / 暂停等,所以很自然的使用 redux 来进行管理,可以清楚的掌握所有改变全局状态的行为。
尽管上手需要掌握一些语法,但是静态类型与自动提示都能提供很大的帮助,在这个并不大的项目中我也体验到了很大的帮助。但是要注意的是 TS 其实并不严格限制对象的类型,只要够懒,遍地 any,就会把 TS 写成 JS,所以为了充分发挥 TS 的威力,一定要有良好的 TS 代码风格。
为了模仿 iOS 端可以通过滑屏切换页面的功能,通过监听 touchStart
,touchMove
,touchEnd
来进行手势的判断并通过 transform
触发模拟滚动实现,在 touchMove
中检测监听滑动的方向及距离,在 touchEnd
中触发路由的切换及页面吸附到整屏的位置。
有这么一个操作需要注意:用户在某歌单往下滑了几下,然后点了某歌播放然后进入了播放器,会发生路由的改变,如果此时从播放器返回,会丢包包括滚动位置在内的歌单页的所有状态丢失(因为 re-mount 了)。
我造了一个轮子来解决这个问题:react-live-route,是对 react-router-v4 中 Route 组件的增强,简单的说就是将歌单页隐藏掉而不是 unmount 掉,具体的解决思路可以参考轮子里的文档。
在 iOS 版的网易云中,可以滑动来切换页面,同时会触发顶部 tab 下的滑块移动。在项目中,滑动页面与滑块分属于两个兄弟组件的子组件且嵌套层次较深,如果直接通过 prop 来传递略显丑陋,有如下解决方案:
-
通过 redux,但是 redux 最好只负责领域数据,这种 UI 的状态就不要往 store 中放了。
-
通过 event-emitter,其实和 redux 差不多,因为 redux 也是基于 event-emitter 实现的, 但是不经过 react-redux 虽然可以实现,但是破坏了 react 整个自顶向下界面更新的原则。
-
通过新的 context API 实现,如下图:
项目中用到的网易云音乐的 API 来自 NeteaseCloudMusicApi。
目前还有一些部分没完成,包括但不限于:
- code splitting
- 组件中有些功能还是有耦合,需要再抽象
- SSR
克隆代码到本地之后,需要在 4000 端口运行 NeteaseCloudMusicApi。