前言:
整理一个年初做的一个APP项目,项目中的核心功能是与【一起练琴】这个 app 较为相似,其中最重点的是乐谱播放和根据乐谱练习评分的功能。今天我们主要介绍下我们选用的一个乐谱播放的方案。
重点调研
经过大量的调研在乐谱可视化方向已经发展多年,有不少成熟的开源解决方案,层层筛选后有三个库比较适合我们的需求(支持乐谱定制、音源定制、musicxml 格式支持):
alphaTab – music notation for everyone 这个是基于 typescript 的实现,主攻吉他谱,当然其他乐谱也支持。
特点:可以识别乐谱和定制音源播放,可以读取 musicxml格式
VexFlow - HTML5 Music Engraving 一个用于渲染音乐符号的JavaScript库,音源播放需要定制开发。
opensheetmusicdisplay 一个开源的乐谱播放器。
特点:简单,支持 musicxml 乐谱文件,支持音源,支持 svg、canvas 两种渲染模式
选型
基于以上三个开源库进行定制开发,发现会有不同的问题:
AlphaTap
缺点:musicxml支持不完善,样式定制较为困难,音源定制不灵活
优点:美观、屏幕自适应完善、定制开发容易
VexFlow
缺点:这个应该也不算是缺点,因为它是一个乐谱绘制库,所以对于musicxml、音源等支持,需要用第三方基于它实现的库,或者我们自己开发,但是当时的项目周期太紧张,最好还是放弃掉了。
优点:强大、与原生 app 结合好、有基于 flutter 的方案
OpenSheetMusicDisplay
缺点:简陋,需要优化的地方较多,音源需要定制
优点:这个是完整的乐谱播放器实现,基础的乐谱播放功能、音源加载、乐谱格式 musicxml 支持都比较完整。
最终我们选择提 OSMD 这个方案,下面讲一下我们是如何使用的
开发
为了兼容移动展示,和最终嵌入到原生 app 中,我们使用了几个库:
$ yarn add vue opensheetmusicdisplay vant vue-webview-js-bridge
vant ui # 适配移动端组件的 UI
vue-webview-js-bridge # 与原生 app 通信
项目结构如下
.
├── LICENSE
├── README.md
├── babel.config.js
├── package-lock.json
├── package.json
├── postcss.config.js
├── public
│ ├── Happy_Birthday.musicxml
│ ├── static
│ └── voice.svg
├── src
│ ├── App.vue
│ ├── assets
│ ├── components
│ ├── layout
│ ├── main.js
│ ├── plugins
│ ├── router
│ ├── scores.js
│ └── views
├── vue.config.js
├── yarn-error.log
└── yarn.lock
可以看出非常简单,就是一个基础的 vue 工程,主要是 view 中的 score.vue,它整合了 OSMD 播放器,和加载乐谱的方式。看下重点代码:
views/score.vue
...
components/Score.vue
以上两个文件的主要作用是,
- score component 用来引入和整合 osmd 模块,包含加载乐谱、加载音频、乐谱渲染
- score view 作为入口页面用来访问乐谱,桥接与原生 app 通信控制
最终效果:
做个样机展示下:
总结
本篇文章主要介绍了乐谱的可视化方案,并将方案应用在实际项目中,并做出示例。可以给有相关需求的朋友做为参考。
相关资源
https://github.com/baisheng/osmd-player 文章中的系统源码
alphaTab – music notation for everyone
VexFlow - HTML5 Music Engraving
opensheetmusicdisplay