@@ -2,18 +2,26 @@ local utils = require('orgmode.utils')
2
2
local TodoKeyword = require (' orgmode.objects.todo_keywords.todo_keyword' )
3
3
4
4
--- @class OrgTodoKeywords
5
- --- @field org_todo_keywords string[]
5
+ --- @field org_todo_keywords string[][] | string[]
6
6
--- @field org_todo_keyword_faces table<string , string>
7
7
--- @field todo_keywords OrgTodoKeyword[]
8
+ --- @field sequences OrgTodoKeyword[][] Array of todo keyword sequences
8
9
local TodoKeywords = {}
9
10
TodoKeywords .__index = TodoKeywords
10
11
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> }
12
13
--- @return OrgTodoKeywords
13
14
function 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
+
14
21
local this = setmetatable ({
15
- org_todo_keywords = opts . org_todo_keywords ,
22
+ org_todo_keywords = normalized_keywords ,
16
23
org_todo_keyword_faces = opts .org_todo_keyword_faces ,
24
+ sequences = {},
17
25
}, self )
18
26
this :_parse ()
19
27
return this
@@ -44,6 +52,19 @@ function TodoKeywords:find(keyword)
44
52
end )
45
53
end
46
54
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
+
47
68
--- @param type OrgTodoKeywordType
48
69
--- @return OrgTodoKeyword
49
70
function TodoKeywords :first_by_type (type )
@@ -60,6 +81,12 @@ function TodoKeywords:all()
60
81
return self .todo_keywords
61
82
end
62
83
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
+
63
90
--- @return OrgTodoKeyword
64
91
function TodoKeywords :first ()
65
92
return self .todo_keywords [1 ]
79
106
80
107
--- @private
81
108
function 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 )
83
162
local list = {}
163
+ local seq_keywords = {}
164
+
84
165
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 )
91
167
table.insert (list , todo_keyword )
168
+ table.insert (seq_keywords , todo_keyword )
92
169
end
93
170
94
171
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 )
101
173
table.insert (list , todo_keyword )
174
+ table.insert (seq_keywords , todo_keyword )
102
175
end
103
176
104
- self . todo_keywords = list
177
+ return list , seq_keywords
105
178
end
106
179
107
180
--- @private
@@ -116,9 +189,9 @@ function TodoKeywords:_get_hl(keyword, type)
116
189
end
117
190
118
191
--- @private
192
+ --- @param keywords string[]
119
193
--- @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 )
122
195
local has_separator = vim .tbl_contains (keywords , ' |' )
123
196
if not has_separator then
124
197
return { unpack (keywords , 1 , # keywords - 1 ) }, { keywords [# keywords ] }
0 commit comments