Compare commits

..

7 Commits

Author SHA1 Message Date
ffd079ae24 fix(deps): update dependency @ffmpeg.wasm/core-mt to v0.13.2 2026-02-17 20:48:37 +00:00
4a4c0fe055 chore(i18n): Translated using Weblate (Spanish)
Currently translated at 100.0% (477 of 477 strings)

Translation: pear-devs/pear-desktop/i18n
Translate-URL: https://hosted.weblate.org/projects/youtube-music/i18n/es/
2026-02-17 11:09:50 +01:00
fd9ae2c23d chore(i18n): Translated using Weblate (French)
Currently translated at 100.0% (477 of 477 strings)

Translation: pear-devs/pear-desktop/i18n
Translate-URL: https://hosted.weblate.org/projects/youtube-music/i18n/fr/
2026-02-17 11:09:47 +01:00
b6d2bcd42f chore(i18n): Translated using Weblate (Portuguese (Brazil))
Currently translated at 100.0% (477 of 477 strings)

Translation: pear-devs/pear-desktop/i18n
Translate-URL: https://hosted.weblate.org/projects/youtube-music/i18n/pt_BR/
2026-02-16 03:09:55 +01:00
021972520a chore(i18n): Translated using Weblate (Chinese (Simplified Han script))
Currently translated at 100.0% (477 of 477 strings)

Translation: pear-devs/pear-desktop/i18n
Translate-URL: https://hosted.weblate.org/projects/youtube-music/i18n/zh_Hans/
2026-02-16 03:09:54 +01:00
90e949d4e9 chore(i18n): Translated using Weblate (Chinese (Traditional Han script))
Currently translated at 100.0% (469 of 469 strings)

