Skip to content

Commit

Permalink
feat(punctuator): recognize number patterns
Browse files Browse the repository at this point in the history
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.
  • Loading branch information
lotem committed Jan 16, 2025
1 parent 2930c26 commit 2f35778
Show file tree
Hide file tree
Showing 2 changed files with 67 additions and 22 deletions.
86 changes: 65 additions & 21 deletions src/rime/gear/punctuator.cc
Original file line number Diff line number Diff line change
Expand Up @@ -58,6 +58,30 @@ static bool punctuation_is_translated(Context* ctx) {
return cand && cand->type() == "punct";
}

inline static bool is_digit_separator(char ch) {
return ch == '.' || ch == ':' || ch == ',' || ch == '\'';
}

inline static bool ends_with_digit(const string& text) {
auto len = text.length();
return len > 0 && isdigit(text[len - 1]);
}

// recognizes patterns like 3.14 12:30 1,000 1'000
static bool is_after_number(Context* ctx) {
const CommitHistory& history = ctx->commit_history();
if (history.empty()) {
return false;
}
const CommitRecord& cr = history.back();
return ends_with_digit(cr.text) & (cr.type == "thru" || cr.type == "raw");
}

static bool is_after_digit_separator(Context* ctx) {
const auto& comp = ctx->composition();
return !comp.empty() && comp.back().HasTag("punct_number");
}

ProcessResult Punctuator::ProcessKeyEvent(const KeyEvent& key_event) {
if (key_event.release() || key_event.ctrl() || key_event.alt() ||
key_event.super())
Expand All @@ -72,29 +96,46 @@ ProcessResult Punctuator::ProcessKeyEvent(const KeyEvent& key_event) {
if (!use_space_ && ch == XK_space && ctx->IsComposing()) {
return kNoop;
}
if (ch == '.' || ch == ':') { // 3.14, 12:30
const CommitHistory& history(ctx->commit_history());
if (!history.empty()) {
const CommitRecord& cr(history.back());
if (cr.type == "thru" && cr.text.length() == 1 && isdigit(cr.text[0])) {
return kRejected;
}
}
if (isdigit(ch) && is_after_digit_separator(ctx)) {
ctx->PushInput(ch) && ctx->Commit();
return kAccepted;
}
config_.LoadConfig(engine_);
string punct_key(1, ch);
auto punct_definition = config_.GetPunctDefinition(punct_key);
string key(1, ch);
auto punct_definition = config_.GetPunctDefinition(key);
if (!punct_definition)
return kNoop;
DLOG(INFO) << "punct key: '" << punct_key << "'";
if (!AlternatePunct(punct_key, punct_definition)) {
ctx->PushInput(ch) && punctuation_is_translated(ctx) &&
(ConfirmUniquePunct(punct_definition) ||
AutoCommitPunct(punct_definition) || PairPunct(punct_definition));
DLOG(INFO) << "punct key: '" << key << "'";
if (AlternatePunct(key, punct_definition)) {
return kAccepted;
}
if (ToggleNumberMode(key) || ctx->PushInput(ch)) {
if (punctuation_is_translated(ctx)) {
ConfirmUniquePunct(punct_definition) ||
AutoCommitPunct(punct_definition) || PairPunct(punct_definition);
}
}
return kAccepted;
}

bool Punctuator::ToggleNumberMode(const string& key) {
Context* ctx = engine_->context();
if (ctx->input() == key) {
Composition& comp = ctx->composition();
if (!comp.empty()) {
Segment& segment(comp.back());
if (segment.HasTag("punct_number")) {
segment.tags.erase("punct_number");
segment.tags.insert("punct");
segment.status = Segment::kVoid;
DLOG(INFO) << "exit number mode, key = " << key;
ctx->update_notifier()(ctx);
return true;
}
}
}
return false;
}

bool Punctuator::AlternatePunct(const string& key,
const an<ConfigItem>& definition) {
if (!As<ConfigList>(definition))
Expand Down Expand Up @@ -170,16 +211,20 @@ bool PunctSegmentor::Proceed(Segmentation* segmentation) {
char ch = input[k];
if (ch < 0x20 || ch >= 0x7f)
return true;
config_.LoadConfig(engine_);
string punct_key(1, ch);
auto punct_definition = config_.GetPunctDefinition(punct_key);
string key(1, ch);
auto punct_definition = config_.GetPunctDefinition(key);
if (!punct_definition)
return true;
{
Segment segment(k, k + 1);
DLOG(INFO) << "add a punctuation segment [" << segment.start << ", "
<< segment.end << ")";
segment.tags.insert("punct");
if (k == 0 && is_digit_separator(ch) &&
is_after_number(engine_->context())) {
segment.tags.insert("punct_number");
} else {
segment.tags.insert("punct");
}
segmentation->AddSegment(segment);
}
return false; // exclusive
Expand Down Expand Up @@ -235,7 +280,6 @@ an<Translation> PunctTranslator::Query(const string& input,
const Segment& segment) {
if (!segment.HasTag("punct"))
return nullptr;
config_.LoadConfig(engine_);
auto definition = config_.GetPunctDefinition(input);
if (!definition)
return nullptr;
Expand Down
3 changes: 2 additions & 1 deletion src/rime/gear/punctuator.h
Original file line number Diff line number Diff line change
Expand Up @@ -35,8 +35,9 @@ class Punctuator : public Processor {
virtual ProcessResult ProcessKeyEvent(const KeyEvent& key_event);

protected:
bool ConfirmUniquePunct(const an<ConfigItem>& definition);
bool ToggleNumberMode(const string& key);
bool AlternatePunct(const string& key, const an<ConfigItem>& definition);
bool ConfirmUniquePunct(const an<ConfigItem>& definition);
bool AutoCommitPunct(const an<ConfigItem>& definition);
bool PairPunct(const an<ConfigItem>& definition);

Expand Down

0 comments on commit 2f35778

Please sign in to comment.