/* * Copyright (C) 2020-2022 Roy Qu (royqh1979@gmail.com) * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see <https://www.gnu.org/licenses/>. */ #include "editorlist.h" #include "editor.h" #include <QMessageBox> #include <QVariant> #include <mainwindow.h> #include <QFileInfo> #include "settings.h" #include "project.h" #include "systemconsts.h" #include <QApplication> EditorList::EditorList(QTabWidget* leftPageWidget, QTabWidget* rightPageWidget, QSplitter* splitter, QWidget* panel, QObject* parent): QObject(parent), mLayout(LayoutShowType::lstLeft), mLeftPageWidget(leftPageWidget), mRightPageWidget(rightPageWidget), mSplitter(splitter), mPanel(panel), mUpdateCount(0) { } Editor* EditorList::newEditor(const QString& filename, const QByteArray& encoding, bool inProject, bool newFile, QTabWidget* page) { QTabWidget * parentPageControl = nullptr; if (page == nullptr) parentPageControl = getNewEditorPageControl(); else parentPageControl = page; if (fileExists(filename)) { pMainWindow->fileSystemWatcher()->addPath(filename); } Editor * e = new Editor(parentPageControl,filename,encoding,inProject,newFile,parentPageControl); connect(e, &Editor::renamed, this, &EditorList::onEditorRenamed); updateLayout(); if (pMainWindow->project()){ PProjectUnit unit = pMainWindow->project()->findUnitByFilename(filename); if (unit) { pMainWindow->project()->associateEditorToUnit(e,unit); e->setInProject(true); } } connect(e,&Editor::fileSaved, pMainWindow, &MainWindow::onFileSaved); return e; } QTabWidget* EditorList::getNewEditorPageControl() const { return getFocusedPageControl(); } QTabWidget* EditorList::getFocusedPageControl() const { //todo: switch(mLayout) { case LayoutShowType::lstLeft: return mLeftPageWidget; case LayoutShowType::lstRight: return mRightPageWidget; case LayoutShowType::lstBoth: { Editor* editor = dynamic_cast<Editor*>(mRightPageWidget->currentWidget()); if (editor && editor->hasFocus()) return mRightPageWidget; return mLeftPageWidget; } default: return nullptr; } } void EditorList::showLayout(LayoutShowType layout) { if (layout == mLayout) return; mLayout = layout; // Apply widths if layout does not change switch(mLayout) { case LayoutShowType::lstLeft: mLeftPageWidget->setVisible(true); mRightPageWidget->setVisible(false); break; case LayoutShowType::lstRight: mLeftPageWidget->setVisible(false); mRightPageWidget->setVisible(true); break; case LayoutShowType::lstBoth: mLeftPageWidget->setVisible(true); mRightPageWidget->setVisible(true); } } void EditorList::onEditorRenamed(const QString &oldFilename, const QString &newFilename, bool firstSave) { emit editorRenamed(oldFilename, newFilename, firstSave); } QTabWidget *EditorList::rightPageWidget() const { return mRightPageWidget; } QTabWidget *EditorList::leftPageWidget() const { return mLeftPageWidget; } Editor* EditorList::getEditor(int index, QTabWidget* tabsWidget) const { QTabWidget* selectedWidget; if (tabsWidget == nullptr) { selectedWidget = getFocusedPageControl(); } else { selectedWidget = tabsWidget; } if (!selectedWidget) return nullptr; if (index == -1) { index = selectedWidget->currentIndex(); } if (index<0 || index >= selectedWidget->count()) { return nullptr; } return (Editor*)selectedWidget->widget(index); } bool EditorList::closeEditor(Editor* editor, bool transferFocus, bool force) { if (editor == NULL) return false; if (force) { editor->save(true,false); } else if ( (editor->modified()) && (!editor->empty())) { // ask user if he wants to save QMessageBox::StandardButton reply; reply = QMessageBox::question(editor,QObject::tr("Save"), QString(QObject::tr("Save changes to %1?")).arg(editor->filename()), QMessageBox::Yes|QMessageBox::No|QMessageBox::Cancel); if (reply == QMessageBox::Cancel) { return false; } else if (reply == QMessageBox::Yes) { if (!editor->save(false,false)) { return false; } } } beginUpdate(); auto end = finally([this] { this->endUpdate(); }); // if (transferFocus && (editor->pageControl()->currentWidget()==editor)) { // //todo: activate & focus the previous editor // } if (editor->inProject() && pMainWindow->project()) { int projIndex = pMainWindow->project()->indexInUnits(editor); if (projIndex>=0) { pMainWindow->project()->closeUnit(projIndex); } } else { if (pSettings->history().addToOpenedFiles(editor->filename())) { pMainWindow->rebuildOpenedFileHisotryMenu(); } delete editor; } updateLayout(); if (!force) { editor = getEditor(); if (transferFocus && editor) editor->activate(); else pMainWindow->updateClassBrowserForEditor(editor); } emit editorClosed(); return true; } bool EditorList::swapEditor(Editor *editor) { Q_ASSERT(editor!=nullptr); beginUpdate(); auto action = finally([this](){ endUpdate(); }); //remember old index QTabWidget* fromPageControl = editor->pageControl(); if (fromPageControl == mLeftPageWidget) { editor->setPageControl(mRightPageWidget); } else { editor->setPageControl(mLeftPageWidget); } updateLayout(); editor->activate(); return true; } void EditorList::beginUpdate() { if (mUpdateCount==0) { mPanel->setUpdatesEnabled(false); } mUpdateCount++; } void EditorList::endUpdate() { mUpdateCount--; if (mUpdateCount==0) { mPanel->setUpdatesEnabled(true); mPanel->update(); } } void EditorList::applySettings() { for (int i=0;i<mLeftPageWidget->count();i++) { Editor* e = static_cast<Editor*>(mLeftPageWidget->widget(i)); e->applySettings(); } for (int i=0;i<mRightPageWidget->count();i++) { Editor* e = static_cast<Editor*>(mRightPageWidget->widget(i)); e->applySettings(); } } void EditorList::applyColorSchemes(const QString& name) { for (int i=0;i<mLeftPageWidget->count();i++) { Editor* e = static_cast<Editor*>(mLeftPageWidget->widget(i)); e->applyColorScheme(name); } for (int i=0;i<mRightPageWidget->count();i++) { Editor* e = static_cast<Editor*>(mRightPageWidget->widget(i)); e->applyColorScheme(name); } } bool EditorList::isFileOpened(const QString &name) { QFileInfo fileInfo(name); QString filename = fileInfo.absoluteFilePath(); for (int i=0;i<mLeftPageWidget->count();i++) { Editor* e = static_cast<Editor*>(mLeftPageWidget->widget(i)); if (e->filename().compare(filename)==0 || e->filename().compare(name)==0) return true; } for (int i=0;i<mRightPageWidget->count();i++) { Editor* e = static_cast<Editor*>(mRightPageWidget->widget(i)); if (e->filename().compare(filename)==0 || e->filename().compare(name)==0) return true; } return false; } int EditorList::pageCount() { return mLeftPageWidget->count()+mRightPageWidget->count(); } void EditorList::selectNextPage() { QTabWidget * pageControl = getFocusedPageControl(); if (pageControl && pageControl->count()>0) { pageControl->setCurrentIndex( (pageControl->currentIndex()+1) % pageControl->count() ); } } void EditorList::selectPreviousPage() { QTabWidget * pageControl = getFocusedPageControl(); if (pageControl && pageControl->count()>0) { pageControl->setCurrentIndex( (pageControl->currentIndex()+pageControl->count()-1) % pageControl->count() ); } } Editor *EditorList::operator[](int index) { if (index>=0 && index<mLeftPageWidget->count()) { return static_cast<Editor*>(mLeftPageWidget->widget(index)); } index -= mLeftPageWidget->count(); if (index>=0 && index<mRightPageWidget->count()) { return static_cast<Editor*>(mRightPageWidget->widget(index)); } return nullptr; } bool EditorList::closeAll(bool force) { // beginUpdate(); // auto end = finally([this] { // this->endUpdate(); // }); while (mLeftPageWidget->count()>0) { if (!closeEditor(getEditor(0,mLeftPageWidget),false,force)) { return false; } } while (mRightPageWidget->count()>0) { if (!closeEditor(getEditor(0,mRightPageWidget),false,force)) { return false; } } return true; } void EditorList::forceCloseEditor(Editor *editor) { beginUpdate(); delete editor; // Force layout update when creating, destroying or moving editors updateLayout(); endUpdate(); emit editorClosed(); } Editor* EditorList::getOpenedEditorByFilename(QString filename) { if (filename.isEmpty()) return nullptr; QFileInfo fileInfo(filename); QString fullname = fileInfo.absoluteFilePath(); for (int i=0;i<mLeftPageWidget->count();i++) { Editor* e = static_cast<Editor*>(mLeftPageWidget->widget(i)); if (e->filename().compare(filename, PATH_SENSITIVITY)==0 || e->filename().compare(fullname, PATH_SENSITIVITY)==0) { return e; } } for (int i=0;i<mRightPageWidget->count();i++) { Editor* e = static_cast<Editor*>(mRightPageWidget->widget(i)); if (e->filename().compare(filename)==0 || e->filename().compare(fullname)==0) { return e; } } return nullptr; } Editor *EditorList::getEditorByFilename(QString filename) { if (filename.isEmpty()) return nullptr; //check if an editor is already openned Editor* e=getOpenedEditorByFilename(filename); if (e!=nullptr) return e; //Todo: check if is in the project //Create a new editor QFileInfo fileInfo(filename); QString fullname = fileInfo.absoluteFilePath(); if (fileInfo.exists() && fileInfo.isFile()) return newEditor(fullname,pSettings->editor().autoDetectFileEncoding()?ENCODING_AUTO_DETECT:pSettings->editor().defaultEncoding(),false,false); return nullptr; } bool EditorList::getContentFromOpenedEditor(const QString &filename, QStringList &buffer) { Editor * e= getOpenedEditorByFilename(filename); if (!e) return false; buffer = e->contents(); return true; } void EditorList::getVisibleEditors(Editor *&left, Editor *&right) { switch(mLayout) { case LayoutShowType::lstLeft: left = getEditor(-1,mLeftPageWidget); right = nullptr; break; case LayoutShowType::lstRight: left = nullptr; right = getEditor(-1,mRightPageWidget); break; case LayoutShowType::lstBoth: left = getEditor(-1,mLeftPageWidget); right = getEditor(-1,mRightPageWidget); break; } } void EditorList::updateLayout() { if (mRightPageWidget->count() == 0) showLayout(LayoutShowType::lstLeft); else if (mLeftPageWidget->count() ==0) showLayout(LayoutShowType::lstRight); else showLayout(LayoutShowType::lstBoth); }