#include "autocompleteplugin.hpp" #include #include #include #include #include #include #include using namespace EE::Graphics; using namespace EE::System; using json = nlohmann::json; namespace ecode { #if EE_PLATFORM != EE_PLATFORM_EMSCRIPTEN || defined( __EMSCRIPTEN_PTHREADS__ ) #define AUTO_COMPLETE_THREADED 1 #else #define AUTO_COMPLETE_THREADED 0 #endif static json getURIJSON( TextDocument* doc, const PluginIDType& id ) { json data; data["uri"] = doc->getURI().toString(); if ( id.isInteger() ) data["id"] = id.asInt(); else data["id"] = id.asString(); return data; } static json getURIAndPositionJSON( UICodeEditor* editor ) { json data; auto doc = editor->getDocumentRef(); auto sel = doc->getSelection(); data["uri"] = doc->getURI().toString(); data["position"] = { { "line", sel.start().line() }, { "character", sel.start().column() } }; return data; } static AutoCompletePlugin::SymbolsList fuzzyMatchSymbols( const std::vector& symbolsVec, const std::string& match, const size_t& max ) { AutoCompletePlugin::SymbolsList matches; matches.reserve( max ); int score; for ( const auto& symbols : symbolsVec ) { for ( const auto& symbol : *symbols ) { if ( ( score = String::fuzzyMatch( symbol.text, match ) ) > 0 ) { if ( std::find( matches.begin(), matches.end(), symbol ) == matches.end() ) { symbol.setScore( score ); matches.push_back( symbol ); } } } if ( matches.size() > max ) break; } std::sort( matches.begin(), matches.end(), []( const AutoCompletePlugin::Suggestion& left, const AutoCompletePlugin::Suggestion& right ) { return left.score > right.score && left.kind != LSPCompletionItemKind::Text; } ); return matches; } UICodeEditorPlugin* AutoCompletePlugin::New( PluginManager* pluginManager ) { return eeNew( AutoCompletePlugin, ( pluginManager ) ); } AutoCompletePlugin::AutoCompletePlugin( PluginManager* pluginManager ) : mManager( pluginManager ), mSymbolPattern( "[%a_ñàáâãäåèéêëìíîïòóôõöùúûüýÿÑÀÁÂÃÄÅÈÉÊËÌÍÎÏÒÓÔÕÖÙÚÛÜÝ][%w_" "ñàáâãäåèéêëìíîïòóôõöùúûüýÿÑÀÁÂÃÄÅÈÉÊËÌÍÎÏÒÓÔÕÖÙÚÛÜÝ]*" ), mBoxPadding( PixelDensity::dpToPx( Rectf( 4, 4, 12, 4 ) ) ), mPool( pluginManager->getThreadPool() ) { mManager->subscribeMessages( this, [&]( const PluginMessage& msg ) -> PluginRequestHandle { return processResponse( msg ); } ); } AutoCompletePlugin::~AutoCompletePlugin() { mClosing = true; mManager->unsubscribeMessages( this ); Lock l( mDocMutex ); Lock l2( mLangSymbolsMutex ); Lock l3( mSuggestionsMutex ); for ( const auto& editor : mEditors ) { for ( auto listener : editor.second ) editor.first->removeEventListener( listener ); editor.first->unregisterPlugin( this ); } } void AutoCompletePlugin::onRegister( UICodeEditor* editor ) { Lock l( mDocMutex ); std::vector listeners; listeners.push_back( editor->addEventListener( Event::OnDocumentLoaded, [&, editor]( const Event* ) { mDirty = true; tryRequestCapabilities( editor ); } ) ); listeners.push_back( editor->addEventListener( Event::OnDocumentClosed, [&]( const Event* event ) { Lock l( mDocMutex ); const DocEvent* docEvent = static_cast( event ); TextDocument* doc = docEvent->getDoc(); mDocs.erase( doc ); mDocCache.erase( doc ); mDirty = true; } ) ); listeners.push_back( editor->addEventListener( Event::OnDocumentChanged, [&, editor]( const Event* ) { TextDocument* oldDoc = mEditorDocs[editor]; TextDocument* newDoc = editor->getDocumentRef().get(); Lock l( mDocMutex ); mDocs.erase( oldDoc ); mDocCache.erase( oldDoc ); mEditorDocs[editor] = newDoc; mDirty = true; } ) ); listeners.push_back( editor->addEventListener( Event::OnCursorPosChange, [&, editor]( const Event* ) { if ( !mReplacing ) resetSuggestions( editor ); } ) ); listeners.push_back( editor->addEventListener( Event::OnFocusLoss, [&]( const Event* ) { resetSignatureHelp(); } ) ); listeners.push_back( editor->addEventListener( Event::OnDocumentUndoRedo, [&]( const Event* ) { resetSignatureHelp(); } ) ); listeners.push_back( editor->addEventListener( Event::OnDocumentSyntaxDefinitionChange, [&]( const Event* ev ) { const DocSyntaxDefEvent* event = static_cast( ev ); std::string oldLang = event->getOldLang(); std::string newLang = event->getNewLang(); #if AUTO_COMPLETE_THREADED mPool->run( [&, oldLang, newLang] { updateLangCache( oldLang ); updateLangCache( newLang ); } ); #else updateLangCache( oldLang ); updateLangCache( newLang ); #endif } ) ); mEditors.insert( { editor, listeners } ); mDocs.insert( editor->getDocumentRef().get() ); mEditorDocs[editor] = editor->getDocumentRef().get(); mDirty = true; } void AutoCompletePlugin::onUnregister( UICodeEditor* editor ) { if ( mClosing ) return; if ( mSuggestionsEditor == editor ) resetSuggestions( editor ); if ( mSignatureHelpEditor == editor ) resetSignatureHelp(); Lock l( mDocMutex ); TextDocument* doc = mEditorDocs[editor]; auto cbs = mEditors[editor]; for ( auto listener : cbs ) editor->removeEventListener( listener ); mEditors.erase( editor ); mEditorDocs.erase( editor ); for ( auto editor : mEditorDocs ) if ( editor.second == doc ) return; mDocs.erase( doc ); mDocCache.erase( doc ); mDirty = true; } bool AutoCompletePlugin::onKeyDown( UICodeEditor* editor, const KeyEvent& event ) { bool ret = false; if ( mSignatureHelpVisible ) { if ( event.getKeyCode() == KEY_ESCAPE ) { resetSignatureHelp(); editor->invalidateDraw(); ret = true; } else if ( event.getKeyCode() == KEY_UP ) { if ( mSignatureHelp.signatures.size() > 1 ) { mSignatureHelpSelected = mSignatureHelpSelected == -1 ? 0 : mSignatureHelpSelected; ++mSignatureHelpSelected; mSignatureHelpSelected = mSignatureHelpSelected % (int)mSignatureHelp.signatures.size(); editor->invalidateDraw(); return true; } else if ( mSuggestions.empty() ) { resetSignatureHelp(); } } else if ( event.getKeyCode() == KEY_DOWN ) { if ( mSignatureHelp.signatures.size() > 1 ) { mSignatureHelpSelected = mSignatureHelpSelected == (int)mSignatureHelp.signatures.size() - 1 ? mSignatureHelp.signatures.size() - 1 : 0; --mSignatureHelpSelected; mSignatureHelpSelected = mSignatureHelpSelected % mSignatureHelp.signatures.size(); editor->invalidateDraw(); return true; } else if ( mSuggestions.empty() ) { resetSignatureHelp(); } } else if ( event.getKeyCode() == EE::Window::KEY_BACKSPACE || event.getKeyCode() == EE::Window::KEY_DELETE ) { auto lang = editor->getDocumentRef()->getSyntaxDefinition().getLSPName(); auto cap = mCapabilities.find( lang ); if ( cap != mCapabilities.end() ) { auto curChar = event.getKeyCode() == EE::Window::KEY_BACKSPACE ? editor->getDocumentRef()->getPrevChar() : editor->getDocumentRef()->getCurrentChar(); const auto& signatureTrigger = cap->second.signatureHelpProvider.triggerCharacters; if ( std::find( signatureTrigger.begin(), signatureTrigger.end(), curChar ) != signatureTrigger.end() ) { resetSignatureHelp(); } } } } if ( !mSuggestions.empty() ) { if ( event.getKeyCode() == KEY_DOWN ) { if ( mSuggestionIndex + 1 < (int)mSuggestions.size() ) { mSuggestionIndex++; if ( mSuggestionIndex < mSuggestionsStartIndex ) mSuggestionsStartIndex = mSuggestionIndex; else if ( mSuggestionIndex > mSuggestionsStartIndex + mSuggestionsMaxVisible - 1 ) { mSuggestionsStartIndex = eemax( 0, mSuggestionIndex - ( mSuggestionsMaxVisible - 1 ) ); } } else { mSuggestionIndex = 0; mSuggestionsStartIndex = 0; } editor->invalidateDraw(); return true; } else if ( event.getKeyCode() == KEY_UP ) { if ( mSuggestionIndex - 1 < 0 ) { mSuggestionIndex = mSuggestions.size() - 1; mSuggestionsStartIndex = eemax( 0, (int)mSuggestions.size() - mSuggestionsMaxVisible ); } else { mSuggestionIndex--; } if ( mSuggestionIndex < (int)mSuggestionsStartIndex ) mSuggestionsStartIndex = mSuggestionIndex; editor->invalidateDraw(); return true; } else if ( event.getKeyCode() == KEY_ESCAPE ) { resetSuggestions( editor ); resetSignatureHelp(); editor->invalidateDraw(); return true; } else if ( event.getKeyCode() == KEY_HOME ) { mSuggestionIndex = 0; mSuggestionsStartIndex = 0; editor->invalidateDraw(); return true; } else if ( event.getKeyCode() == KEY_END ) { mSuggestionIndex = mSuggestions.size() - 1; mSuggestionsStartIndex = eemax( 0, (int)mSuggestions.size() - mSuggestionsMaxVisible ); editor->invalidateDraw(); return true; } else if ( event.getKeyCode() == KEY_PAGEUP ) { if ( mSuggestionIndex - (int)( mSuggestionsMaxVisible - 1 ) >= 0 ) { mSuggestionIndex -= ( mSuggestionsMaxVisible - 1 ); if ( mSuggestionIndex < mSuggestionsStartIndex ) mSuggestionsStartIndex = mSuggestionIndex; } else { mSuggestionIndex = 0; mSuggestionsStartIndex = 0; } editor->invalidateDraw(); return true; } else if ( event.getKeyCode() == KEY_PAGEDOWN ) { if ( mSuggestionIndex + mSuggestionsMaxVisible < (int)mSuggestions.size() ) { mSuggestionIndex += mSuggestionsMaxVisible - 1; } else { mSuggestionIndex = mSuggestions.size() - 1; } mSuggestionsStartIndex = eemax( 0, mSuggestionIndex - ( mSuggestionsMaxVisible - 1 ) ); editor->invalidateDraw(); return true; } else if ( event.getKeyCode() == KEY_TAB || event.getKeyCode() == KEY_RETURN || event.getKeyCode() == KEY_KP_ENTER ) { pickSuggestion( editor ); return true; } } else if ( event.getKeyCode() == KEY_SPACE && ( event.getMod() & KeyMod::getDefaultModifier() ) ) { std::string partialSymbol( getPartialSymbol( &editor->getDocument() ) ); updateSuggestions( partialSymbol, editor ); return true; } return ret; } void AutoCompletePlugin::requestSignatureHelp( UICodeEditor* editor ) { { Lock l( mSignatureHelpEditorMutex ); mSignatureHelpEditor = editor; } auto doc = editor->getDocumentRef(); mSignatureHelpPosition = editor->getDocumentRef()->getSelection().start(); mPool->run( [&, editor]() { json data = getURIAndPositionJSON( editor ); mManager->sendRequest( this, PluginMessageType::SignatureHelp, PluginMessageFormat::JSON, &data ); } ); } void AutoCompletePlugin::requestCodeCompletion( UICodeEditor* editor ) { { Lock l( mHandlesMutex ); auto handleIt = mHandles.find( editor->getDocumentRef().get() ); if ( handleIt != mHandles.end() ) { for ( const PluginIDType& hndl : handleIt->second ) { auto data = getURIJSON( handleIt->first, hndl ); mManager->sendBroadcast( PluginMessageType::CancelRequest, PluginMessageFormat::JSON, &data ); } handleIt->second.clear(); } } json data = getURIAndPositionJSON( editor ); PluginRequestHandle handle( mManager->sendRequest( this, PluginMessageType::CodeCompletion, PluginMessageFormat::JSON, &data ) ); Lock l( mHandlesMutex ); mHandles[editor->getDocumentRef().get()].push_back( handle.id() ); } bool AutoCompletePlugin::onTextInput( UICodeEditor* editor, const TextInputEvent& event ) { std::string partialSymbol( getPartialSymbol( &editor->getDocument() ) ); auto lang = editor->getDocumentRef()->getSyntaxDefinition().getLSPName(); auto cap = mCapabilities.find( lang ); if ( cap != mCapabilities.end() ) { bool requestedSignatureHelp = false; const auto& signatureTrigger = cap->second.signatureHelpProvider.triggerCharacters; if ( std::find( signatureTrigger.begin(), signatureTrigger.end(), event.getChar() ) != signatureTrigger.end() ) { requestSignatureHelp( editor ); requestedSignatureHelp = true; } if ( mSignatureHelpVisible && !requestedSignatureHelp ) { auto doc = editor->getDocumentRef(); auto curPos = doc->getSelection().start(); if ( curPos.line() != mSignatureHelpPosition.line() || curPos < doc->startOfWord( doc->positionOffset( mSignatureHelpPosition, 1 ) ) ) resetSignatureHelp(); } const auto& triggerCharacters = cap->second.completionProvider.triggerCharacters; if ( partialSymbol.size() >= 1 || std::find( triggerCharacters.begin(), triggerCharacters.end(), event.getChar() ) != triggerCharacters.end() ) { updateSuggestions( partialSymbol, editor ); } else { resetSuggestions( editor ); } return false; } if ( partialSymbol.size() >= 3 ) { updateSuggestions( partialSymbol, editor ); } else { resetSuggestions( editor ); } return false; } void AutoCompletePlugin::updateDocCache( TextDocument* doc ) { { Lock lu( mDocsUpdatingMutex ); mDocsUpdating[doc] = true; } Clock clock; std::unordered_map::iterator docCache; { Lock l( mDocMutex ); docCache = mDocCache.find( doc ); if ( docCache == mDocCache.end() || mClosing ) return; } auto changeId = doc->getCurrentChangeId(); auto symbols = getDocumentSymbols( doc ); { Lock l( mDocMutex ); docCache = mDocCache.find( doc ); if ( docCache == mDocCache.end() || mClosing ) return; auto& cache = docCache->second; cache.changeId = changeId; cache.symbols = std::move( symbols ); } std::string langName( doc->getSyntaxDefinition().getLanguageName() ); { Lock l( mLangSymbolsMutex ); auto& lang = mLangCache[langName]; lang.clear(); Lock l2( mDocMutex ); for ( const auto& d : mDocCache ) { if ( d.first->getSyntaxDefinition().getLanguageName() == langName ) lang.insert( lang.end(), d.second.symbols.begin(), d.second.symbols.end() ); } } Log::debug( "Dictionary for %s updated in: %.2fms", doc->getFilename().c_str(), clock.getElapsedTime().asMilliseconds() ); { Lock lu( mDocsUpdatingMutex ); mDocsUpdating[doc] = false; } } void AutoCompletePlugin::updateLangCache( const std::string& langName ) { Clock clock; Lock l( mLangSymbolsMutex ); Lock l2( mDocMutex ); auto& lang = mLangCache[langName]; lang.clear(); for ( const auto& d : mDocCache ) { if ( d.first->getSyntaxDefinition().getLanguageName() == langName ) lang.insert( lang.end(), d.second.symbols.begin(), d.second.symbols.end() ); } Log::debug( "Lang dictionary for %s updated in: %.2fms", langName.c_str(), clock.getElapsedTime().asMilliseconds() ); } void AutoCompletePlugin::pickSuggestion( UICodeEditor* editor ) { mReplacing = true; std::string symbol( getPartialSymbol( editor->getDocumentRef().get() ) ); if ( !symbol.empty() ) editor->getDocument().execute( "delete-to-previous-word" ); editor->getDocument().textInput( mSuggestions[mSuggestionIndex].text ); mReplacing = false; resetSuggestions( editor ); } PluginRequestHandle AutoCompletePlugin::processCodeCompletion( const LSPCompletionList& completion ) { SymbolsList suggestions; for ( const auto& item : completion.items ) { if ( !item.insertText.empty() ) suggestions.push_back( { item.kind, item.insertText, item.detail, item.sortText, item.textEdit.range } ); else if ( !item.textEdit.text.empty() ) suggestions.push_back( { item.kind, item.textEdit.text, item.detail, item.sortText, item.textEdit.range } ); else suggestions.push_back( { item.kind, item.filterText, item.detail, item.sortText } ); } if ( suggestions.empty() || !mSuggestionsEditor ) return {}; UICodeEditor* editor = nullptr; { Lock l( mSuggestionsEditorMutex ); editor = mSuggestionsEditor; } if ( !editor ) return {}; std::string symbol( getPartialSymbol( editor->getDocumentRef().get() ) ); const std::string& lang = editor->getDocument().getSyntaxDefinition().getLanguageName(); bool hasLangSuggestions = false; { Lock l2( mLangSymbolsMutex ); auto langSuggestions = mLangCache.find( lang ); hasLangSuggestions = langSuggestions != mLangCache.end(); } if ( symbol.empty() || !hasLangSuggestions ) { Lock l( mSuggestionsMutex ); mSuggestions = suggestions; } else { SymbolsList fuzzySuggestions; { Lock l2( mLangSymbolsMutex ); auto& symbols = mLangCache[lang]; fuzzySuggestions = fuzzyMatchSymbols( { &suggestions, &symbols }, symbol, eemax( 100UL, suggestions.size() ) ); } Lock l( mSuggestionsMutex ); mSuggestions = fuzzySuggestions; } editor->runOnMainThread( [editor] { editor->invalidateDraw(); } ); return {}; } PluginRequestHandle AutoCompletePlugin::processSignatureHelp( const LSPSignatureHelp& signatureHelp ) { UICodeEditor* editor = nullptr; { Lock l( mSignatureHelpEditorMutex ); editor = mSignatureHelpEditor; } if ( !editor ) return {}; editor->runOnMainThread( [this, editor, signatureHelp] { mSignatureHelpVisible = true; mSignatureHelp = signatureHelp; if ( mSignatureHelp.signatures.empty() ) resetSignatureHelp(); editor->invalidateDraw(); } ); return {}; } PluginRequestHandle AutoCompletePlugin::processResponse( const PluginMessage& msg ) { if ( msg.isResponse() && msg.type == PluginMessageType::CodeCompletion ) { if ( msg.responseID ) { Lock l( mHandlesMutex ); for ( auto& handle : mHandles ) { auto find = std::find( handle.second.begin(), handle.second.end(), msg.responseID ); if ( find != handle.second.end() ) handle.second.erase( find ); } } return processCodeCompletion( msg.asCodeCompletion() ); } else if ( msg.isResponse() && msg.type == PluginMessageType::SignatureHelp ) { return processSignatureHelp( msg.asSignatureHelp() ); } else if ( msg.isBroadcast() && msg.type == PluginMessageType::LanguageServerCapabilities ) { if ( msg.asLanguageServerCapabilities().ready ) { LSPServerCapabilities cap = msg.asLanguageServerCapabilities(); auto& trig = cap.signatureHelpProvider.triggerCharacters; static const std::vector> pairs = { { '(', ')' }, { '{', '}' }, { '<', '>' } }; for ( const auto& pair : pairs ) { if ( std::find( trig.begin(), trig.end(), pair.first ) != trig.end() && std::find( trig.begin(), trig.end(), pair.second ) == trig.end() ) { trig.push_back( pair.second ); } } Lock l( mCapabilitiesMutex ); mCapabilities[cap.language] = std::move( cap ); } } return {}; } bool AutoCompletePlugin::tryRequestCapabilities( UICodeEditor* editor ) { const auto& language = editor->getDocumentRef()->getSyntaxDefinition().getLSPName(); auto it = mCapabilities.find( language ); if ( it != mCapabilities.end() ) return true; json data; data["language"] = language; mManager->sendRequest( this, PluginMessageType::LanguageServerCapabilities, PluginMessageFormat::JSON, &data ); return false; } std::string AutoCompletePlugin::getPartialSymbol( TextDocument* doc ) { TextPosition end = doc->getSelection().end(); TextPosition start = doc->startOfWord( end ); return doc->getText( { start, end } ).toUtf8(); } void AutoCompletePlugin::update( UICodeEditor* ) { if ( mClock.getElapsedTime() >= mUpdateFreq || mDirty ) { mClock.restart(); mDirty = false; Lock l( mDocMutex ); for ( auto& doc : mDocs ) { if ( !doc->isLoading() && mDocCache[doc].changeId != doc->getCurrentChangeId() ) { { Lock lu( mDocsUpdatingMutex ); auto du = mDocsUpdating.find( doc ); // Dont update the document cache if it's still updating the document if ( du != mDocsUpdating.end() && du->second == true ) continue; } #if AUTO_COMPLETE_THREADED mPool->run( [&, doc] { updateDocCache( doc ); } ); #else updateDocCache( doc ); #endif } } } } void AutoCompletePlugin::drawSignatureHelp( UICodeEditor* editor, const Vector2f& startScroll, const Float& lineHeight, bool drawUp ) { TextDocument& doc = editor->getDocument(); Primitives primitives; const SyntaxColorScheme& scheme = editor->getColorScheme(); const auto& normalStyle = scheme.getEditorSyntaxStyle( "suggestion" ); const auto& selectedStyle = scheme.getEditorSyntaxStyle( "suggestion_selected" ); const auto& matchingSelection = scheme.getEditorSyntaxStyle( "matching_selection" ); auto curSigIdx = mSignatureHelpSelected != -1 ? mSignatureHelpSelected : mSignatureHelp.activeSignature; if ( curSigIdx >= (int)mSignatureHelp.signatures.size() ) return; auto curSig = mSignatureHelp.signatures[curSigIdx]; Float vdiff = drawUp ? -mRowHeight : mRowHeight; Vector2f pos( startScroll.x + editor->getXOffsetCol( mSignatureHelpPosition ), startScroll.y + mSignatureHelpPosition.line() * lineHeight + vdiff ); primitives.setColor( Color( selectedStyle.background ).blendAlpha( editor->getAlpha() ) ); String str; if ( mSignatureHelp.signatures.size() > 1 ) { str = String::format( "%s (%d of %zu)", curSig.label.c_str(), mSignatureHelpSelected == -1 ? 1 : mSignatureHelpSelected + 1, mSignatureHelp.signatures.size() ); } else { str = curSig.label; } Rectf boxRect( pos, Sizef( editor->getTextWidth( str ) + mBoxPadding.Left + mBoxPadding.Right, mRowHeight ) ); if ( boxRect.getPosition().x + boxRect.getSize().getWidth() > editor->getScreenPos().x + editor->getPixelsSize().getWidth() ) { boxRect.setPosition( { eefloor( editor->getScreenPos().x + editor->getPixelsSize().getWidth() - boxRect.getSize().getWidth() ), boxRect.getPosition().y } ); if ( boxRect.getPosition().x < editor->getScreenPos().x ) boxRect.setPosition( { eefloor( editor->getScreenPos().x ), boxRect.getPosition().y } ); } bool hasParams = !curSig.parameters.empty(); LSPParameterInformation curParam = hasParams ? curSig.parameters[mSignatureHelp.activeParameter % curSig.parameters.size()] : LSPParameterInformation{ -1, -1 }; Rectf curParamRect; if ( hasParams ) { curParamRect = Rectf( { { boxRect.getPosition().x + mBoxPadding.Left + curParam.start * editor->getGlyphWidth(), boxRect.getPosition().y }, { ( curParam.end - curParam.start ) * editor->getGlyphWidth(), mRowHeight } } ); if ( !editor->getScreenRect().contains( Rectf{ { curParamRect.getPosition().x + ( curParam.end - curParam.start ) * editor->getGlyphWidth(), curParamRect.getPosition().y }, curParamRect.getSize() } ) ) { pos = { startScroll.x - curParam.start * editor->getGlyphWidth() + editor->getXOffsetCol( mSignatureHelpPosition ), startScroll.y + mSignatureHelpPosition.line() * lineHeight + vdiff }; boxRect.setPosition( pos ); curParamRect.setPosition( { boxRect.getPosition().x + mBoxPadding.Left + curParam.start * editor->getGlyphWidth(), boxRect.getPosition().y } ); } } primitives.drawRoundedRectangle( boxRect, 0.f, Vector2f::One, 6 ); if ( hasParams && curParam.end - curParam.start > 0 && curParam.end < (int)str.size() ) { primitives.setColor( matchingSelection.color ); primitives.drawRoundedRectangle( curParamRect, 0.f, Vector2f::One, 6 ); } Text text( "", editor->getFont(), editor->getFontSize() ); text.setFillColor( normalStyle.color ); text.setStyle( normalStyle.style ); text.setString( str ); SyntaxTokenizer::tokenizeText( doc.getSyntaxDefinition(), editor->getColorScheme(), text ); text.draw( boxRect.getPosition().x + mBoxPadding.Left, boxRect.getPosition().y + mBoxPadding.Top ); } void AutoCompletePlugin::postDraw( UICodeEditor* editor, const Vector2f& startScroll, const Float& lineHeight, const TextPosition& cursor ) { bool drawsSuggestions = !( mSuggestions.empty() || !mSuggestionsEditor || mSuggestionsEditor != editor ); bool drawsSignature = mSignatureHelpVisible && mSignatureHelpEditor == editor && !mSignatureHelp.signatures.empty() && mSignatureHelpPosition.isValid(); if ( !drawsSuggestions && !drawsSignature ) return; TextDocument& doc = editor->getDocument(); TextPosition start = doc.startOfWord( editor->getDocument().startOfWord( cursor ) ); Primitives primitives; const SyntaxColorScheme& scheme = editor->getColorScheme(); const auto& normalStyle = scheme.getEditorSyntaxStyle( "suggestion" ); const auto& selectedStyle = scheme.getEditorSyntaxStyle( "suggestion_selected" ); bool drawUp = true; if ( !drawsSuggestions ) { if ( drawsSignature ) drawSignatureHelp( editor, startScroll, lineHeight, drawUp ); return; } SymbolsList suggestions; { Lock l( mSuggestionsMutex ); suggestions = mSuggestions; } Vector2f cursorPos( startScroll.x + editor->getXOffsetCol( start ), startScroll.y + cursor.line() * lineHeight + lineHeight ); size_t largestString = 0; size_t max = eemin( mSuggestionsMaxVisible, suggestions.size() ); mRowHeight = lineHeight + mBoxPadding.Top + mBoxPadding.Bottom; const auto& barStyle = scheme.getEditorSyntaxStyle( "suggestion_scrollbar" ); if ( cursorPos.y + mRowHeight * max > editor->getPixelsSize().getHeight() ) { cursorPos.y -= lineHeight + mRowHeight * max; drawUp = false; } size_t maxIndex = eemin( mSuggestionsStartIndex + mSuggestionsMaxVisible, suggestions.size() ); for ( size_t i = mSuggestionsStartIndex; i < maxIndex; i++ ) largestString = eemax( largestString, editor->getTextWidth( suggestions[i].text ) ); Sizef bar( PixelDensity::dpToPxI( 6 ), eemax( PixelDensity::dpToPx( 8 ), mRowHeight * max * ( mSuggestionsMaxVisible / (Float)suggestions.size() ) ) ); Sizef iconSpace( PixelDensity::dpToPxI( 16 ), mRowHeight ); mBoxRect = Rectf( Vector2f( cursorPos.x, cursorPos.y ) - editor->getScreenPos(), Sizef( largestString + mBoxPadding.Left + mBoxPadding.Right + iconSpace.getWidth() + bar.getWidth(), mRowHeight * max ) ); size_t count = 0; Rectf boxRect( { mBoxRect.getPosition() + editor->getScreenPos(), mBoxRect.getSize() } ); primitives.setColor( Color( normalStyle.background ).blendAlpha( editor->getAlpha() ) ); primitives.drawRoundedRectangle( boxRect, 0.f, Vector2f::One, 6 ); for ( size_t i = mSuggestionsStartIndex; i < maxIndex; i++ ) { if ( mSuggestionIndex == (int)i ) { primitives.setColor( Color( selectedStyle.background ).blendAlpha( editor->getAlpha() ) ); primitives.drawRoundedRectangle( Rectf( Vector2f( cursorPos.x, cursorPos.y + mRowHeight * count ), Sizef( mBoxRect.getWidth(), mRowHeight ) ), 0.f, Vector2f::One, 6 ); } Text text( "", editor->getFont(), editor->getFontSize() ); text.setFillColor( mSuggestionIndex == (int)i ? selectedStyle.color : normalStyle.color ); text.setStyle( mSuggestionIndex == (int)i ? selectedStyle.style : normalStyle.style ); auto nlPos = suggestions[i].text.find_first_of( '\n' ); if ( nlPos == std::string::npos ) { text.setString( suggestions[i].text ); } else { text.setString( suggestions[i].text.substr( 0, nlPos ) ); } text.draw( cursorPos.x + iconSpace.getWidth() + mBoxPadding.Left, cursorPos.y + mRowHeight * count + mBoxPadding.Top ); Drawable* icon = editor->getUISceneNode()->findIconDrawable( LSPCompletionItemHelper::toIconString( suggestions[i].kind ), PixelDensity::dpToPxI( 12 ) ); if ( icon ) { Vector2f padding( eefloor( ( iconSpace.getWidth() - icon->getSize().getWidth() ) * 0.5f ), eefloor( ( iconSpace.getHeight() - icon->getSize().getHeight() ) * 0.5f ) ); icon->draw( { cursorPos.x + padding.x, cursorPos.y + mRowHeight * count + padding.y } ); } count++; } if ( drawsSignature ) drawSignatureHelp( editor, startScroll, lineHeight, drawUp ); if ( max >= suggestions.size() ) return; primitives.setColor( barStyle.color ); Float yPos = mSuggestionsStartIndex > 0 ? mSuggestionsStartIndex / (Float)( suggestions.size() - mSuggestionsMaxVisible ) : 0; Rectf barRect( { Vector2f( cursorPos.x + mBoxRect.getWidth() - bar.getWidth(), cursorPos.y + ( mBoxRect.getHeight() - bar.getHeight() ) * yPos ), bar } ); primitives.drawRoundedRectangle( barRect, 0, Vector2f::One, (int)eefloor( bar.getWidth() * 0.5f ) ); } bool AutoCompletePlugin::onMouseDown( UICodeEditor* editor, const Vector2i& position, const Uint32& flags ) { if ( mSuggestions.empty() || !mSuggestionsEditor || mSuggestionsEditor != editor || !( flags & EE_BUTTON_LMASK ) ) return false; Vector2f localPos( editor->convertToNodeSpace( position.asFloat() ) ); if ( mBoxRect.contains( localPos ) ) { localPos -= { mBoxRect.Left, mBoxRect.Top }; mSuggestionIndex = mSuggestionsStartIndex + localPos.y / mRowHeight; editor->invalidateDraw(); return true; } return false; } bool AutoCompletePlugin::onMouseUp( UICodeEditor* editor, const Vector2i& position, const Uint32& flags ) { if ( mSuggestions.empty() || !mSuggestionsEditor || mSuggestionsEditor != editor ) return false; Vector2f localPos( editor->convertToNodeSpace( position.asFloat() ) ); if ( mBoxRect.contains( localPos ) ) { if ( flags & EE_BUTTON_WUMASK ) { mSuggestionsStartIndex = eemax( 0, mSuggestionsStartIndex - mSuggestionsMaxVisible ); editor->invalidateDraw(); return true; } else if ( flags & EE_BUTTON_WDMASK ) { mSuggestionsStartIndex = eemax( 0, eemin( (int)mSuggestions.size() - mSuggestionsMaxVisible, mSuggestionsStartIndex + mSuggestionsMaxVisible ) ); editor->invalidateDraw(); return true; } } return false; } bool AutoCompletePlugin::onMouseDoubleClick( UICodeEditor* editor, const Vector2i& position, const Uint32& flags ) { if ( mSuggestions.empty() || !mSuggestionsEditor || mSuggestionsEditor != editor || !( flags & EE_BUTTON_LMASK ) ) return false; Vector2f localPos( editor->convertToNodeSpace( position.asFloat() ) ); if ( mBoxRect.contains( localPos ) ) { pickSuggestion( editor ); return true; } return false; } bool AutoCompletePlugin::onMouseMove( UICodeEditor* editor, const Vector2i& position, const Uint32& ) { if ( mSuggestions.empty() || !mSuggestionsEditor || mSuggestionsEditor != editor ) return false; Vector2f localPos( editor->convertToNodeSpace( position.asFloat() ) ); if ( mBoxRect.contains( localPos ) ) { editor->getUISceneNode()->setCursor( Cursor::Hand ); return true; } else { editor->getUISceneNode()->setCursor( !editor->isLocked() ? Cursor::IBeam : Cursor::Arrow ); } return false; } const Rectf& AutoCompletePlugin::getBoxPadding() const { return mBoxPadding; } void AutoCompletePlugin::setBoxPadding( const Rectf& boxPadding ) { mBoxPadding = boxPadding; } const Int32& AutoCompletePlugin::getSuggestionsMaxVisible() const { return mSuggestionsMaxVisible; } void AutoCompletePlugin::setSuggestionsMaxVisible( const Uint32& suggestionsMaxVisible ) { mSuggestionsMaxVisible = suggestionsMaxVisible; } const Time& AutoCompletePlugin::getUpdateFreq() const { return mUpdateFreq; } void AutoCompletePlugin::setUpdateFreq( const Time& updateFreq ) { mUpdateFreq = updateFreq; } const std::string& AutoCompletePlugin::getSymbolPattern() const { return mSymbolPattern; } void AutoCompletePlugin::setSymbolPattern( const std::string& symbolPattern ) { mSymbolPattern = symbolPattern; } bool AutoCompletePlugin::isDirty() const { return mDirty; } void AutoCompletePlugin::setDirty( bool dirty ) { mDirty = dirty; } void AutoCompletePlugin::resetSuggestions( UICodeEditor* editor ) { Lock l( mSuggestionsMutex ); mSuggestionIndex = 0; mSuggestionsStartIndex = 0; { Lock l( mSuggestionsEditorMutex ); mSuggestionsEditor = nullptr; } mSuggestions.clear(); if ( editor && editor->hasFocus() ) editor->getUISceneNode()->setCursor( !editor->isLocked() ? Cursor::IBeam : Cursor::Arrow ); } void AutoCompletePlugin::resetSignatureHelp() { mSignatureHelpVisible = false; mSignatureHelp.signatures.clear(); mSignatureHelp.activeSignature = 0; mSignatureHelp.activeParameter = 0; Lock l( mSignatureHelpEditorMutex ); mSignatureHelpEditor = nullptr; } AutoCompletePlugin::SymbolsList AutoCompletePlugin::getDocumentSymbols( TextDocument* doc ) { LuaPattern pattern( mSymbolPattern ); AutoCompletePlugin::SymbolsList symbols; Int64 lc = doc->linesCount(); if ( lc == 0 || lc > 50000 || mClosing ) return symbols; std::string current( getPartialSymbol( doc ) ); TextPosition end = doc->getSelection().end(); for ( Int64 i = 0; i < lc; i++ ) { const auto& string = doc->line( i ).getText().toUtf8(); for ( auto& match : pattern.gmatch( string ) ) { std::string matchStr( match[0] ); // Ignore the symbol if is actually the current symbol being written if ( matchStr.size() < 3 || ( end.line() == i && current == matchStr ) ) continue; if ( std::none_of( symbols.begin(), symbols.end(), [matchStr]( const Suggestion& suggestion ) { return suggestion.text == matchStr; } ) ) symbols.push_back( std::move( matchStr ) ); } if ( mClosing ) break; } return symbols; } void AutoCompletePlugin::runUpdateSuggestions( const std::string& symbol, const SymbolsList& symbols, UICodeEditor* editor ) { { { Lock l( mSuggestionsEditorMutex ); mSuggestionsEditor = editor; } if ( tryRequestCapabilities( editor ) ) requestCodeCompletion( editor ); if ( symbol.empty() ) return; Lock l( mLangSymbolsMutex ); Lock l2( mSuggestionsMutex ); mSuggestions = fuzzyMatchSymbols( { &symbols }, symbol, mSuggestionsMaxVisible ); } editor->runOnMainThread( [editor] { editor->invalidateDraw(); } ); } void AutoCompletePlugin::updateSuggestions( const std::string& symbol, UICodeEditor* editor ) { const std::string& lang = editor->getDocument().getSyntaxDefinition().getLanguageName(); Lock l( mLangSymbolsMutex ); auto langSuggestions = mLangCache.find( lang ); if ( langSuggestions == mLangCache.end() ) return; auto& symbols = langSuggestions->second; { #if AUTO_COMPLETE_THREADED mPool->run( [this, symbol, symbols, editor] { runUpdateSuggestions( symbol, symbols, editor ); } ); #else runUpdateSuggestions( symbol, symbols, editor ); #endif } } } // namespace ecode