Translation: pear-devs/pear-desktop/i18n
Translate-URL: https://hosted.weblate.org/projects/youtube-music/i18n/zh_Hant/
2026-02-14 18:09:56 +00:00
208e57bdd3 feat(synced-lyrics): Add Simplified/Traditional Chinese converter and improve Romanization to display tone (#4111)
Co-authored-by: github-actions[bot] <41898282+github-actions[bot]@users.noreply.github.com>
Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com>
Co-authored-by: Angelos Bouklis <me@arjix.dev>
Co-authored-by: JellyBrick <shlee1503@naver.com>
2026-02-13 01:27:48 +09:00
13 changed files with 266 additions and 25 deletions

View File

@ -89,6 +89,7 @@
"bgutils-js": "3.2.0", "bgutils-js": "3.2.0",
"butterchurn": "3.0.0-beta.5", "butterchurn": "3.0.0-beta.5",
"butterchurn-presets": "3.0.0-beta.4", "butterchurn-presets": "3.0.0-beta.4",
"chinese-conv": "^4.0.0",
"color": "5.0.3", "color": "5.0.3",
"conf": "15.0.2", "conf": "15.0.2",
"custom-electron-prompt": "1.6.1", "custom-electron-prompt": "1.6.1",
@ -121,6 +122,7 @@
"node-html-parser": "7.0.2", "node-html-parser": "7.0.2",
"node-id3": "0.2.9", "node-id3": "0.2.9",
"peerjs": "1.5.5", "peerjs": "1.5.5",
"pinyin-pro": "^3.27.0",
"semver": "7.7.3", "semver": "7.7.3",
"serve": "14.2.5", "serve": "14.2.5",
"socks": "2.8.7", "socks": "2.8.7",
@ -129,7 +131,6 @@
"solid-js": "1.9.11", "solid-js": "1.9.11",
"solid-styled-components": "0.28.5", "solid-styled-components": "0.28.5",
"solid-transition-group": "0.3.0", "solid-transition-group": "0.3.0",
"tiny-pinyin": "1.3.2",
"tinyld": "1.3.4", "tinyld": "1.3.4",
"virtua": "0.48.5", "virtua": "0.48.5",
"vudio": "2.1.1", "vudio": "2.1.1",

26
pnpm-lock.yaml generated
View File

@ -114,6 +114,9 @@ importers:
butterchurn-presets: butterchurn-presets:
specifier: 3.0.0-beta.4 specifier: 3.0.0-beta.4
version: 3.0.0-beta.4 version: 3.0.0-beta.4
chinese-conv:
specifier: ^4.0.0
version: 4.0.0
color: color:
specifier: 5.0.3 specifier: 5.0.3
version: 5.0.3 version: 5.0.3
@ -210,6 +213,9 @@ importers:
peerjs: peerjs:
specifier: 1.5.5 specifier: 1.5.5
version: 1.5.5 version: 1.5.5
pinyin-pro:
specifier: ^3.27.0
version: 3.27.0
semver: semver:
specifier: 7.7.3 specifier: 7.7.3
version: 7.7.3 version: 7.7.3
@ -234,9 +240,6 @@ importers:
solid-transition-group: solid-transition-group:
specifier: 0.3.0 specifier: 0.3.0
version: 0.3.0(solid-js@1.9.11) version: 0.3.0(solid-js@1.9.11)
tiny-pinyin:
specifier: 1.3.2
version: 1.3.2
tinyld: tinyld:
specifier: 1.3.4 specifier: 1.3.4
version: 1.3.4 version: 1.3.4
@ -1827,6 +1830,10 @@ packages:
resolution: {integrity: sha512-Fo07WOYGqMfCWHOzSXOt2CxDbC6skS/jO9ynEcmpANMoPrD+W1r1K6Vx7iNm+AQmETU1Xr2t+n8nzkV9t6xh3w==} resolution: {integrity: sha512-Fo07WOYGqMfCWHOzSXOt2CxDbC6skS/jO9ynEcmpANMoPrD+W1r1K6Vx7iNm+AQmETU1Xr2t+n8nzkV9t6xh3w==}
engines: {node: ^12.17.0 || ^14.13 || >=16.0.0} engines: {node: ^12.17.0 || ^14.13 || >=16.0.0}
chinese-conv@4.0.0:
resolution: {integrity: sha512-PVBMzvv6CtX1cubaDXfxYscIdbOAHPuY/E2vnfJIzOACX+xIW4NRKRlNsZVI2p5KxGsXyUp7tVHfvQlqZ4yx/w==}
engines: {node: ^20.19.0 || >=22.12.0}
chownr@3.0.0: chownr@3.0.0:
resolution: {integrity: sha512-+IxzY9BZOQd/XuYPRmrvEVjF/nqj5kgT4kEq7VofrDoM1MxoRjEWkrCC3EtLi59TVawxTAn+orJwFQcrqEN1+g==} resolution: {integrity: sha512-+IxzY9BZOQd/XuYPRmrvEVjF/nqj5kgT4kEq7VofrDoM1MxoRjEWkrCC3EtLi59TVawxTAn+orJwFQcrqEN1+g==}
engines: {node: '>=18'} engines: {node: '>=18'}
@ -3737,6 +3744,9 @@ packages:
resolution: {integrity: sha512-5gTmgEY/sqK6gFXLIsQNH19lWb4ebPDLA4SdLP7dsWkIXHWlG66oPuVvXSGFPppYZz8ZDZq0dYYrbHfBCVUb1Q==} resolution: {integrity: sha512-5gTmgEY/sqK6gFXLIsQNH19lWb4ebPDLA4SdLP7dsWkIXHWlG66oPuVvXSGFPppYZz8ZDZq0dYYrbHfBCVUb1Q==}
engines: {node: '>=12'} engines: {node: '>=12'}
pinyin-pro@3.27.0:
resolution: {integrity: sha512-Osdgjwe7Rm17N2paDMM47yW+jUIUH3+0RGo8QP39ZTLpTaJVDK0T58hOLaMQJbcMmAebVuK2ePunTEVEx1clNQ==}
pixelmatch@5.3.0: pixelmatch@5.3.0:
resolution: {integrity: sha512-o8mkY4E/+LNUf6LzX96ht6k6CEDi65k9G2rjMtBe9Oo+VPKSvl+0GKHuH/AlG+GA5LPG/i5hrekkxUc3s2HU+Q==} resolution: {integrity: sha512-o8mkY4E/+LNUf6LzX96ht6k6CEDi65k9G2rjMtBe9Oo+VPKSvl+0GKHuH/AlG+GA5LPG/i5hrekkxUc3s2HU+Q==}
hasBin: true hasBin: true
@ -4291,6 +4301,7 @@ packages:
tar@7.5.7: tar@7.5.7:
resolution: {integrity: sha512-fov56fJiRuThVFXD6o6/Q354S7pnWMJIVlDBYijsTNx6jKSE4pvrDTs6lUnmGvNyfJwFQQwWy3owKz1ucIhveQ==} resolution: {integrity: sha512-fov56fJiRuThVFXD6o6/Q354S7pnWMJIVlDBYijsTNx6jKSE4pvrDTs6lUnmGvNyfJwFQQwWy3owKz1ucIhveQ==}
engines: {node: '>=18'} engines: {node: '>=18'}
deprecated: Old versions of tar are not supported, and contain widely publicized security vulnerabilities, which have been fixed in the current version. Please update. Support for old versions may be purchased (at exorbitant rates) by contacting i@izs.me
temp-file@3.4.0: temp-file@3.4.0:
resolution: {integrity: sha512-C5tjlC/HCtVUOi3KWVokd4vHVViOmGjtLwIh4MuzPo/nMYTV/p1urt3RnMz2IWXDdKEGJH3k5+KPxtqRsUYGtg==} resolution: {integrity: sha512-C5tjlC/HCtVUOi3KWVokd4vHVViOmGjtLwIh4MuzPo/nMYTV/p1urt3RnMz2IWXDdKEGJH3k5+KPxtqRsUYGtg==}
@ -4305,9 +4316,6 @@ packages:
tiny-async-pool@1.3.0: tiny-async-pool@1.3.0:
resolution: {integrity: sha512-01EAw5EDrcVrdgyCLgoSPvqznC0sVxDSVeiOz09FUpjh71G79VCqneOr+xvt7T1r76CF6ZZfPjHorN2+d+3mqA==} resolution: {integrity: sha512-01EAw5EDrcVrdgyCLgoSPvqznC0sVxDSVeiOz09FUpjh71G79VCqneOr+xvt7T1r76CF6ZZfPjHorN2+d+3mqA==}
tiny-pinyin@1.3.2:
resolution: {integrity: sha512-uHNGu4evFt/8eNLldazeAM1M8JrMc1jshhJJfVRARTN3yT8HEEibofeQ7QETWQ5ISBjd6fKtTVBCC/+mGS6FpA==}
tiny-typed-emitter@2.1.0: tiny-typed-emitter@2.1.0:
resolution: {integrity: sha512-qVtvMxeXbVej0cQWKqVSSAHmKZEHAvxdF8HEUBFWts8h+xEo5m/lEiPakuyZ3BnCBjOD8i24kzNOiOLLgsSxhA==} resolution: {integrity: sha512-qVtvMxeXbVej0cQWKqVSSAHmKZEHAvxdF8HEUBFWts8h+xEo5m/lEiPakuyZ3BnCBjOD8i24kzNOiOLLgsSxhA==}
@ -6352,6 +6360,8 @@ snapshots:
chalk@5.0.1: {} chalk@5.0.1: {}
chinese-conv@4.0.0: {}
chownr@3.0.0: {} chownr@3.0.0: {}
chromium-pickle-js@0.2.0: {} chromium-pickle-js@0.2.0: {}
@ -8451,6 +8461,8 @@ snapshots:
picomatch@4.0.3: {} picomatch@4.0.3: {}
pinyin-pro@3.27.0: {}
pixelmatch@5.3.0: pixelmatch@5.3.0:
dependencies: dependencies:
pngjs: 6.0.0 pngjs: 6.0.0
@ -9063,8 +9075,6 @@ snapshots:
dependencies: dependencies:
semver: 5.7.2 semver: 5.7.2
tiny-pinyin@1.3.2: {}
tiny-typed-emitter@2.1.0: {} tiny-typed-emitter@2.1.0: {}
tinycolor2@1.6.0: {} tinycolor2@1.6.0: {}

View File

@ -891,6 +891,24 @@
"show-time-codes": { "show-time-codes": {
"label": "Show time codes", "label": "Show time codes",
"tooltip": "Show the time codes next to the lyrics" "tooltip": "Show the time codes next to the lyrics"
},
"convert-chinese-character": {
"label": "Convert Chinese character",
"submenu": {
"disabled": {
"label": "Disabled",
"tooltip": "Disable Chinese character conversion"
},
"simplified-to-traditional": {
"label": "Simplified to Traditional",
"tooltip": "Convert Simplified Chinese to Traditional Chinese"
},
"traditional-to-simplified": {
"label": "Traditional to Simplified",
"tooltip": "Convert Traditional Chinese to Simplified Chinese"
}
},
"tooltip": "Convert Chinese character to Traditional or Simplified"
} }
}, },
"name": "Synced Lyrics", "name": "Synced Lyrics",

