基于 Vue 3 的网络收音机,编译为桌面应用软件

基于 Vue 3 的网络收音机:从本地 stations.json 加载电台清单,在浏览器或 Electron 桌面端 直连公开直播流(MP3 / AAC 等直链与 HLS),无需自建音频后端。

https://github.com/yichuancq/net-rediohttps://github.com/yichuancq/net-redio

核心功能点

  • 电台列表 :运行时 fetch public/stations.json;支持根为数组,或 { "channels": [...] } / { "stations": [...] }(与 SomaFM channels.json 等字段对齐时可带封面、genre、lastPlaying 等)。
  • 浏览与搜索 :默认「所有电台」按热度排序;侧栏按大洲/国家筛选(countrycode);顶栏搜索匹配名称、国家、标签、genre、描述、DJ、在播曲目等。
  • 收藏 :顶栏星标进入收藏列表;卡片上星标添加/移除;数据存 localStorage(与 stationMergeKey 一致)。
  • 播放 :选中即播;Icecast 类直链与 .m3u8hls.js,Safari 可原生 HLS);底栏音量、静音、播放/暂停。
  • 主界面 :深浅色主题(html.theme-light,本地记忆);侧栏可折叠;上方黑胶舞台(可选唱臂、多色唱片底、转速);封面主色轻染舞台背景;下方网格单独滚动;底栏含示意性电平动画(非 Web Audio 真实频谱)、封面与曲目信息。
  • 设置页 :主题强调色(色板 + 最近使用,写入 --accent 等 CSS 变量);唱片底片颜色/转速;是否显示唱臂、是否显示示波器区域;均持久化 localStorage

主界面截图

项目结构

net-redio/

├── electron/

│ └── main.js # Electron 主进程(窗口、生产环境加载 dist/index.html)

├── public/

│ ├── index.html

│ ├── stations.json # 运行时加载的电台清单(路径与 App.vue 中 fetch 逻辑一致)

│ └── favicon.ico

├── scripts/

│ └── update-somafm-stations.mjs

├── src/

│ ├── App.vue # 根布局、清单/导航/播放/HLS(核心)

│ ├── main.js

│ ├── composables/

│ │ ├── useNetRadioTheme.js # 明暗主题 + html 类名

│ │ ├── useThemeAccentColor.js # 自定义强调色、最近色、CSS 变量

│ │ ├── useVinylAppearance.js # 唱片色/转速、示波器、唱臂;localStorage

│ │ ├── useRadioFavorites.js # 收藏 ID

│ │ └── useStationArtAccent.js # 封面主色采样(canvas,供舞台背景)

│ ├── data/

│ │ ├── stationCatalog.js # 解析、规范化、搜索/排序/按国家

│ │ ├── radioContinentNav.js # 侧栏大洲 → 国家

│ │ ├── continentNavCountries.js # 各洲 ISO2 与中文名

│ │ └── themeAccentPalette.js # 设置页主题色色板

│ ├── styles/

│ │ ├── net-tv-tokens.css # CSS 变量(明暗、强调色默认值等)

│ │ └── net-tv-global.css

│ └── views/net-radio/ # 侧栏、工具栏、舞台、网格、底栏、示波器、设置

├── electron-builder.yml

├── vue.config.js # publicPath、devServer;Electron 构建见上文

├── package.json

└── README.md

实现要点

位置 说明
App.vue stationsCatalog 全量;filteredStationsactiveNavKeyall / favorites / country:xx)与搜索共同决定。stationMergeKey:UUID 优先,否则 u:+URL,作收藏与当前台 id。attachStreamdestroyHls(),再区分 HLS 与直链。
stationCatalog.js normalizeStation 须存在 url_resolvedurl;兼容多种 API 字段;parseStationsPayload 识别数组或 channels / stations
useThemeAccentColor.js 有存储时覆盖 documentElement--accent--accent-dim--brand-end--card-active-border;无存储则沿用 net-tv-tokens.css
useVinylAppearance.js 黑胶渐变背景定义在同文件导出的 VINYL_DISC_BACKGROUNDSNetRadioStage 使用)。
收藏 useRadioFavorites 的 id 与网格 stationMergeKey 一致;收藏列表从全量 stationsCatalog 过滤,避免子列表截断导致「收藏消失」。
相关推荐
甲维斯9 小时前
GLM5.2超过Opus4.8Think,全球第二了!
前端·人工智能·ai编程
by————组态9 小时前
Ricon组态系统 - 新一代Web可视化组态平台
前端·后端·物联网·架构·组态·组态软件
JieE2129 小时前
手把手带你用纯 CSS 实现一个 3D 旋转魔方,这些前端基础你能打几分?
前端·css·html
lichenyang4539 小时前
鸿蒙 Web 容器(二):H5 和 ArkTS 说话前,先定一份「协议」
前端
JYeontu9 小时前
开箱流水加载动画
前端·javascript·css
RANxy9 小时前
AntV 入门系列:G6 图可视化实战
前端
尽欢i9 小时前
Vue3 customRef 封神教程:防抖、本地存储、自动埋点一套搞定,模板干干净净
前端·javascript·vue.js
VOLUN9 小时前
TypeScript封装通用RESTful BaseAPI,后台接口代码精简80%
前端·javascript
胡永双9 小时前
Hexo + GitHub Pages搭建个人Blog教程(三)
前端