Skip to content

Commit e02d6b3

Browse files
committed
feat(punctuator): auto-convert digit separator
fixes rime#972 use ascii punctuation ,.:' after numbers. they are auto-committed if followed by digit. or commit manualy with space key. double strike the key to access the original binding.
1 parent 6f6f540 commit e02d6b3

File tree

2 files changed

+67
-22
lines changed

2 files changed

+67
-22
lines changed

src/rime/gear/punctuator.cc

+65-21
Original file line numberDiff line numberDiff line change
@@ -58,6 +58,30 @@ static bool punctuation_is_translated(Context* ctx) {
5858
return cand && cand->type() == "punct";
5959
}
6060

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+
6185
ProcessResult Punctuator::ProcessKeyEvent(const KeyEvent& key_event) {
6286
if (key_event.release() || key_event.ctrl() || key_event.alt() ||
6387
key_event.super())
@@ -72,29 +96,46 @@ ProcessResult Punctuator::ProcessKeyEvent(const KeyEvent& key_event) {
7296
if (!use_space_ && ch == XK_space && ctx->IsComposing()) {
7397
return kNoop;
7498
}
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;
83102
}
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);
87105
if (!punct_definition)
88106
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+
}
94116
}
95117
return kAccepted;
96118
}
97119

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+
98139
bool Punctuator::AlternatePunct(const string& key,
99140
const an<ConfigItem>& definition) {
100141
if (!As<ConfigList>(definition))
@@ -170,16 +211,20 @@ bool PunctSegmentor::Proceed(Segmentation* segmentation) {
170211
char ch = input[k];
171212
if (ch < 0x20 || ch >= 0x7f)
172213
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);
176216
if (!punct_definition)
177217
return true;
178218
{
179219
Segment segment(k, k + 1);
180220
DLOG(INFO) << "add a punctuation segment [" << segment.start << ", "
181221
<< 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+
}
183228
segmentation->AddSegment(segment);
184229
}
185230
return false; // exclusive
@@ -235,7 +280,6 @@ an<Translation> PunctTranslator::Query(const string& input,
235280
const Segment& segment) {
236281
if (!segment.HasTag("punct"))
237282
return nullptr;
238-
config_.LoadConfig(engine_);
239283
auto definition = config_.GetPunctDefinition(input);
240284
if (!definition)
241285
return nullptr;

src/rime/gear/punctuator.h

+2-1
Original file line numberDiff line numberDiff line change
@@ -35,8 +35,9 @@ class Punctuator : public Processor {
3535
virtual ProcessResult ProcessKeyEvent(const KeyEvent& key_event);
3636

3737
protected:
38-
bool ConfirmUniquePunct(const an<ConfigItem>& definition);
38+
bool ToggleNumberMode(const string& key);
3939
bool AlternatePunct(const string& key, const an<ConfigItem>& definition);
40+
bool ConfirmUniquePunct(const an<ConfigItem>& definition);
4041
bool AutoCommitPunct(const an<ConfigItem>& definition);
4142
bool PairPunct(const an<ConfigItem>& definition);
4243

0 commit comments

Comments
 (0)