/**************************************************************************** ** ** 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://qt.digia.com/licensing. For further information ** use the contact form at http://qt.digia.com/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 as published by the Free Software ** Foundation and appearing in the file LICENSE.LGPL included in the ** packaging of this file. Please review the following information to ** ensure the GNU Lesser General Public License version 2.1 requirements ** will be met: 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 "qmljseditor.h" #include "qmljsautocompleter.h" #include "qmljscompletionassist.h" #include "qmljseditorconstants.h" #include "qmljseditordocument.h" #include "qmljseditorplugin.h" #include "qmljsfindreferences.h" #include "qmljsquickfixassist.h" #include "qmloutlinemodel.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include enum { UPDATE_USES_DEFAULT_INTERVAL = 150, UPDATE_OUTLINE_INTERVAL = 500 // msecs after new semantic info has been arrived / cursor has moved }; using namespace Core; using namespace QmlJS; using namespace QmlJS::AST; using namespace QmlJSTools; using namespace TextEditor; namespace QmlJSEditor { namespace Internal { // // QmlJSEditorWidget // QmlJSEditorWidget::QmlJSEditorWidget() { m_outlineCombo = 0; m_contextPane = 0; m_findReferences = new FindReferences(this); setParenthesesMatchingEnabled(true); setMarksVisible(true); setCodeFoldingSupported(true); setLanguageSettingsId(QmlJSTools::Constants::QML_JS_SETTINGS_ID); } void QmlJSEditorWidget::finalizeInitialization() { m_qmlJsEditorDocument = static_cast(textDocument()); m_updateUsesTimer.setInterval(UPDATE_USES_DEFAULT_INTERVAL); m_updateUsesTimer.setSingleShot(true); connect(&m_updateUsesTimer, &QTimer::timeout, this, &QmlJSEditorWidget::updateUses); connect(this, &QPlainTextEdit::cursorPositionChanged, &m_updateUsesTimer, static_cast(&QTimer::start)); m_updateOutlineIndexTimer.setInterval(UPDATE_OUTLINE_INTERVAL); m_updateOutlineIndexTimer.setSingleShot(true); connect(&m_updateOutlineIndexTimer, &QTimer::timeout, this, &QmlJSEditorWidget::updateOutlineIndexNow); textDocument()->setCodec(QTextCodec::codecForName("UTF-8")); // qml files are defined to be utf-8 m_modelManager = QmlJS::ModelManagerInterface::instance(); m_contextPane = ExtensionSystem::PluginManager::getObject(); m_modelManager->activateScan(); m_contextPaneTimer.setInterval(UPDATE_OUTLINE_INTERVAL); m_contextPaneTimer.setSingleShot(true); connect(&m_contextPaneTimer, &QTimer::timeout, this, &QmlJSEditorWidget::updateContextPane); if (m_contextPane) { connect(this, &QmlJSEditorWidget::cursorPositionChanged, &m_contextPaneTimer, static_cast(&QTimer::start)); connect(m_contextPane, &IContextPane::closed, this, &QmlJSEditorWidget::showTextMarker); } m_oldCursorPosition = -1; connect(this->document(), &QTextDocument::modificationChanged, this, &QmlJSEditorWidget::modificationChanged); connect(m_qmlJsEditorDocument, SIGNAL(updateCodeWarnings(QmlJS::Document::Ptr)), this, SLOT(updateCodeWarnings(QmlJS::Document::Ptr))); connect(m_qmlJsEditorDocument, SIGNAL(semanticInfoUpdated(QmlJSTools::SemanticInfo)), this, SLOT(semanticInfoUpdated(QmlJSTools::SemanticInfo))); setRequestMarkEnabled(true); createToolBar(); } QModelIndex QmlJSEditorWidget::outlineModelIndex() { if (!m_outlineModelIndex.isValid()) { m_outlineModelIndex = indexForPosition(position()); emit outlineModelIndexChanged(m_outlineModelIndex); } return m_outlineModelIndex; } bool QmlJSEditor::open(QString *errorString, const QString &fileName, const QString &realFileName) { bool b = BaseTextEditor::open(errorString, fileName, realFileName); textDocument()->setMimeType(MimeDatabase::findByFile(QFileInfo(fileName)).type()); return b; } static void appendExtraSelectionsForMessages( QList *selections, const QList &messages, const QTextDocument *document) { foreach (const DiagnosticMessage &d, messages) { const int line = d.loc.startLine; const int column = qMax(1U, d.loc.startColumn); QTextEdit::ExtraSelection sel; QTextCursor c(document->findBlockByNumber(line - 1)); sel.cursor = c; sel.cursor.setPosition(c.position() + column - 1); if (d.loc.length == 0) { if (sel.cursor.atBlockEnd()) sel.cursor.movePosition(QTextCursor::StartOfWord, QTextCursor::KeepAnchor); else sel.cursor.movePosition(QTextCursor::EndOfWord, QTextCursor::KeepAnchor); } else { sel.cursor.movePosition(QTextCursor::NextCharacter, QTextCursor::KeepAnchor, d.loc.length); } if (d.isWarning()) sel.format.setUnderlineColor(Qt::darkYellow); else sel.format.setUnderlineColor(Qt::red); sel.format.setUnderlineStyle(QTextCharFormat::WaveUnderline); sel.format.setToolTip(d.message); selections->append(sel); } } void QmlJSEditorWidget::updateCodeWarnings(QmlJS::Document::Ptr doc) { if (doc->ast()) { setExtraSelections(CodeWarningsSelection, QList()); } else if (doc->language().isFullySupportedLanguage()) { // show parsing errors QList selections; appendExtraSelectionsForMessages(&selections, doc->diagnosticMessages(), document()); setExtraSelections(CodeWarningsSelection, selections); } else { setExtraSelections(CodeWarningsSelection, QList()); } } void QmlJSEditorWidget::modificationChanged(bool changed) { if (!changed && m_modelManager) m_modelManager->fileChangedOnDisk(textDocument()->filePath()); } void QmlJSEditorWidget::jumpToOutlineElement(int /*index*/) { QModelIndex index = m_outlineCombo->view()->currentIndex(); AST::SourceLocation location = m_qmlJsEditorDocument->outlineModel()->sourceLocation(index); if (!location.isValid()) return; EditorManager::cutForwardNavigationHistory(); EditorManager::addCurrentPositionToNavigationHistory(); QTextCursor cursor = textCursor(); cursor.setPosition(location.offset); setTextCursor(cursor); setFocus(); } void QmlJSEditorWidget::updateOutlineIndexNow() { if (!m_qmlJsEditorDocument->outlineModel()->document()) return; if (m_qmlJsEditorDocument->outlineModel()->document()->editorRevision() != document()->revision()) { m_updateOutlineIndexTimer.start(); return; } m_outlineModelIndex = QModelIndex(); // invalidate QModelIndex comboIndex = outlineModelIndex(); if (comboIndex.isValid()) { bool blocked = m_outlineCombo->blockSignals(true); // There is no direct way to select a non-root item m_outlineCombo->setRootModelIndex(comboIndex.parent()); m_outlineCombo->setCurrentIndex(comboIndex.row()); m_outlineCombo->setRootModelIndex(QModelIndex()); m_outlineCombo->blockSignals(blocked); } } } // namespace Internal } // namespace QmlJSEditor class QtQuickToolbarMarker {}; Q_DECLARE_METATYPE(QtQuickToolbarMarker) namespace QmlJSEditor { namespace Internal { template static QList removeMarkersOfType(const QList &markers) { QList result; foreach (const RefactorMarker &marker, markers) { if (!marker.data.canConvert()) result += marker; } return result; } void QmlJSEditorWidget::updateContextPane() { const SemanticInfo info = m_qmlJsEditorDocument->semanticInfo(); if (m_contextPane && document() && info.isValid() && document()->revision() == info.document->editorRevision()) { Node *oldNode = info.declaringMemberNoProperties(m_oldCursorPosition); Node *newNode = info.declaringMemberNoProperties(position()); if (oldNode != newNode && m_oldCursorPosition != -1) m_contextPane->apply(this, info.document, 0, newNode, false); if (m_contextPane->isAvailable(this, info.document, newNode) && !m_contextPane->widget()->isVisible()) { QList markers = removeMarkersOfType(refactorMarkers()); if (UiObjectMember *m = newNode->uiObjectMemberCast()) { const int start = qualifiedTypeNameId(m)->identifierToken.begin(); for (UiQualifiedId *q = qualifiedTypeNameId(m); q; q = q->next) { if (! q->next) { const int end = q->identifierToken.end(); if (position() >= start && position() <= end) { RefactorMarker marker; QTextCursor tc(document()); tc.setPosition(end); marker.cursor = tc; marker.tooltip = tr("Show Qt Quick ToolBar"); marker.data = QVariant::fromValue(QtQuickToolbarMarker()); markers.append(marker); } } } } setRefactorMarkers(markers); } else if (oldNode != newNode) { setRefactorMarkers(removeMarkersOfType(refactorMarkers())); } m_oldCursorPosition = position(); setSelectedElements(); } } void QmlJSEditorWidget::showTextMarker() { m_oldCursorPosition = -1; updateContextPane(); } void QmlJSEditorWidget::updateUses() { if (m_qmlJsEditorDocument->isSemanticInfoOutdated()) // will be updated when info is updated return; QList selections; foreach (const AST::SourceLocation &loc, m_qmlJsEditorDocument->semanticInfo().idLocations.value(wordUnderCursor())) { if (! loc.isValid()) continue; QTextEdit::ExtraSelection sel; sel.format = textDocument()->fontSettings().toTextCharFormat(TextEditor::C_OCCURRENCES); sel.cursor = textCursor(); sel.cursor.setPosition(loc.begin()); sel.cursor.setPosition(loc.end(), QTextCursor::KeepAnchor); selections.append(sel); } setExtraSelections(CodeSemanticsSelection, selections); } class SelectedElement: protected Visitor { unsigned m_cursorPositionStart; unsigned m_cursorPositionEnd; QList m_selectedMembers; public: SelectedElement() : m_cursorPositionStart(0), m_cursorPositionEnd(0) {} QList operator()(const Document::Ptr &doc, unsigned startPosition, unsigned endPosition) { m_cursorPositionStart = startPosition; m_cursorPositionEnd = endPosition; m_selectedMembers.clear(); Node::accept(doc->qmlProgram(), this); return m_selectedMembers; } protected: bool isSelectable(UiObjectMember *member) const { UiQualifiedId *id = qualifiedTypeNameId(member); if (id) { const QStringRef &name = id->name; if (!name.isEmpty() && name.at(0).isUpper()) return true; } return false; } inline bool isIdBinding(UiObjectMember *member) const { if (UiScriptBinding *script = cast(member)) { if (! script->qualifiedId) return false; else if (script->qualifiedId->name.isEmpty()) return false; else if (script->qualifiedId->next) return false; const QStringRef &propertyName = script->qualifiedId->name; if (propertyName == QLatin1String("id")) return true; } return false; } inline bool containsCursor(unsigned begin, unsigned end) { return m_cursorPositionStart >= begin && m_cursorPositionEnd <= end; } inline bool intersectsCursor(unsigned begin, unsigned end) { return (m_cursorPositionEnd >= begin && m_cursorPositionStart <= end); } inline bool isRangeSelected() const { return (m_cursorPositionStart != m_cursorPositionEnd); } void postVisit(Node *ast) { if (!isRangeSelected() && !m_selectedMembers.isEmpty()) return; // nothing to do, we already have the results. if (UiObjectMember *member = ast->uiObjectMemberCast()) { unsigned begin = member->firstSourceLocation().begin(); unsigned end = member->lastSourceLocation().end(); if ((isRangeSelected() && intersectsCursor(begin, end)) || (!isRangeSelected() && containsCursor(begin, end))) { if (initializerOfObject(member) && isSelectable(member)) { m_selectedMembers << member; // move start towards end; this facilitates multiselection so that root is usually ignored. m_cursorPositionStart = qMin(end, m_cursorPositionEnd); } } } } }; void QmlJSEditorWidget::setSelectedElements() { if (!receivers(SIGNAL(selectedElementsChanged(QList,QString)))) return; QTextCursor tc = textCursor(); QString wordAtCursor; QList offsets; unsigned startPos; unsigned endPos; if (tc.hasSelection()) { startPos = tc.selectionStart(); endPos = tc.selectionEnd(); } else { tc.movePosition(QTextCursor::StartOfWord); tc.movePosition(QTextCursor::EndOfWord, QTextCursor::KeepAnchor); startPos = textCursor().position(); endPos = textCursor().position(); } if (m_qmlJsEditorDocument->semanticInfo().isValid()) { SelectedElement selectedMembers; QList members = selectedMembers(m_qmlJsEditorDocument->semanticInfo().document, startPos, endPos); if (!members.isEmpty()) { foreach (UiObjectMember *m, members) { offsets << m; } } } wordAtCursor = tc.selectedText(); emit selectedElementsChanged(offsets, wordAtCursor); } void QmlJSEditorWidget::applyFontSettings() { BaseTextEditorWidget::applyFontSettings(); if (!m_qmlJsEditorDocument->isSemanticInfoOutdated()) updateUses(); } QString QmlJSEditorWidget::wordUnderCursor() const { QTextCursor tc = textCursor(); const QChar ch = document()->characterAt(tc.position() - 1); // make sure that we're not at the start of the next word. if (ch.isLetterOrNumber() || ch == QLatin1Char('_')) tc.movePosition(QTextCursor::Left); tc.movePosition(QTextCursor::StartOfWord); tc.movePosition(QTextCursor::EndOfWord, QTextCursor::KeepAnchor); const QString word = tc.selectedText(); return word; } bool QmlJSEditorWidget::isClosingBrace(const QList &tokens) const { if (tokens.size() == 1) { const Token firstToken = tokens.first(); return firstToken.is(Token::RightBrace) || firstToken.is(Token::RightBracket); } return false; } void QmlJSEditorWidget::createToolBar() { m_outlineCombo = new QComboBox; m_outlineCombo->setMinimumContentsLength(22); m_outlineCombo->setModel(m_qmlJsEditorDocument->outlineModel()); QTreeView *treeView = new QTreeView; Utils::AnnotatedItemDelegate *itemDelegate = new Utils::AnnotatedItemDelegate(this); itemDelegate->setDelimiter(QLatin1String(" ")); itemDelegate->setAnnotationRole(QmlOutlineModel::AnnotationRole); treeView->setItemDelegateForColumn(0, itemDelegate); treeView->header()->hide(); treeView->setItemsExpandable(false); treeView->setRootIsDecorated(false); m_outlineCombo->setView(treeView); treeView->expandAll(); //m_outlineCombo->setSizeAdjustPolicy(QComboBox::AdjustToContents); // Make the combo box prefer to expand QSizePolicy policy = m_outlineCombo->sizePolicy(); policy.setHorizontalPolicy(QSizePolicy::Expanding); m_outlineCombo->setSizePolicy(policy); connect(m_outlineCombo, SIGNAL(activated(int)), this, SLOT(jumpToOutlineElement(int))); connect(m_qmlJsEditorDocument->outlineModel(), SIGNAL(updated()), m_outlineCombo->view()/*QTreeView*/, SLOT(expandAll())); connect(m_qmlJsEditorDocument->outlineModel(), SIGNAL(updated()), this, SLOT(updateOutlineIndexNow())); connect(this, &QmlJSEditorWidget::cursorPositionChanged, &m_updateOutlineIndexTimer, static_cast(&QTimer::start)); insertExtraToolBarWidget(BaseTextEditorWidget::Left, m_outlineCombo); } BaseTextEditorWidget::Link QmlJSEditorWidget::findLinkAt(const QTextCursor &cursor, bool /*resolveTarget*/, bool /*inNextSplit*/) { const SemanticInfo semanticInfo = m_qmlJsEditorDocument->semanticInfo(); if (! semanticInfo.isValid()) return Link(); const unsigned cursorPosition = cursor.position(); AST::Node *node = semanticInfo.astNodeAt(cursorPosition); QTC_ASSERT(node, return Link()); if (AST::UiImport *importAst = cast(node)) { // if it's a file import, link to the file foreach (const ImportInfo &import, semanticInfo.document->bind()->imports()) { if (import.ast() == importAst && import.type() == ImportType::File) { BaseTextEditorWidget::Link link(import.path()); link.linkTextStart = importAst->firstSourceLocation().begin(); link.linkTextEnd = importAst->lastSourceLocation().end(); return link; } } return Link(); } // string literals that could refer to a file link to them if (StringLiteral *literal = cast(node)) { const QString &text = literal->value.toString(); BaseTextEditorWidget::Link link; link.linkTextStart = literal->literalToken.begin(); link.linkTextEnd = literal->literalToken.end(); if (semanticInfo.snapshot.document(text)) { link.targetFileName = text; return link; } const QString relative = QString::fromLatin1("%1/%2").arg( semanticInfo.document->path(), text); if (semanticInfo.snapshot.document(relative)) { link.targetFileName = relative; return link; } } const ScopeChain scopeChain = semanticInfo.scopeChain(semanticInfo.rangePath(cursorPosition)); Evaluate evaluator(&scopeChain); const Value *value = evaluator.reference(node); QString fileName; int line = 0, column = 0; if (! (value && value->getSourceLocation(&fileName, &line, &column))) return Link(); BaseTextEditorWidget::Link link; link.targetFileName = fileName; link.targetLine = line; link.targetColumn = column - 1; // adjust the column if (AST::UiQualifiedId *q = AST::cast(node)) { for (AST::UiQualifiedId *tail = q; tail; tail = tail->next) { if (! tail->next && cursorPosition <= tail->identifierToken.end()) { link.linkTextStart = tail->identifierToken.begin(); link.linkTextEnd = tail->identifierToken.end(); return link; } } } else if (AST::IdentifierExpression *id = AST::cast(node)) { link.linkTextStart = id->firstSourceLocation().begin(); link.linkTextEnd = id->lastSourceLocation().end(); return link; } else if (AST::FieldMemberExpression *mem = AST::cast(node)) { link.linkTextStart = mem->lastSourceLocation().begin(); link.linkTextEnd = mem->lastSourceLocation().end(); return link; } return Link(); } void QmlJSEditorWidget::findUsages() { m_findReferences->findUsages(textDocument()->filePath(), textCursor().position()); } void QmlJSEditorWidget::renameUsages() { m_findReferences->renameUsages(textDocument()->filePath(), textCursor().position()); } void QmlJSEditorWidget::showContextPane() { const SemanticInfo info = m_qmlJsEditorDocument->semanticInfo(); if (m_contextPane && info.isValid()) { Node *newNode = info.declaringMemberNoProperties(position()); ScopeChain scopeChain = info.scopeChain(info.rangePath(position())); m_contextPane->apply(this, info.document, &scopeChain, newNode, false, true); m_oldCursorPosition = position(); setRefactorMarkers(removeMarkersOfType(refactorMarkers())); } } void QmlJSEditorWidget::performQuickFix(int index) { QuickFixOperation::Ptr op = m_quickFixes.at(index); op->perform(); } void QmlJSEditorWidget::contextMenuEvent(QContextMenuEvent *e) { QPointer menu(new QMenu(this)); QMenu *refactoringMenu = new QMenu(tr("Refactoring"), menu); QSignalMapper mapper; connect(&mapper, SIGNAL(mapped(int)), this, SLOT(performQuickFix(int))); if (!m_qmlJsEditorDocument->isSemanticInfoOutdated()) { IAssistInterface *interface = createAssistInterface(QuickFix, ExplicitlyInvoked); if (interface) { QScopedPointer processor( QmlJSEditorPlugin::instance()->quickFixAssistProvider()->createProcessor()); QScopedPointer proposal(processor->perform(interface)); if (!proposal.isNull()) { BasicProposalItemListModel *model = static_cast(proposal->model()); for (int index = 0; index < model->size(); ++index) { BasicProposalItem *item = static_cast(model->proposalItem(index)); QuickFixOperation::Ptr op = item->data().value(); m_quickFixes.append(op); QAction *action = refactoringMenu->addAction(op->description()); mapper.setMapping(action, index); connect(action, SIGNAL(triggered()), &mapper, SLOT(map())); } delete model; } } } refactoringMenu->setEnabled(!refactoringMenu->isEmpty()); if (ActionContainer *mcontext = ActionManager::actionContainer(Constants::M_CONTEXT)) { QMenu *contextMenu = mcontext->menu(); foreach (QAction *action, contextMenu->actions()) { menu->addAction(action); if (action->objectName() == QLatin1String(Constants::M_REFACTORING_MENU_INSERTION_POINT)) menu->addMenu(refactoringMenu); if (action->objectName() == QLatin1String(Constants::SHOW_QT_QUICK_HELPER)) { bool enabled = m_contextPane->isAvailable( this, m_qmlJsEditorDocument->semanticInfo().document, m_qmlJsEditorDocument->semanticInfo().declaringMemberNoProperties(position())); action->setEnabled(enabled); } } } appendStandardContextMenuActions(menu); menu->exec(e->globalPos()); if (!menu) return; m_quickFixes.clear(); delete menu; } bool QmlJSEditorWidget::event(QEvent *e) { switch (e->type()) { case QEvent::ShortcutOverride: if (static_cast(e)->key() == Qt::Key_Escape && m_contextPane) { if (hideContextPane()) { e->accept(); return true; } } break; default: break; } return BaseTextEditorWidget::event(e); } void QmlJSEditorWidget::wheelEvent(QWheelEvent *event) { bool visible = false; if (m_contextPane && m_contextPane->widget()->isVisible()) visible = true; BaseTextEditorWidget::wheelEvent(event); if (visible) m_contextPane->apply(this, m_qmlJsEditorDocument->semanticInfo().document, 0, m_qmlJsEditorDocument->semanticInfo().declaringMemberNoProperties(m_oldCursorPosition), false, true); } void QmlJSEditorWidget::resizeEvent(QResizeEvent *event) { BaseTextEditorWidget::resizeEvent(event); hideContextPane(); } void QmlJSEditorWidget::scrollContentsBy(int dx, int dy) { BaseTextEditorWidget::scrollContentsBy(dx, dy); hideContextPane(); } QmlJSEditorDocument *QmlJSEditorWidget::qmlJsEditorDocument() const { return m_qmlJsEditorDocument; } void QmlJSEditorWidget::semanticInfoUpdated(const SemanticInfo &semanticInfo) { if (isVisible()) { // trigger semantic highlighting and model update if necessary textDocument()->triggerPendingUpdates(); } if (m_contextPane) { Node *newNode = semanticInfo.declaringMemberNoProperties(position()); if (newNode) { m_contextPane->apply(this, semanticInfo.document, 0, newNode, true); m_contextPaneTimer.start(); //update text marker } } updateUses(); } void QmlJSEditorWidget::onRefactorMarkerClicked(const RefactorMarker &marker) { if (marker.data.canConvert()) showContextPane(); } QModelIndex QmlJSEditorWidget::indexForPosition(unsigned cursorPosition, const QModelIndex &rootIndex) const { QModelIndex lastIndex = rootIndex; QmlOutlineModel *model = m_qmlJsEditorDocument->outlineModel(); const int rowCount = model->rowCount(rootIndex); for (int i = 0; i < rowCount; ++i) { QModelIndex childIndex = model->index(i, 0, rootIndex); AST::SourceLocation location = model->sourceLocation(childIndex); if ((cursorPosition >= location.offset) && (cursorPosition <= location.offset + location.length)) { lastIndex = childIndex; break; } } if (lastIndex != rootIndex) { // recurse lastIndex = indexForPosition(cursorPosition, lastIndex); } return lastIndex; } bool QmlJSEditorWidget::hideContextPane() { bool b = (m_contextPane) && m_contextPane->widget()->isVisible(); if (b) m_contextPane->apply(this, m_qmlJsEditorDocument->semanticInfo().document, 0, 0, false); return b; } IAssistInterface *QmlJSEditorWidget::createAssistInterface( TextEditor::AssistKind assistKind, TextEditor::AssistReason reason) const { if (assistKind == TextEditor::Completion) { return new QmlJSCompletionAssistInterface(document(), position(), textDocument()->filePath(), reason, m_qmlJsEditorDocument->semanticInfo()); } else if (assistKind == TextEditor::QuickFix) { return new QmlJSQuickFixAssistInterface(const_cast(this), reason); } return 0; } QString QmlJSEditorWidget::foldReplacementText(const QTextBlock &block) const { const int curlyIndex = block.text().indexOf(QLatin1Char('{')); if (curlyIndex != -1 && m_qmlJsEditorDocument->semanticInfo().isValid()) { const int pos = block.position() + curlyIndex; Node *node = m_qmlJsEditorDocument->semanticInfo().rangeAt(pos); const QString objectId = idOfObject(node); if (!objectId.isEmpty()) return QLatin1String("id: ") + objectId + QLatin1String("..."); } return TextEditor::BaseTextEditorWidget::foldReplacementText(block); } // // QmlJSEditor // QmlJSEditor::QmlJSEditor() { addContext(Constants::C_QMLJSEDITOR_ID); addContext(ProjectExplorer::Constants::LANG_QMLJS); setDuplicateSupported(true); setCommentStyle(Utils::CommentDefinition::CppStyle); setCompletionAssistProvider(ExtensionSystem::PluginManager::getObject()); } void QmlJSEditor::finalizeInitialization() { configureCodeAssistant(); } bool QmlJSEditor::isDesignModePreferred() const { // stay in design mode if we are there IMode *mode = ModeManager::currentMode(); return mode && mode->id() == Core::Constants::MODE_DESIGN; } // // QmlJSEditorFactory // QmlJSEditorFactory::QmlJSEditorFactory() { setId(Constants::C_QMLJSEDITOR_ID); setDisplayName(qApp->translate("OpenWith::Editors", Constants::C_QMLJSEDITOR_DISPLAY_NAME)); addMimeType(QmlJSTools::Constants::QML_MIMETYPE); addMimeType(QmlJSTools::Constants::QMLPROJECT_MIMETYPE); addMimeType(QmlJSTools::Constants::QBS_MIMETYPE); addMimeType(QmlJSTools::Constants::QMLTYPES_MIMETYPE); addMimeType(QmlJSTools::Constants::JS_MIMETYPE); addMimeType(QmlJSTools::Constants::JSON_MIMETYPE); setDocumentCreator([]() { return new QmlJSEditorDocument; }); setEditorWidgetCreator([]() { return new QmlJSEditorWidget; }); setEditorCreator([]() { return new QmlJSEditor; }); setAutoCompleterCreator([]() { return new AutoCompleter; }); setEditorActionHandlers(Constants::C_QMLJSEDITOR_ID, TextEditorActionHandler::Format | TextEditorActionHandler::UnCommentSelection | TextEditorActionHandler::UnCollapseAll | TextEditorActionHandler::FollowSymbolUnderCursor); } } // namespace Internal } // namespace QmlJSEditor