mirror of https://gitlab.com/cppit/jucipp
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
173 lines
4.9 KiB
173 lines
4.9 KiB
#include "autocomplete.h" |
|
#include "selection_dialog.h" |
|
|
|
Autocomplete::Autocomplete(Gtk::TextView *view, bool &interactive_completion, guint &last_keyval, bool strip_word) |
|
: view(view), interactive_completion(interactive_completion), strip_word(strip_word), state(State::IDLE) { |
|
view->get_buffer()->signal_changed().connect([this, &last_keyval] { |
|
if(CompletionDialog::get() && CompletionDialog::get()->is_visible()) { |
|
cancel_reparse(); |
|
return; |
|
} |
|
if(!this->view->has_focus()) |
|
return; |
|
if(is_continue_key(last_keyval) && (this->interactive_completion || state != State::IDLE)) |
|
run(); |
|
else { |
|
stop(); |
|
|
|
if(is_restart_key(last_keyval) && this->interactive_completion) |
|
run(); |
|
} |
|
}); |
|
|
|
view->get_buffer()->signal_mark_set().connect([this](const Gtk::TextBuffer::iterator &iterator, const Glib::RefPtr<Gtk::TextBuffer::Mark> &mark) { |
|
if(mark->get_name() == "insert") |
|
stop(); |
|
}); |
|
|
|
view->signal_key_release_event().connect([](GdkEventKey *key) { |
|
if(CompletionDialog::get() && CompletionDialog::get()->is_visible()) { |
|
if(CompletionDialog::get()->on_key_release(key)) |
|
return true; |
|
} |
|
return false; |
|
}, false); |
|
|
|
view->signal_focus_out_event().connect([this](GdkEventFocus *event) { |
|
stop(); |
|
return false; |
|
}); |
|
} |
|
|
|
void Autocomplete::run() { |
|
if(run_check()) { |
|
if(!is_processing()) |
|
return; |
|
|
|
if(state == State::CANCELED) |
|
state = State::RESTARTING; |
|
|
|
if(state != State::IDLE) |
|
return; |
|
|
|
state = State::STARTING; |
|
|
|
before_add_rows(); |
|
|
|
if(thread.joinable()) |
|
thread.join(); |
|
auto buffer = view->get_buffer()->get_text(); |
|
auto iter = view->get_buffer()->get_insert()->get_iter(); |
|
auto line_nr = iter.get_line() + 1; |
|
auto column_nr = iter.get_line_index() + 1; |
|
if(strip_word) { |
|
auto pos = iter.get_offset() - 1; |
|
while(pos >= 0 && ((buffer[pos] >= 'a' && buffer[pos] <= 'z') || (buffer[pos] >= 'A' && buffer[pos] <= 'Z') || |
|
(buffer[pos] >= '0' && buffer[pos] <= '9') || buffer[pos] == '_')) { |
|
buffer.replace(pos, 1, " "); |
|
column_nr--; |
|
pos--; |
|
} |
|
} |
|
thread = std::thread([this, line_nr, column_nr, buffer = std::move(buffer)] { |
|
auto lock = get_parse_lock(); |
|
if(!is_processing()) |
|
return; |
|
stop_parse(); |
|
|
|
auto &buffer_raw = const_cast<std::string &>(buffer.raw()); |
|
rows.clear(); |
|
add_rows(buffer_raw, line_nr, column_nr); |
|
|
|
if(is_processing()) { |
|
dispatcher.post([this]() { |
|
after_add_rows(); |
|
if(state == State::RESTARTING) { |
|
state = State::IDLE; |
|
reparse(); |
|
run(); |
|
} |
|
else if(state == State::CANCELED || rows.empty()) { |
|
state = State::IDLE; |
|
reparse(); |
|
} |
|
else { |
|
auto start_iter = view->get_buffer()->get_insert()->get_iter(); |
|
if(prefix.size() > 0 && !start_iter.backward_chars(prefix.size())) { |
|
state = State::IDLE; |
|
reparse(); |
|
return; |
|
} |
|
CompletionDialog::create(view, view->get_buffer()->create_mark(start_iter)); |
|
setup_dialog(); |
|
for(auto &row : rows) { |
|
CompletionDialog::get()->add_row(row); |
|
row.clear(); |
|
} |
|
state = State::IDLE; |
|
|
|
view->get_buffer()->begin_user_action(); |
|
CompletionDialog::get()->show(); |
|
} |
|
}); |
|
} |
|
else { |
|
dispatcher.post([this] { |
|
state = State::CANCELED; |
|
on_add_rows_error(); |
|
}); |
|
} |
|
}); |
|
} |
|
|
|
if(state != State::IDLE) |
|
cancel_reparse(); |
|
} |
|
|
|
void Autocomplete::stop() { |
|
if(state == State::STARTING || state == State::RESTARTING) |
|
state = State::CANCELED; |
|
} |
|
|
|
void Autocomplete::setup_dialog() { |
|
CompletionDialog::get()->on_show = [this] { |
|
on_show(); |
|
}; |
|
|
|
CompletionDialog::get()->on_hide = [this]() { |
|
view->get_buffer()->end_user_action(); |
|
tooltips.hide(); |
|
tooltips.clear(); |
|
on_hide(); |
|
reparse(); |
|
}; |
|
|
|
CompletionDialog::get()->on_changed = [this](unsigned int index, const std::string &text) { |
|
if(index >= rows.size()) { |
|
tooltips.hide(); |
|
return; |
|
} |
|
|
|
on_changed(index, text); |
|
|
|
auto tooltip = get_tooltip(index); |
|
if(tooltip.empty()) |
|
tooltips.hide(); |
|
else { |
|
tooltips.clear(); |
|
auto iter = CompletionDialog::get()->start_mark->get_iter(); |
|
tooltips.emplace_back(view, view->get_buffer()->create_mark(iter), view->get_buffer()->create_mark(iter), [tooltip = std::move(tooltip)](const Glib::RefPtr<Gtk::TextBuffer> &buffer) { |
|
buffer->insert(buffer->get_insert()->get_iter(), tooltip); |
|
}); |
|
|
|
tooltips.show(true); |
|
} |
|
}; |
|
|
|
CompletionDialog::get()->on_select = [this](unsigned int index, const std::string &text, bool hide_window) { |
|
if(index >= rows.size()) |
|
return; |
|
|
|
on_select(index, text, hide_window); |
|
}; |
|
}
|
|
|