/**************************************************************************** ** ** Copyright (C) 2014 Digia Plc and/or its subsidiary(-ies). ** Contact: http://www.qt-project.org/legal ** ** This file is part of Qt Creator. ** ** Commercial License Usage ** Licensees holding valid commercial Qt licenses may use this file in ** accordance with the commercial license agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and Digia. For licensing terms and ** conditions see http://www.qt.io/licensing. For further information ** use the contact form at http://www.qt.io/contact-us. ** ** GNU Lesser General Public License Usage ** Alternatively, this file may be used under the terms of the GNU Lesser ** General Public License version 2.1 or version 3 as published by the Free ** Software Foundation and appearing in the file LICENSE.LGPLv21 and ** LICENSE.LGPLv3 included in the packaging of this file. Please review the ** following information to ensure the GNU Lesser General Public License ** requirements will be met: https://www.gnu.org/licenses/lgpl.html and ** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. ** ** In addition, as a special exception, Digia gives you certain additional ** rights. These rights are described in the Digia Qt LGPL Exception ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. ** ****************************************************************************/ #include "refactoringchanges.h" #include "basetexteditor.h" #include #include #include #include #include #include #include #include #include #include #include using namespace TextEditor; RefactoringChanges::RefactoringChanges() : m_data(new RefactoringChangesData) {} RefactoringChanges::RefactoringChanges(RefactoringChangesData *data) : m_data(data) {} RefactoringChanges::~RefactoringChanges() {} QList > RefactoringChanges::rangesToSelections(QTextDocument *document, const QList &ranges) { QList > selections; foreach (const Range &range, ranges) { QTextCursor start(document); start.setPosition(range.start); start.setKeepPositionOnInsert(true); QTextCursor end(document); end.setPosition(qMin(range.end, document->characterCount() - 1)); selections.append(qMakePair(start, end)); } return selections; } bool RefactoringChanges::createFile(const QString &fileName, const QString &contents, bool reindent, bool openEditor) const { if (QFile::exists(fileName)) return false; // Create a text document for the new file: QTextDocument *document = new QTextDocument; QTextCursor cursor(document); cursor.beginEditBlock(); cursor.insertText(contents); // Reindent the contents: if (reindent) { cursor.select(QTextCursor::Document); m_data->indentSelection(cursor, fileName, 0); } cursor.endEditBlock(); // Write the file to disk: Utils::TextFileFormat format; format.codec = Core::EditorManager::defaultTextCodec(); QString error; bool saveOk = format.writeFile(fileName, document->toPlainText(), &error); delete document; if (!saveOk) return false; m_data->fileChanged(fileName); if (openEditor) this->openEditor(fileName, /*bool activate =*/ false, -1, -1); return true; } bool RefactoringChanges::removeFile(const QString &fileName) const { if (!QFile::exists(fileName)) return false; // ### implement! qWarning() << "RefactoringChanges::removeFile is not implemented"; return true; } BaseTextEditorWidget *RefactoringChanges::openEditor(const QString &fileName, bool activate, int line, int column) { Core::EditorManager::OpenEditorFlags flags = Core::EditorManager::IgnoreNavigationHistory; if (!activate) flags |= Core::EditorManager::DoNotChangeCurrentEditor; if (line != -1) { // openEditorAt uses a 1-based line and a 0-based column! column -= 1; } Core::IEditor *editor = Core::EditorManager::openEditorAt( fileName, line, column, Core::Id(), flags); if (editor) return qobject_cast(editor->widget()); else return 0; } RefactoringFilePtr RefactoringChanges::file(BaseTextEditorWidget *editor) { return RefactoringFilePtr(new RefactoringFile(editor)); } RefactoringFilePtr RefactoringChanges::file(const QString &fileName) const { return RefactoringFilePtr(new RefactoringFile(fileName, m_data)); } RefactoringFile::RefactoringFile(QTextDocument *document, const QString &fileName) : m_fileName(fileName) , m_document(document) , m_editor(0) , m_openEditor(false) , m_activateEditor(false) , m_editorCursorPosition(-1) , m_appliedOnce(false) { } RefactoringFile::RefactoringFile(BaseTextEditorWidget *editor) : m_fileName(editor->baseTextDocument()->filePath()) , m_document(0) , m_editor(editor) , m_openEditor(false) , m_activateEditor(false) , m_editorCursorPosition(-1) , m_appliedOnce(false) { } RefactoringFile::RefactoringFile(const QString &fileName, const QSharedPointer &data) : m_fileName(fileName) , m_data(data) , m_document(0) , m_editor(0) , m_openEditor(false) , m_activateEditor(false) , m_editorCursorPosition(-1) , m_appliedOnce(false) { QList editors = Core::DocumentModel::editorsForFilePath(fileName); if (!editors.isEmpty()) m_editor = qobject_cast(editors.first()->widget()); } RefactoringFile::~RefactoringFile() { delete m_document; } bool RefactoringFile::isValid() const { if (m_fileName.isEmpty()) return false; return document(); } const QTextDocument *RefactoringFile::document() const { return mutableDocument(); } QTextDocument *RefactoringFile::mutableDocument() const { if (m_editor) return m_editor->document(); if (!m_document) { QString fileContents; if (!m_fileName.isEmpty()) { QString error; QTextCodec *defaultCodec = Core::EditorManager::defaultTextCodec(); Utils::TextFileFormat::ReadResult result = Utils::TextFileFormat::readFile( m_fileName, defaultCodec, &fileContents, &m_textFileFormat, &error); if (result != Utils::TextFileFormat::ReadSuccess) { qWarning() << "Could not read " << m_fileName << ". Error: " << error; m_textFileFormat.codec = 0; } } // always make a QTextDocument to avoid excessive null checks m_document = new QTextDocument(fileContents); } return m_document; } const QTextCursor RefactoringFile::cursor() const { if (m_editor) return m_editor->textCursor(); if (!m_fileName.isEmpty()) { if (QTextDocument *doc = mutableDocument()) return QTextCursor(doc); } return QTextCursor(); } QString RefactoringFile::fileName() const { return m_fileName; } BaseTextEditorWidget *RefactoringFile::editor() const { return m_editor; } int RefactoringFile::position(unsigned line, unsigned column) const { QTC_ASSERT(line != 0, return -1); QTC_ASSERT(column != 0, return -1); if (const QTextDocument *doc = document()) return doc->findBlockByNumber(line - 1).position() + column - 1; return -1; } void RefactoringFile::lineAndColumn(int offset, unsigned *line, unsigned *column) const { QTC_ASSERT(line, return); QTC_ASSERT(column, return); QTC_ASSERT(offset >= 0, return); QTextCursor c(cursor()); c.setPosition(offset); *line = c.blockNumber() + 1; *column = c.positionInBlock() + 1; } QChar RefactoringFile::charAt(int pos) const { if (const QTextDocument *doc = document()) return doc->characterAt(pos); return QChar(); } QString RefactoringFile::textOf(int start, int end) const { QTextCursor c = cursor(); c.setPosition(start); c.setPosition(end, QTextCursor::KeepAnchor); return c.selectedText(); } QString RefactoringFile::textOf(const Range &range) const { return textOf(range.start, range.end); } void RefactoringFile::setChangeSet(const Utils::ChangeSet &changeSet) { if (m_fileName.isEmpty()) return; m_changes = changeSet; } void RefactoringFile::appendIndentRange(const Range &range) { if (m_fileName.isEmpty()) return; m_indentRanges.append(range); } void RefactoringFile::appendReindentRange(const Range &range) { if (m_fileName.isEmpty()) return; m_reindentRanges.append(range); } void RefactoringFile::setOpenEditor(bool activate, int pos) { m_openEditor = true; m_activateEditor = activate; m_editorCursorPosition = pos; } void RefactoringFile::apply() { // test file permissions if (!QFileInfo(fileName()).isWritable()) { const QString &path = fileName(); Core::Internal::ReadOnlyFilesDialog roDialog(path, Core::ICore::mainWindow()); const QString &failDetailText = QApplication::translate("RefactoringFile::apply", "Refactoring cannot be applied."); roDialog.setShowFailWarning(true, failDetailText); if (roDialog.exec() == Core::Internal::ReadOnlyFilesDialog::RO_Cancel) return; } // open / activate / goto position if (m_openEditor && !m_fileName.isEmpty()) { unsigned line = unsigned(-1), column = unsigned(-1); if (m_editorCursorPosition != -1) lineAndColumn(m_editorCursorPosition, &line, &column); m_editor = RefactoringChanges::openEditor(m_fileName, m_activateEditor, line, column); m_openEditor = false; m_activateEditor = false; m_editorCursorPosition = -1; } // apply changes, if any if (m_data && !(m_indentRanges.isEmpty() && m_changes.isEmpty())) { QTextDocument *doc = mutableDocument(); if (doc) { QTextCursor c = cursor(); if (m_appliedOnce) c.joinPreviousEditBlock(); else c.beginEditBlock(); // build indent selections now, applying the changeset will change locations const QList > &indentSelections = RefactoringChanges::rangesToSelections(doc, m_indentRanges); m_indentRanges.clear(); const QList > &reindentSelections = RefactoringChanges::rangesToSelections(doc, m_reindentRanges); m_reindentRanges.clear(); // apply changes and reindent m_changes.apply(&c); m_changes.clear(); indentOrReindent(&RefactoringChangesData::indentSelection, indentSelections); indentOrReindent(&RefactoringChangesData::reindentSelection, reindentSelections); c.endEditBlock(); // if this document doesn't have an editor, write the result to a file if (!m_editor && m_textFileFormat.codec) { QTC_ASSERT(!m_fileName.isEmpty(), return); QString error; if (!m_textFileFormat.writeFile(m_fileName, doc->toPlainText(), &error)) qWarning() << "Could not apply changes to" << m_fileName << ". Error: " << error; } fileChanged(); } } m_appliedOnce = true; } void RefactoringFile::indentOrReindent(void (RefactoringChangesData::*mf)(const QTextCursor &, const QString &, const BaseTextDocument *) const, const QList > &ranges) { typedef QPair CursorPair; foreach (const CursorPair &p, ranges) { QTextCursor selection(p.first.document()); selection.setPosition(p.first.position()); selection.setPosition(p.second.position(), QTextCursor::KeepAnchor); ((*m_data).*(mf))(selection, m_fileName, m_editor ? m_editor->baseTextDocument() : 0); } } void RefactoringFile::fileChanged() { if (!m_fileName.isEmpty()) m_data->fileChanged(m_fileName); } RefactoringChangesData::~RefactoringChangesData() {} void RefactoringChangesData::indentSelection(const QTextCursor &, const QString &, const BaseTextDocument *) const { qWarning() << Q_FUNC_INFO << "not implemented"; } void RefactoringChangesData::reindentSelection(const QTextCursor &, const QString &, const BaseTextDocument *) const { qWarning() << Q_FUNC_INFO << "not implemented"; } void RefactoringChangesData::fileChanged(const QString &) { }