-
Notifications
You must be signed in to change notification settings - Fork 306
/
JapaneseVCVPhonemizer.cs
116 lines (105 loc) · 5.3 KB
/
JapaneseVCVPhonemizer.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
using System.Collections.Generic;
using System.Linq;
using OpenUtau.Api;
using OpenUtau.Core.Ustx;
namespace OpenUtau.Plugin.Builtin {
[Phonemizer("Japanese VCV Phonemizer", "JA VCV", language: "JA")]
public class JapaneseVCVPhonemizer : Phonemizer {
/// <summary>
/// The lookup table to convert a hiragana to its tail vowel.
/// </summary>
static readonly string[] vowels = new string[] {
"a=ぁ,あ,か,が,さ,ざ,た,だ,な,は,ば,ぱ,ま,ゃ,や,ら,わ,ァ,ア,カ,ガ,サ,ザ,タ,ダ,ナ,ハ,バ,パ,マ,ャ,ヤ,ラ,ワ,a",
"e=ぇ,え,け,げ,せ,ぜ,て,で,ね,へ,べ,ぺ,め,れ,ゑ,ェ,エ,ケ,ゲ,セ,ゼ,テ,デ,ネ,ヘ,ベ,ペ,メ,レ,ヱ,e",
"i=ぃ,い,き,ぎ,し,じ,ち,ぢ,に,ひ,び,ぴ,み,り,ゐ,ィ,イ,キ,ギ,シ,ジ,チ,ヂ,ニ,ヒ,ビ,ピ,ミ,リ,ヰ,i",
"o=ぉ,お,こ,ご,そ,ぞ,と,ど,の,ほ,ぼ,ぽ,も,ょ,よ,ろ,を,ォ,オ,コ,ゴ,ソ,ゾ,ト,ド,ノ,ホ,ボ,ポ,モ,ョ,ヨ,ロ,ヲ,o",
"n=ん,n",
"u=ぅ,う,く,ぐ,す,ず,つ,づ,ぬ,ふ,ぶ,ぷ,む,ゅ,ゆ,る,ゥ,ウ,ク,グ,ス,ズ,ツ,ヅ,ヌ,フ,ブ,プ,ム,ュ,ユ,ル,ヴ,u",
"N=ン,ng",
};
static readonly Dictionary<string, string> vowelLookup;
static JapaneseVCVPhonemizer() {
// Converts the lookup table from raw strings to a dictionary for better performance.
vowelLookup = vowels.ToList()
.SelectMany(line => {
var parts = line.Split('=');
return parts[1].Split(',').Select(cv => (cv, parts[0]));
})
.ToDictionary(t => t.Item1, t => t.Item2);
}
private USinger singer;
// Simply stores the singer in a field.
public override void SetSinger(USinger singer) => this.singer = singer;
public override Result Process(Note[] notes, Note? prev, Note? next, Note? prevNeighbour, Note? nextNeighbour, Note[] prevNeighbours) {
var note = notes[0];
var currentLyric = note.lyric.Normalize(); //measures for Unicode
if (!string.IsNullOrEmpty(note.phoneticHint)) {
// If a hint is present, returns the hint.
if (CheckOtoUntilHit(new string[] { note.phoneticHint.Normalize() }, note, out var ph)) {
return new Result {
phonemes = new Phoneme[] {
new Phoneme {
phoneme = ph.Alias,
}
},
};
}
}
// The alias for no previous neighbour note. For example, "- な" for "な".
string[] tests = new string[] { $"- {currentLyric}" , currentLyric};
if (prevNeighbour != null) {
// If there is a previous neighbour note, first get its hint or lyric.
var prevLyric = prevNeighbour.Value.lyric.Normalize();
if (!string.IsNullOrEmpty(prevNeighbour.Value.phoneticHint)) {
prevLyric = prevNeighbour.Value.phoneticHint.Normalize();
}
// Get the last unicode element of the hint or lyric. For example, "ゃ" from "きゃ" or "- きゃ".
var unicode = ToUnicodeElements(prevLyric);
// Look up the trailing vowel. For example "a" for "ゃ".
if (vowelLookup.TryGetValue(unicode.LastOrDefault() ?? string.Empty, out var vow)) {
// Now replace "- な" initially set to "a な".
tests = new string[] { $"{vow} {currentLyric}", $"* {currentLyric}", currentLyric, $"- {currentLyric}" };
}
}
if (CheckOtoUntilHit(tests, note, out var oto)) {
return new Result {
phonemes = new Phoneme[] {
new Phoneme {
phoneme = oto.Alias,
}
},
};
}
return new Result {
phonemes = new Phoneme[] {
new Phoneme {
phoneme = currentLyric,
}
},
};
}
private bool CheckOtoUntilHit(string[] input, Note note, out UOto oto) {
oto = default;
var attr = note.phonemeAttributes?.FirstOrDefault(attr => attr.index == 0) ?? default;
string color = attr.voiceColor ?? "";
var otos = new List<UOto>();
foreach (string test in input) {
if (singer.TryGetMappedOto(test + attr.alternate, note.tone + attr.toneShift, color, out var otoAlt)) {
otos.Add(otoAlt);
} else if (singer.TryGetMappedOto(test, note.tone + attr.toneShift, color, out var otoCandidacy)) {
otos.Add(otoCandidacy);
}
}
if (otos.Count > 0) {
if (otos.Any(oto => (oto.Color ?? string.Empty) == color)) {
oto = otos.Find(oto => (oto.Color ?? string.Empty) == color);
return true;
} else {
oto = otos.First();
return true;
}
}
return false;
}
}
}