@@ -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+ -- Normalize input to always be 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+ -- Track used shortcuts to avoid conflicts
140+ if todo_keyword .has_fast_access then
141+ used_shortcuts [todo_keyword .shortcut ] = true
142+ elseif not used_shortcuts [todo_keyword .shortcut ] and # self .org_todo_keywords > 1 then
143+ -- Auto-assign shortcuts when we have multiple sequences
144+ todo_keyword .has_fast_access = true
145+ used_shortcuts [todo_keyword .shortcut ] = true
146+ end
147+
148+ todo_keyword .hl = self :_get_hl (todo_keyword .value , status_type )
149+ return todo_keyword
150+ end
151+
152+ --- @private
153+ --- @param keywords string[]
154+ --- @param seq_idx number
155+ --- @param used_shortcuts table<string , boolean>
156+ --- @param keyword_offset number
157+ --- @return OrgTodoKeyword[] keywords for the sequence
158+ --- @return OrgTodoKeyword[] seq_keywords keywords in this sequence
159+ function TodoKeywords :_parse_sequence (keywords , seq_idx , used_shortcuts , keyword_offset )
160+ keyword_offset = keyword_offset or 0
161+ local todo , done = self :_split_todo_and_done (keywords )
83162 local list = {}
163+ local seq_keywords = {}
164+
84165 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' )
166+ local todo_keyword = self :_create_keyword (keyword , ' TODO' , keyword_offset + i , seq_idx , used_shortcuts )
91167 table.insert (list , todo_keyword )
168+ table.insert (seq_keywords , todo_keyword )
92169 end
93170
94171 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' )
172+ local todo_keyword = self :_create_keyword (keyword , ' DONE' , keyword_offset + # todo + i , seq_idx , used_shortcuts )
101173 table.insert (list , todo_keyword )
174+ table.insert (seq_keywords , todo_keyword )
102175 end
103176
104- self . todo_keywords = list
177+ return list , seq_keywords
105178end
106179
107180--- @private
@@ -116,9 +189,9 @@ function TodoKeywords:_get_hl(keyword, type)
116189end
117190
118191--- @private
192+ --- @param keywords string[]
119193--- @return string[] , string[]
120- function TodoKeywords :_split_todo_and_done ()
121- local keywords = self .org_todo_keywords
194+ function TodoKeywords :_split_todo_and_done (keywords )
122195 local has_separator = vim .tbl_contains (keywords , ' |' )
123196 if not has_separator then
124197 return { unpack (keywords , 1 , # keywords - 1 ) }, { keywords [# keywords ] }
0 commit comments