Skip to content
New issue

Have a question about this project? # for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “#”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? # to your account

feat: support updating a headline's cookie from multiple lists #925

Merged
merged 2 commits into from
Mar 12, 2025
Merged
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion lua/orgmode/files/elements/listitem.lua
Original file line number Diff line number Diff line change
@@ -71,7 +71,7 @@ function Listitem:update_checkbox(action)
else
local parent_headline = self.file:get_closest_headline_or_nil()
if parent_headline then
parent_headline:update_cookie(parent_list)
parent_headline:update_cookie()
end
end
end
30 changes: 23 additions & 7 deletions lua/orgmode/files/headline.lua
Original file line number Diff line number Diff line change
@@ -890,22 +890,38 @@ function Headline:get_cookie()
return self:_parse_title_part('%[%d?%d?%d?%%%]')
end

function Headline:update_cookie(list_node)
local total_boxes = self:child_checkboxes(list_node)
local checked_boxes = vim.tbl_filter(function(box)
return box:match('%[%w%]')
end, total_boxes)
function Headline:update_cookie()
local section = self:node():parent()
if not section then
return self
end

-- Go through all the lists in this headline and gather checked_boxes
local num_boxes, num_checked_boxes = 0, 0
local body = section:field('body')[1]
for node in body:iter_children() do
if node:type() == 'list' then
local boxes = self:child_checkboxes(node)
num_boxes = num_boxes + #boxes
local checked_boxes = vim.tbl_filter(function(box)
return box:match('%[%w%]')
end, boxes)
num_checked_boxes = num_checked_boxes + #checked_boxes
end
end

-- Update the cookie
local cookie = self:get_cookie()
if cookie then
local new_cookie_val
if self.file:get_node_text(cookie):find('%%') then
new_cookie_val = ('[%d%%]'):format((#checked_boxes / #total_boxes) * 100)
new_cookie_val = ('[%d%%]'):format((num_checked_boxes / num_boxes) * 100)
else
new_cookie_val = ('[%d/%d]'):format(#checked_boxes, #total_boxes)
new_cookie_val = ('[%d/%d]'):format(num_checked_boxes, num_boxes)
end
return self:_set_node_text(cookie, new_cookie_val)
end
return self
end

function Headline:child_checkboxes(list_node)
21 changes: 21 additions & 0 deletions tests/plenary/ui/mappings/checkbox_spec.lua
Original file line number Diff line number Diff line change
@@ -151,4 +151,25 @@ describe('Checkbox mappings', function()
'- [ ] checkbox item',
}, vim.api.nvim_buf_get_lines(0, 0, 6, false))
end)

it('should update headline cookies from multiple lists', function()
helpers.create_file({
'* Test orgmode [/]',
'First List',
'- [ ] checkbox item',
'- [ ] checkbox item',
'Second List',
'- [ ] checkbox item',
})
vim.fn.cursor(4, 1)
vim.cmd([[exe "norm \<C-space>"]])
assert.are.same({
'* Test orgmode [1/3]',
'First List',
'- [ ] checkbox item',
'- [X] checkbox item',
'Second List',
'- [ ] checkbox item',
}, vim.api.nvim_buf_get_lines(0, 0, 6, false))
end)
end)