|
|
|
|
#include "window.h"
|
|
|
|
|
#include "config.h"
|
|
|
|
|
#include "menu.h"
|
|
|
|
|
#include "notebook.h"
|
|
|
|
|
#include "directories.h"
|
|
|
|
|
#include "dialogs.h"
|
|
|
|
|
#include "filesystem.h"
|
|
|
|
|
#include "project.h"
|
|
|
|
|
#include "entrybox.h"
|
|
|
|
|
#include "info.h"
|
|
|
|
|
#include "ctags.h"
|
|
|
|
|
|
|
|
|
|
namespace sigc {
|
|
|
|
|
#ifndef SIGC_FUNCTORS_DEDUCE_RESULT_TYPE_WITH_DECLTYPE
|
|
|
|
|
template <typename Functor>
|
|
|
|
|
struct functor_trait<Functor, false> {
|
|
|
|
|
typedef decltype (::sigc::mem_fun(std::declval<Functor&>(),
|
|
|
|
|
&Functor::operator())) _intermediate;
|
|
|
|
|
typedef typename _intermediate::result_type result_type;
|
|
|
|
|
typedef Functor functor_type;
|
|
|
|
|
};
|
|
|
|
|
#else
|
|
|
|
|
SIGC_FUNCTORS_DEDUCE_RESULT_TYPE_WITH_DECLTYPE
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Window::Window() {
|
|
|
|
|
set_title("juCi++");
|
|
|
|
|
set_events(Gdk::POINTER_MOTION_MASK|Gdk::FOCUS_CHANGE_MASK|Gdk::SCROLL_MASK|Gdk::LEAVE_NOTIFY_MASK);
|
|
|
|
|
|
|
|
|
|
set_menu_actions();
|
|
|
|
|
configure();
|
|
|
|
|
activate_menu_items();
|
|
|
|
|
|
|
|
|
|
set_default_size(Config::get().window.default_size.first, Config::get().window.default_size.second);
|
|
|
|
|
|
|
|
|
|
auto directories_scrolled_window=Gtk::manage(new Gtk::ScrolledWindow());
|
|
|
|
|
directories_scrolled_window->add(Directories::get());
|
|
|
|
|
|
|
|
|
|
auto notebook_vbox=Gtk::manage(new Gtk::VBox());
|
|
|
|
|
notebook_vbox->pack_start(Notebook::get());
|
|
|
|
|
notebook_vbox->pack_end(EntryBox::get(), Gtk::PACK_SHRINK);
|
|
|
|
|
|
|
|
|
|
auto terminal_scrolled_window=Gtk::manage(new Gtk::ScrolledWindow());
|
|
|
|
|
terminal_scrolled_window->add(Terminal::get());
|
|
|
|
|
|
|
|
|
|
auto notebook_and_terminal_vpaned=Gtk::manage(new Gtk::VPaned());
|
|
|
|
|
notebook_and_terminal_vpaned->set_position(static_cast<int>(0.75*Config::get().window.default_size.second));
|
|
|
|
|
notebook_and_terminal_vpaned->pack1(*notebook_vbox, Gtk::SHRINK);
|
|
|
|
|
notebook_and_terminal_vpaned->pack2(*terminal_scrolled_window, Gtk::SHRINK);
|
|
|
|
|
|
|
|
|
|
auto hpaned=Gtk::manage(new Gtk::HPaned());
|
|
|
|
|
hpaned->set_position(static_cast<int>(0.2*Config::get().window.default_size.first));
|
|
|
|
|
hpaned->pack1(*directories_scrolled_window, Gtk::SHRINK);
|
|
|
|
|
hpaned->pack2(*notebook_and_terminal_vpaned, Gtk::SHRINK);
|
|
|
|
|
|
|
|
|
|
auto status_hbox=Gtk::manage(new Gtk::HBox());
|
|
|
|
|
status_hbox->set_homogeneous(true);
|
|
|
|
|
status_hbox->pack_start(*Gtk::manage(new Gtk::HBox()));
|
|
|
|
|
auto status_right_hbox=Gtk::manage(new Gtk::HBox());
|
|
|
|
|
status_right_hbox->pack_end(Notebook::get().status_state, Gtk::PACK_SHRINK);
|
|
|
|
|
auto status_right_overlay=Gtk::manage(new Gtk::Overlay());
|
|
|
|
|
status_right_overlay->add(*status_right_hbox);
|
|
|
|
|
status_right_overlay->add_overlay(Notebook::get().status_diagnostics);
|
|
|
|
|
status_hbox->pack_end(*status_right_overlay);
|
|
|
|
|
|
|
|
|
|
auto status_overlay=Gtk::manage(new Gtk::Overlay());
|
|
|
|
|
status_overlay->add(*status_hbox);
|
|
|
|
|
auto status_file_info_hbox=Gtk::manage(new Gtk::HBox);
|
|
|
|
|
status_file_info_hbox->pack_start(Notebook::get().status_file_path, Gtk::PACK_SHRINK);
|
|
|
|
|
status_file_info_hbox->pack_start(Notebook::get().status_branch, Gtk::PACK_SHRINK);
|
|
|
|
|
status_file_info_hbox->pack_start(Notebook::get().status_location, Gtk::PACK_SHRINK);
|
|
|
|
|
status_overlay->add_overlay(*status_file_info_hbox);
|
|
|
|
|
status_overlay->add_overlay(Project::debug_status_label());
|
|
|
|
|
|
|
|
|
|
auto vbox=Gtk::manage(new Gtk::VBox());
|
|
|
|
|
vbox->pack_start(*hpaned);
|
|
|
|
|
vbox->pack_start(*status_overlay, Gtk::PACK_SHRINK);
|
|
|
|
|
|
|
|
|
|
auto overlay_vbox=Gtk::manage(new Gtk::VBox());
|
|
|
|
|
auto overlay_hbox=Gtk::manage(new Gtk::HBox());
|
|
|
|
|
overlay_vbox->set_hexpand(false);
|
|
|
|
|
overlay_vbox->set_halign(Gtk::Align::ALIGN_START);
|
|
|
|
|
overlay_vbox->pack_start(Info::get(), Gtk::PACK_SHRINK, 20);
|
|
|
|
|
overlay_hbox->set_hexpand(false);
|
|
|
|
|
overlay_hbox->set_halign(Gtk::Align::ALIGN_END);
|
|
|
|
|
overlay_hbox->pack_end(*overlay_vbox, Gtk::PACK_SHRINK, 20);
|
|
|
|
|
|
|
|
|
|
auto overlay=Gtk::manage(new Gtk::Overlay());
|
|
|
|
|
overlay->add(*vbox);
|
|
|
|
|
overlay->add_overlay(*overlay_hbox);
|
|
|
|
|
#if GTKMM_MAJOR_VERSION>3 || (GTKMM_MAJOR_VERSION==3 && GTKMM_MINOR_VERSION>=18)
|
|
|
|
|
overlay->set_overlay_pass_through(*overlay_hbox, true);
|
|
|
|
|
#endif
|
|
|
|
|
add(*overlay);
|
|
|
|
|
|
|
|
|
|
show_all_children();
|
|
|
|
|
Info::get().hide();
|
|
|
|
|
|
|
|
|
|
//Scroll to end of terminal whenever info is printed
|
|
|
|
|
Terminal::get().signal_size_allocate().connect([terminal_scrolled_window](Gtk::Allocation& allocation){
|
|
|
|
|
auto adjustment=terminal_scrolled_window->get_vadjustment();
|
|
|
|
|
adjustment->set_value(adjustment->get_upper()-adjustment->get_page_size());
|
|
|
|
|
Terminal::get().queue_draw();
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
EntryBox::get().signal_show().connect([this, hpaned, notebook_and_terminal_vpaned, notebook_vbox](){
|
|
|
|
|
hpaned->set_focus_chain({notebook_and_terminal_vpaned});
|
|
|
|
|
notebook_and_terminal_vpaned->set_focus_chain({notebook_vbox});
|
|
|
|
|
notebook_vbox->set_focus_chain({&EntryBox::get()});
|
|
|
|
|
});
|
|
|
|
|
EntryBox::get().signal_hide().connect([this, hpaned, notebook_and_terminal_vpaned, notebook_vbox](){
|
|
|
|
|
hpaned->unset_focus_chain();
|
|
|
|
|
notebook_and_terminal_vpaned->unset_focus_chain();
|
|
|
|
|
notebook_vbox->unset_focus_chain();
|
|
|
|
|
});
|
|
|
|
|
EntryBox::get().signal_hide().connect([this]() {
|
|
|
|
|
if(auto view=Notebook::get().get_current_view())
|
|
|
|
|
view->grab_focus();
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
Notebook::get().on_change_page=[this](Source::View *view) {
|
|
|
|
|
if(search_entry_shown && EntryBox::get().labels.size()>0) {
|
|
|
|
|
view->update_search_occurrences=[this](int number){
|
|
|
|
|
EntryBox::get().labels.begin()->update(0, std::to_string(number));
|
|
|
|
|
};
|
|
|
|
|
view->search_highlight(last_search, case_sensitive_search, regex_search);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
activate_menu_items();
|
|
|
|
|
|
|
|
|
|
Directories::get().select(view->file_path);
|
|
|
|
|
|
|
|
|
|
if(view->full_reparse_needed)
|
|
|
|
|
view->full_reparse();
|
|
|
|
|
else if(view->soft_reparse_needed)
|
|
|
|
|
view->soft_reparse();
|
|
|
|
|
|
|
|
|
|
Notebook::get().update_status(view);
|
|
|
|
|
|
|
|
|
|
#ifdef JUCI_ENABLE_DEBUG
|
|
|
|
|
if(Project::debugging)
|
|
|
|
|
Project::debug_update_stop();
|
|
|
|
|
#endif
|
|
|
|
|
};
|
|
|
|
|
Notebook::get().on_close_page=[this](Source::View *view) {
|
|
|
|
|
#ifdef JUCI_ENABLE_DEBUG
|
|
|
|
|
if(Project::current && Project::debugging) {
|
|
|
|
|
auto iter=view->get_buffer()->begin();
|
|
|
|
|
while(view->get_source_buffer()->forward_iter_to_source_mark(iter, "debug_breakpoint") ||
|
|
|
|
|
view->get_source_buffer()->get_source_marks_at_iter(iter, "debug_breakpoint").size()) {
|
|
|
|
|
auto end_iter=view->get_iter_at_line_end(iter.get_line());
|
|
|
|
|
view->get_source_buffer()->remove_source_marks(iter, end_iter, "debug_breakpoint");
|
|
|
|
|
Project::current->debug_remove_breakpoint(view->file_path, iter.get_line()+1, view->get_buffer()->get_line_count()+1);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
EntryBox::get().hide();
|
|
|
|
|
if(auto view=Notebook::get().get_current_view())
|
|
|
|
|
Notebook::get().update_status(view);
|
|
|
|
|
else {
|
|
|
|
|
Notebook::get().clear_status();
|
|
|
|
|
|
|
|
|
|
activate_menu_items();
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
signal_focus_out_event().connect([](GdkEventFocus *event) {
|
|
|
|
|
if(auto view=Notebook::get().get_current_view()) {
|
|
|
|
|
view->hide_tooltips();
|
|
|
|
|
view->hide_dialogs();
|
|
|
|
|
}
|
|
|
|
|
return false;
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
Gtk::Settings::get_default()->connect_property_changed("gtk-theme-name", [this] {
|
|
|
|
|
Directories::get().update();
|
|
|
|
|
if(auto view=Notebook::get().get_current_view())
|
|
|
|
|
Notebook::get().update_status(view);
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
about.signal_response().connect([this](int d){
|
|
|
|
|
about.hide();
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
about.set_version(Config::get().window.version);
|
|
|
|
|
about.set_authors({"(in order of appearance)",
|
|
|
|
|
"Ted Johan Kristoffersen",
|
|
|
|
|
"Jørgen Lien Sellæg",
|
|
|
|
|
"Geir Morten Larsen",
|
|
|
|
|
"Ole Christian Eidheim"});
|
|
|
|
|
about.set_name("About juCi++");
|
|
|
|
|
about.set_program_name("juCi++");
|
|
|
|
|
about.set_comments("This is an open source IDE with high-end features to make your programming experience juicy");
|
|
|
|
|
about.set_license_type(Gtk::License::LICENSE_MIT_X11);
|
|
|
|
|
about.set_transient_for(*this);
|
|
|
|
|
} // Window constructor
|
|
|
|
|
|
|
|
|
|
void Window::configure() {
|
|
|
|
|
Config::get().load();
|
|
|
|
|
auto screen = Gdk::Screen::get_default();
|
|
|
|
|
if(css_provider)
|
|
|
|
|
Gtk::StyleContext::remove_provider_for_screen(screen, css_provider);
|
|
|
|
|
if(Config::get().window.theme_name.empty()) {
|
|
|
|
|
css_provider=Gtk::CssProvider::create();
|
|
|
|
|
Gtk::Settings::get_default()->property_gtk_application_prefer_dark_theme()=(Config::get().window.theme_variant=="dark");
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
css_provider=Gtk::CssProvider::get_named(Config::get().window.theme_name, Config::get().window.theme_variant);
|
|
|
|
|
//TODO: add check if theme exists, or else write error to terminal
|
|
|
|
|
Gtk::StyleContext::add_provider_for_screen(screen, css_provider, GTK_STYLE_PROVIDER_PRIORITY_SETTINGS);
|
|
|
|
|
Menu::get().set_keys();
|
|
|
|
|
Terminal::get().configure();
|
|
|
|
|
Directories::get().update();
|
|
|
|
|
if(auto view=Notebook::get().get_current_view())
|
|
|
|
|
Notebook::get().update_status(view);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Window::set_menu_actions() {
|
|
|
|
|
auto &menu = Menu::get();
|
|
|
|
|
|
|
|
|
|
menu.add_action("about", [this]() {
|
|
|
|
|
about.show();
|
|
|
|
|
about.present();
|
|
|
|
|
});
|
|
|
|
|
menu.add_action("preferences", [this]() {
|
|
|
|
|
Notebook::get().open(Config::get().home_juci_path/"config"/"config.json");
|
|
|
|
|
});
|
|
|
|
|
menu.add_action("quit", [this]() {
|
|
|
|
|
close();
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
menu.add_action("new_file", [this]() {
|
|
|
|
|
boost::filesystem::path path = Dialog::new_file(Notebook::get().get_current_folder());
|
|
|
|
|
if(path!="") {
|
|
|
|
|
if(boost::filesystem::exists(path)) {
|
|
|
|
|
Terminal::get().print("Error: "+path.string()+" already exists.\n", true);
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
if(filesystem::write(path)) {
|
|
|
|
|
if(Directories::get().path!="")
|
|
|
|
|
Directories::get().update();
|
|
|
|
|
Notebook::get().open(path);
|
|
|
|
|
if(Directories::get().path!="")
|
|
|
|
|
Directories::get().on_save_file(path);
|
|
|
|
|
Terminal::get().print("New file "+path.string()+" created.\n");
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
Terminal::get().print("Error: could not create new file "+path.string()+".\n", true);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
menu.add_action("new_folder", [this]() {
|
|
|
|
|
auto time_now=std::chrono::system_clock::to_time_t(std::chrono::system_clock::now());
|
|
|
|
|
boost::filesystem::path path = Dialog::new_folder(Notebook::get().get_current_folder());
|
|
|
|
|
if(path!="" && boost::filesystem::exists(path)) {
|
|
|
|
|
boost::system::error_code ec;
|
|
|
|
|
auto last_write_time=boost::filesystem::last_write_time(path, ec);
|
|
|
|
|
if(!ec && last_write_time>=time_now) {
|
|
|
|
|
if(Directories::get().path!="")
|
|
|
|
|
Directories::get().update();
|
|
|
|
|
Terminal::get().print("New folder "+path.string()+" created.\n");
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
Terminal::get().print("Error: "+path.string()+" already exists.\n", true);
|
|
|
|
|
Directories::get().select(path);
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
menu.add_action("new_project_c", [this]() {
|
|
|
|
|
boost::filesystem::path project_path = Dialog::new_folder(Notebook::get().get_current_folder());
|
|
|
|
|
if(project_path!="") {
|
|
|
|
|
auto project_name=project_path.filename().string();
|
|
|
|
|
for(size_t c=0;c<project_name.size();c++) {
|
|
|
|
|
if(project_name[c]==' ')
|
|
|
|
|
project_name[c]='_';
|
|
|
|
|
}
|
|
|
|
|
auto cmakelists_path=project_path;
|
|
|
|
|
cmakelists_path/="CMakeLists.txt";
|
|
|
|
|
auto c_main_path=project_path;
|
|
|
|
|
c_main_path/="main.c";
|
|
|
|
|
if(boost::filesystem::exists(cmakelists_path)) {
|
|
|
|
|
Terminal::get().print("Error: "+cmakelists_path.string()+" already exists.\n", true);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if(boost::filesystem::exists(c_main_path)) {
|
|
|
|
|
Terminal::get().print("Error: "+c_main_path.string()+" already exists.\n", true);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
std::string cmakelists="cmake_minimum_required(VERSION 2.8)\n\nproject("+project_name+")\n\nset(CMAKE_C_FLAGS \"${CMAKE_C_FLAGS} -std=c11 -Wall -Wextra\")\n\nadd_executable("+project_name+" main.c)\n";
|
|
|
|
|
std::string c_main="#include <stdio.h>\n\nint main() {\n printf(\"Hello World!\\n\");\n}\n";
|
|
|
|
|
if(filesystem::write(cmakelists_path, cmakelists) && filesystem::write(c_main_path, c_main)) {
|
|
|
|
|
Directories::get().open(project_path);
|
|
|
|
|
Notebook::get().open(c_main_path);
|
|
|
|
|
Directories::get().update();
|
|
|
|
|
Terminal::get().print("C project "+project_name+" created.\n");
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
Terminal::get().print("Error: Could not create project "+project_path.string()+"\n", true);
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
menu.add_action("new_project_cpp", [this]() {
|
|
|
|
|
boost::filesystem::path project_path = Dialog::new_folder(Notebook::get().get_current_folder());
|
|
|
|
|
if(project_path!="") {
|
|
|
|
|
auto project_name=project_path.filename().string();
|
|
|
|
|
for(size_t c=0;c<project_name.size();c++) {
|
|
|
|
|
if(project_name[c]==' ')
|
|
|
|
|
project_name[c]='_';
|
|
|
|
|
}
|
|
|
|
|
auto cmakelists_path=project_path;
|
|
|
|
|
cmakelists_path/="CMakeLists.txt";
|
|
|
|
|
auto cpp_main_path=project_path;
|
|
|
|
|
cpp_main_path/="main.cpp";
|
|
|
|
|
if(boost::filesystem::exists(cmakelists_path)) {
|
|
|
|
|
Terminal::get().print("Error: "+cmakelists_path.string()+" already exists.\n", true);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if(boost::filesystem::exists(cpp_main_path)) {
|
|
|
|
|
Terminal::get().print("Error: "+cpp_main_path.string()+" already exists.\n", true);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
std::string cmakelists="cmake_minimum_required(VERSION 2.8)\n\nproject("+project_name+")\n\nset(CMAKE_CXX_FLAGS \"${CMAKE_CXX_FLAGS} -std=c++1y -Wall -Wextra -Wno-unused-parameter\")\n\nadd_executable("+project_name+" main.cpp)\n";
|
|
|
|
|
std::string cpp_main="#include <iostream>\n\nint main() {\n std::cout << \"Hello World!\\n\";\n}\n";
|
|
|
|
|
if(filesystem::write(cmakelists_path, cmakelists) && filesystem::write(cpp_main_path, cpp_main)) {
|
|
|
|
|
Directories::get().open(project_path);
|
|
|
|
|
Notebook::get().open(cpp_main_path);
|
|
|
|
|
Directories::get().update();
|
|
|
|
|
Terminal::get().print("C++ project "+project_name+" created.\n");
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
Terminal::get().print("Error: Could not create project "+project_path.string()+"\n", true);
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
menu.add_action("open_file", [this]() {
|
|
|
|
|
auto folder_path=Notebook::get().get_current_folder();
|
|
|
|
|
if(auto view=Notebook::get().get_current_view()) {
|
|
|
|
|
if(!Directories::get().path.empty() && !filesystem::file_in_path(view->file_path, Directories::get().path))
|
|
|
|
|
folder_path=view->file_path.parent_path();
|
|
|
|
|
}
|
|
|
|
|
auto path=Dialog::open_file(folder_path);
|
|
|
|
|
if(path!="")
|
|
|
|
|
Notebook::get().open(path);
|
|
|
|
|
});
|
|
|
|
|
menu.add_action("open_folder", [this]() {
|
|
|
|
|
auto path = Dialog::open_folder(Notebook::get().get_current_folder());
|
|
|
|
|
if (path!="" && boost::filesystem::exists(path))
|
|
|
|
|
Directories::get().open(path);
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
menu.add_action("save", [this]() {
|
|
|
|
|
if(auto view=Notebook::get().get_current_view()) {
|
|
|
|
|
if(Notebook::get().save_current()) {
|
|
|
|
|
if(view->file_path==Config::get().home_juci_path/"config"/"config.json") {
|
|
|
|
|
configure();
|
|
|
|
|
for(size_t c=0;c<Notebook::get().size();c++) {
|
|
|
|
|
Notebook::get().get_view(c)->configure();
|
|
|
|
|
Notebook::get().configure(c);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
menu.add_action("save_as", [this]() {
|
|
|
|
|
if(auto view=Notebook::get().get_current_view()) {
|
|
|
|
|
auto path = Dialog::save_file_as(view->file_path);
|
|
|
|
|
if(path!="") {
|
|
|
|
|
std::ofstream file(path, std::ofstream::binary);
|
|
|
|
|
if(file) {
|
|
|
|
|
file << view->get_buffer()->get_text().raw();
|
|
|
|
|
file.close();
|
|
|
|
|
if(Directories::get().path!="")
|
|
|
|
|
Directories::get().update();
|
|
|
|
|
Notebook::get().open(path);
|
|
|
|
|
Terminal::get().print("File saved to: " + Notebook::get().get_current_view()->file_path.string()+"\n");
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
Terminal::get().print("Error saving file\n", true);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
menu.add_action("print", [this]() {
|
|
|
|
|
if(auto view=Notebook::get().get_current_view()) {
|
|
|
|
|
auto print_operation=Gtk::PrintOperation::create();
|
|
|
|
|
auto print_compositor=Gsv::PrintCompositor::create(*view);
|
|
|
|
|
|
|
|
|
|
print_operation->set_job_name(view->file_path.filename().string());
|
|
|
|
|
print_compositor->set_wrap_mode(Gtk::WrapMode::WRAP_WORD_CHAR);
|
|
|
|
|
|
|
|
|
|
print_operation->signal_begin_print().connect([print_operation, print_compositor](const Glib::RefPtr<Gtk::PrintContext>& print_context) {
|
|
|
|
|
while(!print_compositor->paginate(print_context));
|
|
|
|
|
print_operation->set_n_pages(print_compositor->get_n_pages());
|
|
|
|
|
});
|
|
|
|
|
print_operation->signal_draw_page().connect([print_compositor](const Glib::RefPtr<Gtk::PrintContext>& print_context, int page_nr) {
|
|
|
|
|
print_compositor->draw_page(print_context, page_nr);
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
print_operation->run(Gtk::PRINT_OPERATION_ACTION_PRINT_DIALOG, *this);
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
menu.add_action("edit_undo", [this]() {
|
|
|
|
|
if(auto view=Notebook::get().get_current_view()) {
|
|
|
|
|
auto undo_manager = view->get_source_buffer()->get_undo_manager();
|
|
|
|
|
if (undo_manager->can_undo()) {
|
|
|
|
|
undo_manager->undo();
|
|
|
|
|
view->scroll_to(view->get_buffer()->get_insert());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
menu.add_action("edit_redo", [this]() {
|
|
|
|
|
if(auto view=Notebook::get().get_current_view()) {
|
|
|
|
|
auto undo_manager = view->get_source_buffer()->get_undo_manager();
|
|
|
|
|
if(undo_manager->can_redo()) {
|
|
|
|
|
undo_manager->redo();
|
|
|
|
|
view->scroll_to(view->get_buffer()->get_insert());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
menu.add_action("edit_cut", [this]() {
|
|
|
|
|
auto widget=get_focus();
|
|
|
|
|
if(auto entry=dynamic_cast<Gtk::Entry*>(widget))
|
|
|
|
|
entry->cut_clipboard();
|
|
|
|
|
else if(auto view=Notebook::get().get_current_view())
|
|
|
|
|
view->get_buffer()->cut_clipboard(Gtk::Clipboard::get());
|
|
|
|
|
});
|
|
|
|
|
menu.add_action("edit_copy", [this]() {
|
|
|
|
|
auto widget=get_focus();
|
|
|
|
|
if(auto entry=dynamic_cast<Gtk::Entry*>(widget))
|
|
|
|
|
entry->copy_clipboard();
|
|
|
|
|
else if(auto text_view=dynamic_cast<Gtk::TextView*>(widget))
|
|
|
|
|
text_view->get_buffer()->copy_clipboard(Gtk::Clipboard::get());
|
|
|
|
|
});
|
|
|
|
|
menu.add_action("edit_paste", [this]() {
|
|
|
|
|
auto widget=get_focus();
|
|
|
|
|
if(auto entry=dynamic_cast<Gtk::Entry*>(widget))
|
|
|
|
|
entry->paste_clipboard();
|
|
|
|
|
else if(auto view=Notebook::get().get_current_view())
|
|
|
|
|
view->paste();
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
menu.add_action("edit_find", [this]() {
|
|
|
|
|
search_and_replace_entry();
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
menu.add_action("source_spellcheck", [this]() {
|
|
|
|
|
if(auto view=Notebook::get().get_current_view()) {
|
|
|
|
|
view->remove_spellcheck_errors();
|
|
|
|
|
view->spellcheck();
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
menu.add_action("source_spellcheck_clear", [this]() {
|
|
|
|
|
if(auto view=Notebook::get().get_current_view())
|
|
|
|
|
view->remove_spellcheck_errors();
|
|
|
|
|
});
|
|
|
|
|
menu.add_action("source_spellcheck_next_error", [this]() {
|
|
|
|
|
if(auto view=Notebook::get().get_current_view())
|
|
|
|
|
view->goto_next_spellcheck_error();
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
menu.add_action("source_git_next_diff", [this]() {
|
|
|
|
|
if(auto view=Notebook::get().get_current_view())
|
|
|
|
|
view->git_goto_next_diff();
|
|
|
|
|
});
|
|
|
|
|
menu.add_action("source_git_show_diff", [this]() {
|
|
|
|
|
if(auto view=Notebook::get().get_current_view()) {
|
|
|
|
|
auto diff_details=view->git_get_diff_details();
|
|
|
|
|
if(diff_details.empty())
|
|
|
|
|
return;
|
|
|
|
|
std::string postfix;
|
|
|
|
|
if(diff_details.size()>2) {
|
|
|
|
|
size_t pos=diff_details.find("@@", 2);
|
|
|
|
|
if(pos!=std::string::npos)
|
|
|
|
|
postfix=diff_details.substr(0, pos+2);
|
|
|
|
|
}
|
|
|
|
|
Notebook::get().open(view->file_path.string()+postfix+".diff");
|
|
|
|
|
if(auto new_view=Notebook::get().get_current_view()) {
|
|
|
|
|
if(new_view->get_buffer()->get_text().empty()) {
|
|
|
|
|
new_view->get_source_buffer()->begin_not_undoable_action();
|
|
|
|
|
new_view->get_buffer()->set_text(diff_details);
|
|
|
|
|
new_view->get_source_buffer()->end_not_undoable_action();
|
|
|
|
|
new_view->get_buffer()->set_modified(false);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
menu.add_action("source_indentation_set_buffer_tab", [this]() {
|
|
|
|
|
set_tab_entry();
|
|
|
|
|
});
|
|
|
|
|
menu.add_action("source_indentation_auto_indent_buffer", [this]() {
|
|
|
|
|
auto view=Notebook::get().get_current_view();
|
|
|
|
|
if(view && view->format_style)
|
|
|
|
|
view->format_style();
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
menu.add_action("source_goto_line", [this]() {
|
|
|
|
|
goto_line_entry();
|
|
|
|
|
});
|
|
|
|
|
menu.add_action("source_center_cursor", [this]() {
|
|
|
|
|
if(auto view=Notebook::get().get_current_view())
|
|
|
|
|
view->scroll_to_cursor_delayed(view, true, false);
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
menu.add_action("source_find_symbol_ctags", [this]() {
|
|
|
|
|
if(auto view=Notebook::get().get_current_view()) {
|
|
|
|
|
auto pair=Ctags::get_result(view->file_path.parent_path());
|
|
|
|
|
auto path=std::move(pair.first);
|
|
|
|
|
auto stream=std::move(pair.second);
|
|
|
|
|
stream->seekg(0, std::ios::end);
|
|
|
|
|
if(stream->tellg()==0) {
|
|
|
|
|
Info::get().print("No symbols found in current project");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
stream->seekg(0, std::ios::beg);
|
|
|
|
|
|
|
|
|
|
auto dialog_iter=view->get_iter_for_dialog();
|
|
|
|
|
view->selection_dialog=std::make_unique<SelectionDialog>(*view, view->get_buffer()->create_mark(dialog_iter), true, true);
|
|
|
|
|
auto rows=std::make_shared<std::unordered_map<std::string, Source::Offset> >();
|
|
|
|
|
|
|
|
|
|
std::string line;
|
|
|
|
|
while(std::getline(*stream, line)) {
|
|
|
|
|
auto location=Ctags::get_location(line, true);
|
|
|
|
|
|
|
|
|
|
std::string row=location.file_path.string()+":"+std::to_string(location.line+1)+": "+location.source;
|
|
|
|
|
(*rows)[row]=Source::Offset(location.line, location.index, location.file_path);
|
|
|
|
|
view->selection_dialog->add_row(row);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(rows->size()==0)
|
|
|
|
|
return;
|
|
|
|
|
view->selection_dialog->on_select=[this, rows, path](const std::string &selected, bool hide_window) {
|
|
|
|
|
auto offset=rows->at(selected);
|
|
|
|
|
boost::filesystem::path declaration_file;
|
|
|
|
|
boost::system::error_code ec;
|
|
|
|
|
declaration_file=boost::filesystem::canonical(path/offset.file_path, ec);
|
|
|
|
|
if(ec)
|
|
|
|
|
return;
|
|
|
|
|
Notebook::get().open(declaration_file);
|
|
|
|
|
auto view=Notebook::get().get_current_view();
|
|
|
|
|
view->place_cursor_at_line_index(offset.line, offset.index);
|
|
|
|
|
view->scroll_to_cursor_delayed(view, true, false);
|
|
|
|
|
view->hide_tooltips();
|
|
|
|
|
};
|
|
|
|
|
view->hide_tooltips();
|
|
|
|
|
view->selection_dialog->show();
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
menu.add_action("source_comments_toggle", [this]() {
|
|
|
|
|
if(auto view=Notebook::get().get_current_view()) {
|
|
|
|
|
if(view->toggle_comments) {
|
|
|
|
|
view->toggle_comments();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
menu.add_action("source_comments_add_documentation", [this]() {
|
|
|
|
|
if(auto view=Notebook::get().get_current_view()) {
|
|
|
|
|
if(view->add_documentation) {
|
|
|
|
|
view->add_documentation();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
menu.add_action("source_find_documentation", [this]() {
|
|
|
|
|
if(auto view=Notebook::get().get_current_view()) {
|
|
|
|
|
if(view->get_token_data) {
|
|
|
|
|
auto data=view->get_token_data();
|
|
|
|
|
if(data.size()>0) {
|
|
|
|
|
auto documentation_search=Config::get().source.documentation_searches.find(data[0]);
|
|
|
|
|
if(documentation_search!=Config::get().source.documentation_searches.end()) {
|
|
|
|
|
std::string token_query;
|
|
|
|
|
for(size_t c=1;c<data.size();c++) {
|
|
|
|
|
if(data[c].size()>0) {
|
|
|
|
|
if(token_query.size()>0)
|
|
|
|
|
token_query+=documentation_search->second.separator;
|
|
|
|
|
token_query+=data[c];
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if(token_query.size()>0) {
|
|
|
|
|
std::unordered_map<std::string, std::string>::iterator query;
|
|
|
|
|
if(data[1].size()>0)
|
|
|
|
|
query=documentation_search->second.queries.find(data[1]);
|
|
|
|
|
else
|
|
|
|
|
query=documentation_search->second.queries.find("@empty");
|
|
|
|
|
if(query==documentation_search->second.queries.end())
|
|
|
|
|
query=documentation_search->second.queries.find("@any");
|
|
|
|
|
|
|
|
|
|
if(query!=documentation_search->second.queries.end()) {
|
|
|
|
|
std::string uri=query->second+token_query;
|
|
|
|
|
#ifdef __APPLE__
|
|
|
|
|
Terminal::get().process("open \""+uri+"\"");
|
|
|
|
|
#else
|
|
|
|
|
GError* error=nullptr;
|
|
|
|
|
gtk_show_uri(nullptr, uri.c_str(), GDK_CURRENT_TIME, &error);
|
|
|
|
|
g_clear_error(&error);
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
menu.add_action("source_goto_declaration", [this]() {
|
|
|
|
|
if(auto view=Notebook::get().get_current_view()) {
|
|
|
|
|
if(view->get_declaration_location) {
|
|
|
|
|
auto location=view->get_declaration_location();
|
|
|
|
|
if(location) {
|
|
|
|
|
boost::filesystem::path declaration_file;
|
|
|
|
|
boost::system::error_code ec;
|
|
|
|
|
declaration_file=boost::filesystem::canonical(location.file_path, ec);
|
|
|
|
|
if(ec)
|
|
|
|
|
return;
|
|
|
|
|
Notebook::get().open(declaration_file);
|
|
|
|
|
auto view=Notebook::get().get_current_view();
|
|
|
|
|
auto line=static_cast<int>(location.line);
|
|
|
|
|
auto index=static_cast<int>(location.index);
|
|
|
|
|
view->place_cursor_at_line_index(line, index);
|
|
|
|
|
view->scroll_to_cursor_delayed(view, true, false);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
menu.add_action("source_goto_implementation", [this]() {
|
|
|
|
|
if(auto view=Notebook::get().get_current_view()) {
|
|
|
|
|
if(view->get_implementation_locations) {
|
|
|
|
|
auto locations=view->get_implementation_locations(Notebook::get().get_views());
|
|
|
|
|
if(!locations.empty()) {
|
|
|
|
|
auto dialog_iter=view->get_iter_for_dialog();
|
|
|
|
|
view->selection_dialog=std::make_unique<SelectionDialog>(*view, view->get_buffer()->create_mark(dialog_iter), true, true);
|
|
|
|
|
auto rows=std::make_shared<std::unordered_map<std::string, Source::Offset> >();
|
|
|
|
|
auto project_path=Project::Build::create(view->file_path)->project_path;
|
|
|
|
|
if(project_path.empty()) {
|
|
|
|
|
if(!Directories::get().path.empty())
|
|
|
|
|
project_path=Directories::get().path;
|
|
|
|
|
else
|
|
|
|
|
project_path=view->file_path.parent_path();
|
|
|
|
|
}
|
|
|
|
|
for(auto &location: locations) {
|
|
|
|
|
boost::filesystem::path implementation_path;
|
|
|
|
|
boost::system::error_code ec;
|
|
|
|
|
implementation_path=boost::filesystem::canonical(location.file_path, ec);
|
|
|
|
|
if(!ec) {
|
|
|
|
|
location.file_path=implementation_path;
|
|
|
|
|
auto path=filesystem::get_relative_path(location.file_path, project_path);
|
|
|
|
|
if(path.empty())
|
|
|
|
|
path=location.file_path.filename();
|
|
|
|
|
auto row=path.string()+":"+std::to_string(location.line+1);
|
|
|
|
|
(*rows)[row]=location;
|
|
|
|
|
view->selection_dialog->add_row(row);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(rows->size()==0)
|
|
|
|
|
return;
|
|
|
|
|
else if(rows->size()==1) {
|
|
|
|
|
auto location=*rows->begin();
|
|
|
|
|
Notebook::get().open(location.second.file_path);
|
|
|
|
|
auto view=Notebook::get().get_current_view();
|
|
|
|
|
auto line=static_cast<int>(location.second.line);
|
|
|
|
|
auto index=static_cast<int>(location.second.index);
|
|
|
|
|
view->place_cursor_at_line_index(line, index);
|
|
|
|
|
view->scroll_to_cursor_delayed(view, true, false);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
view->selection_dialog->on_select=[this, rows](const std::string &selected, bool hide_window) {
|
|
|
|
|
auto location=rows->at(selected);
|
|
|
|
|
Notebook::get().open(location.file_path);
|
|
|
|
|
auto view=Notebook::get().get_current_view();
|
|
|
|
|
view->place_cursor_at_line_index(location.line, location.index);
|
|
|
|
|
view->scroll_to_cursor_delayed(view, true, false);
|
|
|
|
|
view->hide_tooltips();
|
|
|
|
|
};
|
|
|
|
|
view->hide_tooltips();
|
|
|
|
|
view->selection_dialog->show();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
menu.add_action("source_goto_usage", [this]() {
|
|
|
|
|
if(auto view=Notebook::get().get_current_view()) {
|
|
|
|
|
if(view->get_usages) {
|
|
|
|
|
auto usages=view->get_usages(Notebook::get().get_views());
|
|
|
|
|
if(!usages.empty()) {
|
|
|
|
|
auto dialog_iter=view->get_iter_for_dialog();
|
|
|
|
|
view->selection_dialog=std::make_unique<SelectionDialog>(*view, view->get_buffer()->create_mark(dialog_iter), true, true);
|
|
|
|
|
auto rows=std::make_shared<std::unordered_map<std::string, Source::Offset> >();
|
|
|
|
|
|
|
|
|
|
auto iter=view->get_buffer()->get_insert()->get_iter();
|
|
|
|
|
for(auto &usage: usages) {
|
|
|
|
|
std::string row;
|
|
|
|
|
bool current_page=true;
|
|
|
|
|
//add file name if usage is not in current page
|
|
|
|
|
if(view->file_path!=usage.first.file_path) {
|
|
|
|
|
row=usage.first.file_path.filename().string()+":";
|
|
|
|
|
current_page=false;
|
|
|
|
|
}
|
|
|
|
|
row+=std::to_string(usage.first.line+1)+": "+usage.second;
|
|
|
|
|
(*rows)[row]=usage.first;
|
|
|
|
|
view->selection_dialog->add_row(row);
|
|
|
|
|
|
|
|
|
|
//Set dialog cursor to the last row if the textview cursor is at the same line
|
|
|
|
|
if(current_page &&
|
|
|
|
|
iter.get_line()==static_cast<int>(usage.first.line) && iter.get_line_index()>=static_cast<int>(usage.first.index)) {
|
|
|
|
|
view->selection_dialog->set_cursor_at_last_row();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(rows->size()==0)
|
|
|
|
|
return;
|
|
|
|
|
view->selection_dialog->on_select=[this, rows](const std::string &selected, bool hide_window) {
|
|
|
|
|
auto offset=rows->at(selected);
|
|
|
|
|
boost::filesystem::path declaration_file;
|
|
|
|
|
boost::system::error_code ec;
|
|
|
|
|
declaration_file=boost::filesystem::canonical(offset.file_path, ec);
|
|
|
|
|
if(ec)
|
|
|
|
|
return;
|
|
|
|
|
Notebook::get().open(declaration_file);
|
|
|
|
|
auto view=Notebook::get().get_current_view();
|
|
|
|
|
view->place_cursor_at_line_index(offset.line, offset.index);
|
|
|
|
|
view->scroll_to(view->get_buffer()->get_insert(), 0.0, 1.0, 0.5);
|
|
|
|
|
view->hide_tooltips();
|
|
|
|
|
};
|
|
|
|
|
view->hide_tooltips();
|
|
|
|
|
view->selection_dialog->show();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
menu.add_action("source_goto_method", [this]() {
|
|
|
|
|
if(auto view=Notebook::get().get_current_view()) {
|
|
|
|
|
if(view->get_methods) {
|
|
|
|
|
auto methods=Notebook::get().get_current_view()->get_methods();
|
|
|
|
|
if(!methods.empty()) {
|
|
|
|
|
auto dialog_iter=view->get_iter_for_dialog();
|
|
|
|
|
view->selection_dialog=std::make_unique<SelectionDialog>(*view, view->get_buffer()->create_mark(dialog_iter), true, true);
|
|
|
|
|
auto rows=std::make_shared<std::unordered_map<std::string, Source::Offset> >();
|
|
|
|
|
auto iter=view->get_buffer()->get_insert()->get_iter();
|
|
|
|
|
for(auto &method: methods) {
|
|
|
|
|
(*rows)[method.second]=method.first;
|
|
|
|
|
view->selection_dialog->add_row(method.second);
|
|
|
|
|
if(iter.get_line()>=static_cast<int>(method.first.line))
|
|
|
|
|
view->selection_dialog->set_cursor_at_last_row();
|
|
|
|
|
}
|
|
|
|
|
view->selection_dialog->on_select=[view, rows](const std::string& selected, bool hide_window) {
|
|
|
|
|
auto offset=rows->at(selected);
|
|
|
|
|
view->get_buffer()->place_cursor(view->get_buffer()->get_iter_at_line_index(offset.line, offset.index));
|
|
|
|
|
view->scroll_to(view->get_buffer()->get_insert(), 0.0, 1.0, 0.5);
|
|
|
|
|
view->hide_tooltips();
|
|
|
|
|
};
|
|
|
|
|
view->hide_tooltips();
|
|
|
|
|
view->selection_dialog->show();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
menu.add_action("source_rename", [this]() {
|
|
|
|
|
rename_token_entry();
|
|
|
|
|
});
|
|
|
|
|
menu.add_action("source_implement_method", [this]() {
|
|
|
|
|
const static std::string button_text="Insert Method Implementation";
|
|
|
|
|
|
|
|
|
|
if(auto view=Notebook::get().get_current_view()) {
|
|
|
|
|
if(view->get_method) {
|
|
|
|
|
auto iter=view->get_buffer()->get_insert()->get_iter();
|
|
|
|
|
if(!EntryBox::get().buttons.empty() && EntryBox::get().buttons.back().get_label()==button_text &&
|
|
|
|
|
iter.ends_line() && iter.starts_line()) {
|
|
|
|
|
EntryBox::get().buttons.back().activate();
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
auto method=std::make_shared<std::string>(view->get_method());
|
|
|
|
|
if(method->empty())
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
EntryBox::get().clear();
|
|
|
|
|
EntryBox::get().labels.emplace_back();
|
|
|
|
|
EntryBox::get().labels.back().set_text(*method);
|
|
|
|
|
EntryBox::get().buttons.emplace_back(button_text, [this, method](){
|
|
|
|
|
if(auto view=Notebook::get().get_current_view()) {
|
|
|
|
|
view->get_buffer()->insert_at_cursor(*method);
|
|
|
|
|
EntryBox::get().clear();
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
EntryBox::get().show();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
menu.add_action("source_goto_next_diagnostic", [this]() {
|
|
|
|
|
if(auto view=Notebook::get().get_current_view()) {
|
|
|
|
|
if(view->goto_next_diagnostic) {
|
|
|
|
|
view->goto_next_diagnostic();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
menu.add_action("source_apply_fix_its", [this]() {
|
|
|
|
|
if(auto view=Notebook::get().get_current_view()) {
|
|
|
|
|
if(view->get_fix_its) {
|
|
|
|
|
auto buffer=view->get_buffer();
|
|
|
|
|
auto fix_its=view->get_fix_its();
|
|
|
|
|
std::vector<std::pair<Glib::RefPtr<Gtk::TextMark>, Glib::RefPtr<Gtk::TextMark> > > fix_it_marks;
|
|
|
|
|
for(auto &fix_it: fix_its) {
|
|
|
|
|
auto start_iter=buffer->get_iter_at_line_index(fix_it.offsets.first.line, fix_it.offsets.first.index);
|
|
|
|
|
auto end_iter=buffer->get_iter_at_line_index(fix_it.offsets.second.line, fix_it.offsets.second.index);
|
|
|
|
|
fix_it_marks.emplace_back(buffer->create_mark(start_iter), buffer->create_mark(end_iter));
|
|
|
|
|
}
|
|
|
|
|
size_t c=0;
|
|
|
|
|
buffer->begin_user_action();
|
|
|
|
|
for(auto &fix_it: fix_its) {
|
|
|
|
|
if(fix_it.type==Source::FixIt::Type::INSERT) {
|
|
|
|
|
buffer->insert(fix_it_marks[c].first->get_iter(), fix_it.source);
|
|
|
|
|
}
|
|
|
|
|
if(fix_it.type==Source::FixIt::Type::REPLACE) {
|
|
|
|
|
buffer->erase(fix_it_marks[c].first->get_iter(), fix_it_marks[c].second->get_iter());
|
|
|
|
|
buffer->insert(fix_it_marks[c].first->get_iter(), fix_it.source);
|
|
|
|
|
}
|
|
|
|
|
if(fix_it.type==Source::FixIt::Type::ERASE) {
|
|
|
|
|
buffer->erase(fix_it_marks[c].first->get_iter(), fix_it_marks[c].second->get_iter());
|
|
|
|
|
}
|
|
|
|
|
c++;
|
|
|
|
|
}
|
|
|
|
|
for(auto &mark_pair: fix_it_marks) {
|
|
|
|
|
buffer->delete_mark(mark_pair.first);
|
|
|
|
|
buffer->delete_mark(mark_pair.second);
|
|
|
|
|
}
|
|
|
|
|
buffer->end_user_action();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
menu.add_action("project_set_run_arguments", [this]() {
|
|
|
|
|
auto project=Project::create();
|
|
|
|
|
auto run_arguments=std::make_shared<std::pair<std::string, std::string> >(project->get_run_arguments());
|
|
|
|
|
if(run_arguments->second.empty())
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
EntryBox::get().clear();
|
|
|
|
|
EntryBox::get().labels.emplace_back();
|
|
|
|
|
auto label_it=EntryBox::get().labels.begin();
|
|
|
|
|
label_it->update=[label_it](int state, const std::string& message){
|
|
|
|
|
label_it->set_text("Set empty to let juCi++ deduce executable");
|
|
|
|
|
};
|
|
|
|
|
label_it->update(0, "");
|
|
|
|
|
EntryBox::get().entries.emplace_back(run_arguments->second, [this, run_arguments](const std::string& content){
|
|
|
|
|
Project::run_arguments[run_arguments->first]=content;
|
|
|
|
|
EntryBox::get().hide();
|
|
|
|
|
}, 50);
|
|
|
|
|
auto entry_it=EntryBox::get().entries.begin();
|
|
|
|
|
entry_it->set_placeholder_text("Project: Set Run Arguments");
|
|
|
|
|
EntryBox::get().buttons.emplace_back("Project: set run arguments", [this, entry_it](){
|
|
|
|
|
entry_it->activate();
|
|
|
|
|
});
|
|
|
|
|
EntryBox::get().show();
|
|
|
|
|
});
|
|
|
|
|
menu.add_action("compile_and_run", [this]() {
|
|
|
|
|
if(Project::compiling || Project::debugging) {
|
|
|
|
|
Info::get().print("Compile or debug in progress");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Project::current=Project::create();
|
|
|
|
|
|
|
|
|
|
if(Config::get().project.save_on_compile_or_run)
|
|
|
|
|
Project::save_files(Project::current->build->project_path);
|
|
|
|
|
|
|
|
|
|
Project::current->compile_and_run();
|
|
|
|
|
});
|
|
|
|
|
menu.add_action("compile", [this]() {
|
|
|
|
|
if(Project::compiling || Project::debugging) {
|
|
|
|
|
Info::get().print("Compile or debug in progress");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Project::current=Project::create();
|
|
|
|
|
|
|
|
|
|
if(Config::get().project.save_on_compile_or_run)
|
|
|
|
|
Project::save_files(Project::current->build->project_path);
|
|
|
|
|
|
|
|
|
|
Project::current->compile();
|
|
|
|
|
});
|
|
|
|
|
menu.add_action("project_recreate_build", [this]() {
|
|
|
|
|
if(Project::compiling || Project::debugging) {
|
|
|
|
|
Info::get().print("Compile or debug in progress");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Project::current=Project::create();
|
|
|
|
|
|
|
|
|
|
Project::current->recreate_build();
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
menu.add_action("run_command", [this]() {
|
|
|
|
|
EntryBox::get().clear();
|
|
|
|
|
EntryBox::get().labels.emplace_back();
|
|
|
|
|
auto label_it=EntryBox::get().labels.begin();
|
|
|
|
|
label_it->update=[label_it](int state, const std::string& message){
|
|
|
|
|
label_it->set_text("Run Command directory order: opened directory, file path, current directory");
|
|
|
|
|
};
|
|
|
|
|
label_it->update(0, "");
|
|
|
|
|
EntryBox::get().entries.emplace_back(last_run_command, [this](const std::string& content){
|
|
|
|
|
if(content!="") {
|
|
|
|
|
last_run_command=content;
|
|
|
|
|
auto run_path=Notebook::get().get_current_folder();
|
|
|
|
|
Terminal::get().async_print("Running: "+content+'\n');
|
|
|
|
|
|
|
|
|
|
Terminal::get().async_process(content, run_path, [this, content](int exit_status){
|
|
|
|
|
Terminal::get().async_print(content+" returned: "+std::to_string(exit_status)+'\n');
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
EntryBox::get().hide();
|
|
|
|
|
}, 30);
|
|
|
|
|
auto entry_it=EntryBox::get().entries.begin();
|
|
|
|
|
entry_it->set_placeholder_text("Command");
|
|
|
|
|
EntryBox::get().buttons.emplace_back("Run command", [this, entry_it](){
|
|
|
|
|
entry_it->activate();
|
|
|
|
|
});
|
|
|
|
|
EntryBox::get().show();
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
menu.add_action("kill_last_running", [this]() {
|
|
|
|
|
Terminal::get().kill_last_async_process();
|
|
|
|
|
});
|
|
|
|
|
menu.add_action("force_kill_last_running", [this]() {
|
|
|
|
|
Terminal::get().kill_last_async_process(true);
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
#ifdef JUCI_ENABLE_DEBUG
|
|
|
|
|
menu.add_action("debug_set_run_arguments", [this]() {
|
|
|
|
|
auto project=Project::create();
|
|
|
|
|
auto run_arguments=std::make_shared<std::pair<std::string, std::string> >(project->debug_get_run_arguments());
|
|
|
|
|
if(run_arguments->second.empty())
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
EntryBox::get().clear();
|
|
|
|
|
EntryBox::get().labels.emplace_back();
|
|
|
|
|
auto label_it=EntryBox::get().labels.begin();
|
|
|
|
|
label_it->update=[label_it](int state, const std::string& message){
|
|
|
|
|
label_it->set_text("Set empty to let juCi++ deduce executable");
|
|
|
|
|
};
|
|
|
|
|
label_it->update(0, "");
|
|
|
|
|
EntryBox::get().entries.emplace_back(run_arguments->second, [this, run_arguments](const std::string& content){
|
|
|
|
|
Project::debug_run_arguments[run_arguments->first]=content;
|
|
|
|
|
EntryBox::get().hide();
|
|
|
|
|
}, 50);
|
|
|
|
|
auto entry_it=EntryBox::get().entries.begin();
|
|
|
|
|
entry_it->set_placeholder_text("Debug: Set Run Arguments");
|
|
|
|
|
|
|
|
|
|
if(auto options=project->debug_get_options()) {
|
|
|
|
|
EntryBox::get().buttons.emplace_back("", [this, options]() {
|
|
|
|
|
options->set_visible(true);
|
|
|
|
|
});
|
|
|
|
|
EntryBox::get().buttons.back().set_image_from_icon_name("preferences-system");
|
|
|
|
|
EntryBox::get().buttons.back().set_always_show_image(true);
|
|
|
|
|
EntryBox::get().buttons.back().set_tooltip_text("Additional Options");
|
|
|
|
|
options->set_relative_to(EntryBox::get().buttons.back());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
EntryBox::get().buttons.emplace_back("Debug: set run arguments", [this, entry_it](){
|
|
|
|
|
entry_it->activate();
|
|
|
|
|
});
|
|
|
|
|
EntryBox::get().show();
|
|
|
|
|
});
|
|
|
|
|
menu.add_action("debug_start_continue", [this](){
|
|
|
|
|
if(Project::compiling) {
|
|
|
|
|
Info::get().print("Compile in progress");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
else if(Project::debugging) {
|
|
|
|
|
Project::current->debug_continue();
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Project::current=Project::create();
|
|
|
|
|
|
|
|
|
|
if(Config::get().project.save_on_compile_or_run)
|
|
|
|
|
Project::save_files(Project::current->build->project_path);
|
|
|
|
|
|
|
|
|
|
Project::current->debug_start();
|
|
|
|
|
});
|
|
|
|
|
menu.add_action("debug_stop", [this]() {
|
|
|
|
|
if(Project::current)
|
|
|
|
|
Project::current->debug_stop();
|
|
|
|
|
});
|
|
|
|
|
menu.add_action("debug_kill", [this]() {
|
|
|
|
|
if(Project::current)
|
|
|
|
|
Project::current->debug_kill();
|
|
|
|
|
});
|
|
|
|
|
menu.add_action("debug_step_over", [this]() {
|
|
|
|
|
if(Project::current)
|
|
|
|
|
Project::current->debug_step_over();
|
|
|
|
|
});
|
|
|
|
|
menu.add_action("debug_step_into", [this]() {
|
|
|
|
|
if(Project::current)
|
|
|
|
|
Project::current->debug_step_into();
|
|
|
|
|
});
|
|
|
|
|
menu.add_action("debug_step_out", [this]() {
|
|
|
|
|
if(Project::current)
|
|
|
|
|
Project::current->debug_step_out();
|
|
|
|
|
});
|
|
|
|
|
menu.add_action("debug_backtrace", [this]() {
|
|
|
|
|
if(Project::current)
|
|
|
|
|
Project::current->debug_backtrace();
|
|
|
|
|
});
|
|
|
|
|
menu.add_action("debug_show_variables", [this]() {
|
|
|
|
|
if(Project::current)
|
|
|
|
|
Project::current->debug_show_variables();
|
|
|
|
|
});
|
|
|
|
|
menu.add_action("debug_run_command", [this]() {
|
|
|
|
|
EntryBox::get().clear();
|
|
|
|
|
EntryBox::get().entries.emplace_back(last_run_debug_command, [this](const std::string& content){
|
|
|
|
|
if(content!="") {
|
|
|
|
|
if(Project::current)
|
|
|
|
|
Project::current->debug_run_command(content);
|
|
|
|
|
last_run_debug_command=content;
|
|
|
|
|
}
|
|
|
|
|
EntryBox::get().hide();
|
|
|
|
|
}, 30);
|
|
|
|
|
auto entry_it=EntryBox::get().entries.begin();
|
|
|
|
|
entry_it->set_placeholder_text("Debug Command");
|
|
|
|
|
EntryBox::get().buttons.emplace_back("Run debug command", [this, entry_it](){
|
|
|
|
|
entry_it->activate();
|
|
|
|
|
});
|
|
|
|
|
EntryBox::get().show();
|
|
|
|
|
});
|
|
|
|
|
menu.add_action("debug_toggle_breakpoint", [this](){
|
|
|
|
|
if(auto view=Notebook::get().get_current_view()) {
|
|
|
|
|
if(view->toggle_breakpoint)
|
|
|
|
|
view->toggle_breakpoint(view->get_buffer()->get_insert()->get_iter().get_line());
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
menu.add_action("debug_goto_stop", [this](){
|
|
|
|
|
if(Project::debugging) {
|
|
|
|
|
if(!Project::debug_stop.first.empty()) {
|
|
|
|
|
Notebook::get().open(Project::debug_stop.first);
|
|
|
|
|
if(auto view=Notebook::get().get_current_view()) {
|
|
|
|
|
int line=Project::debug_stop.second.first;
|
|
|
|
|
int index=Project::debug_stop.second.second;
|
|
|
|
|
view->place_cursor_at_line_index(line, index);
|
|
|
|
|
view->scroll_to_cursor_delayed(view, true, true);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
Project::debug_update_status("");
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
menu.add_action("next_tab", [this]() {
|
|
|
|
|
Notebook::get().next();
|
|
|
|
|
});
|
|
|
|
|
menu.add_action("previous_tab", [this]() {
|
|
|
|
|
Notebook::get().previous();
|
|
|
|
|
});
|
|
|
|
|
menu.add_action("close_tab", [this]() {
|
|
|
|
|
if(Notebook::get().get_current_view())
|
|
|
|
|
Notebook::get().close_current();
|
|
|
|
|
});
|
|
|
|
|
menu.add_action("window_toggle_split", [this] {
|
|
|
|
|
Notebook::get().toggle_split();
|
|
|
|
|
});
|
|
|
|
|
menu.add_action("window_clear_terminal", [this] {
|
|
|
|
|
Terminal::get().clear();
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Window::activate_menu_items() {
|
|
|
|
|
auto &menu = Menu::get();
|
|
|
|
|
auto view=Notebook::get().get_current_view();
|
|
|
|
|
|
|
|
|
|
menu.actions["source_spellcheck"]->set_enabled(view);
|
|
|
|
|
menu.actions["source_spellcheck_clear"]->set_enabled(view);
|
|
|
|
|
menu.actions["source_spellcheck_next_error"]->set_enabled(view);
|
|
|
|
|
menu.actions["source_git_next_diff"]->set_enabled(view);
|
|
|
|
|
menu.actions["source_git_show_diff"]->set_enabled(view);
|
|
|
|
|
menu.actions["source_indentation_set_buffer_tab"]->set_enabled(view);
|
|
|
|
|
menu.actions["source_goto_line"]->set_enabled(view);
|
|
|
|
|
menu.actions["source_center_cursor"]->set_enabled(view);
|
|
|
|
|
|
|
|
|
|
menu.actions["source_indentation_auto_indent_buffer"]->set_enabled(view && static_cast<bool>(view->format_style));
|
|
|
|
|
menu.actions["source_find_symbol_ctags"]->set_enabled(view);
|
|
|
|
|
menu.actions["source_comments_toggle"]->set_enabled(view && static_cast<bool>(view->toggle_comments));
|
|
|
|
|
menu.actions["source_comments_add_documentation"]->set_enabled(view && static_cast<bool>(view->add_documentation));
|
|
|
|
|
menu.actions["source_find_documentation"]->set_enabled(view && static_cast<bool>(view->get_token_data));
|
|
|
|
|
menu.actions["source_goto_declaration"]->set_enabled(view && static_cast<bool>(view->get_declaration_location));
|
|
|
|
|
menu.actions["source_goto_implementation"]->set_enabled(view && static_cast<bool>(view->get_implementation_locations));
|
|
|
|
|
menu.actions["source_goto_usage"]->set_enabled(view && static_cast<bool>(view->get_usages));
|
|
|
|
|
menu.actions["source_goto_method"]->set_enabled(view && static_cast<bool>(view->get_methods));
|
|
|
|
|
menu.actions["source_rename"]->set_enabled(view && static_cast<bool>(view->rename_similar_tokens));
|
|
|
|
|
menu.actions["source_implement_method"]->set_enabled(view && static_cast<bool>(view->get_method));
|
|
|
|
|
menu.actions["source_goto_next_diagnostic"]->set_enabled(view && static_cast<bool>(view->goto_next_diagnostic));
|
|
|
|
|
menu.actions["source_apply_fix_its"]->set_enabled(view && static_cast<bool>(view->get_fix_its));
|
|
|
|
|
#ifdef JUCI_ENABLE_DEBUG
|
|
|
|
|
Project::debug_activate_menu_items();
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool Window::on_key_press_event(GdkEventKey *event) {
|
|
|
|
|
if(event->keyval==GDK_KEY_Escape) {
|
|
|
|
|
EntryBox::get().hide();
|
|
|
|
|
}
|
|
|
|
|
#ifdef __APPLE__ //For Apple's Command-left, right, up, down keys
|
|
|
|
|
else if((event->state & GDK_META_MASK)>0 && (event->state & GDK_MOD1_MASK)==0) {
|
|
|
|
|
if(event->keyval==GDK_KEY_Left || event->keyval==GDK_KEY_KP_Left) {
|
|
|
|
|
event->keyval=GDK_KEY_Home;
|
|
|
|
|
event->state=event->state & GDK_SHIFT_MASK;
|
|
|
|
|
event->hardware_keycode=115;
|
|
|
|
|
}
|
|
|
|
|
else if(event->keyval==GDK_KEY_Right || event->keyval==GDK_KEY_KP_Right) {
|
|
|
|
|
event->keyval=GDK_KEY_End;
|
|
|
|
|
event->state=event->state & GDK_SHIFT_MASK;
|
|
|
|
|
event->hardware_keycode=119;
|
|
|
|
|
}
|
|
|
|
|
else if(event->keyval==GDK_KEY_Up || event->keyval==GDK_KEY_KP_Up) {
|
|
|
|
|
event->keyval=GDK_KEY_Home;
|
|
|
|
|
event->state=event->state & GDK_SHIFT_MASK;
|
|
|
|
|
event->state+=GDK_CONTROL_MASK;
|
|
|
|
|
event->hardware_keycode=115;
|
|
|
|
|
}
|
|
|
|
|
else if(event->keyval==GDK_KEY_Down || event->keyval==GDK_KEY_KP_Down) {
|
|
|
|
|
event->keyval=GDK_KEY_End;
|
|
|
|
|
event->state=event->state & GDK_SHIFT_MASK;
|
|
|
|
|
event->state+=GDK_CONTROL_MASK;
|
|
|
|
|
event->hardware_keycode=119;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
return Gtk::ApplicationWindow::on_key_press_event(event);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool Window::on_delete_event(GdkEventAny *event) {
|
|
|
|
|
Notebook::get().save_session();
|
|
|
|
|
|
|
|
|
|
for(size_t c=Notebook::get().size()-1;c!=static_cast<size_t>(-1);--c) {
|
|
|
|
|
if(!Notebook::get().close(c))
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
Terminal::get().kill_async_processes();
|
|
|
|
|
#ifdef JUCI_ENABLE_DEBUG
|
|
|
|
|
if(Project::current)
|
|
|
|
|
Project::current->debug_cancel();
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Window::search_and_replace_entry() {
|
|
|
|
|
EntryBox::get().clear();
|
|
|
|
|
EntryBox::get().labels.emplace_back();
|
|
|
|
|
auto label_it=EntryBox::get().labels.begin();
|
|
|
|
|
label_it->update=[label_it](int state, const std::string& message){
|
|
|
|
|
if(state==0) {
|
|
|
|
|
try {
|
|
|
|
|
auto number = stoi(message);
|
|
|
|
|
if(number==0)
|
|
|
|
|
label_it->set_text("");
|
|
|
|
|
else if(number==1)
|
|
|
|
|
label_it->set_text("1 result found");
|
|
|
|
|
else if(number>1)
|
|
|
|
|
label_it->set_text(message+" results found");
|
|
|
|
|
}
|
|
|
|
|
catch(const std::exception &e) {}
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
EntryBox::get().entries.emplace_back(last_search, [this](const std::string& content){
|
|
|
|
|
if(auto view=Notebook::get().get_current_view())
|
|
|
|
|
view->search_forward();
|
|
|
|
|
});
|
|
|
|
|
auto search_entry_it=EntryBox::get().entries.begin();
|
|
|
|
|
search_entry_it->set_placeholder_text("Find");
|
|
|
|
|
if(auto view=Notebook::get().get_current_view()) {
|
|
|
|
|
view->update_search_occurrences=[label_it](int number){
|
|
|
|
|
label_it->update(0, std::to_string(number));
|
|
|
|
|
};
|
|
|
|
|
view->search_highlight(search_entry_it->get_text(), case_sensitive_search, regex_search);
|
|
|
|
|
}
|
|
|
|
|
search_entry_it->signal_key_press_event().connect([this](GdkEventKey* event){
|
|
|
|
|
if((event->keyval==GDK_KEY_Return || event->keyval==GDK_KEY_KP_Enter) && (event->state&GDK_SHIFT_MASK)>0) {
|
|
|
|
|
if(auto view=Notebook::get().get_current_view())
|
|
|
|
|
view->search_backward();
|
|
|
|
|
}
|
|
|
|
|
return false;
|
|
|
|
|
});
|
|
|
|
|
search_entry_it->signal_changed().connect([this, search_entry_it](){
|
|
|
|
|
last_search=search_entry_it->get_text();
|
|
|
|
|
if(auto view=Notebook::get().get_current_view())
|
|
|
|
|
view->search_highlight(search_entry_it->get_text(), case_sensitive_search, regex_search);
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
EntryBox::get().entries.emplace_back(last_replace, [this](const std::string &content){
|
|
|
|
|
if(auto view=Notebook::get().get_current_view())
|
|
|
|
|
view->replace_forward(content);
|
|
|
|
|
});
|
|
|
|
|
auto replace_entry_it=EntryBox::get().entries.begin();
|
|
|
|
|
replace_entry_it++;
|
|
|
|
|
replace_entry_it->set_placeholder_text("Replace");
|
|
|
|
|
replace_entry_it->signal_key_press_event().connect([this, replace_entry_it](GdkEventKey* event){
|
|
|
|
|
if((event->keyval==GDK_KEY_Return || event->keyval==GDK_KEY_KP_Enter) && (event->state&GDK_SHIFT_MASK)>0) {
|
|
|
|
|
if(auto view=Notebook::get().get_current_view())
|
|
|
|
|
view->replace_backward(replace_entry_it->get_text());
|
|
|
|
|
}
|
|
|
|
|
return false;
|
|
|
|
|
});
|
|
|
|
|
replace_entry_it->signal_changed().connect([this, replace_entry_it](){
|
|
|
|
|
last_replace=replace_entry_it->get_text();
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
EntryBox::get().buttons.emplace_back("↑", [this](){
|
|
|
|
|
if(auto view=Notebook::get().get_current_view())
|
|
|
|
|
view->search_backward();
|
|
|
|
|
});
|
|
|
|
|
EntryBox::get().buttons.back().set_tooltip_text("Find Previous\n\nShortcut: Shift+Enter in the Find entry field");
|
|
|
|
|
EntryBox::get().buttons.emplace_back("⇄", [this, replace_entry_it](){
|
|
|
|
|
if(auto view=Notebook::get().get_current_view()) {
|
|
|
|
|
view->replace_forward(replace_entry_it->get_text());
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
EntryBox::get().buttons.back().set_tooltip_text("Replace Next\n\nShortcut: Enter in the Replace entry field");
|
|
|
|
|
EntryBox::get().buttons.emplace_back("↓", [this](){
|
|
|
|
|
if(auto view=Notebook::get().get_current_view())
|
|
|
|
|
view->search_forward();
|
|
|
|
|
});
|
|
|
|
|
EntryBox::get().buttons.back().set_tooltip_text("Find Next\n\nShortcut: Enter in the Find entry field");
|
|
|
|
|
EntryBox::get().buttons.emplace_back("Replace All", [this, replace_entry_it](){
|
|
|
|
|
if(auto view=Notebook::get().get_current_view())
|
|
|
|
|
view->replace_all(replace_entry_it->get_text());
|
|
|
|
|
});
|
|
|
|
|
EntryBox::get().buttons.back().set_tooltip_text("Replace All");
|
|
|
|
|
|
|
|
|
|
EntryBox::get().toggle_buttons.emplace_back("Aa");
|
|
|
|
|
EntryBox::get().toggle_buttons.back().set_tooltip_text("Match Case");
|
|
|
|
|
EntryBox::get().toggle_buttons.back().set_active(case_sensitive_search);
|
|
|
|
|
EntryBox::get().toggle_buttons.back().on_activate=[this, search_entry_it](){
|
|
|
|
|
case_sensitive_search=!case_sensitive_search;
|
|
|
|
|
if(auto view=Notebook::get().get_current_view())
|
|
|
|
|
view->search_highlight(search_entry_it->get_text(), case_sensitive_search, regex_search);
|
|
|
|
|
};
|
|
|
|
|
EntryBox::get().toggle_buttons.emplace_back(".*");
|
|
|
|
|
EntryBox::get().toggle_buttons.back().set_tooltip_text("Use Regex");
|
|
|
|
|
EntryBox::get().toggle_buttons.back().set_active(regex_search);
|
|
|
|
|
EntryBox::get().toggle_buttons.back().on_activate=[this, search_entry_it](){
|
|
|
|
|
regex_search=!regex_search;
|
|
|
|
|
if(auto view=Notebook::get().get_current_view())
|
|
|
|
|
view->search_highlight(search_entry_it->get_text(), case_sensitive_search, regex_search);
|
|
|
|
|
};
|
|
|
|
|
EntryBox::get().signal_hide().connect([this]() {
|
|
|
|
|
for(size_t c=0;c<Notebook::get().size();c++) {
|
|
|
|
|
Notebook::get().get_view(c)->update_search_occurrences=nullptr;
|
|
|
|
|
Notebook::get().get_view(c)->search_highlight("", case_sensitive_search, regex_search);
|
|
|
|
|
}
|
|
|
|
|
search_entry_shown=false;
|
|
|
|
|
});
|
|
|
|
|
search_entry_shown=true;
|
|
|
|
|
EntryBox::get().show();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Window::set_tab_entry() {
|
|
|
|
|
EntryBox::get().clear();
|
|
|
|
|
if(auto view=Notebook::get().get_current_view()) {
|
|
|
|
|
auto tab_char_and_size=view->get_tab_char_and_size();
|
|
|
|
|
|
|
|
|
|
EntryBox::get().labels.emplace_back();
|
|
|
|
|
auto label_it=EntryBox::get().labels.begin();
|
|
|
|
|
|
|
|
|
|
EntryBox::get().entries.emplace_back(std::to_string(tab_char_and_size.second));
|
|
|
|
|
auto entry_tab_size_it=EntryBox::get().entries.begin();
|
|
|
|
|
entry_tab_size_it->set_placeholder_text("Tab size");
|
|
|
|
|
|
|
|
|
|
char tab_char=tab_char_and_size.first;
|
|
|
|
|
std::string tab_char_string;
|
|
|
|
|
if(tab_char==' ')
|
|
|
|
|
tab_char_string="space";
|
|
|
|
|
else if(tab_char=='\t')
|
|
|
|
|
tab_char_string="tab";
|
|
|
|
|
|
|
|
|
|
EntryBox::get().entries.emplace_back(tab_char_string);
|
|
|
|
|
auto entry_tab_char_it=EntryBox::get().entries.rbegin();
|
|
|
|
|
entry_tab_char_it->set_placeholder_text("Tab char");
|
|
|
|
|
|
|
|
|
|
const auto activate_function=[this, entry_tab_char_it, entry_tab_size_it, label_it](const std::string& content){
|
|
|
|
|
if(auto view=Notebook::get().get_current_view()) {
|
|
|
|
|
char tab_char=0;
|
|
|
|
|
unsigned tab_size=0;
|
|
|
|
|
try {
|
|
|
|
|
tab_size = static_cast<unsigned>(std::stoul(entry_tab_size_it->get_text()));
|
|
|
|
|
std::string tab_char_string=entry_tab_char_it->get_text();
|
|
|
|
|
std::transform(tab_char_string.begin(), tab_char_string.end(), tab_char_string.begin(), ::tolower);
|
|
|
|
|
if(tab_char_string=="space")
|
|
|
|
|
tab_char=' ';
|
|
|
|
|
else if(tab_char_string=="tab")
|
|
|
|
|
tab_char='\t';
|
|
|
|
|
}
|
|
|
|
|
catch(const std::exception &e) {}
|
|
|
|
|
|
|
|
|
|
if(tab_char!=0 && tab_size>0) {
|
|
|
|
|
view->set_tab_char_and_size(tab_char, tab_size);
|
|
|
|
|
EntryBox::get().hide();
|
|
|
|
|
}
|
|
|
|
|
else {
|
|
|
|
|
label_it->set_text("Tab size must be >0 and tab char set to either 'space' or 'tab'");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
entry_tab_char_it->on_activate=activate_function;
|
|
|
|
|
entry_tab_size_it->on_activate=activate_function;
|
|
|
|
|
|
|
|
|
|
EntryBox::get().buttons.emplace_back("Set tab in current buffer", [this, entry_tab_char_it](){
|
|
|
|
|
entry_tab_char_it->activate();
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
EntryBox::get().show();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Window::goto_line_entry() {
|
|
|
|
|
EntryBox::get().clear();
|
|
|
|
|
if(Notebook::get().get_current_view()) {
|
|
|
|
|
EntryBox::get().entries.emplace_back("", [this](const std::string& content){
|
|
|
|
|
if(auto view=Notebook::get().get_current_view()) {
|
|
|
|
|
try {
|
|
|
|
|
view->place_cursor_at_line_index(stoi(content)-1, 0);
|
|
|
|
|
view->scroll_to_cursor_delayed(view, true, false);
|
|
|
|
|
}
|
|
|
|
|
catch(const std::exception &e) {}
|
|
|
|
|
EntryBox::get().hide();
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
auto entry_it=EntryBox::get().entries.begin();
|
|
|
|
|
entry_it->set_placeholder_text("Line number");
|
|
|
|
|
EntryBox::get().buttons.emplace_back("Go to line", [this, entry_it](){
|
|
|
|
|
entry_it->activate();
|
|
|
|
|
});
|
|
|
|
|
EntryBox::get().show();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void Window::rename_token_entry() {
|
|
|
|
|
EntryBox::get().clear();
|
|
|
|
|
if(auto view=Notebook::get().get_current_view()) {
|
|
|
|
|
if(view->get_token_spelling && view->rename_similar_tokens) {
|
|
|
|
|
auto spelling=std::make_shared<std::string>(view->get_token_spelling());
|
|
|
|
|
if(!spelling->empty()) {
|
|
|
|
|
EntryBox::get().labels.emplace_back();
|
|
|
|
|
auto label_it=EntryBox::get().labels.begin();
|
|
|
|
|
label_it->update=[label_it](int state, const std::string& message){
|
|
|
|
|
label_it->set_text("Warning: only opened files will be refactored, and altered files will be saved");
|
|
|
|
|
};
|
|
|
|
|
label_it->update(0, "");
|
|
|
|
|
auto iter=std::make_shared<Gtk::TextIter>(view->get_buffer()->get_insert()->get_iter());
|
|
|
|
|
EntryBox::get().entries.emplace_back(*spelling, [this, view, spelling, iter](const std::string& content){
|
|
|
|
|
//TODO: gtk needs a way to check if iter is valid without dumping g_error message
|
|
|
|
|
//iter->get_buffer() will print such a message, but no segfault will occur
|
|
|
|
|
if(Notebook::get().get_current_view()==view && content!=*spelling && iter->get_buffer() && view->get_buffer()->get_insert()->get_iter()==*iter) {
|
|
|
|
|
auto renamed_pairs=view->rename_similar_tokens(Notebook::get().get_views(), content);
|
|
|
|
|
for(auto &renamed: renamed_pairs)
|
|
|
|
|
Terminal::get().print("Replaced "+std::to_string(renamed.second)+" occurrence"+(renamed.second>1?"s":"")+" in file "+renamed.first.string()+"\n");
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
Info::get().print("Operation canceled");
|
|
|
|
|
EntryBox::get().hide();
|
|
|
|
|
});
|
|
|
|
|
auto entry_it=EntryBox::get().entries.begin();
|
|
|
|
|
entry_it->set_placeholder_text("New name");
|
|
|
|
|
EntryBox::get().buttons.emplace_back("Rename", [this, entry_it](){
|
|
|
|
|
entry_it->activate();
|
|
|
|
|
});
|
|
|
|
|
EntryBox::get().show();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|