@@ -58,6 +58,30 @@ static bool punctuation_is_translated(Context* ctx) {
58
58
return cand && cand->type () == " punct" ;
59
59
}
60
60
61
+ inline static bool is_digit_separator (char ch) {
62
+ return ch == ' .' || ch == ' :' || ch == ' ,' || ch == ' \' ' ;
63
+ }
64
+
65
+ inline static bool ends_with_digit (const string& text) {
66
+ auto len = text.length ();
67
+ return len > 0 && isdigit (text[len - 1 ]);
68
+ }
69
+
70
+ // recognizes patterns like 3.14 12:30 1,000 1'000
71
+ static bool is_after_number (Context* ctx) {
72
+ const CommitHistory& history = ctx->commit_history ();
73
+ if (history.empty ()) {
74
+ return false ;
75
+ }
76
+ const CommitRecord& cr = history.back ();
77
+ return ends_with_digit (cr.text ) & (cr.type == " thru" || cr.type == " raw" );
78
+ }
79
+
80
+ static bool is_after_digit_separator (Context* ctx) {
81
+ const auto & comp = ctx->composition ();
82
+ return !comp.empty () && comp.back ().HasTag (" punct_number" );
83
+ }
84
+
61
85
ProcessResult Punctuator::ProcessKeyEvent (const KeyEvent& key_event) {
62
86
if (key_event.release () || key_event.ctrl () || key_event.alt () ||
63
87
key_event.super ())
@@ -72,29 +96,46 @@ ProcessResult Punctuator::ProcessKeyEvent(const KeyEvent& key_event) {
72
96
if (!use_space_ && ch == XK_space && ctx->IsComposing ()) {
73
97
return kNoop ;
74
98
}
75
- if (ch == ' .' || ch == ' :' ) { // 3.14, 12:30
76
- const CommitHistory& history (ctx->commit_history ());
77
- if (!history.empty ()) {
78
- const CommitRecord& cr (history.back ());
79
- if (cr.type == " thru" && cr.text .length () == 1 && isdigit (cr.text [0 ])) {
80
- return kRejected ;
81
- }
82
- }
99
+ if (isdigit (ch) && is_after_digit_separator (ctx)) {
100
+ ctx->PushInput (ch) && ctx->Commit ();
101
+ return kAccepted ;
83
102
}
84
- config_.LoadConfig (engine_);
85
- string punct_key (1 , ch);
86
- auto punct_definition = config_.GetPunctDefinition (punct_key);
103
+ string key (1 , ch);
104
+ auto punct_definition = config_.GetPunctDefinition (key);
87
105
if (!punct_definition)
88
106
return kNoop ;
89
- DLOG (INFO) << " punct key: '" << punct_key << " '" ;
90
- if (!AlternatePunct (punct_key, punct_definition)) {
91
- ctx->PushInput (ch) && punctuation_is_translated (ctx) &&
92
- (ConfirmUniquePunct (punct_definition) ||
93
- AutoCommitPunct (punct_definition) || PairPunct (punct_definition));
107
+ DLOG (INFO) << " punct key: '" << key << " '" ;
108
+ if (AlternatePunct (key, punct_definition)) {
109
+ return kAccepted ;
110
+ }
111
+ if (ToggleNumberMode (key) || ctx->PushInput (ch)) {
112
+ if (punctuation_is_translated (ctx)) {
113
+ ConfirmUniquePunct (punct_definition) ||
114
+ AutoCommitPunct (punct_definition) || PairPunct (punct_definition);
115
+ }
94
116
}
95
117
return kAccepted ;
96
118
}
97
119
120
+ bool Punctuator::ToggleNumberMode (const string& key) {
121
+ Context* ctx = engine_->context ();
122
+ if (ctx->input () == key) {
123
+ Composition& comp = ctx->composition ();
124
+ if (!comp.empty ()) {
125
+ Segment& segment (comp.back ());
126
+ if (segment.HasTag (" punct_number" )) {
127
+ segment.tags .erase (" punct_number" );
128
+ segment.tags .insert (" punct" );
129
+ segment.status = Segment::kVoid ;
130
+ DLOG (INFO) << " exit number mode, key = " << key;
131
+ ctx->update_notifier ()(ctx);
132
+ return true ;
133
+ }
134
+ }
135
+ }
136
+ return false ;
137
+ }
138
+
98
139
bool Punctuator::AlternatePunct (const string& key,
99
140
const an<ConfigItem>& definition) {
100
141
if (!As<ConfigList>(definition))
@@ -170,16 +211,20 @@ bool PunctSegmentor::Proceed(Segmentation* segmentation) {
170
211
char ch = input[k];
171
212
if (ch < 0x20 || ch >= 0x7f )
172
213
return true ;
173
- config_.LoadConfig (engine_);
174
- string punct_key (1 , ch);
175
- auto punct_definition = config_.GetPunctDefinition (punct_key);
214
+ string key (1 , ch);
215
+ auto punct_definition = config_.GetPunctDefinition (key);
176
216
if (!punct_definition)
177
217
return true ;
178
218
{
179
219
Segment segment (k, k + 1 );
180
220
DLOG (INFO) << " add a punctuation segment [" << segment.start << " , "
181
221
<< segment.end << " )" ;
182
- segment.tags .insert (" punct" );
222
+ if (k == 0 && is_digit_separator (ch) &&
223
+ is_after_number (engine_->context ())) {
224
+ segment.tags .insert (" punct_number" );
225
+ } else {
226
+ segment.tags .insert (" punct" );
227
+ }
183
228
segmentation->AddSegment (segment);
184
229
}
185
230
return false ; // exclusive
@@ -235,7 +280,6 @@ an<Translation> PunctTranslator::Query(const string& input,
235
280
const Segment& segment) {
236
281
if (!segment.HasTag (" punct" ))
237
282
return nullptr ;
238
- config_.LoadConfig (engine_);
239
283
auto definition = config_.GetPunctDefinition (input);
240
284
if (!definition)
241
285
return nullptr ;
0 commit comments