#include #include "notebook.h" #include "logging.h" Notebook::Model::Model() { cc_extension_ = ".cc"; h_extension_ = ".h"; scrollvalue_ = 50; } Notebook::View::View() : notebook_() { view_.pack2(notebook_); view_.set_position(120); } Notebook::Controller::Controller(Gtk::Window* window, Keybindings::Controller& keybindings, Source::Config& source_cfg, Directories::Config& dir_cfg) : source_config_(source_cfg), directories_(dir_cfg), index_(0, 1) { window_ = window; OnNewPage("juCi++"); refClipboard_ = Gtk::Clipboard::get(); ispopup = false; view().pack1(directories_.widget(), true, true); CreateKeybindings(keybindings); INFO("Notebook Controller Success"); } // Constructor void Notebook::Controller::CreateKeybindings(Keybindings::Controller &keybindings) { INFO("Notebook create signal handlers"); directories().m_TreeView.signal_row_activated() .connect(sigc::mem_fun(*this, &Notebook::Controller::OnDirectoryNavigation)); keybindings.action_group_menu()-> add(Gtk::Action::create("FileMenu", Gtk::Stock::FILE)); keybindings.action_group_menu()-> add(Gtk::Action::create("FileNewStandard", Gtk::Stock::NEW, "New empty file", "Create a new file"), [this]() { is_new_file_ = true; OnFileNewEmptyfile(); }); keybindings.action_group_menu()-> add(Gtk::Action::create("FileNewCC", "New cc file"), Gtk::AccelKey(keybindings.config_ .key_map()["new_cc_file"]), [this]() { is_new_file_ = true; OnFileNewCCFile(); }); keybindings.action_group_menu()-> add(Gtk::Action::create("FileNewH", "New h file"), Gtk::AccelKey(keybindings.config_ .key_map()["new_h_file"]), [this]() { is_new_file_ = true; OnFileNewHeaderFile(); }); keybindings.action_group_menu()-> add(Gtk::Action::create("WindowCloseTab", "Close tab"), Gtk::AccelKey(keybindings.config_ .key_map()["close_tab"]), [this]() { OnCloseCurrentPage(); }); keybindings.action_group_menu()-> add(Gtk::Action::create("EditFind", Gtk::Stock::FIND), [this]() { is_new_file_ = false; OnEditSearch(); // TODO(Oyvang) Zalox, Forgi)Create function OnEditFind(); }); keybindings.action_group_menu()-> add(Gtk::Action::create("EditCopy", Gtk::Stock::COPY), [this]() { OnEditCopy(); }); keybindings.action_group_menu()-> add(Gtk::Action::create("EditCut", Gtk::Stock::CUT), [this]() { OnEditCut(); }); keybindings.action_group_menu()-> add(Gtk::Action::create("EditPaste", Gtk::Stock::PASTE), [this]() { OnEditPaste(); }); keybindings.action_group_menu()-> add(Gtk::Action::create("EditUndo", "Undo"), Gtk::AccelKey(keybindings.config_ .key_map()["edit_undo"]), [this]() { //OnUndo(); }); entry_.view_.entry().signal_activate(). connect( [this]() { if (is_new_file_) { OnNewPage(entry_.text()); entry_.OnHideEntries(is_new_file_); } else { Search(true); } }); entry_.button_apply().signal_clicked(). connect( [this]() { OnNewPage(entry_.text()); entry_.OnHideEntries(is_new_file_); }); entry_.button_close().signal_clicked(). connect( [this]() { entry_.OnHideEntries(is_new_file_); }); entry_.button_next().signal_clicked(). connect( [this]() { Search(true); }); entry_.button_prev().signal_clicked(). connect( [this]() { Search(false); }); INFO("Notebook signal handlers sucsess"); } bool Notebook::Controller:: OnMouseRelease(GdkEventButton* button) { if (button->button == 1 && ispopup) { popup_.response(Gtk::RESPONSE_DELETE_EVENT); return true; } return false; } bool Notebook::Controller::OnKeyRelease(GdkEventKey* key) { return GeneratePopup(key->keyval); } bool Notebook::Controller::GeneratePopup(int key_id) { INFO("Notebook genereate popup, getting iters"); std::string path = text_vec_.at(CurrentPage())->path(); if (!LegalExtension(path.substr(path.find_last_of(".") + 1))) return false; // Get function to fill popup with suggests item vector under is for testing Gtk::TextIter beg = CurrentTextView().get_buffer()->get_insert()->get_iter(); Gtk::TextIter end = CurrentTextView().get_buffer()->get_insert()->get_iter(); Gtk::TextIter tmp = CurrentTextView().get_buffer()->get_insert()->get_iter(); Gtk::TextIter tmp1 = CurrentTextView().get_buffer()->get_insert()->get_iter(); Gtk::TextIter line = CurrentTextView().get_buffer()->get_iter_at_line(tmp.get_line()); if (end.backward_char() && end.backward_char()) { bool illegal_chars = end.backward_search("\"", Gtk::TEXT_SEARCH_VISIBLE_ONLY, tmp, tmp1, line) || end.backward_search("//", Gtk::TEXT_SEARCH_VISIBLE_ONLY, tmp, tmp1, line); INFO("Notebook genereate popup, checking key_id"); if (illegal_chars) { return false; } std::string c = text_vec_[CurrentPage()]->buffer()->get_text(end, beg); switch (key_id) { case 46: break; case 58: if (c != "::") return false; break; case 60: if (c != "->") return false; break; case 62: if (c != "->") return false; break; default: return false; } } else { return false; } INFO("Notebook genereate popup, getting autocompletions"); std::vector acdata; text_vec_.at(CurrentPage())-> GetAutoCompleteSuggestions(beg.get_line()+1, beg.get_line_offset()+2, &acdata); std::map items; for (auto &data : acdata) { std::stringstream ss; std::string return_value; for (auto &chunk : data.chunks_) { switch (chunk.kind()) { case clang::CompletionChunk_ResultType: return_value = chunk.chunk(); break; case clang::CompletionChunk_Informative: break; default: ss << chunk.chunk(); break; } } items[ss.str() + " --> " + return_value] = ss.str(); } // Replace over with get suggestions from zalox! OVER IS JUST FOR TESTING Gtk::ScrolledWindow popup_scroll_; Gtk::ListViewText listview_(1, false, Gtk::SelectionMode::SELECTION_SINGLE); popup_scroll_.set_policy(Gtk::PolicyType::POLICY_NEVER, Gtk::PolicyType::POLICY_NEVER); listview_.set_enable_search(true); listview_.set_headers_visible(false); listview_.set_hscroll_policy(Gtk::ScrollablePolicy::SCROLL_NATURAL); listview_.set_activate_on_single_click(true); if (items.empty()) { items["No suggestions found..."] = ""; } for (auto &i : items) { listview_.append(i.first); } popup_scroll_.add(listview_); popup_.get_vbox()->pack_start(popup_scroll_); popup_.set_transient_for(*window_); popup_.show_all(); INFO("Notebook genereate popup, moving popup"); int popup_x = popup_.get_width(); int popup_y = items.size() * 20; PopupSetSize(popup_scroll_, popup_x, popup_y); int x, y; FindPopupPosition(CurrentTextView(), popup_x, popup_y, x, y); popup_.move(x, y+15); INFO("Notebook genereate popup, create handler"); PopupSelectHandler(popup_, listview_, &items); ispopup = true; INFO("Notebook genereate popup, run popup"); popup_.run(); INFO("Notebook genereate popup, hide popup"); popup_.hide(); ispopup = false; return true; } bool Notebook::Controller::ScrollEventCallback(GdkEventScroll* scroll_event) { int page = CurrentPage(); int direction_y = scroll_event->delta_y; int direction_x = scroll_event->delta_x; Glib::RefPtr adj = scrolledtext_vec_.at(page)-> get_vscrollbar()->get_adjustment(); if ( direction_y != 0 ) { int dir_val = direction_y == -1 ? -model_.scrollvalue_:+model_.scrollvalue_; adj->set_value(adj->get_value()+dir_val); text_vec_.at(page)->view().set_vadjustment(adj); linenumbers_vec_.at(page)->view().set_vadjustment(adj); } return true; } Notebook::Controller::~Controller() { INFO("Notebook destructor"); for (auto &i : text_vec_) delete i; for (auto &i : linenumbers_vec_) delete i; for (auto &i : editor_vec_) delete i; for (auto &i : scrolledtext_vec_) delete i; for (auto &i : scrolledline_vec_) delete i; } Gtk::Paned& Notebook::Controller::view() { return view_.view(); } Gtk::Box& Notebook::Controller::entry_view() { return entry_.view(); } void Notebook::Controller::OnNewPage(std::string name) { INFO("Notebook Generate new page"); OnCreatePage(); text_vec_.back()->OnNewEmptyFile(); Notebook().append_page(*editor_vec_.back(), name); Notebook().show_all_children(); Notebook().set_current_page(Pages()-1); Notebook().set_focus_child(text_vec_.at(Pages()-1)->view()); } void Notebook::Controller:: MapBuffers(std::map *buffers) { for (auto &buffer : text_vec_) { buffers->operator[](buffer->model().file_path()) = buffer->buffer()->get_text().raw(); } } void Notebook::Controller::OnOpenFile(std::string path) { INFO("Notebook open file"); OnCreatePage(); text_vec_.back()->OnOpenFile(path); text_vec_.back()->set_is_saved(true); unsigned pos = path.find_last_of("/\\"); Notebook().append_page(*editor_vec_.back(), path.substr(pos+1)); Notebook().show_all_children(); Notebook().set_current_page(Pages()-1); Notebook().set_focus_child(text_vec_.back()->view()); OnBufferChange(); text_vec_.back()->set_is_changed(false); } void Notebook::Controller::OnCreatePage() { INFO("Notebook create page"); text_vec_.push_back(new Source::Controller(source_config(), this)); linenumbers_vec_.push_back(new Source::Controller(source_config(), this)); scrolledline_vec_.push_back(new Gtk::ScrolledWindow()); scrolledtext_vec_.push_back(new Gtk::ScrolledWindow()); editor_vec_.push_back(new Gtk::HBox()); scrolledtext_vec_.back()->add(text_vec_.back()->view()); scrolledline_vec_.back()->add(linenumbers_vec_.back()->view()); linenumbers_vec_.back()->view().get_buffer()->set_text("1 "); linenumbers_vec_.back()->view().override_color(Gdk::RGBA("Black")); linenumbers_vec_.back()-> view().set_justification(Gtk::Justification::JUSTIFY_RIGHT); scrolledline_vec_.back()->get_vscrollbar()->hide(); linenumbers_vec_.back()->view().set_editable(false); linenumbers_vec_.back()->view().set_sensitive(false); editor_vec_.back()->pack_start(*scrolledline_vec_.back(), false, false); editor_vec_.back()->pack_start(*scrolledtext_vec_.back(), true, true); TextViewHandlers(text_vec_.back()->view()); } void Notebook::Controller::OnCloseCurrentPage() { INFO("Notebook close page"); if (Pages() != 0) { if(text_vec_.back()->is_changed()){ AskToSaveDialog(); } int page = CurrentPage(); Notebook().remove_page(page); delete text_vec_.at(page); delete linenumbers_vec_.at(page); delete scrolledtext_vec_.at(page); delete scrolledline_vec_.at(page); delete editor_vec_.at(page); text_vec_.erase(text_vec_.begin()+ page); linenumbers_vec_.erase(linenumbers_vec_.begin()+page); scrolledtext_vec_.erase(scrolledtext_vec_.begin()+page); scrolledline_vec_.erase(scrolledline_vec_.begin()+page); editor_vec_.erase(editor_vec_.begin()+page); } } void Notebook::Controller::OnFileNewEmptyfile() { entry_.OnShowSetFilenName(""); } void Notebook::Controller::OnFileNewCCFile() { entry_.OnShowSetFilenName(model_.cc_extension_); } void Notebook::Controller::OnFileNewHeaderFile() { entry_.OnShowSetFilenName(model_.h_extension_); } void Notebook::Controller::OnEditCopy() { if (Pages() != 0) { Buffer(text_vec_.at(CurrentPage()))->copy_clipboard(refClipboard_); } } void Notebook::Controller::OnEditPaste() { if (Pages() != 0) { Buffer(text_vec_.at(CurrentPage()))->paste_clipboard(refClipboard_); } } void Notebook::Controller::OnEditCut() { if (Pages() != 0) { Buffer(text_vec_.at(CurrentPage()))->cut_clipboard(refClipboard_); } } std::string Notebook::Controller::GetCursorWord() { INFO("Notebook get cursor word"); int page = CurrentPage(); std::string word; Gtk::TextIter start, end; start = Buffer(text_vec_.at(page))->get_insert()->get_iter(); end = Buffer(text_vec_.at(page))->get_insert()->get_iter(); if (!end.ends_line()) { while (!end.ends_word()) { end.forward_char(); } } if (!start.starts_line()) { while (!start.starts_word()) { start.backward_char(); } } word = Buffer(text_vec_.at(page))->get_text(start, end); // TODO(Oyvang) fix selected text return word; } void Notebook::Controller::OnEditSearch() { search_match_end_ = Buffer(text_vec_.at(CurrentPage()))->get_iter_at_offset(0); entry_.OnShowSearch(GetCursorWord()); } void Notebook::Controller::Search(bool forward) { INFO("Notebook search"); int page = CurrentPage(); std::string search_word; search_word = entry_.text(); Gtk::TextIter test; if ( !forward ) { if ( search_match_start_ == 0 || search_match_start_.get_line_offset() == 0) { search_match_start_ = Buffer(text_vec_.at(CurrentPage()))->end(); } search_match_start_. backward_search(search_word, Gtk::TextSearchFlags::TEXT_SEARCH_TEXT_ONLY | Gtk::TextSearchFlags::TEXT_SEARCH_VISIBLE_ONLY, search_match_start_, search_match_end_); } else { if ( search_match_end_ == 0 ) { search_match_end_ = Buffer(text_vec_.at(CurrentPage()))->begin(); } search_match_end_. forward_search(search_word, Gtk::TextSearchFlags::TEXT_SEARCH_TEXT_ONLY | Gtk::TextSearchFlags::TEXT_SEARCH_VISIBLE_ONLY, search_match_start_, search_match_end_); } } void Notebook::Controller::OnBufferChange() { int page = CurrentPage(); int text_nr = Buffer(text_vec_.at(page))->get_line_count(); int line_nr = Buffer(linenumbers_vec_.at(page))->get_line_count(); while (line_nr < text_nr) { line_nr++; Buffer(linenumbers_vec_.at(page))-> insert(Buffer(linenumbers_vec_.at(page))->end(), "\n"+std::to_string(line_nr)+" "); } while (line_nr > text_nr) { Gtk::TextIter iter = Buffer(linenumbers_vec_.at(page))->get_iter_at_line(line_nr); iter.backward_char(); line_nr--; Buffer(linenumbers_vec_.at(page))-> erase(iter, Buffer(linenumbers_vec_.at(page))->end()); } if (Buffer(text_vec_.at(page))->get_insert()->get_iter().starts_line() && Buffer(text_vec_.at(page))->get_insert()->get_iter().get_line() == Buffer(text_vec_.at(page))->end().get_line()) { GdkEventScroll* scroll = new GdkEventScroll; scroll->delta_y = 1.0; scroll->delta_x = 0.0; ScrollEventCallback(scroll); delete scroll; } text_vec_.at(page)->set_is_changed(true); } void Notebook::Controller ::OnDirectoryNavigation(const Gtk::TreeModel::Path& path, Gtk::TreeViewColumn* column) { INFO("Notebook directory navigation"); Gtk::TreeModel::iterator iter = directories().m_refTreeModel->get_iter(path); if (iter) { Gtk::TreeModel::Row row = *iter; std::string upath = Glib::ustring(row[directories().view().m_col_path]); boost::filesystem::path fs_path(upath); if (boost::filesystem::is_directory(fs_path)) { directories().m_TreeView.row_expanded(path) ? directories().m_TreeView.collapse_row(path) : directories().m_TreeView.expand_row(path, false); } else { std::stringstream sstm; sstm << row[directories().view().m_col_path]; std::string file = sstm.str(); OnOpenFile(file); } } } Gtk::TextView& Notebook::Controller::CurrentTextView() { return text_vec_.at(CurrentPage())->view(); } int Notebook::Controller::CurrentPage() { return Notebook().get_current_page(); } Glib::RefPtr Notebook::Controller::Buffer(Source::Controller *source) { return source->view().get_buffer(); } int Notebook::Controller::Pages() { return Notebook().get_n_pages(); } Gtk::Notebook& Notebook::Controller::Notebook() { return view_.notebook(); } void Notebook::Controller::BufferChangeHandler(Glib::RefPtr buffer) { buffer->signal_changed().connect( [this]() { OnBufferChange(); }); buffer->signal_end_user_action().connect( [this]() { //UpdateHistory(); }); } void Notebook::Controller::TextViewHandlers(Gtk::TextView& textview) { textview.get_buffer()->signal_changed().connect( [this]() { OnBufferChange(); }); textview.signal_button_release_event(). connect(sigc::mem_fun(*this, &Notebook::Controller::OnMouseRelease), false); textview.signal_key_release_event(). connect(sigc::mem_fun(*this, &Notebook::Controller::OnKeyRelease), false); } void Notebook::Controller::PopupSelectHandler(Gtk::Dialog &popup, Gtk::ListViewText &listview, std::map *items) { listview.signal_row_activated(). connect([this, &listview, &popup, items](const Gtk::TreeModel::Path& path, Gtk::TreeViewColumn*) { std::string selected = items-> at(listview.get_text(listview.get_selected()[0])); CurrentTextView().get_buffer()->insert_at_cursor(selected); popup.response(Gtk::RESPONSE_DELETE_EVENT); }); } void Notebook::Controller::PopupSetSize(Gtk::ScrolledWindow &scroll, int ¤t_x, int ¤t_y) { INFO("Notebook popup set size"); int textview_x = CurrentTextView().get_width(); int textview_y = 150; bool is_never_scroll_x = true; bool is_never_scroll_y = true; if (current_x > textview_x) { current_x = textview_x; is_never_scroll_x = false; } if (current_y > textview_y) { current_y = textview_y; is_never_scroll_y = false; } scroll.set_size_request(current_x, current_y); if (!is_never_scroll_x && !is_never_scroll_y) { scroll.set_policy(Gtk::PolicyType::POLICY_AUTOMATIC, Gtk::PolicyType::POLICY_AUTOMATIC); } else if (!is_never_scroll_x && is_never_scroll_y) { scroll.set_policy(Gtk::PolicyType::POLICY_AUTOMATIC, Gtk::PolicyType::POLICY_NEVER); } else if (is_never_scroll_x && !is_never_scroll_y) { scroll.set_policy(Gtk::PolicyType::POLICY_NEVER, Gtk::PolicyType::POLICY_AUTOMATIC); } } std::string Notebook::Controller::CurrentPagePath(){ return text_vec_.at(CurrentPage())->path(); } void Notebook::Controller::FindPopupPosition(Gtk::TextView& textview, int popup_x, int popup_y, int &x, int &y) { INFO("Notebook popup find position"); Gdk::Rectangle temp1, temp2; textview.get_cursor_locations( CurrentTextView(). get_buffer()->get_insert()-> get_iter(), temp1, temp2); int textview_edge_x = 0; int textview_edge_y = 0; textview.buffer_to_window_coords( Gtk::TextWindowType::TEXT_WINDOW_WIDGET, temp1.get_x(), temp1.get_y(), x, y); Glib::RefPtr gdkw = CurrentTextView().get_window(Gtk::TextWindowType::TEXT_WINDOW_WIDGET); gdkw->get_origin(textview_edge_x, textview_edge_y); x += textview_edge_x; y += textview_edge_y; if ((textview_edge_x-x)*-1 > textview.get_width()-popup_x) { x -= popup_x; if (x < textview_edge_x) x = textview_edge_x; } if ((textview_edge_y-y)*-1 > textview.get_height()-popup_y) { y -= (popup_y+14) + 15; if (x < textview_edge_y) y = textview_edge_y +15; } } void Notebook::Controller:: OnSaveFile() { INFO("Notebook save file"); if (text_vec_.at(CurrentPage())->is_saved()) { std::ofstream file; file.open (text_vec_.at(CurrentPage())->path()); file << CurrentTextView().get_buffer()->get_text(); file.close(); } else { std::string path = OnSaveFileAs(); if (path != "") { std::ofstream file; file.open (path); file << CurrentTextView().get_buffer()->get_text(); file.close(); text_vec_.at(CurrentPage())->set_file_path(path); text_vec_.at(CurrentPage())->set_is_saved(true); } } } std::string Notebook::Controller::OnSaveFileAs(){ INFO("Notebook save as"); Gtk::FileChooserDialog dialog("Please choose a file", Gtk::FILE_CHOOSER_ACTION_SAVE); DEBUG("SET TRANSISTEN FPR"); dialog.set_transient_for(*window_); dialog.set_position(Gtk::WindowPosition::WIN_POS_CENTER_ALWAYS); dialog.add_button("_Cancel", Gtk::RESPONSE_CANCEL); dialog.add_button("_Save", Gtk::RESPONSE_OK); //dialog.set_current_name("Untitled"); DEBUG("RUN DIALOG"); int result = dialog.run(); DEBUG("DIALOG RUNNING"); switch (result) { case(Gtk::RESPONSE_OK): { DEBUG("get_filename()"); std::string path = dialog.get_filename(); DEBUG_VAR(path); unsigned pos = path.find_last_of("/\\"); std::cout << path<< std::endl; //notebook_.OnSaveFile(path); return path; break; } case(Gtk::RESPONSE_CANCEL): { break; } default: { std::cout << "Unexpected button clicked." << std::endl; break; } } return ""; } void Notebook::Controller::AskToSaveDialog() { INFO("AskToSaveDialog"); DEBUG("AskToSaveDialog: Finding file path"); Gtk::MessageDialog dialog(*window_, "Save file!", false, Gtk::MESSAGE_QUESTION, Gtk::BUTTONS_YES_NO); dialog.set_secondary_text( "Do you want to save: " + text_vec_.at(CurrentPage())->path()+" ?"); DEBUG("AskToSaveDialog: run dialog"); int result = dialog.run(); //Handle the response: DEBUG("AskToSaveDialog: switch response"); switch(result) { case(Gtk::RESPONSE_YES): { DEBUG("AskToSaveDialog: save file: yes, trying to save file"); OnSaveFile(); DEBUG("AskToSaveDialog: save file: yes, saved sucess"); break; } case(Gtk::RESPONSE_NO): { DEBUG("AskToSaveDialog: save file: no"); break; } default: { DEBUG("AskToSaveDialog: unexpected action: Default switch"); break; } } } bool Notebook::Controller::LegalExtension(std::string e) { std::transform(e.begin(), e.end(),e.begin(), ::tolower); std::vector extensions = source_config().extensiontable(); if (find(extensions.begin(), extensions.end(), e) != extensions.end()) { DEBUG("Legal extension"); return true; } DEBUG("Ilegal extension"); return false; }