diff --git a/.github/workflows/docgen.yml b/.github/workflows/docgen.yml index 014375f06..43cb76922 100644 --- a/.github/workflows/docgen.yml +++ b/.github/workflows/docgen.yml @@ -4,6 +4,9 @@ on: push: branches: - master + paths: + - docs/** + - lua/orgmode/api/** jobs: docgen: diff --git a/docs/configuration.org b/docs/configuration.org index 69f9394c3..01c6818e6 100644 --- a/docs/configuration.org +++ b/docs/configuration.org @@ -2838,3 +2838,25 @@ require('orgmode').setup({ } }) #+end_src + +*** Input +:PROPERTIES: +:CUSTOM_ID: input +:END: +- Type: =boolean= +- Default: =false= + +By default, Vim's built-in =input()= function is used for input prompts. +To use Neovim's =vim.ui.input()= function, add this to config: + +#+begin_src lua +require('orgmode').setup({ + ui = { + input = { + use_vim_ui = true + } + } +}) +#+end_src + +📝 NOTE: If you are using a plugin for =vim.ui.input=, make sure it supports autocompletion for better experience. [[https://github.com/folke/snacks.nvim][snacks.nvim]] input module supports autocompletion. diff --git a/lua/orgmode/agenda/init.lua b/lua/orgmode/agenda/init.lua index 74cad80ff..308dc08cd 100644 --- a/lua/orgmode/agenda/init.lua +++ b/lua/orgmode/agenda/init.lua @@ -7,6 +7,7 @@ local AgendaFilter = require('orgmode.agenda.filter') local Menu = require('orgmode.ui.menu') local Promise = require('orgmode.utils.promise') local AgendaTypes = require('orgmode.agenda.types') +local Input = require('orgmode.ui.input') ---@class OrgAgenda ---@field highlights table[] @@ -48,7 +49,25 @@ function Agenda:open_view(type, opts) return end self.views = { view } - return self:render() + return self:prepare_and_render() +end + +function Agenda:prepare_and_render() + return Promise.map(function(view) + return view:prepare() + end, self.views):next(function(views) + local valid_views = vim.tbl_filter(function(view) + return view ~= false + end, views) + + -- Some of the views returned false, abort render + if #valid_views ~= #self.views then + return + end + + self.views = views + return self:render() + end) end function Agenda:render() @@ -71,6 +90,10 @@ function Agenda:render() vim.w.org_window_pos = nil end end + + if #self.views > 1 then + vim.fn.cursor({ 1, 0 }) + end end function Agenda:agenda(opts) @@ -151,11 +174,7 @@ function Agenda:_build_custom_commands() table.insert(views, AgendaTypes[agenda_type.type]:new(opts)) end self.views = views - local result = self:render() - if #self.views > 1 then - vim.fn.cursor({ 1, 0 }) - end - return result + return self:prepare_and_render() end, }) end @@ -271,16 +290,21 @@ end ---@param source? string function Agenda:redo(source, preserve_cursor_pos) self:_call_all_views('redo') - return self.files:load(true):next(vim.schedule_wrap(function() - local save_view = preserve_cursor_pos and vim.fn.winsaveview() - if source == 'mapping' then - self:_call_view_and_render('redraw') - end - self:render() - if save_view then - vim.fn.winrestview(save_view) - end - end)) + local save_view = preserve_cursor_pos and vim.fn.winsaveview() + return self.files + :load(true) + :next(function() + if source == 'mapping' then + return self:_call_view_async('redraw') + end + return true + end) + :next(function() + self:render() + if save_view then + vim.fn.winrestview(save_view) + end + end) end function Agenda:advance_span(direction) @@ -499,14 +523,15 @@ end function Agenda:filter() local this = self self.filters:parse_available_filters(self.views) - local filter_term = utils.input('Filter [+cat-tag/regexp/]: ', self.filters.value, function(arg_lead) + return Input.open('Filter [+cat-tag/regexp/]: ', self.filters.value, function(arg_lead) return utils.prompt_autocomplete(arg_lead, this.filters:get_completion_list(), { '+', '-' }) + end):next(function(value) + if not value or value == self.filters.value then + return false + end + self.filters:parse(value) + return self:redo('filter', true) end) - if filter_term == self.filters.value then - return - end - self.filters:parse(filter_term) - return self:redo('filter', true) end ---@param opts table @@ -576,6 +601,22 @@ function Agenda:_call_view(method, ...) return executed end +function Agenda:_call_view_async(method, ...) + local args = { ... } + return Promise.map(function(view) + if view[method] and view.view:is_in_range() then + return view[method](view, unpack(args)) + end + end, self.views):next(function(views) + for _, view in ipairs(views) do + if view then + return true + end + end + return false + end) +end + function Agenda:_call_all_views(method, ...) local executed = false for _, view in ipairs(self.views) do diff --git a/lua/orgmode/agenda/types/agenda.lua b/lua/orgmode/agenda/types/agenda.lua index ac3556bce..061f06d1f 100644 --- a/lua/orgmode/agenda/types/agenda.lua +++ b/lua/orgmode/agenda/types/agenda.lua @@ -9,13 +9,7 @@ local AgendaLineToken = require('orgmode.agenda.view.token') local ClockReport = require('orgmode.clock.report') local utils = require('orgmode.utils') local SortingStrategy = require('orgmode.agenda.sorting_strategy') - ----@class OrgAgendaViewType ----@field render fun(self: OrgAgendaViewType, bufnr:number, current_line?: number): OrgAgendaView ----@field get_lines fun(self: OrgAgendaViewType): OrgAgendaLine | OrgAgendaLine[] ----@field get_line fun(self: OrgAgendaViewType, line_nr: number): OrgAgendaLine | nil ----@field rerender_agenda_line fun(self: OrgAgendaViewType, agenda_line: OrgAgendaLine, headline: OrgHeadline): OrgAgendaLine | nil ----@field view OrgAgendaView +local Promise = require('orgmode.utils.promise') ---@class OrgAgendaTypeOpts ---@field files OrgFiles @@ -97,6 +91,10 @@ function OrgAgendaType:new(opts) return this end +function OrgAgendaType:prepare() + return Promise.resolve(self) +end + function OrgAgendaType:redo() if self.agenda_files then self.files:load_sync(true) diff --git a/lua/orgmode/agenda/types/init.lua b/lua/orgmode/agenda/types/init.lua index e1caf3770..1ac7104f3 100644 --- a/lua/orgmode/agenda/types/init.lua +++ b/lua/orgmode/agenda/types/init.lua @@ -1,3 +1,13 @@ +---@class OrgAgendaViewType +---@field render fun(self: OrgAgendaViewType, bufnr:number, current_line?: number): OrgAgendaView +---@field get_lines fun(self: OrgAgendaViewType): OrgAgendaLine | OrgAgendaLine[] +---@field get_line fun(self: OrgAgendaViewType, line_nr: number): OrgAgendaLine | nil +---@field rerender_agenda_line fun(self: OrgAgendaViewType, agenda_line: OrgAgendaLine, headline: OrgHeadline): OrgAgendaLine | nil +---@field view OrgAgendaView +---@field prepare fun(self: OrgAgendaViewType): OrgPromise +---@field redraw? fun(self: OrgAgendaViewType): OrgPromise +---@field redo? fun(self: OrgAgendaViewType): OrgPromise + ---@alias OrgAgendaTypes 'agenda' | 'todo' | 'tags' | 'tags_todo' | 'search' return { agenda = require('orgmode.agenda.types.agenda'), diff --git a/lua/orgmode/agenda/types/search.lua b/lua/orgmode/agenda/types/search.lua index de3ca8b5b..54900d03e 100644 --- a/lua/orgmode/agenda/types/search.lua +++ b/lua/orgmode/agenda/types/search.lua @@ -1,6 +1,6 @@ -local utils = require('orgmode.utils') ---@diagnostic disable: inject-field local OrgAgendaTodosType = require('orgmode.agenda.types.todo') +local Input = require('orgmode.ui.input') ---@class OrgAgendaSearchTypeOpts:OrgAgendaTodosTypeOpts ---@field headline_query? string @@ -18,18 +18,27 @@ function OrgAgendaSearchType:new(opts) local obj = OrgAgendaTodosType:new(opts) setmetatable(obj, self) obj.headline_query = self.headline_query - if not opts.headline_query or opts.headline_query == '' then - obj.headline_query = self:get_search_term() - end return obj end +function OrgAgendaSearchType:prepare() + if not self.headline_query or self.headline_query == '' then + return self:get_search_term() + end +end + function OrgAgendaSearchType:get_file_headlines(file) return file:find_headlines_matching_search_term(self.headline_query or '', false, true) end function OrgAgendaSearchType:get_search_term() - return utils.input('Enter search term: ', self.headline_query or '') + return Input.open('Enter search term: ', self.headline_query or ''):next(function(value) + if not value then + return false + end + self.headline_query = value + return self + end) end function OrgAgendaSearchType:redraw() @@ -37,8 +46,7 @@ function OrgAgendaSearchType:redraw() if self.id then return self end - self.headline_query = self:get_search_term() - return self + return self:get_search_term() end return OrgAgendaSearchType diff --git a/lua/orgmode/agenda/types/tags.lua b/lua/orgmode/agenda/types/tags.lua index edcd9f299..23be657a4 100644 --- a/lua/orgmode/agenda/types/tags.lua +++ b/lua/orgmode/agenda/types/tags.lua @@ -4,6 +4,7 @@ local config = require('orgmode.config') local utils = require('orgmode.utils') local Search = require('orgmode.files.elements.search') local OrgAgendaTodosType = require('orgmode.agenda.types.todo') +local Input = require('orgmode.ui.input') ---@alias OrgAgendaTodoIgnoreDeadlinesTypes 'all' | 'near' | 'far' | 'past' | 'future' ---@alias OrgAgendaTodoIgnoreScheduledTypes 'all' | 'past' | 'future' @@ -27,24 +28,31 @@ function OrgAgendaTagsType:new(opts) if not opts.id then opts.subheader = 'Press "r" to update search' end - local match_query = opts.match_query - if not opts.id and (not match_query or match_query == '') then - match_query = self:get_tags(opts.files) - if not match_query then - return nil - end - end - setmetatable(self, { __index = OrgAgendaTodosType }) local obj = OrgAgendaTodosType:new(opts) setmetatable(obj, self) - obj.match_query = match_query or '' + obj.match_query = opts.match_query or '' obj.todo_ignore_deadlines = opts.todo_ignore_deadlines obj.todo_ignore_scheduled = opts.todo_ignore_scheduled - obj.header = opts.header or ('Headlines with TAGS match: ' .. obj.match_query) return obj end +function OrgAgendaTagsType:_get_header() + if self.header then + return self.header + end + + return 'Headlines with TAGS match: ' .. (self.match_query or '') +end + +function OrgAgendaTagsType:prepare() + if self.id or self.match_query and self.match_query ~= '' then + return self + end + + return self:get_tags() +end + function OrgAgendaTagsType:get_file_headlines(file) local headlines = file:apply_search(Search:new(self.match_query), self.todo_only) if self.todo_ignore_deadlines then @@ -94,15 +102,20 @@ function OrgAgendaTagsType:get_file_headlines(file) return headlines end ----@param files? OrgFiles -function OrgAgendaTagsType:get_tags(files) - local tags = utils.input('Match: ', self.match_query or '', function(arg_lead) - return utils.prompt_autocomplete(arg_lead, (files or self.files):get_tags()) +function OrgAgendaTagsType:get_tags() + return Input.open('Match: ', self.match_query or '', function(arg_lead) + return utils.prompt_autocomplete(arg_lead, self.files:get_tags()) + end):next(function(tags) + if not tags then + return false + end + if vim.trim(tags) == '' then + utils.echo_warning('Invalid tag.') + return false + end + self.match_query = tags + return self end) - if vim.trim(tags) == '' then - return utils.echo_warning('Invalid tag.') - end - return tags end function OrgAgendaTagsType:redraw() @@ -110,9 +123,7 @@ function OrgAgendaTagsType:redraw() if self.id then return self end - self.match_query = self:get_tags() or '' - self.header = 'Headlines with TAGS match: ' .. self.match_query - return self + return self:get_tags() end return OrgAgendaTagsType diff --git a/lua/orgmode/agenda/types/todo.lua b/lua/orgmode/agenda/types/todo.lua index 06656234e..66be36b7e 100644 --- a/lua/orgmode/agenda/types/todo.lua +++ b/lua/orgmode/agenda/types/todo.lua @@ -8,6 +8,7 @@ local utils = require('orgmode.utils') local agenda_highlights = require('orgmode.colors.highlights') local hl_map = agenda_highlights.get_agenda_hl_map() local SortingStrategy = require('orgmode.agenda.sorting_strategy') +local Promise = require('orgmode.utils.promise') ---@class OrgAgendaTodosTypeOpts ---@field files OrgFiles @@ -74,6 +75,10 @@ function OrgAgendaTodosType:new(opts) return this end +function OrgAgendaTodosType:prepare() + return Promise.resolve(self) +end + function OrgAgendaTodosType:_setup_agenda_files() if not self.agenda_files then return @@ -90,6 +95,13 @@ function OrgAgendaTodosType:redo() end end +function OrgAgendaTodosType:_get_header() + if self.header then + return self.header + end + return 'Global list of TODO items of type: ALL' +end + ---@param bufnr? number function OrgAgendaTodosType:render(bufnr) self.bufnr = bufnr or 0 @@ -97,7 +109,7 @@ function OrgAgendaTodosType:render(bufnr) local agendaView = AgendaView:new({ bufnr = self.bufnr, highlighter = self.highlighter }) agendaView:add_line(AgendaLine:single_token({ - content = self.header or 'Global list of TODO items of type: ALL', + content = self:_get_header(), hl_group = '@org.agenda.header', })) if self.subheader then diff --git a/lua/orgmode/api/init.lua b/lua/orgmode/api/init.lua index 11fb7be5a..ec5ba9f94 100644 --- a/lua/orgmode/api/init.lua +++ b/lua/orgmode/api/init.lua @@ -3,6 +3,7 @@ local OrgFile = require('orgmode.api.file') local OrgHeadline = require('orgmode.api.headline') local orgmode = require('orgmode') local validator = require('orgmode.utils.validator') +local Promise = require('orgmode.utils.promise') ---@class OrgApiRefileOpts ---@field source OrgApiHeadline @@ -54,7 +55,7 @@ end ---Refile headline to another file or headline ---If executed from capture buffer, it will close the capture buffer ---@param opts OrgApiRefileOpts ----@return boolean +---@return OrgPromise function OrgApi.refile(opts) validator.validate({ source = { opts.source, 'table' }, @@ -91,13 +92,16 @@ function OrgApi.refile(opts) refile_opts.template = orgmode.capture._window.template end - if is_capture then - orgmode.capture:_refile_from_capture_buffer(refile_opts) - else - orgmode.capture:_refile_from_org_file(refile_opts) - end - - return true + return Promise.resolve() + :next(function() + if is_capture then + return orgmode.capture:_refile_from_capture_buffer(refile_opts) + end + return orgmode.capture:_refile_from_org_file(refile_opts) + end) + :next(function() + return true + end) end --- Insert a link to a given location at the current cursor position @@ -108,10 +112,9 @@ end --- If is *, is used as prefilled description for the link. --- If is id, this format can also be used to pass a prefilled description. --- @param link_location string ---- @return boolean +--- @return OrgPromise function OrgApi.insert_link(link_location) - orgmode.links:insert_link(link_location) - return true + return orgmode.links:insert_link(link_location) end return OrgApi diff --git a/lua/orgmode/capture/init.lua b/lua/orgmode/capture/init.lua index efc368433..6184eee5e 100644 --- a/lua/orgmode/capture/init.lua +++ b/lua/orgmode/capture/init.lua @@ -8,6 +8,8 @@ local Range = require('orgmode.files.elements.range') local CaptureWindow = require('orgmode.capture.window') local Date = require('orgmode.objects.date') local Datetree = require('orgmode.capture.template.datetree') +local Input = require('orgmode.ui.input') +local Promise = require('orgmode.utils.promise') ---@alias OrgOnCaptureClose fun(capture:OrgCapture, opts:OrgProcessCaptureOpts) ---@alias OrgOnCaptureCancel fun(capture:OrgCapture) @@ -130,14 +132,18 @@ end function Capture:refile_to_destination() local source_file = self.files:get_current_file() local source_headline = source_file:get_headlines()[1] - local destination = self:get_destination() - self:_refile_from_capture_buffer({ - template = self._window.template, - source_file = source_file, - source_headline = source_headline, - destination_file = destination.file, - destination_headline = destination.headline, - }) + return self:get_destination():next(function(destination) + if not destination then + return false + end + return self:_refile_from_capture_buffer({ + template = self._window.template, + source_file = source_file, + source_headline = source_headline, + destination_file = destination.file, + destination_headline = destination.headline, + }) + end) end ---Triggered from org file when we want to refile headline @@ -204,54 +210,68 @@ end ---Refile a headline from a regular org file (non-capture) ---@private ---@param opts OrgProcessRefileOpts ----@return number +---@return OrgPromise function Capture:_refile_from_org_file(opts) local source_headline = opts.source_headline local source_file = source_headline.file local destination_file = opts.destination_file local destination_headline = opts.destination_headline - if not opts.destination_file then - local destination = self:get_destination() - destination_file = destination.file - destination_headline = destination.headline - end - assert(destination_file) + return Promise.resolve() + :next(function() + if not opts.destination_file then + return self:get_destination():next(function(destination) + if not destination then + return false + end + destination_file = destination.file + destination_headline = destination.headline + return destination + end) + end + end) + :next(function() + if not destination_file then + return false + end - local is_same_file = source_file.filename == destination_file.filename + local is_same_file = source_file.filename == destination_file.filename - local target_level = 0 - local target_line = -1 + local target_level = 0 + local target_line = -1 - if destination_headline then - target_level = destination_headline:get_level() - target_line = destination_headline:get_range().end_line - end + if destination_headline then + target_level = destination_headline:get_level() + target_line = destination_headline:get_range().end_line + end - local lines = source_headline:get_lines() + local lines = source_headline:get_lines() - if destination_headline or source_headline:get_level() > 1 then - lines = self:_adapt_headline_level(source_headline, target_level, is_same_file) - end + if destination_headline or source_headline:get_level() > 1 then + lines = self:_adapt_headline_level(source_headline, target_level, is_same_file) + end - destination_file:update_sync(function(file) - if is_same_file then - local item_range = source_headline:get_range() - return vim.cmd(string.format('silent! %d,%d move %s', item_range.start_line, item_range.end_line, target_line)) - end + destination_file:update_sync(function() + if is_same_file then + local item_range = source_headline:get_range() + return vim.cmd( + string.format('silent! %d,%d move %s', item_range.start_line, item_range.end_line, target_line) + ) + end - local range = self:_get_destination_range_without_empty_lines(Range.from_line(target_line)) - target_line = range.start_line - vim.api.nvim_buf_set_lines(0, range.start_line, range.end_line, false, lines) - end) + local range = self:_get_destination_range_without_empty_lines(Range.from_line(target_line)) + target_line = range.start_line + vim.api.nvim_buf_set_lines(0, range.start_line, range.end_line, false, lines) + end) - if not is_same_file and source_file.filename == utils.current_file_path() then - local item_range = source_headline:get_range() - vim.api.nvim_buf_set_lines(0, item_range.start_line - 1, item_range.end_line, false, {}) - end + if not is_same_file and source_file.filename == utils.current_file_path() then + local item_range = source_headline:get_range() + vim.api.nvim_buf_set_lines(0, item_range.start_line - 1, item_range.end_line, false, {}) + end - utils.echo_info(opts.message or ('Wrote %s'):format(destination_file.filename)) - return target_line + 1 + utils.echo_info(opts.message or ('Wrote %s'):format(destination_file.filename)) + return target_line + 1 + end) end ---@param headline OrgHeadline @@ -278,24 +298,26 @@ function Capture:refile_file_headline_to_archive(headline) local destination_file = self.files:get(archive_location) local todo_state = headline:get_todo() - local target_line = self:_refile_from_org_file({ - source_headline = headline, - destination_file = destination_file, - message = ('Archived to %s'):format(destination_file.filename), - }) - - destination_file = self.files:get(archive_location) - destination_file:update(function(archive_file) - local archived_headline = archive_file:get_closest_headline({ target_line, 0 }) - archived_headline:set_property('ARCHIVE_TIME', Date.now():to_string()) - archived_headline:set_property('ARCHIVE_FILE', file.filename) - local outline_path = headline:get_outline_path() - if outline_path ~= '' then - archived_headline:set_property('ARCHIVE_OLPATH', outline_path) - end - archived_headline:set_property('ARCHIVE_CATEGORY', headline:get_category()) - archived_headline:set_property('ARCHIVE_TODO', todo_state or '') - end) + return self + :_refile_from_org_file({ + source_headline = headline, + destination_file = destination_file, + message = ('Archived to %s'):format(destination_file.filename), + }) + :next(function(target_line) + destination_file = self.files:get(archive_location) + return destination_file:update(function(archive_file) + local archived_headline = archive_file:get_closest_headline({ target_line, 0 }) + archived_headline:set_property('ARCHIVE_TIME', Date.now():to_string()) + archived_headline:set_property('ARCHIVE_FILE', file.filename) + local outline_path = headline:get_outline_path() + if outline_path ~= '' then + archived_headline:set_property('ARCHIVE_OLPATH', outline_path) + end + archived_headline:set_property('ARCHIVE_CATEGORY', headline:get_category()) + archived_headline:set_property('ARCHIVE_TODO', todo_state or '') + end) + end) end ---@param item OrgHeadline @@ -359,51 +381,53 @@ function Capture:_get_destination_range_without_empty_lines(range) end --- Prompt for file (and headline) where to refile to ---- @return { file: OrgFile, headline?: OrgHeadline} +--- @return OrgPromise<{ file: OrgFile, headline?: OrgHeadline}> function Capture:get_destination() local valid_destinations = self:_get_autocompletion_files() - local destination = utils.input('Enter destination: ', '', function(arg_lead) + return Input.open('Enter destination: ', '', function(arg_lead) return self:autocomplete_refile(arg_lead, valid_destinations) - end) + end):next(function(destination) + if not destination then + return false + end - local path = destination:match('^.*%.org/') - local headline_title = path and destination:sub(#path + 1) or '' + local path = destination:match('^.*%.org/') + local headline_title = path and destination:sub(#path + 1) or '' - if not valid_destinations[path] then - utils.echo_error( - ('"%s" is not a is not a file specified in the "org_agenda_files" setting. Refiling cancelled.'):format( - destination[1] + if not valid_destinations[path] then + utils.echo_error( + ('"%s" is not a is not a file specified in the "org_agenda_files" setting. Refiling cancelled.'):format(path) ) - ) - return {} - end + return false + end - local destination_file = valid_destinations[path] - local result = { - file = destination_file, - } + local destination_file = valid_destinations[path] + local result = { + file = destination_file, + } - if not headline_title or vim.trim(headline_title) == '' then - return result - end + if not headline_title or vim.trim(headline_title) == '' then + return result + end - local headlines = vim.tbl_filter(function(item) - local pattern = '^' .. vim.pesc(headline_title:lower()) .. '$' - return item:get_title():lower():match(pattern) - end, destination_file:get_opened_unfinished_headlines()) + local headlines = vim.tbl_filter(function(item) + local pattern = '^' .. vim.pesc(headline_title:lower()) .. '$' + return item:get_title():lower():match(pattern) + end, destination_file:get_opened_unfinished_headlines()) - if not headlines[1] then - utils.echo_error( - ("'%s' is not a valid headline in '%s'. Refiling cancelled."):format(headline_title, destination_file.filename) - ) - return {} - end + if not headlines[1] then + utils.echo_error( + ("'%s' is not a valid headline in '%s'. Refiling cancelled."):format(headline_title, destination_file.filename) + ) + return {} + end - return { - file = destination_file, - headline = headlines[1], - } + return { + file = destination_file, + headline = headlines[1], + } + end) end ---@param arg_lead string diff --git a/lua/orgmode/capture/template/init.lua b/lua/orgmode/capture/template/init.lua index acee3f12e..c22d49002 100644 --- a/lua/orgmode/capture/template/init.lua +++ b/lua/orgmode/capture/template/init.lua @@ -4,6 +4,7 @@ local utils = require('orgmode.utils') local validator = require('orgmode.utils.validator') local Calendar = require('orgmode.objects.calendar') local Promise = require('orgmode.utils.promise') +local Input = require('orgmode.ui.input') local expansions = { ['%%f'] = function() @@ -353,34 +354,48 @@ function Template:_compile_dates(content) end ---@param content string ----@return string +---@return OrgPromise function Template:_compile_prompts(content) + local prepared_inputs = {} for exp in content:gmatch('%%%^%b{}') do local details = exp:match('%{(.*)%}') local parts = vim.split(details, '|') local title, default = parts[1], parts[2] - local response + local input = { + fallback_value = default, + exp = exp, + } if #parts > 2 then - local completion_items = vim.list_slice(parts, 3, #parts) - local prompt = string.format('%s [%s]: ', title, default) - response = utils.input(prompt, '', function(arg_lead) - return vim.tbl_filter(function(v) - return v:match('^' .. vim.pesc(arg_lead)) - end, completion_items) - end) + input.prompt = string.format('%s [%s]: ', title, default) + input.completion = function() + local completion_items = vim.list_slice(parts, 3, #parts) + return function(arg_lead) + return vim.tbl_filter(function(v) + return v:match('^' .. vim.pesc(arg_lead)) + end, completion_items) + end + end else - local prompt = default and string.format('%s [%s]:', title, default) or title .. ': ' - response = vim.trim(vim.fn.input({ - prompt = prompt, - cancelreturn = default or '', - })) + input.prompt = default and string.format('%s [%s]:', title, default) or title .. ': ' end - if #response == 0 and default then - response = default - end - content = content:gsub(vim.pesc(exp), response) + table.insert(prepared_inputs, input) end - return content + + if #prepared_inputs == 0 then + return Promise.resolve(content) + end + + return Promise.mapSeries(function(prepared_input) + return Input.open(prepared_input.prompt, '', prepared_input.completion and prepared_input.completion() or nil) + :next(function(response) + if not response or #response == 0 then + response = prepared_input.fallback_value + end + content = content:gsub(vim.pesc(prepared_input.exp), response) + end) + end, prepared_inputs):next(function() + return content + end) end function Template:_compile_expressions(content) diff --git a/lua/orgmode/clock/init.lua b/lua/orgmode/clock/init.lua index 450eeaa18..339345d96 100644 --- a/lua/orgmode/clock/init.lua +++ b/lua/orgmode/clock/init.lua @@ -1,6 +1,7 @@ local Duration = require('orgmode.objects.duration') local utils = require('orgmode.utils') local Promise = require('orgmode.utils.promise') +local Input = require('orgmode.ui.input') ---@class OrgClock ---@field files OrgFiles @@ -99,12 +100,17 @@ function Clock:org_set_effort() local item = self.files:get_closest_headline() -- TODO: Add Effort_ALL property as autocompletion local current_effort = item:get_property('Effort') - local effort = utils.input('Effort: ', current_effort or '') - local duration = Duration.parse(effort) - if duration == nil then - return utils.echo_error('Invalid duration format: ' .. effort) - end - item:set_property('Effort', effort) + return Input.open('Effort: ', current_effort or ''):next(function(effort) + if not effort then + return false + end + local duration = Duration.parse(effort) + if duration == nil then + return utils.echo_error('Invalid duration format: ' .. effort) + end + item:set_property('Effort', effort) + return item + end) end function Clock:get_statusline() diff --git a/lua/orgmode/config/_meta.lua b/lua/orgmode/config/_meta.lua index 09b51df9e..fdfe59b1e 100644 --- a/lua/orgmode/config/_meta.lua +++ b/lua/orgmode/config/_meta.lua @@ -169,6 +169,7 @@ ---@class OrgUiConfig ---@field folds? { colored: boolean } Should folds be colored or use the default folding highlight. Default: { colored: true } ---@field menu? { handler: fun() | nil } Menu configuration +---@field input? { use_vim_ui: boolean } Input configuration ---@class OrgMappingsConfig ---@field disable_all? boolean Disable all mappings. Default: false diff --git a/lua/orgmode/config/defaults.lua b/lua/orgmode/config/defaults.lua index 383f57f3a..60a28c632 100644 --- a/lua/orgmode/config/defaults.lua +++ b/lua/orgmode/config/defaults.lua @@ -213,6 +213,9 @@ local DefaultConfig = { menu = { handler = nil, }, + input = { + use_vim_ui = false, + }, }, } diff --git a/lua/orgmode/org/hyperlinks/init.lua b/lua/orgmode/org/hyperlinks/init.lua index 54d69601d..de509b245 100644 --- a/lua/orgmode/org/hyperlinks/init.lua +++ b/lua/orgmode/org/hyperlinks/init.lua @@ -223,49 +223,7 @@ function Hyperlinks.get_link_under_cursor() end function Hyperlinks.insert_link(link_location) - local selected_link = Link:new(link_location) - local desc = selected_link.url:get_target_value() - - if selected_link.url:is_id() then - link_location = ('id:%s'):format(selected_link.url:get_id()) - end - - local link_description = vim.trim(utils.input('Description: ', desc or '')) - - link_location = '[' .. vim.trim(link_location) .. ']' - - if link_description ~= '' then - link_description = '[' .. link_description .. ']' - end - - local insert_from - local insert_to - local target_col = #link_location + #link_description + 2 - - -- check if currently on link - local link, position = Hyperlinks.get_link_under_cursor() - if link and position then - insert_from = position.from - 1 - insert_to = position.to + 1 - target_col = target_col + position.from - else - local colnr = vim.fn.col('.') - insert_from = colnr - insert_to = colnr + 1 - target_col = target_col + colnr - end - - local linenr = vim.fn.line('.') or 0 - local curr_line = vim.fn.getline(linenr) - local new_line = string.sub(curr_line, 0, insert_from) - .. '[' - .. link_location - .. link_description - .. ']' - .. string.sub(curr_line, insert_to, #curr_line) - - vim.fn.setline(linenr, new_line) - vim.fn.cursor(linenr, target_col) + return org.links:insert_link(link_location) end return Hyperlinks diff --git a/lua/orgmode/org/links/init.lua b/lua/orgmode/org/links/init.lua index d1f60af27..e25c2fc27 100644 --- a/lua/orgmode/org/links/init.lua +++ b/lua/orgmode/org/links/init.lua @@ -2,6 +2,7 @@ local config = require('orgmode.config') local utils = require('orgmode.utils') local OrgLinkUrl = require('orgmode.org.links.url') local OrgHyperlink = require('orgmode.org.links.hyperlink') +local Input = require('orgmode.ui.input') ---@class OrgLinks:OrgLinkType ---@field private files OrgFiles @@ -107,42 +108,46 @@ function OrgLinks:insert_link(link_location, desc) link_location = ('id:%s'):format(selected_link.url:get_path()) end - local link_description = vim.trim(utils.input('Description: ', desc or '')) - - link_location = '[' .. vim.trim(link_location) .. ']' + return Input.open('Description: ', desc or ''):next(function(link_description) + if not link_description then + return false + end + link_location = '[' .. vim.trim(link_location) .. ']' - if link_description ~= '' then - link_description = '[' .. link_description .. ']' - end + if link_description ~= '' then + link_description = '[' .. link_description .. ']' + end - local insert_from - local insert_to - local target_col = #link_location + #link_description + 2 - - -- check if currently on link - local link, position = OrgHyperlink.at_cursor() - if link and position then - insert_from = position.from - 1 - insert_to = position.to + 1 - target_col = target_col + position.from - else - local colnr = vim.fn.col('.') - insert_from = colnr - insert_to = colnr + 1 - target_col = target_col + colnr - end + local insert_from + local insert_to + local target_col = #link_location + #link_description + 2 + + -- check if currently on link + local link, position = OrgHyperlink.at_cursor() + if link and position then + insert_from = position.from - 1 + insert_to = position.to + 1 + target_col = target_col + position.from + else + local colnr = vim.fn.col('.') + insert_from = colnr + insert_to = colnr + 1 + target_col = target_col + colnr + end - local linenr = vim.fn.line('.') or 0 - local curr_line = vim.fn.getline(linenr) - local new_line = string.sub(curr_line, 0, insert_from) - .. '[' - .. link_location - .. link_description - .. ']' - .. string.sub(curr_line, insert_to, #curr_line) - - vim.fn.setline(linenr, new_line) - vim.fn.cursor(linenr, target_col) + local linenr = vim.fn.line('.') or 0 + local curr_line = vim.fn.getline(linenr) + local new_line = string.sub(curr_line, 0, insert_from) + .. '[' + .. link_location + .. link_description + .. ']' + .. string.sub(curr_line, insert_to, #curr_line) + + vim.fn.setline(linenr, new_line) + vim.fn.cursor(linenr, target_col) + return true + end) end ---@param link_type OrgLinkType diff --git a/lua/orgmode/org/mappings.lua b/lua/orgmode/org/mappings.lua index 82a2bdaed..f3905c103 100644 --- a/lua/orgmode/org/mappings.lua +++ b/lua/orgmode/org/mappings.lua @@ -13,6 +13,8 @@ local Table = require('orgmode.files.elements.table') local EventManager = require('orgmode.events') local events = EventManager.event local Babel = require('orgmode.babel') +local Promise = require('orgmode.utils.promise') +local Input = require('orgmode.ui.input') ---@class OrgMappings ---@field capture OrgCapture @@ -46,15 +48,26 @@ function OrgMappings:set_tags(tags) local headline_tags = headline:get_own_tags() local current_tags = utils.tags_to_string(headline_tags) - if not tags then - tags = utils.input('Tags: ', current_tags, function(arg_lead) - return utils.prompt_autocomplete(arg_lead, self.files:get_tags()) + return Promise.resolve() + :next(function() + if not tags then + return Input.open('Tags: ', current_tags, function(arg_lead) + return utils.prompt_autocomplete(arg_lead, self.files:get_tags()) + end) + end + if type(tags) == 'table' then + tags = string.format(':%s:', table.concat(tags, ':')) + end + + return tags end) - elseif type(tags) == 'table' then - tags = string.format(':%s:', table.concat(tags, ':')) - end + :next(function(new_tags) + if not new_tags then + return + end - return headline:set_tags(tags) + return headline:set_tags(new_tags) + end) end function OrgMappings:toggle_archive_tag() @@ -791,15 +804,20 @@ end -- currently on function OrgMappings:insert_link() local link = OrgHyperlink.at_cursor() - local link_location = utils.input('Links: ', link and link.url:to_string() or '', function(arg_lead) + return Input.open('Links: ', link and link.url:to_string() or '', function(arg_lead) return self.links:autocomplete(arg_lead) - end) - if vim.trim(link_location) == '' then - utils.echo_warning('No Link selected') - return - end + end):next(function(link_location) + if not link_location then + return false + end - self.links:insert_link(link_location, link and link.desc) + if vim.trim(link_location) == '' then + utils.echo_warning('No Link selected') + return false + end + + return self.links:insert_link(link_location, link and link.desc) + end) end function OrgMappings:store_link() diff --git a/lua/orgmode/ui/input.lua b/lua/orgmode/ui/input.lua new file mode 100644 index 000000000..aef15a927 --- /dev/null +++ b/lua/orgmode/ui/input.lua @@ -0,0 +1,43 @@ +local config = require('orgmode.config') +local Promise = require('orgmode.utils.promise') +local utils = require('orgmode.utils') + +---@class OrgInput +local OrgInput = {} + +---@param prompt string +---@param default? string +---@param completion? fun(arg_lead: string): string[] +---@return OrgPromise +function OrgInput.open(prompt, default, completion) + _G.orgmode.__input_completion = completion + local opts = { + prompt = prompt, + default = default or '', + } + if completion then + opts.completion = 'customlist,v:lua.orgmode.__input_completion' + end + + return Promise.new(function(resolve, reject) + if config.ui.input.use_vim_ui then + return vim.ui.input(opts, function(value) + if value == nil then + return reject('Canceled') + end + return resolve(value) + end) + end + + opts.cancelreturn = vim.NIL + local value = vim.fn.input(opts) + if value == vim.NIL then + return reject('Canceled') + end + return resolve(value) + end):catch(function() + utils.echo_error('Input canceled') + end) +end + +return OrgInput diff --git a/lua/orgmode/utils/init.lua b/lua/orgmode/utils/init.lua index d8525db18..727eceacd 100644 --- a/lua/orgmode/utils/init.lua +++ b/lua/orgmode/utils/init.lua @@ -616,17 +616,6 @@ function utils.sorted_pairs(t) end end ----@param prompt string ----@param default? any ----@param completion_fn? fun(arg_lead: string): string[] -function utils.input(prompt, default, completion_fn) - local result = vim.fn.OrgmodeInput(prompt, default, completion_fn) - if result == vim.NIL then - error('Canceled.', 0) - end - return result -end - ---@param headline OrgHeadline function utils.goto_headline(headline) local current_file_path = utils.current_file_path() diff --git a/lua/orgmode/utils/promise.lua b/lua/orgmode/utils/promise.lua index 75adedc97..be08fcf38 100644 --- a/lua/orgmode/utils/promise.lua +++ b/lua/orgmode/utils/promise.lua @@ -380,6 +380,14 @@ function Promise.map(callback, list, concurrency) end) end +--- Equivalents to JavaScript's Promise.mapSeries +--- @param callback fun(value: any, index: number): any +--- @param list any[]: promise or non-promise values +--- @return OrgPromise +function Promise.mapSeries(callback, list) + return Promise.map(callback, list, 1) +end + --- Equivalents to JavaScript's Promise.race. --- @param list any[]: promise or non-promise values --- @return OrgPromise diff --git a/tests/plenary/api/api_spec.lua b/tests/plenary/api/api_spec.lua index 38667c57d..16636d31e 100644 --- a/tests/plenary/api/api_spec.lua +++ b/tests/plenary/api/api_spec.lua @@ -442,10 +442,12 @@ describe('Api', function() '* TODO Some task', }) - api.refile({ - source = api.current().headlines[2], - destination = api.load(destination_file.filename), - }) + api + .refile({ + source = api.current().headlines[2], + destination = api.load(destination_file.filename), + }) + :wait() assert.are.same(vim.api.nvim_buf_get_name(0), source_file.filename) vim.cmd('e' .. destination_file.filename) @@ -480,10 +482,12 @@ describe('Api', function() assert.are.same(vim.api.nvim_buf_get_name(0), source_file.filename) - api.refile({ - source = api.current().headlines[2], - destination = api.load(destination_file.filename).headlines[2], - }) + api + .refile({ + source = api.current().headlines[2], + destination = api.load(destination_file.filename).headlines[2], + }) + :wait() vim.cmd('e' .. destination_file.filename) @@ -517,10 +521,12 @@ describe('Api', function() ' DEADLINE: <2021-07-21 Wed 22:02>', }) - api.refile({ - source = api.current().headlines[1], - destination = api.load(destination_file.filename), - }) + api + .refile({ + source = api.current().headlines[1], + destination = api.load(destination_file.filename), + }) + :wait() assert.are.Not.same(vim.api.nvim_buf_get_name(0), source_file) @@ -554,10 +560,12 @@ describe('Api', function() ' DEADLINE: <2021-07-21 Wed 22:02>', }) - api.refile({ - source = api.current().headlines[1], - destination = api.load(destination_file.filename).headlines[2], - }) + api + .refile({ + source = api.current().headlines[1], + destination = api.load(destination_file.filename).headlines[2], + }) + :wait() assert.are.Not.same(vim.api.nvim_buf_get_name(0), source_file) diff --git a/tests/plenary/capture/capture_spec.lua b/tests/plenary/capture/capture_spec.lua index 65046cfdb..0a2f4467f 100644 --- a/tests/plenary/capture/capture_spec.lua +++ b/tests/plenary/capture/capture_spec.lua @@ -140,10 +140,12 @@ describe('Refile', function() local source_headline = capture_file:get_headlines()[2] ---@diagnostic disable-next-line: invisible - org.capture:_refile_from_org_file({ - source_headline = source_headline, - destination_file = destination_file, - }) + org.capture + :_refile_from_org_file({ + source_headline = source_headline, + destination_file = destination_file, + }) + :wait() vim.cmd('edit' .. vim.fn.fnameescape(destination_file.filename)) assert.are.same({ '* foo', @@ -169,10 +171,12 @@ describe('Refile', function() local item = capture_file:get_headlines()[2] ---@diagnostic disable-next-line: invisible - org.capture:_refile_from_org_file({ - destination_file = destination_file, - source_headline = item, - }) + org.capture + :_refile_from_org_file({ + destination_file = destination_file, + source_headline = item, + }) + :wait() vim.cmd('edit' .. vim.fn.fnameescape(destination_file.filename)) assert.are.same({ '* foobar', @@ -197,11 +201,13 @@ describe('Refile', function() local item = capture_file:get_headlines()[1] ---@diagnostic disable-next-line: invisible - org.capture:_refile_from_org_file({ - destination_file = destination_file, - source_headline = item, - destination_headline = destination_file:get_headlines()[1], - }) + org.capture + :_refile_from_org_file({ + destination_file = destination_file, + source_headline = item, + destination_headline = destination_file:get_headlines()[1], + }) + :wait() vim.cmd('edit' .. vim.fn.fnameescape(destination_file.filename)) assert.are.same({ '* foobar',