-- Copyright 2011-14 Paul Kulchenko, ZeroBrane LLC -- authors: Lomtik Software (J. Winwood & John Labenski) -- Luxinia Dev (Eike Decker & Christoph Kubisch) --------------------------------------------------------- local editorID = 100 -- window id to create editor pages with, incremented for new editors local openDocuments = ide.openDocuments local statusBar = ide.frame.statusBar local notebook = ide.frame.notebook local edcfg = ide.config.editor local styles = ide.config.styles local unpack = table.unpack or unpack local q = EscapeMagic local margin = { LINENUMBER = 0, MARKER = 1, FOLD = 2 } local linenummask = "99999" local foldtypes = { [0] = { wxstc.wxSTC_MARKNUM_FOLDEROPEN, wxstc.wxSTC_MARKNUM_FOLDER, wxstc.wxSTC_MARKNUM_FOLDERSUB, wxstc.wxSTC_MARKNUM_FOLDERTAIL, wxstc.wxSTC_MARKNUM_FOLDEREND, wxstc.wxSTC_MARKNUM_FOLDEROPENMID, wxstc.wxSTC_MARKNUM_FOLDERMIDTAIL, }, box = { wxstc.wxSTC_MARK_BOXMINUS, wxstc.wxSTC_MARK_BOXPLUS, wxstc.wxSTC_MARK_VLINE, wxstc.wxSTC_MARK_LCORNER, wxstc.wxSTC_MARK_BOXPLUSCONNECTED, wxstc.wxSTC_MARK_BOXMINUSCONNECTED, wxstc.wxSTC_MARK_TCORNER, }, circle = { wxstc.wxSTC_MARK_CIRCLEMINUS, wxstc.wxSTC_MARK_CIRCLEPLUS, wxstc.wxSTC_MARK_VLINE, wxstc.wxSTC_MARK_LCORNERCURVE, wxstc.wxSTC_MARK_CIRCLEPLUSCONNECTED, wxstc.wxSTC_MARK_CIRCLEMINUSCONNECTED, wxstc.wxSTC_MARK_TCORNERCURVE, }, plus = { wxstc.wxSTC_MARK_MINUS, wxstc.wxSTC_MARK_PLUS }, arrow = { wxstc.wxSTC_MARK_ARROWDOWN, wxstc.wxSTC_MARK_ARROW }, } -- ---------------------------------------------------------------------------- -- Update the statusbar text of the frame using the given editor. -- Only update if the text has changed. local statusTextTable = { "OVR?", "R/O?", "Cursor Pos" } local function updateStatusText(editor) local texts = { "", "", "" } if ide.frame and editor then local pos = editor:GetCurrentPos() local selected = #editor:GetSelectedText() local selections = ide.wxver >= "2.9.5" and editor:GetSelections() or 1 texts = { iff(editor:GetOvertype(), TR("OVR"), TR("INS")), iff(editor:GetReadOnly(), TR("R/O"), TR("R/W")), table.concat({ TR("Ln: %d"):format(editor:LineFromPosition(pos) + 1), TR("Col: %d"):format(editor:GetColumn(pos) + 1), selected > 0 and TR("Sel: %d/%d"):format(selected, selections) or "", }, ' ')} end if ide.frame then for n in ipairs(texts) do if (texts[n] ~= statusTextTable[n]) then statusBar:SetStatusText(texts[n], n+1) statusTextTable[n] = texts[n] end end end end local function updateBraceMatch(editor) local pos = editor:GetCurrentPos() local posp = pos > 0 and pos-1 local char = editor:GetCharAt(pos) local charp = posp and editor:GetCharAt(posp) local match = { [string.byte("<")] = true, [string.byte(">")] = true, [string.byte("(")] = true, [string.byte(")")] = true, [string.byte("{")] = true, [string.byte("}")] = true, [string.byte("[")] = true, [string.byte("]")] = true, } pos = (match[char] and pos) or (charp and match[charp] and posp) if (pos) then -- don't match brackets in markup comments local style = bit.band(editor:GetStyleAt(pos), 31) if (MarkupIsSpecial and MarkupIsSpecial(style) or editor.spec.iscomment[style]) then return end local pos2 = editor:BraceMatch(pos) if (pos2 == wxstc.wxSTC_INVALID_POSITION) then editor:BraceBadLight(pos) else editor:BraceHighlight(pos,pos2) end editor.matchon = true elseif(editor.matchon) then editor:BraceBadLight(wxstc.wxSTC_INVALID_POSITION) editor:BraceHighlight(wxstc.wxSTC_INVALID_POSITION,-1) editor.matchon = false end end -- Check if file is altered, show dialog to reload it local function isFileAlteredOnDisk(editor) if not editor then return end local id = editor:GetId() if openDocuments[id] then local filePath = openDocuments[id].filePath local fileName = openDocuments[id].fileName local oldModTime = openDocuments[id].modTime if filePath and (string.len(filePath) > 0) and oldModTime and oldModTime:IsValid() then local modTime = GetFileModTime(filePath) if modTime == nil then openDocuments[id].modTime = nil wx.wxMessageBox( TR("File '%s' no longer exists."):format(fileName), GetIDEString("editormessage"), wx.wxOK + wx.wxCENTRE, ide.frame) elseif not editor:GetReadOnly() and modTime:IsValid() and oldModTime:IsEarlierThan(modTime) then local ret = (edcfg.autoreload and (not EditorIsModified(editor)) and wx.wxYES) or wx.wxMessageBox( TR("File '%s' has been modified on disk."):format(fileName) .."\n"..TR("Do you want to reload it?"), GetIDEString("editormessage"), wx.wxYES_NO + wx.wxCENTRE, ide.frame) if ret ~= wx.wxYES or ReLoadFile(filePath, editor, true) then openDocuments[id].modTime = GetFileModTime(filePath) end end end end end local function navigateToPosition(editor, fromPosition, toPosition, length) table.insert(editor.jumpstack, fromPosition) editor:GotoPosEnforcePolicy(toPosition) if length then editor:SetAnchor(toPosition + length) end end local function navigateBack(editor) if #editor.jumpstack == 0 then return end local pos = table.remove(editor.jumpstack) editor:GotoPosEnforcePolicy(pos) return true end -- ---------------------------------------------------------------------------- -- Get/Set notebook editor page, use nil for current page, returns nil if none function GetEditor(selection) if selection == nil then selection = notebook:GetSelection() end local editor if (selection >= 0) and (selection < notebook:GetPageCount()) and (notebook:GetPage(selection):GetClassInfo():GetClassName()=="wxStyledTextCtrl") then editor = notebook:GetPage(selection):DynamicCast("wxStyledTextCtrl") end return editor end -- init new notebook page selection, use nil for current page function SetEditorSelection(selection) local editor = GetEditor(selection) updateStatusText(editor) -- update even if nil statusBar:SetStatusText("",1) ide.frame:SetTitle(ExpandPlaceholders(ide.config.format.apptitle)) if editor then editor:SetFocus() editor:SetSTCFocus(true) local id = editor:GetId() FileTreeMarkSelected(openDocuments[id] and openDocuments[id].filePath or '') AddToFileHistory(openDocuments[id] and openDocuments[id].filePath) else FileTreeMarkSelected('') end SetAutoRecoveryMark() end function GetEditorFileAndCurInfo(nochecksave) local editor = GetEditor() if (not (editor and (nochecksave or SaveIfModified(editor)))) then return end local id = editor:GetId() local filepath = openDocuments[id].filePath if not filepath then return end local fn = wx.wxFileName(filepath) fn:Normalize() local info = {} info.pos = editor:GetCurrentPos() info.line = editor:GetCurrentLine() info.sel = editor:GetSelectedText() info.sel = info.sel and info.sel:len() > 0 and info.sel or nil info.selword = info.sel and info.sel:match("([^a-zA-Z_0-9]+)") or info.sel return fn,info end -- Set if the document is modified and update the notebook page text function SetDocumentModified(id, modified, text) local modpref, doc = '* ', openDocuments[id] if not doc then return end local pageText = text or notebook:GetPageText(doc.index):gsub("^"..EscapeMagic(modpref), "") if modified then pageText = modpref..pageText end openDocuments[id].isModified = modified notebook:SetPageText(doc.index, pageText) end function EditorAutoComplete(editor) if not (editor and editor.spec) then return end local pos = editor:GetCurrentPos() -- don't do auto-complete in comments or strings. -- the current position and the previous one have default style (0), -- so we need to check two positions back. local style = pos >= 2 and bit.band(editor:GetStyleAt(pos-2),31) or 0 if editor.spec.iscomment[style] or editor.spec.isstring[style] then return end -- retrieve the current line and get a string to the current cursor position in the line local line = editor:GetCurrentLine() local linetx = editor:GetLine(line) local linestart = editor:PositionFromLine(line) local localpos = pos-linestart local lt = linetx:sub(1,localpos) lt = lt:gsub("%s*(["..editor.spec.sep.."])%s*", "%1") -- strip closed brace scopes lt = lt:gsub("%b()","") lt = lt:gsub("%b{}","") lt = lt:gsub("%b[]",".0") -- remove everything that can't be auto-completed lt = lt:match("[%w_"..q(editor.spec.sep).."]*$") -- know now which string is to be completed local userList = CreateAutoCompList(editor, lt, pos) -- remove any suggestions that match the word the cursor is on -- for example, if typing 'foo' in front of 'bar', 'foobar' is not offered local right = linetx:sub(localpos+1,#linetx):match("^([%a_]+[%w_]*)") if userList and right then userList = userList:gsub("%f[%w_]"..lt..right.."%f[%W]",""):gsub(" +"," ") end -- don't show the list if it only suggests what's already typed if userList and #userList > 0 and not lt:find(userList.."$") then editor:UserListShow(1, userList) elseif editor:AutoCompActive() then editor:AutoCompCancel() end end local ident = "([a-zA-Z_][a-zA-Z_0-9%.%:]*)" local function getValAtPosition(editor, pos) local line = editor:LineFromPosition(pos) local linetx = editor:GetLine(line) local linestart = editor:PositionFromLine(line) local localpos = pos-linestart local selected = editor:GetSelectionStart() ~= editor:GetSelectionEnd() and pos >= editor:GetSelectionStart() and pos <= editor:GetSelectionEnd() -- check if we have a selected text or an identifier. -- for an identifier, check fragments on the left and on the right. -- this is to match 'io' in 'i^o.print' and 'io.print' in 'io.pr^int'. -- remove square brackets to make tbl[index].x show proper values. local start = linetx:sub(1,localpos) :gsub("%b[]", function(s) return ("."):rep(#s) end) :find(ident.."$") local right, funccall = linetx:sub(localpos+1,#linetx):match("^([a-zA-Z_0-9]*)%s*(['\"{%(]?)") local var = selected -- GetSelectedText() returns concatenated text when multiple instances -- are selected, so get the selected text based on start/end and editor:GetTextRange(editor:GetSelectionStart(), editor:GetSelectionEnd()) or (start and linetx:sub(start,localpos):gsub(":",".")..right or nil) -- since this function can be called in different contexts, we need -- to detect function call of different types: -- 1. foo.b^ar(... -- the cursor (pos) is on the function name -- 2. foo.bar(..^. -- the cursor (pos) is on the parameter list -- "var" has value for #1 and the following fragment checks for #2 -- check if the style is the right one; this is to ignore -- comments, strings, numbers (to avoid '1 = 1'), keywords, and such local goodpos = true if start and not selected then local style = bit.band(editor:GetStyleAt(linestart+start),31) if editor.spec.iscomment[style] or (MarkupIsAny and MarkupIsAny(style)) -- markup in comments or editor.spec.isstring[style] or style == wxstc.wxSTC_LUA_NUMBER or style == wxstc.wxSTC_LUA_WORD then goodpos = false end end local linetxtopos = linetx:sub(1,localpos) funccall = (#funccall > 0) and goodpos and var or (linetxtopos..")"):match(ident .. "%s*%b()$") or (linetxtopos.."}"):match(ident .. "%s*%b{}$") or (linetxtopos.."'"):match(ident .. "%s*'[^']*'$") or (linetxtopos..'"'):match(ident .. '%s*"[^"]*"$') or nil -- don't do anything for strings or comments or numbers if not goodpos then return nil, funccall end return var, funccall end local function formatUpToX(s) local x = math.max(20, ide.config.acandtip.width) local splitstr = "([ \t]*)(%S*)([ \t]*)(\n?)" local t = {""} for prefix, word, suffix, newline in s:gmatch(splitstr) do if #(t[#t]) + #prefix + #word > x and #t > 0 then table.insert(t, word..suffix) else t[#t] = t[#t]..prefix..word..suffix end if #newline > 0 then table.insert(t, "") end end return table.concat(t, "\n") end local function callTipFitAndShow(editor, pos, tip) local point = editor:PointFromPosition(pos) local height = editor:TextHeight(pos) local maxlines = math.max(1, math.floor( math.max(editor:GetSize():GetHeight()-point:GetY()-height, point:GetY())/height-1 )) -- cut the tip to not exceed the number of maxlines. -- move the position to the left if needed to fit. -- find the longest line in terms of width in pixels. local maxwidth = 0 local lines = {} for line in formatUpToX(tip):gmatch("[^\n]*\n?") do local width = editor:TextWidth(wxstc.wxSTC_STYLE_DEFAULT, line) if width > maxwidth then maxwidth = width end table.insert(lines, line) if #lines >= maxlines then lines[#lines] = lines[#lines]:gsub("%s*\n$","")..'...' break end end tip = table.concat(lines, '') local startpos = editor:PositionFromLine(editor:LineFromPosition(pos)) local afterwidth = editor:GetSize():GetWidth()-point:GetX() if maxwidth > afterwidth then local charwidth = editor:TextWidth(wxstc.wxSTC_STYLE_DEFAULT, 'A') pos = math.max(startpos, pos - math.floor((maxwidth - afterwidth) / charwidth)) end editor:CallTipShow(pos, tip) end function EditorCallTip(editor, pos, x, y) -- don't show anything if the calltip/auto-complete is active; -- this may happen after typing function name, while the mouse is over -- a different function or when auto-complete is on for a parameter. if editor:CallTipActive() or editor:AutoCompActive() then return end -- don't activate if the window itself is not active (in the background) if not ide.frame:IsActive() then return end local var, funccall = editor:ValueFromPosition(pos) -- if this is a value type rather than a function/method call, then use -- full match to avoid calltip about coroutine.status for "status" vars local tip = GetTipInfo(editor, funccall or var, false, not funccall) local limit = ide.config.acandtip.maxlength if ide.debugger and ide.debugger.server then if var then ide.debugger.quickeval(var, function(val) if #val > limit then val = val:sub(1, limit-3).."..." end -- check if the mouse position is specified and the mouse has moved, -- then don't show the tooltip as it's already too late for it. if x and y then local mpos = wx.wxGetMousePosition() if mpos.x ~= x or mpos.y ~= y then return end end if PackageEventHandle("onEditorCallTip", editor, val, funccall or var, true) ~= false then callTipFitAndShow(editor, pos, val) end end) end elseif tip then local oncalltip = PackageEventHandle("onEditorCallTip", editor, tip, funccall or var, false) -- only shorten if shown on mouse-over. Use shortcut to get full info. local showtooltip = ide.frame.menuBar:FindItem(ID_SHOWTOOLTIP) local suffix = "...\n" ..TR("Use '%s' to see full description."):format(showtooltip:GetLabel()) if x and y and #tip > limit then tip = tip:sub(1, limit-#suffix):gsub("%W*%w*$","")..suffix end if oncalltip ~= false then callTipFitAndShow(editor, pos, tip) end end end function EditorIsModified(editor) local modified = false if editor then local id = editor:GetId() modified = openDocuments[id] and (openDocuments[id].isModified or not openDocuments[id].filePath) end return modified end -- Indicator handling for functions and local/global variables -- indicator.MASKED is handled separately, so don't include in MAX local indicator = {FNCALL = 0, LOCAL = 1, GLOBAL = 2, MASKING = 3, MASKED = 4, MAX = 3} function IndicateFunctionsOnly(editor, lines, linee) local sindic = styles.indicator if not (edcfg.showfncall and editor.spec and editor.spec.isfncall) or not (sindic and sindic.fncall and sindic.fncall.st ~= wxstc.wxSTC_INDIC_HIDDEN) then return end local lines = lines or 0 local linee = linee or editor:GetLineCount()-1 if (lines < 0) then return end local isfncall = editor.spec.isfncall local isinvalid = {} for i,v in pairs(editor.spec.iscomment) do isinvalid[i] = v end for i,v in pairs(editor.spec.iskeyword0) do isinvalid[i] = v end for i,v in pairs(editor.spec.isstring) do isinvalid[i] = v end editor:SetIndicatorCurrent(indicator.FNCALL) for line=lines,linee do local tx = editor:GetLine(line) local ls = editor:PositionFromLine(line) editor:IndicatorClearRange(ls, #tx) local from = 1 local off = -1 while from do tx = from==1 and tx or string.sub(tx,from) local f,t,w = isfncall(tx) if (f) then local p = ls+f+off local s = bit.band(editor:GetStyleAt(p),31) if not isinvalid[s] then editor:IndicatorFillRange(p, #w) end off = off + t end from = t and (t+1) end end end local delayed = {} function IndicateIfNeeded() local editor = GetEditor() -- do the current one first if delayed[editor] then return IndicateAll(editor) end for ed in pairs(delayed) do return IndicateAll(ed) end end -- find all instances of a symbol at pos -- return table with [0] as the definition position (if local) local function indicateFindInstances(editor, name, pos) local tokens = editor:GetTokenList() local instances = {{[-1] = 1}} local this for _, token in ipairs(tokens) do local op = token[1] if op == 'EndScope' then -- EndScope has "new" level, so need +1 if this and token.fpos > pos and this == token.at+1 then break end if #instances > 1 and instances[#instances][-1] == token.at+1 then table.remove(instances) end elseif token.name == name then if op == 'Id' then table.insert(instances[#instances], token.fpos) elseif op:find("^Var") then if this and this == token.at then break end -- if new Var is defined at the same level, replace the current frame; -- if not, add a new one; skip implicit definition of "self" variable. instances[#instances + (token.at > instances[#instances][-1] and 1 or 0)] = {[0] = (not token.self and token.fpos or nil), [-1] = token.at} end if token.fpos <= pos and pos <= token.fpos+#name then this = instances[#instances][-1] end end end instances[#instances][-1] = nil -- remove the current level -- only return the list if "this" instance has been found; -- this is to avoid reporting (improper) instances when checking for -- comments, strings, table fields, etc. return this and instances[#instances] or {} end function IndicateAll(editor, lines) if not ide.config.autoanalyzer then return end local d = delayed[editor] delayed[editor] = nil -- assume this can be finished for now -- this function can be called for an editor tab that is already closed -- when there are still some pending events for it, so handle it. if not pcall(function() editor:GetId() end) then return end -- if markvars is not set in the spec, nothing else to do if not (editor.spec and editor.spec.marksymbols) then return end local indic = styles.indicator or {} local pos, vars = d and d[1] or 1, d and d[2] or nil local start = lines and editor:PositionFromLine(lines)+1 or nil if d and start and pos >= start then -- ignore delayed processing as the change is earlier in the text pos, vars = 1, nil end local tokens = editor:GetTokenList() if start then -- if the range is specified local curindic = editor:GetIndicatorCurrent() editor:SetIndicatorCurrent(indicator.MASKED) for n = #tokens, 1, -1 do local token = tokens[n] -- find the last token before the range if not token.nobreak and token.name and token.fpos+#token.name < start then pos, vars = token.fpos+#token.name, token.context break end -- unmask all variables from the rest of the list if token[1] == 'Masked' then editor:IndicatorClearRange(token.fpos-1, #token.name) end -- trim the list as it will be re-generated table.remove(tokens, n) end -- Clear masked indicators from the current position to the end as these -- will be re-calculated and re-applied based on masking variables. -- This step is needed as some positions could have shifted after updates. editor:IndicatorClearRange(pos-1, editor:GetLength()-pos+1) editor:SetIndicatorCurrent(curindic) -- need to cleanup vars as they may include variables from later -- fragments (because the cut-point was arbitrary). Also need -- to clean variables in other scopes, hence getmetatable use. local vars = vars while vars do for name, var in pairs(vars) do -- remove all variables that are created later than the current pos -- skip all non-variable elements from the vars table if type(name) == 'string' then while type(var) == 'table' and var.fpos and (var.fpos > pos) do var = var.masked -- restored a masked var vars[name] = var end end end vars = getmetatable(vars) and getmetatable(vars).__index end else if pos == 1 then -- if not continuing, then trim the list tokens = editor:ResetTokenList() end end local cleared = {} for indic = 0, indicator.MAX do cleared[indic] = pos end local function IndicateOne(indic, pos, length) editor:SetIndicatorCurrent(indic) editor:IndicatorClearRange(cleared[indic]-1, pos-cleared[indic]) editor:IndicatorFillRange(pos-1, length) cleared[indic] = pos+length end local s = TimeGet() local canwork = start and 0.010 or 0.100 -- use shorter interval when typing local f = editor.spec.marksymbols(editor:GetText(), pos, vars) while true do local op, name, lineinfo, vars, at, nobreak = f() if not op then break end local var = vars and vars[name] local token = {op, name=name, fpos=lineinfo, at=at, context=vars, self = (op == 'VarSelf') or nil, nobreak=nobreak} if op == 'Function' then vars['function'] = (vars['function'] or 0) + 1 end if op == 'FunctionCall' then if indic.fncall and edcfg.showfncall then IndicateOne(indicator.FNCALL, lineinfo, #name) end elseif op ~= 'VarNext' and op ~= 'VarInside' and op ~= 'Statement' and op ~= 'String' then table.insert(tokens, token) end -- indicate local/global variables if op == 'Id' and (var and indic.varlocal or not var and indic.varglobal) then IndicateOne(var and indicator.LOCAL or indicator.GLOBAL, lineinfo, #name) end -- indicate masked values at the same level if op == 'Var' and var and (var.masked and at == var.masked.at) then local fpos = var.masked.fpos -- indicate masked if it's not implicit self if indic.varmasked and not var.masked.self then editor:SetIndicatorCurrent(indicator.MASKED) editor:IndicatorFillRange(fpos-1, #name) table.insert(tokens, {"Masked", name=name, fpos=fpos, nobreak=nobreak}) end if indic.varmasking then IndicateOne(indicator.MASKING, lineinfo, #name) end end if lineinfo and not nobreak and (op == 'Statement' or op == 'String') and TimeGet()-s > canwork then delayed[editor] = {lineinfo, vars} break end end -- clear indicators till the end of processed fragment pos = delayed[editor] and delayed[editor][1] or editor:GetLength()+1 -- don't clear "masked" indicators as those can be set out of order (so -- last updated fragment is not always the last in terms of its position); -- these indicators should be up-to-date to the end of the code fragment. -- also don't clear "funccall" indicators as those can be set based on -- IndicateFunctionsOnly processing, which is dealt with separately local funconly = ide.config.editor.showfncall and editor.spec.isfncall for indic = funconly and indicator.LOCAL or indicator.FNCALL, indicator.MAX do IndicateOne(indic, pos, 0) end local needmore = delayed[editor] ~= nil if ide.config.outlineinactivity then if needmore then ide.timers.outline:Stop() else ide.timers.outline:Start(ide.config.outlineinactivity*1000, wx.wxTIMER_ONE_SHOT) end end return needmore -- request more events if still need to work end -- ---------------------------------------------------------------------------- -- Create an editor function CreateEditor(bare) local editor = wxstc.wxStyledTextCtrl(notebook, editorID, wx.wxDefaultPosition, wx.wxSize(0, 0), wx.wxBORDER_NONE) editorID = editorID + 1 -- increment so they're always unique editor.matchon = false editor.assignscache = false editor.autocomplete = false editor.bom = false editor.jumpstack = {} editor.ctrlcache = {} editor.tokenlist = {} -- populate cache with Ctrl- combinations for workaround on Linux -- http://wxwidgets.10942.n7.nabble.com/Menu-shortcuts-inconsistentcy-issue-td85065.html for id, shortcut in pairs(ide.config.keymap) do local key = shortcut:match('^Ctrl[-+](.)$') if key then editor.ctrlcache[key:byte()] = id end end -- populate editor keymap with configured combinations for _, map in ipairs(edcfg.keymap or {}) do local key, mod, cmd, os = unpack(map) if not os or os == ide.osname then if cmd then editor:CmdKeyAssign(key, mod, cmd) else editor:CmdKeyClear(key, mod) end end end editor:SetBufferedDraw(not ide.config.hidpi and true or false) editor:StyleClearAll() editor:SetFont(ide.font.eNormal) editor:StyleSetFont(wxstc.wxSTC_STYLE_DEFAULT, ide.font.eNormal) editor:SetTabWidth(tonumber(edcfg.tabwidth) or 2) editor:SetIndent(tonumber(edcfg.tabwidth) or 2) editor:SetUseTabs(edcfg.usetabs and true or false) editor:SetIndentationGuides(edcfg.indentguide and true or false) editor:SetViewWhiteSpace(edcfg.whitespace and true or false) if (edcfg.usewrap) then editor:SetWrapMode(wxstc.wxSTC_WRAP_WORD) editor:SetWrapStartIndent(0) if ide.wxver >= "2.9.5" then if edcfg.wrapflags then editor:SetWrapVisualFlags(tonumber(edcfg.wrapflags) or wxstc.wxSTC_WRAPVISUALFLAG_NONE) end if edcfg.wrapstartindent then editor:SetWrapStartIndent(tonumber(edcfg.wrapstartindent) or 0) end if edcfg.wrapindentmode then editor:SetWrapIndentMode(edcfg.wrapindentmode) end end else editor:SetScrollWidth(100) -- set default width editor:SetScrollWidthTracking(1) -- enable width auto-adjustment end if edcfg.defaulteol == wxstc.wxSTC_EOL_CRLF or edcfg.defaulteol == wxstc.wxSTC_EOL_LF then editor:SetEOLMode(edcfg.defaulteol) -- else: keep wxStyledTextCtrl default behavior (CRLF on Windows, LF on Unix) end editor:SetCaretLineVisible(edcfg.caretline and true or false) editor:SetVisiblePolicy(wxstc.wxSTC_VISIBLE_STRICT, 3) editor:SetMarginType(margin.LINENUMBER, wxstc.wxSTC_MARGIN_NUMBER) editor:SetMarginMask(margin.LINENUMBER, 0) editor:SetMarginWidth(margin.LINENUMBER, editor:TextWidth(wxstc.wxSTC_STYLE_DEFAULT, linenummask)) editor:SetMarginWidth(margin.MARKER, 18) editor:SetMarginType(margin.MARKER, wxstc.wxSTC_MARGIN_SYMBOL) editor:SetMarginMask(margin.MARKER, 0xffffffff - wxstc.wxSTC_MASK_FOLDERS) editor:SetMarginSensitive(margin.MARKER, true) editor:MarkerDefine(StylesGetMarker("currentline")) editor:MarkerDefine(StylesGetMarker("breakpoint")) editor:MarkerDefine(StylesGetMarker("bookmark")) if edcfg.fold then editor:SetMarginWidth(margin.FOLD, 18) editor:SetMarginType(margin.FOLD, wxstc.wxSTC_MARGIN_SYMBOL) editor:SetMarginMask(margin.FOLD, wxstc.wxSTC_MASK_FOLDERS) editor:SetMarginSensitive(margin.FOLD, true) end editor:SetFoldFlags(tonumber(edcfg.foldflags) or wxstc.wxSTC_FOLDFLAG_LINEAFTER_CONTRACTED) if ide.wxver >= "2.9.5" then -- allow multiple selection and multi-cursor editing if supported editor:SetMultipleSelection(1) editor:SetAdditionalCaretsBlink(1) editor:SetAdditionalSelectionTyping(1) -- allow extra ascent/descent editor:SetExtraAscent(tonumber(edcfg.extraascent) or 0) editor:SetExtraDescent(tonumber(edcfg.extradescent) or 0) end do local fg, bg = wx.wxWHITE, wx.wxColour(128, 128, 128) local foldtype = foldtypes[edcfg.foldtype] or foldtypes.box local foldmarkers = foldtypes[0] for m = 1, #foldmarkers do editor:MarkerDefine(foldmarkers[m], foldtype[m] or wxstc.wxSTC_MARK_EMPTY, fg, bg) end bg:delete() end if edcfg.calltipdelay and edcfg.calltipdelay > 0 then editor:SetMouseDwellTime(edcfg.calltipdelay) end editor:AutoCompSetIgnoreCase(ide.config.acandtip.ignorecase) if (ide.config.acandtip.strategy > 0) then editor:AutoCompSetAutoHide(0) editor:AutoCompStops([[ \n\t=-+():.,;*/!"'$%&~'#°^@?´`<>][|}{]]) end function editor:GotoPosEnforcePolicy(pos) self:GotoPos(pos) self:EnsureVisibleEnforcePolicy(self:LineFromPosition(pos)) end local function getMarginWidth(editor) local width = 0 for m = 0, 7 do width = width + editor:GetMarginWidth(m) end return width end function editor:ShowPosEnforcePolicy(pos) local line = self:LineFromPosition(pos) self:EnsureVisibleEnforcePolicy(line) if edcfg.usewrap then return end -- skip the rest if line wrapping is on local xwidth = self:GetClientSize():GetWidth() - getMarginWidth(editor) local xoffset = self:GetTextExtent(self:GetLine(line):sub(1, pos-self:PositionFromLine(line)+1)) self:SetXOffset(xoffset > xwidth and xoffset-xwidth or 0) end function editor:GetTokenList() return self.tokenlist end function editor:ResetTokenList() self.tokenlist = {}; return self.tokenlist end function editor:SetupKeywords(...) return SetupKeywords(self, ...) end function editor:ValueFromPosition(pos) return getValAtPosition(self, pos) end -- GotoPos should work by itself, but it doesn't (wx 2.9.5). -- This is likely because the editor window hasn't been refreshed yet, -- so its LinesOnScreen method returns 0/-1, which skews the calculations. -- To avoid this, the caret line is made visible at the first opportunity. do local redolater function editor:GotoPosDelayed(pos) local badtime = self:LinesOnScreen() <= 0 -- -1 on OSX, 0 on Windows if pos then if badtime then redolater = pos -- without this GotoPos the content is not scrolled correctly on -- Windows, but with this it's not scrolled correctly on OSX. if ide.osname ~= 'Macintosh' then self:GotoPos(pos) end else redolater = nil self:GotoPosEnforcePolicy(pos) end elseif not badtime and redolater then -- reset the left margin first to make sure that the position -- is set "from the left" to get the best content displayed. self:SetXOffset(0) self:GotoPosEnforcePolicy(redolater) redolater = nil end end end if bare then return editor end -- bare editor doesn't have any event handlers editor.ev = {} editor:Connect(wxstc.wxEVT_STC_MARGINCLICK, function (event) local line = editor:LineFromPosition(event:GetPosition()) local marginno = event:GetMargin() if marginno == margin.MARKER then DebuggerToggleBreakpoint(editor, line) elseif marginno == margin.FOLD then if wx.wxGetKeyState(wx.WXK_SHIFT) and wx.wxGetKeyState(wx.WXK_CONTROL) then FoldSome() else local level = editor:GetFoldLevel(line) if HasBit(level, wxstc.wxSTC_FOLDLEVELHEADERFLAG) then editor:ToggleFold(line) end end end end) editor:Connect(wxstc.wxEVT_STC_MODIFIED, function (event) if (editor.assignscache and editor:GetCurrentLine() ~= editor.assignscache.line) then editor.assignscache = false end local evtype = event:GetModificationType() local inserted = bit.band(evtype, wxstc.wxSTC_MOD_INSERTTEXT) ~= 0 local deleted = bit.band(evtype, wxstc.wxSTC_MOD_DELETETEXT) ~= 0 if (inserted or deleted) then SetAutoRecoveryMark() local firstLine = editor:LineFromPosition(event:GetPosition()) local linesChanged = inserted and event:GetLinesAdded() or 0 table.insert(editor.ev, {event:GetPosition(), linesChanged}) DynamicWordsAdd(editor, nil, firstLine, linesChanged) end local beforeInserted = bit.band(evtype,wxstc.wxSTC_MOD_BEFOREINSERT) ~= 0 local beforeDeleted = bit.band(evtype,wxstc.wxSTC_MOD_BEFOREDELETE) ~= 0 if (beforeInserted or beforeDeleted) then -- unfold the current line being changed if folded local firstLine = editor:LineFromPosition(event:GetPosition()) if not editor:GetFoldExpanded(firstLine) then editor:ToggleFold(firstLine) end end -- hide calltip/auto-complete after undo/redo/delete local undodelete = (wxstc.wxSTC_MOD_DELETETEXT + wxstc.wxSTC_PERFORMED_UNDO + wxstc.wxSTC_PERFORMED_REDO) if bit.band(evtype, undodelete) ~= 0 then if editor:CallTipActive() then editor:CallTipCancel() end if editor:AutoCompActive() then editor:AutoCompCancel() end end if ide.config.acandtip.nodynwords then return end -- only required to track changes if beforeDeleted then local pos = event:GetPosition() local text = editor:GetTextRange(pos, pos+event:GetLength()) local _, numlines = text:gsub("\r?\n","%1") DynamicWordsRem(editor,nil,editor:LineFromPosition(pos), numlines) end if beforeInserted then DynamicWordsRem(editor,nil,editor:LineFromPosition(event:GetPosition()), 0) end end) editor:Connect(wxstc.wxEVT_STC_CHARADDED, function (event) local LF = string.byte("\n") local ch = event:GetKey() local pos = editor:GetCurrentPos() local line = editor:GetCurrentLine() local linetx = editor:GetLine(line) local linestart = editor:PositionFromLine(line) local localpos = pos-linestart local linetxtopos = linetx:sub(1,localpos) if PackageEventHandle("onEditorCharAdded", editor, event) == false then -- this event has already been handled elseif (ch == LF) then -- auto-indent if (line > 0) then local indent = editor:GetLineIndentation(line - 1) local linedone = editor:GetLine(line - 1) -- if the indentation is 0 and the current line is not empty, -- but the previous line is empty, then take indentation from the -- current line (instead of the previous one). This may happen when -- CR is hit at the beginning of a line (rather than at the end). if indent == 0 and not linetx:match("^[\010\013]*$") and linedone:match("^[\010\013]*$") then indent = editor:GetLineIndentation(line) end local ut = editor:GetUseTabs() local tw = ut and editor:GetTabWidth() or editor:GetIndent() local style = bit.band(editor:GetStyleAt(editor:PositionFromLine(line-1)), 31) if edcfg.smartindent -- don't apply smartindent to multi-line comments or strings and not (editor.spec.iscomment[style] or editor.spec.isstring[style]) and editor.spec.isdecindent and editor.spec.isincindent then local closed, blockend = editor.spec.isdecindent(linedone) local opened = editor.spec.isincindent(linedone) -- if the current block is already indented, skip reverse indenting if (line > 1) and (closed > 0 or blockend > 0) and editor:GetLineIndentation(line-2) > indent then -- adjust opened first; this is needed when use ENTER after }) if blockend == 0 then opened = opened + closed end closed, blockend = 0, 0 end editor:SetLineIndentation(line-1, indent - tw * closed) indent = indent + tw * (opened - blockend) if indent < 0 then indent = 0 end end editor:SetLineIndentation(line, indent) indent = ut and (indent / tw) or indent editor:GotoPos(editor:GetCurrentPos()+indent) end elseif ch == ("("):byte() then local tip = GetTipInfo(editor,linetxtopos,ide.config.acandtip.shorttip) if tip then if editor:CallTipActive() then editor:CallTipCancel() end if PackageEventHandle("onEditorCallTip", editor, tip) ~= false then callTipFitAndShow(editor, pos, tip) end end elseif ide.config.autocomplete then -- code completion prompt local trigger = linetxtopos:match("["..editor.spec.sep.."%w_]+$") -- make sure .autocomplete is never `nil` or editor.autocomplete fails editor.autocomplete = trigger and (#trigger > 1 or trigger:match("["..editor.spec.sep.."]")) and true or false end end) editor:Connect(wxstc.wxEVT_STC_DWELLSTART, function (event) -- on Linux DWELLSTART event seems to be generated even for those -- editor windows that are not active. What's worse, when generated -- the event seems to report "old" position when retrieved using -- event:GetX and event:GetY, so instead we use wxGetMousePosition. local linux = ide.osname == 'Unix' if linux and editor ~= GetEditor() then return end -- check if this editor has focus; it may not when Stack/Watch window -- is on top, but DWELL events are still triggered in this case. -- Don't want to show calltip as it is still shown when the focus -- is switched to a different application. local focus = editor:FindFocus() if focus and focus:GetId() ~= editor:GetId() then return end -- event:GetX() and event:GetY() positions don't correspond to -- the correct positions calculated using ScreenToClient (at least -- on Windows and Linux), so use what's calculated. local mpos = wx.wxGetMousePosition() local cpos = editor:ScreenToClient(mpos) local position = editor:PositionFromPointClose(cpos.x, cpos.y) if position ~= wxstc.wxSTC_INVALID_POSITION then EditorCallTip(editor, position, mpos.x, mpos.y) end event:Skip() end) editor:Connect(wxstc.wxEVT_STC_DWELLEND, function (event) if editor:CallTipActive() then editor:CallTipCancel() end event:Skip() end) editor:Connect(wx.wxEVT_KILL_FOCUS, function (event) if editor:AutoCompActive() then editor:AutoCompCancel() end PackageEventHandle("onEditorFocusLost", editor) event:Skip() end) editor:Connect(wxstc.wxEVT_STC_USERLISTSELECTION, function (event) if PackageEventHandle("onEditorUserlistSelection", editor, event) == false then return end if ide.wxver >= "2.9.5" and editor:GetSelections() > 1 then local text = event:GetText() -- capture all positions as the selection may change local positions = {} for s = 0, editor:GetSelections()-1 do table.insert(positions, editor:GetSelectionNCaret(s)) end -- process all selections from last to first table.sort(positions) local mainpos = editor:GetSelectionNCaret(editor:GetMainSelection()) editor:BeginUndoAction() for s = #positions, 1, -1 do local pos = positions[s] local start_pos = editor:WordStartPosition(pos, true) editor:SetSelection(start_pos, pos) editor:ReplaceSelection(text) -- if this is the main position, save new cursor position to restore if pos == mainpos then mainpos = editor:GetCurrentPos() elseif pos < mainpos then -- adjust main position as earlier changes may affect it mainpos = mainpos + #text - (pos - start_pos) end end editor:EndUndoAction() editor:GotoPos(mainpos) else local pos = editor:GetCurrentPos() local start_pos = editor:WordStartPosition(pos, true) editor:SetSelection(start_pos, pos) editor:ReplaceSelection(event:GetText()) end end) editor:Connect(wxstc.wxEVT_STC_SAVEPOINTREACHED, function () SetDocumentModified(editor:GetId(), false) end) editor:Connect(wxstc.wxEVT_STC_SAVEPOINTLEFT, function () SetDocumentModified(editor:GetId(), true) end) -- "updateStatusText" should be called in UPDATEUI event, but it creates -- several performance problems on Windows (using wx2.9.5+) when -- brackets or backspace is used (very slow screen repaint with 0.5s delay). -- Moving it to PAINTED event creates problems on OSX (using wx2.9.5+), -- where refresh of R/W and R/O status in the status bar is delayed. editor:Connect(wxstc.wxEVT_STC_PAINTED, function (event) PackageEventHandle("onEditorPainted", editor, event) if ide.osname == 'Windows' then -- STC_PAINTED is called on multiple editors when they point to -- the same document; only update status for the active one if notebook:GetSelection() == notebook:GetPageIndex(editor) then updateStatusText(editor) end if edcfg.usewrap ~= true and editor:AutoCompActive() then -- showing auto-complete list leaves artifacts on the screen, -- which can only be fixed by a forced refresh. -- shows with wxSTC 3.21 and both wxwidgets 2.9.5 and 3.1 editor:Update() editor:Refresh() end end end) local alreadyProcessed = 0 editor:Connect(wxstc.wxEVT_STC_UPDATEUI, function (event) -- some of UPDATEUI events are triggered by blinking cursor, and since -- there are no changes, the rest of the processing can be skipped; -- the reason for `alreadyProcessed` is that it is not possible -- to completely skip all of these updates as this causes the issue -- of markup styling becoming visible after text deletion by Backspace. -- to avoid this, we allow the first update after any updates caused -- by real changes; the rest of UPDATEUI events are skipped. if event:GetUpdated() == wxstc.wxSTC_UPDATE_CONTENT and not next(editor.ev) then if alreadyProcessed > 1 then return end else alreadyProcessed = 0 end alreadyProcessed = alreadyProcessed + 1 PackageEventHandle("onEditorUpdateUI", editor, event) if ide.osname ~= 'Windows' then updateStatusText(editor) end editor:GotoPosDelayed() updateBraceMatch(editor) local minupdated for _,iv in ipairs(editor.ev) do local line = editor:LineFromPosition(iv[1]) if not minupdated or line < minupdated then minupdated = line end local ok, res = pcall(IndicateAll, editor, line) if not ok then DisplayOutputLn("Internal error: ",res,line,line+iv[2]) end IndicateFunctionsOnly(editor,line,line+iv[2]) end local firstvisible = editor:DocLineFromVisible(editor:GetFirstVisibleLine()) local lastline = math.min(editor:GetLineCount(), firstvisible + editor:LinesOnScreen()) -- lastline - editor:LinesOnScreen() can get negative; fix it local firstline = math.min(math.max(0, lastline - editor:LinesOnScreen()), firstvisible) MarkupStyle(editor,minupdated or firstline,lastline) editor.ev = {} end) editor:Connect(wx.wxEVT_IDLE, function (event) -- show auto-complete if needed if editor.autocomplete then EditorAutoComplete(editor) editor.autocomplete = false end end) editor:Connect(wx.wxEVT_LEFT_DOWN, function (event) if MarkupHotspotClick then local position = editor:PositionFromPointClose(event:GetX(),event:GetY()) if position ~= wxstc.wxSTC_INVALID_POSITION then if MarkupHotspotClick(position, editor) then return end end end if event:ControlDown() and event:AltDown() -- ide.wxver >= "2.9.5"; fix after GetModifiers is added to wxMouseEvent in wxlua and not event:ShiftDown() and not event:MetaDown() then local point = event:GetPosition() local pos = editor:PositionFromPointClose(point.x, point.y) local value = pos ~= wxstc.wxSTC_INVALID_POSITION and editor:ValueFromPosition(pos) or nil local instances = value and indicateFindInstances(editor, value, pos+1) if instances and instances[0] then navigateToPosition(editor, pos, instances[0]-1, #value) return end end event:Skip() end) if edcfg.nomousezoom then -- disable zoom using mouse wheel as it triggers zooming when scrolling -- on OSX with kinetic scroll and then pressing CMD. editor:Connect(wx.wxEVT_MOUSEWHEEL, function (event) if wx.wxGetKeyState(wx.WXK_CONTROL) then return end event:Skip() end) end local inhandler = false editor:Connect(wx.wxEVT_SET_FOCUS, function (event) event:Skip() if inhandler or ide.exitingProgram then return end inhandler = true PackageEventHandle("onEditorFocusSet", editor) isFileAlteredOnDisk(editor) inhandler = false end) editor:Connect(wx.wxEVT_KEY_DOWN, function (event) local keycode = event:GetKeyCode() local mod = event:GetModifiers() local first, last = 0, notebook:GetPageCount()-1 if PackageEventHandle("onEditorKeyDown", editor, event) == false then -- this event has already been handled elseif keycode == wx.WXK_ESCAPE and ide.frame:IsFullScreen() then ShowFullScreen(false) -- Ctrl-Home and Ctrl-End don't work on OSX with 2.9.5+; fix it elseif ide.osname == 'Macintosh' and ide.wxver >= "2.9.5" and (mod == wx.wxMOD_RAW_CONTROL or mod == (wx.wxMOD_RAW_CONTROL + wx.wxMOD_SHIFT)) and (keycode == wx.WXK_HOME or keycode == wx.WXK_END) then local pos = keycode == wx.WXK_HOME and 0 or editor:GetLength() if event:ShiftDown() -- mark selection and scroll to caret then editor:SetCurrentPos(pos) editor:EnsureCaretVisible() else editor:GotoPos(pos) end elseif mod == wx.wxMOD_RAW_CONTROL and keycode == wx.WXK_PAGEUP or mod == (wx.wxMOD_RAW_CONTROL + wx.wxMOD_SHIFT) and keycode == wx.WXK_TAB then if notebook:GetSelection() == first then notebook:SetSelection(last) else notebook:AdvanceSelection(false) end elseif mod == wx.wxMOD_RAW_CONTROL and (keycode == wx.WXK_PAGEDOWN or keycode == wx.WXK_TAB) then if notebook:GetSelection() == last then notebook:SetSelection(first) else notebook:AdvanceSelection(true) end elseif (keycode == wx.WXK_DELETE or keycode == wx.WXK_BACK) and (mod == wx.wxMOD_NONE) then -- Delete and Backspace behave the same way for selected text if #(editor:GetSelectedText()) > 0 then local length = editor:GetLength() local selections = ide.wxver >= "2.9.5" and editor:GetSelections() or 1 editor:Clear() -- remove selected fragments -- check if the modification has failed, which may happen -- if there is "invisible" text in the selected fragment. -- if there is only one selection, then delete manually. if length == editor:GetLength() and selections == 1 then editor:SetTargetStart(editor:GetSelectionStart()) editor:SetTargetEnd(editor:GetSelectionEnd()) editor:ReplaceTarget("") end else local pos = editor:GetCurrentPos() if keycode == wx.WXK_BACK then pos = pos - 1 if pos < 0 then return end end -- check if the modification is to one of "invisible" characters. -- if not, proceed with "normal" processing as there are other -- events that may depend on Backspace, for example, re-calculating -- auto-complete suggestions. local style = bit.band(editor:GetStyleAt(pos), 31) if not MarkupIsSpecial or not MarkupIsSpecial(style) then -- if BACKSPACE is used at tab stop, with spaces for indentation, -- and only whilespaces on the left, reduce indent if edcfg.backspaceunindent and keycode == wx.WXK_BACK and not editor:GetUseTabs() then -- get the line number from the *current* position of the cursor local line = editor:LineFromPosition(pos+1) local text = editor:GetLine(line):sub(1, pos-editor:PositionFromLine(line)+1) local tw = editor:GetIndent() -- if on the tab stop position and only white spaces on the left if text:find('^%s+$') and #text % tw == 0 then editor:SetLineIndentation(line, editor:GetLineIndentation(line) - tw) editor:GotoPos(pos+1-tw) return end end event:Skip() return end editor:SetTargetStart(pos) editor:SetTargetEnd(pos+1) editor:ReplaceTarget("") end elseif mod == wx.wxMOD_ALT and keycode == wx.WXK_LEFT then -- if no "jump back" is needed, then do normal processing as this -- combination can be mapped to some action if not navigateBack(editor) then event:Skip() end elseif (keycode == wx.WXK_DELETE and mod == wx.wxMOD_SHIFT) or (keycode == wx.WXK_INSERT and mod == wx.wxMOD_CONTROL) then ide.frame:AddPendingEvent(wx.wxCommandEvent( wx.wxEVT_COMMAND_MENU_SELECTED, keycode == wx.WXK_INSERT and ID_COPY or ID_CUT)) elseif ide.osname == "Unix" and ide.wxver >= "2.9.5" and mod == wx.wxMOD_CONTROL and editor.ctrlcache[keycode] then ide.frame:AddPendingEvent(wx.wxCommandEvent( wx.wxEVT_COMMAND_MENU_SELECTED, editor.ctrlcache[keycode])) else if ide.osname == 'Macintosh' and mod == wx.wxMOD_META then return -- ignore a key press if Command key is also pressed end event:Skip() end end) local function selectAllInstances(instances, name, curpos) local this local idx = 0 for _, pos in pairs(instances) do pos = pos - 1 -- positions are 0-based in Scintilla if idx == 0 then -- clear selections first as there seems to be a bug (Scintilla 3.2.3) -- that doesn't reset selection after right mouse click. editor:ClearSelections() editor:SetSelection(pos, pos+#name) else editor:AddSelection(pos+#name, pos) end -- check if this is the current selection if curpos >= pos and curpos <= pos+#name then this = idx end idx = idx + 1 end if this then editor:SetMainSelection(this) end end editor:Connect(wxstc.wxEVT_STC_DOUBLECLICK, function(event) -- only activate selection of instances on Ctrl/Cmd-DoubleClick if event:GetModifiers() == wx.wxMOD_CONTROL then local pos = event:GetPosition() local value = pos ~= wxstc.wxSTC_INVALID_POSITION and editor:ValueFromPosition(pos) or nil local instances = value and indicateFindInstances(editor, value, pos+1) if instances and (instances[0] or #instances > 0) then selectAllInstances(instances, value, pos) return end end event:Skip() end) editor:Connect(wxstc.wxEVT_STC_ZOOM, function(event) editor:SetMarginWidth(margin.LINENUMBER, editor:TextWidth(wxstc.wxSTC_STYLE_DEFAULT, linenummask)) -- if Shift+Zoom is used, then zoom all editors, not just the current one if wx.wxGetKeyState(wx.WXK_SHIFT) then local zoom = editor:GetZoom() for _, doc in pairs(openDocuments) do -- check the editor zoom level to avoid recursion if doc.editor:GetZoom() ~= zoom then doc.editor:SetZoom(zoom) end end end event:Skip() end) local pos, value, instances editor:Connect(wx.wxEVT_CONTEXT_MENU, function (event) local point = editor:ScreenToClient(event:GetPosition()) pos = editor:PositionFromPointClose(point.x, point.y) value = pos ~= wxstc.wxSTC_INVALID_POSITION and editor:ValueFromPosition(pos) or nil instances = value and indicateFindInstances(editor, value, pos+1) local occurrences = (not instances or #instances == 0) and "" or (" (%d)"):format(#instances+(instances[0] and 1 or 0)) local line = instances and instances[0] and editor:LineFromPosition(instances[0]-1)+1 local def = line and " ("..TR("on line %d"):format(line)..")" or "" local selections = ide.wxver >= "2.9.5" and editor:GetSelections() or 1 local menu = wx.wxMenu { { ID_UNDO, TR("&Undo") }, { ID_REDO, TR("&Redo") }, { }, { ID_CUT, TR("Cu&t") }, { ID_COPY, TR("&Copy") }, { ID_PASTE, TR("&Paste") }, { ID_SELECTALL, TR("Select &All") }, { }, { ID_GOTODEFINITION, TR("Go To Definition")..def }, { ID_RENAMEALLINSTANCES, TR("Rename All Instances")..occurrences }, { ID_REPLACEALLSELECTIONS, TR("Replace All Selections") }, { }, { ID_QUICKADDWATCH, TR("Add Watch Expression") }, { ID_QUICKEVAL, TR("Evaluate In Console") }, { ID_ADDTOSCRATCHPAD, TR("Add To Scratchpad") }, } menu:Enable(ID_GOTODEFINITION, instances and instances[0]) menu:Enable(ID_RENAMEALLINSTANCES, instances and (instances[0] or #instances > 0) or editor:GetSelectionStart() ~= editor:GetSelectionEnd()) menu:Enable(ID_REPLACEALLSELECTIONS, selections > 1) menu:Enable(ID_QUICKADDWATCH, value ~= nil) menu:Enable(ID_QUICKEVAL, value ~= nil) local debugger = ide.debugger menu:Enable(ID_ADDTOSCRATCHPAD, debugger.scratchpad and debugger.scratchpad.editors and not debugger.scratchpad.editors[editor]) -- disable calltips that could open over the menu local dwelltime = editor:GetMouseDwellTime() editor:SetMouseDwellTime(0) -- disable dwelling -- cancel calltip if it's already shown as it interferes with popup menu if editor:CallTipActive() then editor:CallTipCancel() end PackageEventHandle("onMenuEditor", menu, editor, event) editor:PopupMenu(menu) editor:SetMouseDwellTime(dwelltime) -- restore dwelling end) editor:Connect(ID_GOTODEFINITION, wx.wxEVT_COMMAND_MENU_SELECTED, function(event) if value and instances[0] then navigateToPosition(editor, editor:GetCurrentPos(), instances[0]-1, #value) end end) editor:Connect(ID_RENAMEALLINSTANCES, wx.wxEVT_COMMAND_MENU_SELECTED, function(event) if value and pos then if not (instances and (instances[0] or #instances > 0)) then -- if multiple instances (of a variable) are not detected, -- then simply find all instances of (selected) `value` instances = {} local length, pos = editor:GetLength(), 0 while true do editor:SetTargetStart(pos) editor:SetTargetEnd(length) pos = editor:SearchInTarget(value) if pos == -1 then break end table.insert(instances, pos+1) pos = pos + #value end end selectAllInstances(instances, value, pos) end end) editor:Connect(ID_REPLACEALLSELECTIONS, wx.wxEVT_COMMAND_MENU_SELECTED, function(event) local main = editor:GetMainSelection() local text = wx.wxGetTextFromUser( TR("Enter replacement text"), TR("Replace All Selections"), editor:GetTextRange(editor:GetSelectionNStart(main), editor:GetSelectionNEnd(main)) ) if not text or text == "" then return end editor:BeginUndoAction() for s = 0, editor:GetSelections()-1 do local selst, selend = editor:GetSelectionNStart(s), editor:GetSelectionNEnd(s) editor:SetTargetStart(selst) editor:SetTargetEnd(selend) editor:ReplaceTarget(text) editor:SetSelectionNStart(s, selst) editor:SetSelectionNEnd(s, selst+#text) end editor:EndUndoAction() editor:SetMainSelection(main) end) editor:Connect(ID_QUICKADDWATCH, wx.wxEVT_COMMAND_MENU_SELECTED, function(event) ide:AddWatch(value) end) editor:Connect(ID_QUICKEVAL, wx.wxEVT_COMMAND_MENU_SELECTED, function(event) ShellExecuteCode(value) end) editor:Connect(ID_ADDTOSCRATCHPAD, wx.wxEVT_COMMAND_MENU_SELECTED, function(event) DebuggerScratchpadOn(editor) end) return editor end -- ---------------------------------------------------------------------------- -- Add an editor to the notebook function AddEditor(editor, name) assert(notebook:GetPageIndex(editor) == -1, "Editor being added is not in the notebook: failed") -- set the document properties local id = editor:GetId() local document = setmetatable({}, ide.proto.Document) document.editor = editor document.fileName = name document.filePath = nil document.modTime = nil document.isModified = false openDocuments[id] = document -- add page only after document is created as there may be handlers -- that expect the document (for example, onEditorFocusSet) if not notebook:AddPage(editor, name, true) then openDocuments[id] = nil return else document.index = notebook:GetPageIndex(editor) return document end end function GetSpec(ext,forcespec) local spec = forcespec -- search proper spec -- allow forcespec for "override" if ext and not spec then for _,curspec in pairs(ide.specs) do local exts = curspec.exts if (exts) then for _,curext in ipairs(exts) do if (curext == ext) then spec = curspec break end end if (spec) then break end end end end return spec end function SetupKeywords(editor, ext, forcespec, styles, font, fontitalic) local lexerstyleconvert = nil local spec = forcespec or GetSpec(ext) -- found a spec setup lexers and keywords if spec then editor:SetLexer(spec.lexer or wxstc.wxSTC_LEX_NULL) lexerstyleconvert = spec.lexerstyleconvert if (spec.keywords) then for i,words in ipairs(spec.keywords) do editor:SetKeyWords(i-1,words) end end editor.api = GetApi(spec.apitype or "none") editor.spec = spec else editor:SetLexer(wxstc.wxSTC_LEX_NULL) editor:SetKeyWords(0, "") editor.api = GetApi("none") editor.spec = ide.specs.none end -- need to set folding property after lexer is set, otherwise -- the folds are not shown (wxwidgets 2.9.5) if edcfg.fold then editor:SetProperty("fold", "1") editor:SetProperty("fold.html", "1") editor:SetProperty("fold.compact", edcfg.foldcompact and "1" or "0") editor:SetProperty("fold.comment", "1") end -- quickfix to prevent weird looks, otherwise need to update styling mechanism for cpp -- cpp "greyed out" styles are styleid + 64 editor:SetProperty("lexer.cpp.track.preprocessor", "0") editor:SetProperty("lexer.cpp.update.preprocessor", "0") -- create italic font if only main font is provided if font and not fontitalic then fontitalic = wx.wxFont(font) fontitalic:SetStyle(wx.wxFONTSTYLE_ITALIC) end StylesApplyToEditor(styles or ide.config.styles, editor, font or ide.font.eNormal,fontitalic or ide.font.eItalic,lexerstyleconvert) end