@@ -2,18 +2,26 @@ local utils = require('orgmode.utils')
22local TodoKeyword = require (' orgmode.objects.todo_keywords.todo_keyword' )
33
44--- @class OrgTodoKeywords
5- --- @field org_todo_keywords string[]
5+ --- @field org_todo_keywords string[][] | string[]
66--- @field org_todo_keyword_faces table<string , string>
77--- @field todo_keywords OrgTodoKeyword[]
8+ --- @field sequences OrgTodoKeyword[][] Array of todo keyword sequences
89local TodoKeywords = {}
910TodoKeywords .__index = TodoKeywords
1011
11- --- @param opts { org_todo_keywords : string[] , org_todo_keyword_faces : table<string , string> }
12+ --- @param opts { org_todo_keywords : string[][] | string[] , org_todo_keyword_faces : table<string , string> }
1213--- @return OrgTodoKeywords
1314function TodoKeywords :new (opts )
15+ -- Support both single sequence (string[]) and multiple sequences (string[][])
16+ local normalized_keywords = opts .org_todo_keywords
17+ if type (normalized_keywords [1 ]) ~= ' table' then
18+ normalized_keywords = { normalized_keywords }
19+ end
20+
1421 local this = setmetatable ({
15- org_todo_keywords = opts . org_todo_keywords ,
22+ org_todo_keywords = normalized_keywords ,
1623 org_todo_keyword_faces = opts .org_todo_keyword_faces ,
24+ sequences = {},
1725 }, self )
1826 this :_parse ()
1927 return this
@@ -44,6 +52,19 @@ function TodoKeywords:find(keyword)
4452 end )
4553end
4654
55+ --- @param keyword string
56+ --- @return number | nil sequence index this keyword belongs to
57+ function TodoKeywords :find_sequence_index (keyword )
58+ for seq_idx , seq in ipairs (self .sequences ) do
59+ for _ , todo_keyword in ipairs (seq ) do
60+ if todo_keyword .value == keyword then
61+ return seq_idx
62+ end
63+ end
64+ end
65+ return nil
66+ end
67+
4768--- @param type OrgTodoKeywordType
4869--- @return OrgTodoKeyword
4970function TodoKeywords :first_by_type (type )
@@ -60,6 +81,12 @@ function TodoKeywords:all()
6081 return self .todo_keywords
6182end
6283
84+ --- @param sequence_idx ? number
85+ --- @return OrgTodoKeyword[]
86+ function TodoKeywords :sequence (sequence_idx )
87+ return self .sequences [sequence_idx or 1 ] or {}
88+ end
89+
6390--- @return OrgTodoKeyword
6491function TodoKeywords :first ()
6592 return self .todo_keywords [1 ]
79106
80107--- @private
81108function TodoKeywords :_parse ()
82- local todo , done = self :_split_todo_and_done ()
109+ self .todo_keywords = {}
110+ self .sequences = {}
111+ local used_shortcuts = {}
112+
113+ for seq_idx , sequence in ipairs (self .org_todo_keywords ) do
114+ local keyword_offset = # self .todo_keywords
115+ local keywords , seq_keywords = self :_parse_sequence (sequence , seq_idx , used_shortcuts , keyword_offset )
116+
117+ for _ , keyword in ipairs (keywords ) do
118+ table.insert (self .todo_keywords , keyword )
119+ end
120+ table.insert (self .sequences , seq_keywords )
121+ end
122+ end
123+
124+ --- @private
125+ --- @param keyword string
126+ --- @param status_type string ' TODO' or ' DONE'
127+ --- @param index number
128+ --- @param seq_idx number
129+ --- @param used_shortcuts table<string , boolean>
130+ --- @return OrgTodoKeyword
131+ function TodoKeywords :_create_keyword (keyword , status_type , index , seq_idx , used_shortcuts )
132+ local todo_keyword = TodoKeyword :new ({
133+ type = status_type ,
134+ keyword = keyword ,
135+ index = index ,
136+ sequence_index = seq_idx ,
137+ })
138+
139+ if todo_keyword .has_fast_access then
140+ used_shortcuts [todo_keyword .shortcut ] = true
141+ elseif not used_shortcuts [todo_keyword .shortcut ] and # self .org_todo_keywords > 1 then
142+ -- Enable fast access for all keywords when multiple sequences exist
143+ todo_keyword .has_fast_access = true
144+ used_shortcuts [todo_keyword .shortcut ] = true
145+ end
146+
147+ todo_keyword .hl = self :_get_hl (todo_keyword .value , status_type )
148+ return todo_keyword
149+ end
150+
151+ --- @private
152+ --- @param keywords string[]
153+ --- @param seq_idx number
154+ --- @param used_shortcuts table<string , boolean>
155+ --- @param keyword_offset number
156+ --- @return OrgTodoKeyword[] keywords for the sequence
157+ --- @return OrgTodoKeyword[] seq_keywords keywords in this sequence
158+ function TodoKeywords :_parse_sequence (keywords , seq_idx , used_shortcuts , keyword_offset )
159+ keyword_offset = keyword_offset or 0
160+ local todo , done = self :_split_todo_and_done (keywords )
83161 local list = {}
162+ local seq_keywords = {}
163+
84164 for i , keyword in ipairs (todo ) do
85- local todo_keyword = TodoKeyword :new ({
86- type = ' TODO' ,
87- keyword = keyword ,
88- index = i ,
89- })
90- todo_keyword .hl = self :_get_hl (todo_keyword .value , ' TODO' )
165+ local todo_keyword = self :_create_keyword (keyword , ' TODO' , keyword_offset + i , seq_idx , used_shortcuts )
91166 table.insert (list , todo_keyword )
167+ table.insert (seq_keywords , todo_keyword )
92168 end
93169
94170 for i , keyword in ipairs (done ) do
95- local todo_keyword = TodoKeyword :new ({
96- type = ' DONE' ,
97- keyword = keyword ,
98- index = # todo + i ,
99- })
100- todo_keyword .hl = self :_get_hl (todo_keyword .value , ' DONE' )
171+ local todo_keyword = self :_create_keyword (keyword , ' DONE' , keyword_offset + # todo + i , seq_idx , used_shortcuts )
101172 table.insert (list , todo_keyword )
173+ table.insert (seq_keywords , todo_keyword )
102174 end
103175
104- self . todo_keywords = list
176+ return list , seq_keywords
105177end
106178
107179--- @private
@@ -116,9 +188,9 @@ function TodoKeywords:_get_hl(keyword, type)
116188end
117189
118190--- @private
191+ --- @param keywords string[]
119192--- @return string[] , string[]
120- function TodoKeywords :_split_todo_and_done ()
121- local keywords = self .org_todo_keywords
193+ function TodoKeywords :_split_todo_and_done (keywords )
122194 local has_separator = vim .tbl_contains (keywords , ' |' )
123195 if not has_separator then
124196 return { unpack (keywords , 1 , # keywords - 1 ) }, { keywords [# keywords ] }
0 commit comments