View File

@ -161,7 +161,8 @@
"default": "Predeterminado", "default": "Predeterminado",
"force-show": "Forzar la visualización", "force-show": "Forzar la visualización",
"hide": "Ocultar", "hide": "Ocultar",
"label": "Botones de \"Me Gusta\"" "label": "Botones de \"Me Gusta\"",
"swap": "Intercambiar el orden de los botones de \"Me Gusta\""
}, },
"remove-upgrade-button": "Eliminar el botón de Actualización", "remove-upgrade-button": "Eliminar el botón de Actualización",
"theme": { "theme": {
@ -322,10 +323,10 @@
"label": "Nombre del host" "label": "Nombre del host"
}, },
"https": { "https": {
"label": "HTTPS & Certificados", "label": "HTTPS y Certificados",
"submenu": { "submenu": {
"cert": { "cert": {
"dialogTitle": "Selecciona el archivo de certificado HTTPS", "dialogTitle": "Seleccione el archivo de certificado HTTPS",
"label": "Archivo de certificado (.crt/.pem)" "label": "Archivo de certificado (.crt/.pem)"
}, },
"enable-https": { "enable-https": {
@ -412,6 +413,17 @@
"no-captions": "Sin subtítulos para ésta canción" "no-captions": "Sin subtítulos para ésta canción"
} }
}, },
"clock": {
"description": "Añade un reloj a la barra de navegación",
"menu": {
"format": {
"24-hour-format": "Formato 24 horas",
"display-seconds": "Mostrar segundos",
"label": "Formato"
}
},
"name": "Reloj"
},
"compact-sidebar": { "compact-sidebar": {
"description": "Establecer siempre la barra lateral en modo compacto", "description": "Establecer siempre la barra lateral en modo compacto",
"name": "Barra lateral compacta" "name": "Barra lateral compacta"
@ -830,6 +842,24 @@
"not-found": "⚠️ No se han encontrado letras para esta canción." "not-found": "⚠️ No se han encontrado letras para esta canción."
}, },
"menu": { "menu": {
"convert-chinese-character": {
"label": "Convertir carácter Chino",
"submenu": {
"disabled": {
"label": "Deshabilitado",
"tooltip": "Deshabilitar conversión de caracteres Chinos"
},
"simplified-to-traditional": {
"label": "Simplificar a Tradicional",
"tooltip": "Convertir Chino Simplifcado en Chino Tradicional"
},
"traditional-to-simplified": {
"label": "Tradicional a Simplificado",
"tooltip": "Convertir Chino Tradicional a Chino Simplificado"
}
},
"tooltip": "Convertir carácter Chino a Tradicional o Simplificado"
},
"default-text-string": { "default-text-string": {
"label": "Carácter predeterminado entre letras", "label": "Carácter predeterminado entre letras",
"tooltip": "Elige el carácter predeterminado que se utilizará para el espacio entre letras" "tooltip": "Elige el carácter predeterminado que se utilizará para el espacio entre letras"

View File

@ -842,6 +842,24 @@
"not-found": "⚠️ Aucune paroles trouvées pour ce titre." "not-found": "⚠️ Aucune paroles trouvées pour ce titre."
}, },
"menu": { "menu": {
"convert-chinese-character": {
"label": "Convertir les caractères Chinois",
"submenu": {
"disabled": {
"label": "Désactivé",
"tooltip": "Désactiver la conversion des caractères Chinois"
},
"simplified-to-traditional": {
"label": "Simplifié a Traditionnel",
"tooltip": "Convertir le Chinois Simplifié au Chinois Traditionnel"
},
"traditional-to-simplified": {
"label": "Traditionnel a Simplifié",
"tooltip": "Convertir le Chinois Traditionnel au Chinois Simplifié"
}
},
"tooltip": "Convertir les caractères Chinois en Traditionnel ou Simplifié"
},
"default-text-string": { "default-text-string": {
"label": "Caractère par défaut entre les paroles", "label": "Caractère par défaut entre les paroles",
"tooltip": "Choisi le caractère par défaut à utiliser pour les blancs entre les paroles" "tooltip": "Choisi le caractère par défaut à utiliser pour les blancs entre les paroles"

View File

@ -842,6 +842,24 @@
"not-found": "⚠️ Nenhuma letra encontrada para esta música." "not-found": "⚠️ Nenhuma letra encontrada para esta música."
}, },
"menu": { "menu": {
"convert-chinese-character": {
"label": "Converter caracteres Chineses",
"submenu": {
"disabled": {
"label": "Desativado",
"tooltip": "Desativar conversão de caracteres Chineses"
},
"simplified-to-traditional": {
"label": "Simplificado para Tradicional",
"tooltip": "Converter Chinês Simplificado para Chinês Tradicional"
},
"traditional-to-simplified": {
"label": "Tradicional para Simplificado",
"tooltip": "Converter Chinês Tradicional para Chinês Simplificado"
}
},
"tooltip": "Converter caractere Chinês para Tradicional ou Simplificado"
},
"default-text-string": { "default-text-string": {
"label": "Caractere padrão entre letras", "label": "Caractere padrão entre letras",
"tooltip": "Escolha o caractere padrão a ser usado para o intervalo entre as letras" "tooltip": "Escolha o caractere padrão a ser usado para o intervalo entre as letras"

View File

@ -842,6 +842,24 @@
"not-found": "⚠️ 未找到此歌曲的歌词。" "not-found": "⚠️ 未找到此歌曲的歌词。"
}, },
"menu": { "menu": {
"convert-chinese-character": {
"label": "转换中文字符",
"submenu": {
"disabled": {
"label": "已停用",
"tooltip": "禁用中文字符转换"
},
"simplified-to-traditional": {
"label": "简体到繁体",
"tooltip": "转换简体中文到繁体中文"
},
"traditional-to-simplified": {
"label": "繁体到简体",
"tooltip": "转换繁体中文到简体中文"
}
},
"tooltip": "转换简繁体中文字符"
},
"default-text-string": { "default-text-string": {
"label": "默认的歌词行间字符", "label": "默认的歌词行间字符",
"tooltip": "选择在歌词间隙期间默认显示的字符" "tooltip": "选择在歌词间隙期间默认显示的字符"

View File

@ -161,7 +161,8 @@
"default": "預設", "default": "預設",
"force-show": "強制顯示", "force-show": "強制顯示",
"hide": "隱藏", "hide": "隱藏",
"label": "按讚按鈕" "label": "按讚按鈕",
"swap": "交換讚及倒讚的按鈕位置"
}, },
"remove-upgrade-button": "移除升級按鈕", "remove-upgrade-button": "移除升級按鈕",
"theme": { "theme": {
@ -321,6 +322,22 @@
"hostname": { "hostname": {
"label": "主機名稱" "label": "主機名稱"
}, },
"https": {
"label": "HTTPS 及 憑證",
"submenu": {
"cert": {
"dialogTitle": "選擇憑證檔案",
"label": "憑證檔案(.crt/.pem"
},
"enable-https": {
"label": "啟用 HTTPS"
},
"key": {
"dialogTitle": "選擇私人金鑰檔案",
"label": "私人金鑰檔案(.key/.pem"
}
}
},
"port": { "port": {
"label": "連接埠" "label": "連接埠"
} }
@ -396,6 +413,17 @@
"no-captions": "該首歌曲無可用的字幕" "no-captions": "該首歌曲無可用的字幕"
} }
}, },
"clock": {
"description": "新增時鐘至應用程式上方",
"menu": {
"format": {
"24-hour-format": "24 小時制",
"display-seconds": "顯示秒數",
"label": "時間格式"
}
},
"name": "時鐘"
},
"compact-sidebar": { "compact-sidebar": {
"description": "永遠讓側邊欄保持收合狀態", "description": "永遠讓側邊欄保持收合狀態",
"name": "收合側邊欄" "name": "收合側邊欄"
@ -462,8 +490,8 @@
"set-status-display-type": { "set-status-display-type": {
"label": "狀態樣式", "label": "狀態樣式",
"submenu": { "submenu": {
"artist": "正在聆聽 {artist} 的歌曲",
"application": "正在聆聽 {{applicationName}}", "application": "正在聆聽 {{applicationName}}",
"artist": "正在聆聽 {artist} 的歌曲",
"title": "正在聆聽 {song title}" "title": "正在聆聽 {song title}"
} }
} }

View File

@ -153,6 +153,62 @@ export const menu = async (
}); });
}, },
}, },
{
label: t('plugins.synced-lyrics.menu.convert-chinese-character.label'),
toolTip: t(
'plugins.synced-lyrics.menu.convert-chinese-character.tooltip',
),
type: 'submenu',
submenu: [
{
label: t(
'plugins.synced-lyrics.menu.convert-chinese-character.submenu.disabled.label',
),
toolTip: t(
'plugins.synced-lyrics.menu.convert-chinese-character.submenu.disabled.tooltip',
),
type: 'radio',
checked:
config.convertChineseCharacter === 'disabled' ||
config.convertChineseCharacter === undefined,
click() {
ctx.setConfig({
convertChineseCharacter: 'disabled',
});
},
},
{
label: t(
'plugins.synced-lyrics.menu.convert-chinese-character.submenu.simplified-to-traditional.label',
),
toolTip: t(
'plugins.synced-lyrics.menu.convert-chinese-character.submenu.simplified-to-traditional.tooltip',
),
type: 'radio',
checked: config.convertChineseCharacter === 'simplifiedToTraditional',
click() {
ctx.setConfig({
convertChineseCharacter: 'simplifiedToTraditional',
});
},
},
{
label: t(
'plugins.synced-lyrics.menu.convert-chinese-character.submenu.traditional-to-simplified.label',
),
toolTip: t(
'plugins.synced-lyrics.menu.convert-chinese-character.submenu.traditional-to-simplified.tooltip',
),
type: 'radio',
checked: config.convertChineseCharacter === 'traditionalToSimplified',
click() {
ctx.setConfig({
convertChineseCharacter: 'traditionalToSimplified',
});
},
},
],
},
{ {
label: t('plugins.synced-lyrics.menu.show-time-codes.label'), label: t('plugins.synced-lyrics.menu.show-time-codes.label'),
toolTip: t('plugins.synced-lyrics.menu.show-time-codes.tooltip'), toolTip: t('plugins.synced-lyrics.menu.show-time-codes.tooltip'),

View File

@ -1,6 +1,11 @@
import { createEffect, createSignal, Show } from 'solid-js'; import { createEffect, createMemo, createSignal, Show } from 'solid-js';
import { canonicalize, romanize, simplifyUnicode } from '../utils'; import {
canonicalize,
convertChineseCharacter,
romanize,
simplifyUnicode,
} from '../utils';
import { config } from '../renderer'; import { config } from '../renderer';
interface PlainLyricsProps { interface PlainLyricsProps {
@ -9,11 +14,19 @@ interface PlainLyricsProps {
export const PlainLyrics = (props: PlainLyricsProps) => { export const PlainLyrics = (props: PlainLyricsProps) => {
const [romanization, setRomanization] = createSignal(''); const [romanization, setRomanization] = createSignal('');
const text = createMemo(() => {
let line = props.line;
const convertChineseText = config()?.convertChineseCharacter;
if (convertChineseText && convertChineseText !== 'disabled') {
line = convertChineseCharacter(line, convertChineseText);
}
return line;
});
createEffect(() => { createEffect(() => {
if (!config()?.romanization) return; if (!config()?.romanization) return;
const input = canonicalize(props.line); const input = canonicalize(text());
romanize(input).then((result) => { romanize(input).then((result) => {
setRomanization(canonicalize(result)); setRomanization(canonicalize(result));
}); });
@ -31,13 +44,13 @@ export const PlainLyrics = (props: PlainLyricsProps) => {
> >
<yt-formatted-string <yt-formatted-string
text={{ text={{
runs: [{ text: props.line }], runs: [{ text: text() }],
}} }}
/> />
<Show <Show
when={ when={
config()?.romanization && config()?.romanization &&
simplifyUnicode(props.line) !== simplifyUnicode(romanization()) simplifyUnicode(text()) !== simplifyUnicode(romanization())
} }
> >
<yt-formatted-string <yt-formatted-string

View File

@ -7,7 +7,12 @@ import { type LineLyrics } from '@/plugins/synced-lyrics/types';
import { config, currentTime } from '../renderer'; import { config, currentTime } from '../renderer';
import { _ytAPI } from '..'; import { _ytAPI } from '..';
import { canonicalize, romanize, simplifyUnicode } from '../utils'; import {
canonicalize,
convertChineseCharacter,
romanize,
simplifyUnicode,
} from '../utils';
interface SyncedLineProps { interface SyncedLineProps {
scroller: VirtualizerHandle; scroller: VirtualizerHandle;
@ -81,7 +86,14 @@ const EmptyLine = (props: SyncedLineProps) => {
}; };
export const SyncedLine = (props: SyncedLineProps) => { export const SyncedLine = (props: SyncedLineProps) => {
const text = createMemo(() => props.line.text.trim()); const text = createMemo(() => {
let line = props.line.text;
const convertChineseText = config()?.convertChineseCharacter;
if (convertChineseText && convertChineseText !== 'disabled') {
line = convertChineseCharacter(line, convertChineseText);
}
return line.trim();
});
const [romanization, setRomanization] = createSignal(''); const [romanization, setRomanization] = createSignal('');
createEffect(() => { createEffect(() => {

View File

@ -3,10 +3,11 @@ import KuromojiAnalyzer from 'kuroshiro-analyzer-kuromoji';
import Kuroshiro from 'kuroshiro'; import Kuroshiro from 'kuroshiro';
import { romanize as esHangulRomanize } from 'es-hangul'; import { romanize as esHangulRomanize } from 'es-hangul';
import hanja from 'hanja'; import hanja from 'hanja';
import * as pinyin from 'tiny-pinyin'; import { pinyin } from 'pinyin-pro';
import { romanize as romanizeThaiFrag } from '@dehoist/romanize-thai'; import { romanize as romanizeThaiFrag } from '@dehoist/romanize-thai';
import { lazy } from 'lazy-var'; import { lazy } from 'lazy-var';
import { detect } from 'tinyld'; import { detect } from 'tinyld';
import { sify, tify } from 'chinese-conv';
import Sanscript from '@indic-transliteration/sanscript'; import Sanscript from '@indic-transliteration/sanscript';
import { waitForElement } from '@/utils/wait-for-element'; import { waitForElement } from '@/utils/wait-for-element';
@ -85,6 +86,20 @@ export const canonicalize = (text: string) => {
); );
}; };
export const convertChineseCharacter = (
text: string,
mode: 'simplifiedToTraditional' | 'traditionalToSimplified',
) => {
if (!hasChinese([text])) return text;
switch (mode) {
case 'simplifiedToTraditional':
return tify(text);
case 'traditionalToSimplified':
return sify(text);
}
};
export const simplifyUnicode = (text?: string) => export const simplifyUnicode = (text?: string) =>
text text
? text ? text
@ -172,9 +187,9 @@ export const romanizeHangul = (line: string) =>
esHangulRomanize(hanja.translate(line, 'SUBSTITUTION')); esHangulRomanize(hanja.translate(line, 'SUBSTITUTION'));
export const romanizeChinese = (line: string) => { export const romanizeChinese = (line: string) => {
return line.replaceAll(/[\u4E00-\u9FFF]+/g, (match) => return line.replaceAll(/[\u4E00-\u9FFF]+/g, (match) => {
pinyin.convertToPinyin(match, ' ', true), return pinyin(match, { separator: ' ' });
); });
}; };
const thaiSegmenter = Intl.Segmenter.supportedLocalesOf('th').includes('th') const thaiSegmenter = Intl.Segmenter.supportedLocalesOf('th').includes('th')

View File

@ -10,6 +10,10 @@ export type SyncedLyricsPluginConfig = {
showLyricsEvenIfInexact: boolean; showLyricsEvenIfInexact: boolean;
lineEffect: LineEffect; lineEffect: LineEffect;
romanization: boolean; romanization: boolean;
convertChineseCharacter?:
| 'simplifiedToTraditional'
| 'traditionalToSimplified'
| 'disabled';
}; };
export type LineLyricsStatus = 'previous' | 'current' | 'upcoming'; export type LineLyricsStatus = 'previous' | 'current' | 'upcoming';