aboutsummaryrefslogtreecommitdiff
path: root/modules/textadept/command_entry.lua
blob: f0c771ed65729bca120458b9a4fa9caad1c3c061 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
-- Copyright 2007-2011 Mitchell mitchell<att>caladbolg.net. See LICENSE.
-- Modified by Jay Gould.

local L = locale.localize
local events = events

-- Environment for abbreviated commands.
-- @class table
-- @name env
local env = setmetatable({}, {
  __index = function(t, k)
    local f = buffer[k]
    if f and type(f) == 'function' then
      f = function(...) buffer[k](buffer, ...) end
    elseif f == nil then
      f = view[k] or gui[k] or _G[k]
    end
    return f
  end,
  __newindex = function(t, k, v)
    for _, t2 in ipairs{ buffer, view, gui } do
      if t2[k] ~= nil then
        t2[k] = v
        return
      end
    end
    rawset(t, k, v)
  end,
})

-- Execute a Lua command.
events.connect(events.COMMAND_ENTRY_COMMAND, function(command)
  local f, err = loadstring(command)
  if err then error(err) end
  gui.command_entry.focus() -- toggle focus to hide
  setfenv(f, env)
  f()
  events.emit(events.UPDATE_UI)
end)

events.connect(events.COMMAND_ENTRY_KEYPRESS, function(code)
  local ce = gui.command_entry
  local KEYSYMS = keys.KEYSYMS
  if KEYSYMS[code] == 'esc' then
    ce.focus() -- toggle focus to hide
    return true
  elseif KEYSYMS[code] == '\t' then
    local substring = ce.entry_text:match('[%w_.:]+$') or ''
    local path, o, prefix = substring:match('^([%w_.:]-)([.:]?)([%w_]*)$')
    local f, err = loadstring('return ('..path..')')
    if type(f) == "function" then setfenv(f, env) end
    local ok, tbl = pcall(f)
    local cmpls = {}
    prefix = '^'..prefix
    if not ok then -- shorthand notation
      for _, t in ipairs{ buffer, view, gui, _G } do
        for k in pairs(t) do
          if type(k) == 'string' and k:find(prefix) then
            cmpls[#cmpls + 1] = k
          end
        end
      end
      for f in pairs(_SCINTILLA.functions) do
        if f:find(prefix) then cmpls[#cmpls + 1] = f end
      end
      for p in pairs(_SCINTILLA.properties) do
        if p:find(prefix) then cmpls[#cmpls + 1] = p end
      end
    else
      if type(tbl) ~= 'table' then return end
      for k in pairs(tbl) do
        if type(k) == 'string' and k:find(prefix) then cmpls[#cmpls + 1] = k end
      end
      if path == 'buffer' then
        if o == ':' then
          for f in pairs(_SCINTILLA.functions) do
            if f:find(prefix) then cmpls[#cmpls + 1] = f end
          end
        else
          for p in pairs(_SCINTILLA.properties) do
            if p:find(prefix) then cmpls[#cmpls + 1] = p end
          end
        end
      end
    end
    table.sort(cmpls)
    ce.show_completions(cmpls)
    return true
  end
end)