-- Copyright 2007-2020 Mitchell mitchell.att.foicica.com. See LICENSE. package.path = table.concat({ _USERHOME .. '/modules/?.lua', _USERHOME .. '/modules/?/init.lua', _HOME .. '/modules/?.lua', _HOME .. '/modules/?/init.lua', package.path }, ';'); package.cpath = table.concat({ string.format('%s/modules/?.%s', _USERHOME, not WIN32 and 'so' or 'dll'), string.format('%s/modules/?.%s', _HOME, not WIN32 and 'so' or 'dll'), package.cpath }, ';') -- Populate initial `_G.buffer` with temporarily exported io functions now that -- it exists. This is needed for menus and key bindings. for _, name in ipairs{'reload', 'save', 'save_as', 'close'} do buffer[name], io['_' .. name] = io['_' .. name], nil end textadept = require('textadept') -- Documentation is in core/.view.luadoc. local function set_theme(view, name, props) if not assert_type(name, 'string', 2):find('[/\\]') then name = package.searchpath(name, string.format( '%s/themes/?.lua;%s/themes/?.lua', _USERHOME, _HOME)) end if not name or not lfs.attributes(name) then return end if not assert_type(props, 'table/nil', 3) then props = {} end local orig_view = _G.view if view ~= orig_view then ui.goto_view(view) end dofile(name) for prop, value in pairs(props) do view.property[prop] = value end -- Force reload of all styles since the current lexer may have defined its own -- styles. (The LPeg lexer has only refreshed default lexer styles.) -- Note: cannot use `buffer.set_lexer()` because it may not exist yet. local SETLEXERLANGUAGE = _SCINTILLA.properties.lexer_language[2] buffer:private_lexer_call(SETLEXERLANGUAGE, buffer._lexer or 'text') if view ~= orig_view then ui.goto_view(orig_view) end end events.connect(events.VIEW_NEW, function() view.set_theme = set_theme end) view.set_theme = set_theme -- needed for the first view -- On reset, _LOADED['lexer'] is removed. Force a reload in order for set_theme -- to work properly. if not arg then view:goto_buffer(buffer) end -- Temporary compatibility. function lfs.dir_foreach(dir, f, filter, n, include_dirs) ui.dialogs.msgbox{text='Compatibility issue',text='Please change your use of "lfs.dir_foreach()" to "for filename in lfs.walk()"'}; for filename in lfs.walk(dir, filter, n, include_dirs) do if f(filename) == false then return end end end setmetatable(_L, {__index = function(t, k) return rawget(t, k:gsub('_', '')) or 'No Localization:'..k end}) setmetatable(textadept.snippets, {__index = function(t, k) return rawget(t, k:gsub('^_', '')) end}) buffer.set_theme = function(...) view:set_theme(select(2, ...)); events.connect(events.INITIALIZED, function() ui.dialogs.msgbox{title='Compatibility issue',text='Please change your use of "buffer:set_theme()" to "view:set_theme()"'} end) end -- The remainder of this file defines default buffer properties and applies them -- to subsequent buffers. Normally, a setting like `buffer.use_tabs = false` -- only applies to the current (initial) buffer. However, temporarily tap into -- buffer's metatable in order to capture these initial buffer settings (both -- from Textadept's init.lua and from the user's init.lua). local settings = {} local buffer_mt, view_mt = getmetatable(buffer), getmetatable(view) local function repr(v) return string.format(type(v) == 'string' and '%q' or '%s', v) end for _, mt in ipairs{buffer_mt, view_mt} do mt.__orig_index, mt.__orig_newindex = mt.__index, mt.__newindex mt.__index = function(t, k) local v = mt.__orig_index(t, k) if type(v) == 'function' then return function(...) local args = {...} if type(args[1]) == 'table' then table.remove(args, 1) end -- self for i = 1, #args do args[i] = repr(args[i]) end settings[#settings + 1] = string.format( 'buffer:%s(%s)', k, table.concat(args, ',')) return v(...) end elseif type(v) == 'table' then local property_mt = getmetatable(v) setmetatable(v, { __index = property_mt.__index, __newindex = function(property, k2, v2) settings[#settings + 1] = string.format( 'buffer.%s[%s]=%s', k, repr(k2), repr(v2)) property_mt.__newindex(property, k2, v2) end }) end return v end mt.__newindex = function(t, k, v) settings[#settings + 1] = string.format('buffer[%s]=%s', repr(k), repr(v)) mt.__orig_newindex(t, k, v) end end -- Default buffer and view settings. local buffer, view = buffer, view view:set_theme(not CURSES and 'light' or 'term') -- Multiple Selection and Virtual Space buffer.multiple_selection = true buffer.additional_selection_typing = true buffer.multi_paste = buffer.MULTIPASTE_EACH --buffer.virtual_space_options = buffer.VS_RECTANGULARSELECTION | -- buffer.VS_USERACCESSIBLE view.rectangular_selection_modifier = view.MOD_ALT view.mouse_selection_rectangular_switch = true --view.additional_carets_blink = false --view.additional_carets_visible = false -- Scrolling. view:set_x_caret_policy(view.CARET_SLOP, 20) view:set_y_caret_policy( view.CARET_SLOP | view.CARET_STRICT | view.CARET_EVEN, 1) view:set_visible_policy(view.VISIBLE_SLOP | view.VISIBLE_STRICT, 5) --view.h_scroll_bar = CURSES --view.v_scroll_bar = false if CURSES and not (WIN32 or LINUX or BSD) then view.v_scroll_bar = false end --view.scroll_width = --view.scroll_width_tracking = true --view.end_at_last_line = false -- Whitespace view.view_ws = view.WS_INVISIBLE --view.whitespace_size = --view.extra_ascent = --view.extra_descent = -- Line Endings view.view_eol = false -- Styling if not CURSES then view.idle_styling = view.IDLESTYLING_ALL end -- Caret and Selection Styles. --view.sel_eol_filled = true view.caret_line_visible = not CURSES --view.caret_line_visible_always = true --view.caret_period = 0 --view.caret_style = view.CARETSTYLE_BLOCK --view.caret_width = --buffer.caret_sticky = buffer.CARETSTICKY_ON -- Margins. --view.margin_left = --view.margin_right = -- Line Number Margin. view.margin_type_n[1] = view.MARGIN_NUMBER local function resize_line_number_margin() -- This needs to be evaluated dynamically since themes/styles can change. local buffer, view = _G.buffer, _G.view local width = math.max(4, #tostring(buffer.line_count)) * view:text_width(view.STYLE_LINENUMBER, '9') + (not CURSES and 4 or 0) view.margin_width_n[1] = math.max(view.margin_width_n[1], width) end events.connect(events.BUFFER_NEW, resize_line_number_margin) events.connect(events.VIEW_NEW, resize_line_number_margin) events.connect(events.FILE_OPENED, resize_line_number_margin) -- Marker Margin. view.margin_width_n[2] = not CURSES and 4 or 1 -- Fold Margin. view.margin_width_n[3] = not CURSES and 12 or 1 view.margin_mask_n[3] = view.MASK_FOLDERS -- Other Margins. for i = 2, view.margins do view.margin_type_n[i] = view.MARGIN_SYMBOL view.margin_sensitive_n[i] = true view.margin_cursor_n[i] = view.CURSORARROW if i > 3 then view.margin_width_n[i] = 0 end end -- Annotations. view.annotation_visible = view.ANNOTATION_BOXED -- Other. buffer.buffered_draw = not CURSES and not OSX -- Quartz buffers drawing on OSX --buffer.word_chars = --buffer.whitespace_chars = --buffer.punctuation_chars = -- Tabs and Indentation Guides. -- Note: tab and indentation settings apply to individual buffers. buffer.tab_width = 2 buffer.use_tabs = false --buffer.indent = 2 buffer.tab_indents = true buffer.back_space_un_indents = true view.indentation_guides = not CURSES and view.IV_LOOKBOTH or view.IV_NONE -- Margin Markers. view:marker_define(textadept.bookmarks.MARK_BOOKMARK, view.MARK_FULLRECT) view:marker_define(textadept.run.MARK_WARNING, view.MARK_FULLRECT) view:marker_define(textadept.run.MARK_ERROR, view.MARK_FULLRECT) -- Arrow Folding Symbols. --view:marker_define(buffer.MARKNUM_FOLDEROPEN, view.MARK_ARROWDOWN) --view:marker_define(buffer.MARKNUM_FOLDER, view.MARK_ARROW) --view:marker_define(buffer.MARKNUM_FOLDERSUB, view.MARK_EMPTY) --view:marker_define(buffer.MARKNUM_FOLDERTAIL, view.MARK_EMPTY) --view:marker_define(buffer.MARKNUM_FOLDEREND, view.MARK_EMPTY) --view:marker_define(buffer.MARKNUM_FOLDEROPENMID, view.MARK_EMPTY) --view:marker_define(buffer.MARKNUM_FOLDERMIDTAIL, view.MARK_EMPTY) -- Plus/Minus Folding Symbols. --view:marker_define(buffer.MARKNUM_FOLDEROPEN, view.MARK_MINUS) --view:marker_define(buffer.MARKNUM_FOLDER, view.MARK_PLUS) --view:marker_define(buffer.MARKNUM_FOLDERSUB, view.MARK_EMPTY) --view:marker_define(buffer.MARKNUM_FOLDERTAIL, view.MARK_EMPTY) --view:marker_define(buffer.MARKNUM_FOLDEREND, view.MARK_EMPTY) --view:marker_define(buffer.MARKNUM_FOLDEROPENMID, view.MARK_EMPTY) --view:marker_define(buffer.MARKNUM_FOLDERMIDTAIL, view.MARK_EMPTY) -- Circle Tree Folding Symbols. --view:marker_define(buffer.MARKNUM_FOLDEROPEN, view.MARK_CIRCLEMINUS) --view:marker_define(buffer.MARKNUM_FOLDER, view.MARK_CIRCLEPLUS) --view:marker_define(buffer.MARKNUM_FOLDERSUB, view.MARK_VLINE) --view:marker_define(buffer.MARKNUM_FOLDERTAIL, view.MARK_LCORNERCURVE) --view:marker_define( -- buffer.MARKNUM_FOLDEREND, view.MARK_CIRCLEPLUSCONNECTED) --view:marker_define( -- buffer.MARKNUM_FOLDEROPENMID, view.MARK_CIRCLEMINUSCONNECTED) --view:marker_define(buffer.MARKNUM_FOLDERMIDTAIL, view.MARK_TCORNERCURVE) -- Box Tree Folding Symbols. view:marker_define(buffer.MARKNUM_FOLDEROPEN, view.MARK_BOXMINUS) view:marker_define(buffer.MARKNUM_FOLDER, view.MARK_BOXPLUS) view:marker_define(buffer.MARKNUM_FOLDERSUB, view.MARK_VLINE) view:marker_define(buffer.MARKNUM_FOLDERTAIL, view.MARK_LCORNER) view:marker_define(buffer.MARKNUM_FOLDEREND, view.MARK_BOXPLUSCONNECTED) view:marker_define( buffer.MARKNUM_FOLDEROPENMID, view.MARK_BOXMINUSCONNECTED) view:marker_define(buffer.MARKNUM_FOLDERMIDTAIL, view.MARK_TCORNER) --view:marker_enable_highlight(true) -- Indicators. view.indic_style[ui.find.INDIC_FIND] = view.INDIC_ROUNDBOX if not CURSES then view.indic_under[ui.find.INDIC_FIND] = true end local INDIC_BRACEMATCH = textadept.editing.INDIC_BRACEMATCH view.indic_style[INDIC_BRACEMATCH] = view.INDIC_BOX view:brace_highlight_indicator(not CURSES, INDIC_BRACEMATCH) local INDIC_HIGHLIGHT = textadept.editing.INDIC_HIGHLIGHT view.indic_style[INDIC_HIGHLIGHT] = view.INDIC_ROUNDBOX if not CURSES then view.indic_under[INDIC_HIGHLIGHT] = true end local INDIC_PLACEHOLDER = textadept.snippets.INDIC_PLACEHOLDER view.indic_style[INDIC_PLACEHOLDER] = not CURSES and view.INDIC_DOTBOX or view.INDIC_STRAIGHTBOX -- Autocompletion. --buffer.auto_c_separator = --buffer.auto_c_cancel_at_start = false --buffer.auto_c_fill_ups = '(' buffer.auto_c_choose_single = true --buffer.auto_c_ignore_case = true --buffer.auto_c_case_insensitive_behaviour = -- buffer.CASEINSENSITIVEBEHAVIOUR_IGNORECASE buffer.auto_c_multi = buffer.MULTIAUTOC_EACH --buffer.auto_c_auto_hide = false --buffer.auto_c_drop_rest_of_word = true --buffer.auto_c_type_separator = --view.auto_c_max_height = --view.auto_c_max_width = -- Call Tips. view.call_tip_use_style = buffer.tab_width * view:text_width(view.STYLE_CALLTIP, ' ') --view.call_tip_position = true -- Folding. view.property['fold'] = '1' --view.property['fold.by.indentation'] = '1' --view.property['fold.line.comments'] = '1' --view.property['fold.on.zero.sum.lines'] = '1' --view.property['fold.compact'] = '1' view.automatic_fold = view.AUTOMATICFOLD_SHOW | view.AUTOMATICFOLD_CLICK | view.AUTOMATICFOLD_CHANGE view.fold_flags = not CURSES and view.FOLDFLAG_LINEAFTER_CONTRACTED or 0 view.fold_display_text_style = view.FOLDDISPLAYTEXT_BOXED -- Line Wrapping. view.wrap_mode = view.WRAP_NONE --view.wrap_visual_flags = view.WRAPVISUALFLAG_MARGIN --view.wrap_visual_flags_location = view.WRAPVISUALFLAGLOC_END_BY_TEXT --view.wrap_indent_mode = view.WRAPINDENT_SAME --view.wrap_start_indent = -- Long Lines. --view.edge_mode = not CURSES and view.EDGE_LINE or view.EDGE_BACKGROUND --view.edge_column = 80 -- Accessibility. buffer.accessibility = buffer.ACCESSIBILITY_DISABLED -- Load user init file, which may also define default buffer settings. local user_init = _USERHOME .. '/init.lua' if lfs.attributes(user_init) then dofile(user_init) end -- Generate default buffer settings for subsequent buffers and remove temporary -- buffer metatable listener. local load_settings = load(table.concat(settings, '\n')) for _, mt in ipairs{buffer_mt, view_mt} do mt.__index, mt.__newindex = mt.__orig_index, mt.__orig_newindex end local SETLEXERLANGUAGE = _SCINTILLA.properties.lexer_language[2] -- Sets default properties for a Scintilla document. events.connect(events.BUFFER_NEW, function() local buffer = _G.buffer local SETDIRECTFUNCTION = _SCINTILLA.properties.direct_function[1] local SETDIRECTPOINTER = _SCINTILLA.properties.doc_pointer[2] local SETLUASTATE = _SCINTILLA.functions.change_lexer_state[1] local LOADLEXERLIBRARY = _SCINTILLA.functions.load_lexer_library[1] buffer.lexer_language = 'lpeg' buffer:private_lexer_call(SETDIRECTFUNCTION, buffer.direct_function) buffer:private_lexer_call(SETDIRECTPOINTER, buffer.direct_pointer) buffer:private_lexer_call(SETLUASTATE, _LUA) buffer:private_lexer_call(LOADLEXERLIBRARY, _USERHOME .. '/lexers') buffer:private_lexer_call(LOADLEXERLIBRARY, _HOME .. '/lexers') load_settings() buffer:private_lexer_call(SETLEXERLANGUAGE, 'text') if buffer == ui.command_entry then ui.command_entry.caret_line_visible = false end end, 1) -- Sets default properties for a Scintilla window. events.connect(events.VIEW_NEW, function() local buffer, view = _G.buffer, _G.view -- Allow redefinitions of these Scintilla key commands. local ctrl_keys = { '[', ']', '/', '\\', 'Z', 'Y', 'X', 'C', 'V', 'A', 'L', 'T', 'D', 'U' } for _, key in ipairs(ctrl_keys) do view:clear_cmd_key(string.byte(key) | view.MOD_CTRL << 16) end for _, key in ipairs{'L', 'T', 'U', 'Z'} do -- ctrl+shift keys view:clear_cmd_key( string.byte(key) | (view.MOD_CTRL | view.MOD_SHIFT) << 16) end -- Since BUFFER_NEW loads themes and settings on startup, only load them for -- subsequent views. if #_VIEWS == 1 then return end load_settings() -- Refresh styles since the user may have altered style settings. -- When load_settings() calls `view.property['style.default'] = ...`, the -- LPeg lexer resets all styles to that default. However, load_settings() may -- later call a user's `view.property['fontsize'] = ...`, which -- 'style.default' references. Styles are now stale and need refreshing. This -- is not an issue in BUFFER_NEW since a lexer is set immediately afterwards, -- which refreshes styles. -- Note: `buffer:set_lexer()` is insufficient for some reason. buffer:private_lexer_call(SETLEXERLANGUAGE, buffer._lexer or 'text') end, 1)