AI prompts
base on Windows端本地音乐播放器,使用Material You配色。Dart (Flutter) + Rust (lofty, windows-rs) + C (bass lib) 多语言项目。绝赞开发中。 # Coriander Player:一款使用 Material You 配色的本地音乐播放器。

## [更多软件截图在下面(点我滚动到下面)](#软件截图)
**该播放器发行版已经附带桌面歌词组件。项目仓库请见 [desktop_lyric](https://github.com/Ferry-200/desktop_lyric.git)**
## 安装
1. 下载 [Release](https://github.com/Ferry-200/coriander_player/releases/latest) 里文件安装
2. **(已过时,现在的体验版已经落后于正式版)** 你也可以到 [Action 构建版本(体验版)介绍](https://github.com/Ferry-200/coriander_player/issues/49) 下载体验版 :)
3. 通过 scoop 安装,使用此 [bucket](https://github.com/jinzhongjia/scoop-bucket)
```sh
# 添加 bucket
scoop bucket add jin https://github.com/jinzhongjia/scoop-bucket
# 安装
scoop install jin/coriander_player
```
## 支持
<div style="display:flex">
<image style="height: 300px" src="https://github.com/user-attachments/assets/1565d416-d453-4e7a-bdaa-29af2f802024"></image>
<image style="height: 300px" src="https://github.com/user-attachments/assets/cd3a624c-99f7-4f56-811c-36a15f50ba35"></image>
</div>
## 软件内快捷键
页面中有文本框且处于输入状态时会自动忽略快捷键操作。如果要使用快捷键,可以点击输入框以外的地方,然后再次使用。
- Esc:返回上一级
- 空格:暂停/播放
- Ctrl + 左方向键:上一曲
- Ctrl + 右方向键:下一曲
## 支持播放的音乐格式
- mp3, mp2, mp1
- ogg
- wav, wave
- aif, aiff, aifc
- asf, wma
- aac, adts
- m4a
- ac3
- amr, 3ga
- flac
- mpc
- mid
- wv, wvc
- opus
- dsf, dff
- ape
## 支持下列音乐格式的内嵌歌词
- aac
- aiff
- flac
- m4a
- mp3
- ogg
- opus
- wav(标签必须用 UTF-8 编码)
其他格式的只支持同目录的 lrc 文件或者是网络歌词
## 外挂 LRC 支持编码
- utf-8
- utf-16
## 选择默认歌词
默认情况下,软件会先读取本地歌词。如果没有,则匹配在线歌词。
你可以在正在播放界面的歌词切换按钮展开的菜单中进入选择默认歌词的页面。

在这个界面中,你可以在本地歌词(如果有)和几个匹配程度高的在线歌词中选择一个作为默认歌词。之后再播放这首音乐时,软件会加载你指定的歌词。
## 提供建议、提交 Bug 或者提 PR
我正处于学习和适应 Github 工作流的阶段,所以目前不设置太多的要求。你只需要注意以下几点:
1. 如果要提交 Bug,请创建一个新的 issue。尽可能说明复现步骤并提供截图。
2. 如果你提交 PR,由于我正在学习相关知识,可能会在处理 PR 时和你沟通如何操作分支之类的问题。
## 编译
1. 开发 flutter 需要的环境
2. 需要编译 Coriander Player(本仓库) 软件本体和 desktop_lyric。[desktop_lyric](https://github.com/Ferry-200/desktop_lyric.git) 也是 Flutter 应用,直接编译即可
3. 要把得到的 desktop_lyric 产物放在软件目录的 `desktop_lyric/` 目录下
4. 编译后要把 BASS 库的 64 位的 `bass.dll`, `bassape.dll`, `bassdsd.dll`, `bassflac.dll`, `bassmidi.dll`, `bassopus.dll`, `basswv.dll`, `basswasapi.dll` 放在软件目录的 `BASS` 文件夹下
## 歌词特性解释
1. lrc歌词的间奏识别
在一些lrc歌词中,会使用 **只有时间标签而内容为空** 的一行来表示上一行的结束。如:
```
[02:32.57]光は やさしく抱きしめた
[02:32.57]那天没能放声大哭的我
[02:39.94]
[02:55.18]照らされた世界 咲き誇る大切な人
[02:55.18]光芒普照整个世界 珍重之人绽放于心
```
如果这一行(第三行)的时间戳和下一行的时间戳之间大于5s,就把这两行之间的时间作为间奏时长
**所以,不是所有lrc歌词在间奏时都能显示间奏动画。**
2. 逐字歌词的间奏识别
逐字歌词都会给出每一行的开始时间和持续时间,所以识别间奏会简单得多。如
```
[5905,5466]<0,217,0>世<217,383,0>界<600,495,0>は<1095,272,0>と<1367,328,0>て<1695,343,0>も<2038,616,0>綺<2654,752,0>麗<3406,276,0>だ<3682,276,0>っ<3958,504,0>た<4462,1004,0>な
[23037,5254]<0,255,0>書<255,280,0>架<535,312,0>の<847,592,0>隙<1439,312,0>間<1751,223,0>に<1974,160,0>住<2134,144,0>ま<2278,352,0>う<2630,640,0>一<3270,640,0>輪<3910,190,0>の<4100,680,0>花<4780,474,0>は
```
第一行的开始时间是 5905ms,持续 5466ms;第二行则是 23037ms和 5254ms。可见 5905 + 5466 = 11371,与 23037相差超过 5000ms,所以这两行时间可以插入表示间奏的空白行
## 感谢
- [music_api](https://github.com/yhsj0919/music_api.git):实现歌曲的匹配和歌词的获取
- [Lofty](https://crates.io/crates/lofty):歌曲标签获取
- [BASS](https://www.un4seen.com/bass.html):播放乐曲
- [flutter_rust_bridge](https://pub.dev/packages/flutter_rust_bridge):实现许多 Windows 原生交互
- [Silicon7921](https://github.com/Silicon7921):绘制了新图标
## 软件截图






.png)
.png)
.png)
.png)

.png)
.png)
.png)
", Assign "at most 3 tags" to the expected json: {"id":"9788","tags":[]} "only from the tags list I provide: [{"id":77,"name":"3d"},{"id":89,"name":"agent"},{"id":17,"name":"ai"},{"id":54,"name":"algorithm"},{"id":24,"name":"api"},{"id":44,"name":"authentication"},{"id":3,"name":"aws"},{"id":27,"name":"backend"},{"id":60,"name":"benchmark"},{"id":72,"name":"best-practices"},{"id":39,"name":"bitcoin"},{"id":37,"name":"blockchain"},{"id":1,"name":"blog"},{"id":45,"name":"bundler"},{"id":58,"name":"cache"},{"id":21,"name":"chat"},{"id":49,"name":"cicd"},{"id":4,"name":"cli"},{"id":64,"name":"cloud-native"},{"id":48,"name":"cms"},{"id":61,"name":"compiler"},{"id":68,"name":"containerization"},{"id":92,"name":"crm"},{"id":34,"name":"data"},{"id":47,"name":"database"},{"id":8,"name":"declarative-gui "},{"id":9,"name":"deploy-tool"},{"id":53,"name":"desktop-app"},{"id":6,"name":"dev-exp-lib"},{"id":59,"name":"dev-tool"},{"id":13,"name":"ecommerce"},{"id":26,"name":"editor"},{"id":66,"name":"emulator"},{"id":62,"name":"filesystem"},{"id":80,"name":"finance"},{"id":15,"name":"firmware"},{"id":73,"name":"for-fun"},{"id":2,"name":"framework"},{"id":11,"name":"frontend"},{"id":22,"name":"game"},{"id":81,"name":"game-engine "},{"id":23,"name":"graphql"},{"id":84,"name":"gui"},{"id":91,"name":"http"},{"id":5,"name":"http-client"},{"id":51,"name":"iac"},{"id":30,"name":"ide"},{"id":78,"name":"iot"},{"id":40,"name":"json"},{"id":83,"name":"julian"},{"id":38,"name":"k8s"},{"id":31,"name":"language"},{"id":10,"name":"learning-resource"},{"id":33,"name":"lib"},{"id":41,"name":"linter"},{"id":28,"name":"lms"},{"id":16,"name":"logging"},{"id":76,"name":"low-code"},{"id":90,"name":"message-queue"},{"id":42,"name":"mobile-app"},{"id":18,"name":"monitoring"},{"id":36,"name":"networking"},{"id":7,"name":"node-version"},{"id":55,"name":"nosql"},{"id":57,"name":"observability"},{"id":46,"name":"orm"},{"id":52,"name":"os"},{"id":14,"name":"parser"},{"id":74,"name":"react"},{"id":82,"name":"real-time"},{"id":56,"name":"robot"},{"id":65,"name":"runtime"},{"id":32,"name":"sdk"},{"id":71,"name":"search"},{"id":63,"name":"secrets"},{"id":25,"name":"security"},{"id":85,"name":"server"},{"id":86,"name":"serverless"},{"id":70,"name":"storage"},{"id":75,"name":"system-design"},{"id":79,"name":"terminal"},{"id":29,"name":"testing"},{"id":12,"name":"ui"},{"id":50,"name":"ux"},{"id":88,"name":"video"},{"id":20,"name":"web-app"},{"id":35,"name":"web-server"},{"id":43,"name":"webassembly"},{"id":69,"name":"workflow"},{"id":87,"name":"yaml"}]" returns me the "expected json"