-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathnotr.py
1067 lines (858 loc) · 37.5 KB
/
notr.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
import os
import sys
import re
import glob
import random
import json
import pathlib
import time
import dataclasses
import sublime
import sublime_plugin
try:
from . import sbot_common as sc # normal import
except:
import sbot_common as sc # unittest import
# FUTURE embed pics, other docs, ...
# FUTURE keep a backup or track changes?
# Known file types.
IMAGE_TYPES = ['.jpg', '.jpeg', '.png', '.bmp', '.gif']
#--------------------------- Types -------------------------------------------------
# One target: section or file/uri.
@dataclasses.dataclass(order=True)
class Target:
sort_index: str = dataclasses.field(init=False)
name: str # section title or description
ttype: str # 'section', 'uri', 'image', 'path' - discriminate file and dir?
category: str # 'sticky', 'mru', 'none'
level: int # for section only
tags: list # tags for targets
resource: str # what ttype points to
file: str # .ntr file path
line: int # .ntr file line
def __post_init__(self):
self.sort_index = self.name
# A reference to a Target.
@dataclasses.dataclass(order=True)
class Ref:
sort_index: str = dataclasses.field(init=False)
name: str # "target#name"
file: str # .ntr file path
line: int # .ntr file line
def __post_init__(self):
self.sort_index = self.name
#---------------------------- Data -----------------------------------------------
# Current project file (*.nproj) contents.
_current_project = None
# The whole store file (Notr.store) contents. None means uninitialized.
_store = None
# Persisted mru.
_current_mru = []
# All Targets found in all ntr files. They are ordered as:
# for dir in project.notr_paths
# for notr_file in dir
# for section in notr_file
# _targets.add(section)
_targets = []
# All Refs found in all ntr files.
_refs = []
# Parse errors to report to user. Tuples of (path, line, msg)
_parse_errors = []
#-----------------------------------------------------------------------------------
def plugin_loaded():
'''Called per plugin instance.'''
sc.debug(f'plugin_loaded() {__package__}')
#-----------------------------------------------------------------------------------
def plugin_unloaded():
'''Ditto.'''
pass
#-----------------------------------------------------------------------------------
#-------------------------- EventListener ------------------------------------------
#-----------------------------------------------------------------------------------
#-----------------------------------------------------------------------------------
class NotrEvent(sublime_plugin.EventListener):
''' Process view events. '''
def on_init(self, views):
''' First thing that happens when plugin/window created. Initialize everything. '''
global _store
fn = sc.get_settings_fn()
settings = sublime.load_settings(sc.get_settings_fn())
random.seed()
# Check user project files.
valid_projects = []
project_files = settings.get('project_files')
if project_files is not None:
for p in project_files: # pyright: ignore
sp = sc.expand_vars(p)
if sp is not None and os.path.isfile(sp):
valid_projects.append(sc.expand_vars(p))
else:
sc.error(f'Invalid project file in your settings: {p}')
# Get persisted store info.
_store = None
store_fn = sc.get_store_fn()
if os.path.isfile(store_fn):
try:
with open(store_fn, 'r') as fp:
_store = json.load(fp)
except Exception as e:
sc.error(f'Error processing {store_fn}: {e}', e.__traceback__)
if _store is None: # Assume new file with default fields.
_store = {}
for p in valid_projects:
_store[sc.expand_vars(p)] = {'active': {len(_store) == 0}, 'mru': []}
# Remove dead/invalid projects. Determine project file. Ensure one only active.
dead = []
project_fn = None
for path, v in _store.items():
if not os.path.isfile(path) or path not in valid_projects:
dead.append(path)
elif not project_fn and v['active'] is True:
project_fn = path
else:
v['active'] = False
for p in dead:
del _store[p]
if project_fn is None and len(valid_projects) > 0:
project_fn = valid_projects[0]
if project_fn is not None:
_open_project(project_fn)
_process_notr_files(views[0].window())
# Views are all valid now so init them.
for view in views:
self._init_fixed_hl(view)
else:
sc.error(f'No valid project files in your settings - please edit')
def on_load(self, view):
''' Loaded a new file. '''
# View is valid so init it.
self._init_fixed_hl(view)
def on_pre_close(self, view):
''' Save anything. '''
del view
_write_store()
def on_post_save(self, view):
''' Called after a view has been saved.
If it is a notr file, reload all ntr files. Seems a bit brute force, how else?
If it is a notr project, reload the project.'''
if view.syntax().name == 'Notr':
_process_notr_files(view.window())
elif _current_project is not None and view.file_name() == _current_project['_fn']:
_open_project(view.file_name())
_process_notr_files(view.window)
def _init_fixed_hl(self, view):
''' Add any highlights. '''
if (_current_project is not None and
view.is_scratch() is False and
view.file_name() is not None and
view.syntax() is not None and
view.syntax().name == 'Notr'):
settings = sublime.load_settings(sc.get_settings_fn())
whole_word = settings.get('fixed_hl_whole_word')
fixed_hl = _current_project['fixed_hl']
if fixed_hl is None:
return
hl_info = sc.get_highlight_info('fixed')
for hl_index in range(len(fixed_hl)):
hl = hl_info[hl_index]
# Clean first.
view.erase_regions(hl.region_name)
# New ones.
hl_regions = []
# Colorize one token.
for token in fixed_hl[hl_index]:
escaped = re.escape(token)
if whole_word: # and escaped[0].isalnum():
escaped = r'\b%s\b' % escaped
regs = view.find_all(escaped) if whole_word else view.find_all(token, sublime.LITERAL)
# print(escaped, len(regs))
hl_regions.extend(regs)
if len(hl_regions) > 0:
view.add_regions(key=hl.region_name, regions=hl_regions, scope=hl.scope_name,
flags=sublime.RegionFlags.DRAW_STIPPLED_UNDERLINE)
#-----------------------------------------------------------------------------------
#-------------------------- WindowCommands -----------------------------------------
#-----------------------------------------------------------------------------------
#-----------------------------------------------------------------------------------
class NotrOpenProjectCommand(sublime_plugin.WindowCommand):
''' User selects an existing project. '''
panel_items = []
def run(self):
''' Populate selector with user project files. '''
self.panel_items.clear()
if _store is not None and len(_store) > 0:
for path, _ in _store.items():
projfn = pathlib.Path(path).stem
self.panel_items.append(sublime.QuickPanelItem(trigger=projfn, details=path, kind=sublime.KIND_AMBIGUOUS))
self.window.show_quick_panel(self.panel_items, on_select=self.on_sel_project)
def on_sel_project(self, *args, **kwargs):
del kwargs
if len(args) > 0 and args[0] >= 0:
_open_project(self.panel_items[args[0]].details)
_process_notr_files(self.window)
def is_visible(self):
return True
#-----------------------------------------------------------------------------------
class NotrEditProjectCommand(sublime_plugin.WindowCommand):
''' Open the file in a new view for editing.'''
def run(self):
# Open the file in a new view.
vnew = None
if _current_project is not None:
fn = _current_project['_fn']
try:
vnew = self.window.open_file(fn)
vnew.assign_syntax('Packages/JSON/JSON.sublime-syntax')
except Exception as e:
sc.error(f'Failed to open {fn}: {e}', e.__traceback__)
vnew = None
def is_visible(self):
return _store is not None and len(_store) > 0 and _current_project is not None
#-----------------------------------------------------------------------------------
class NotrReloadCommand(sublime_plugin.WindowCommand):
''' Reload after editing. '''
def run(self):
# Open and process notr files.
_process_notr_files(self.window)
def is_visible(self):
return _current_project is not None
#-----------------------------------------------------------------------------------
class NotrDumpCommand(sublime_plugin.WindowCommand):
''' Diagnostic. '''
def run(self):
text = []
def do_one(name, coll):
text.append(f'\n========== {name} ==========')
text.extend([str(x) for x in coll])
if len(_parse_errors) > 0:
text.append('\n========== !! errors below !! ==========')
do_one('targets', _targets)
do_one('refs', _refs)
do_one('tags', _get_all_tags())
do_one('ntr_files', _get_all_ntr_files())
if len(_parse_errors) > 0:
text.append('\n========== errors ==========')
text.extend([f'{p[0]}({p[1]}): {p[2]}' for p in _parse_errors])
sc.create_new_view(self.window, '\n'.join(text))
def is_visible(self):
return True
#-----------------------------------------------------------------------------------
class NotrFindInFilesCommand(sublime_plugin.WindowCommand):
''' Search in the .ntr files. '''
def run(self):
''' Assemble the search locations from users paths. Index directory also? '''
if _current_project is None:
return
paths = ["*.ntr", "-<open files>"] # always
notr_paths = _current_project['notr_paths']
if notr_paths is not None:
for npath in notr_paths: # pyright: ignore
expath = sc.expand_vars(npath)
if expath is not None and os.path.exists(expath):
paths.append(expath)
# Show it so the user can enter the pattern.
# https://github.com/SublimeText/PackageDev/blob/master/plugins/command_completions/builtin_commands_meta_data.yaml
self.window.run_command("show_panel", {
"panel": "find_in_files", "where": ', '.join(paths), "case_sensitive": True, "pattern": '',
"whole_word": False, "preserve_case": True, "show_context": False, "use_buffer": True,
"replace": '', "regex": False})
def is_visible(self):
return True
#-----------------------------------------------------------------------------------
class NotrPublishCommand(sublime_plugin.WindowCommand):
''' TODO A research project to publish the .ntr files somehow/somewhere. '''
def run(self):
onedrive_path = os.path.expandvars('$OneDrive')
docs_path = os.path.join(onedrive_path, 'OneDriveDocuments')
pub_path = os.path.join(onedrive_path, '_notr')
notes_path = os.path.join(docs_path, 'notes')
notes_to_pub = [
'activities.ntr',
'shop.ntr',
'todo.ntr',
]
# Gen the pub files.
for p in notes_to_pub:
src = os.path.join(notes_path, p)
dest = os.path.join(pub_path, p.replace('.ntr', '.html'))
sc.debug(f'{src}->{dest}')
# need to manage files in/out. Maybe use clipboard?
# self.window.run_command("sbot_render_to_html", {"group": 0})
time.sleep(0.5) # let st finish this first
def is_visible(self):
return False
#-----------------------------------------------------------------------------------
#-------------------------- TextCommands -------------------------------------------
#-----------------------------------------------------------------------------------
#-----------------------------------------------------------------------------------
class NotrGotoTargetCommand(sublime_plugin.TextCommand):
''' List all the tag(s) and/or target(s) for user selection then open corresponding file. '''
# Prepared lists for quick panel.
_tags = []
def run(self, edit, filter_by_tag):
del edit
# Determine if user has selected a specific ref or link to follow, otherwise show the list of all.
valid = True # default
tref = _get_selection_for_scope(self.view, 'markup.link.refname.notr')
tlink = _get_selection_for_scope(self.view, 'markup.underline.link.notr')
if tlink is None:
tlink = _get_selection_for_scope(self.view, 'markup.link.target.notr')
# Explicit ref to selected element - do immediate.
if tref is not None:
# Get the corresponding target spec.
for target in _targets:
valid = False
if target.name == tref:
if target.ttype == 'section':
# Open the notr file and position it.
sc.wait_load_file(self.view.window(), target.file, target.line)
valid = True
else: # 'image', 'uri', 'path'
valid = sc.open_path(target.resource)
break
if not valid:
sc.error(f'Invalid reference in {self.view.file_name()}: [{tref}]')
# Explicit link. do immediate.
elif tlink is not None:
valid = True
fn = sc.expand_vars(tlink)
if fn is None:
valid = False
else:
valid = sc.open_path(fn)
if not valid:
sc.error(f'Invalid link: [{tlink}]')
# Show a quickpanel of all target names.
else:
if filter_by_tag:
self._tags = _get_all_tags()
if len(self._tags) > 0:
panel_items = []
for tag in self._tags:
panel_items.append(sublime.QuickPanelItem(trigger=tag, kind=sublime.KIND_AMBIGUOUS))
win = self.view.window()
if win is not None:
win.show_quick_panel(panel_items, on_select=self.on_sel_tag)
else:
sublime.status_message('No tags in project')
else:
targets = _filter_order_targets(mru_first=True, current_file=self.view.file_name())
self.show_targets(targets)
def on_sel_tag(self, *args, **kwargs):
del kwargs
if len(args) > 0 and args[0] >= 0:
# Hide current quick panel.
win = self.view.window()
if win is not None:
win.run_command("hide_overlay")
# Make a selector with target names, current file's first.
sel_tag = self._tags[args[0]]
# Filter per tag selection.
tag_targets = _filter_order_targets(sort=False, mru_first=True, tags=[sel_tag])
if len(tag_targets) > 0:
self.show_targets(tag_targets)
else:
sublime.status_message('No targets with that tag')
def show_targets(self, targets):
'''Present target options to user.'''
self._targets_to_select = targets
panel_items = _build_selector(self._targets_to_select)
# for pi in panel_items:
# print(pi.trigger)
win = self.view.window()
if win is not None:
win.show_quick_panel(panel_items, on_select=self.on_sel_target)
def on_sel_target(self, *args, **kwargs):
del kwargs
if len(args) > 0 and args[0] >= 0:
# Locate the target record.
target = self._targets_to_select[args[0]]
_update_mru(target.name)
if target.ttype == 'section':
# Open the notr file and position it.
sc.wait_load_file(self.view.window(), target.file, target.line)
else: # 'image', 'uri', 'path'
sc.open_path(target.resource)
def is_visible(self):
return True
#-----------------------------------------------------------------------------------
class NotrGotoSectionCommand(sublime_plugin.TextCommand):
''' Go to next or previous section.'''
def run(self, edit, where):
del edit
next = where == 'next'
view = self.view
fn = view.file_name()
if fn is None:
return # --- early return
caret = sc.get_single_caret(view)
if caret is None:
return # -- early return
sel_row, _ = view.rowcol(caret) # current selected row
sel_line = sel_row + 1
# Create a list of all section line numbers in file.
# Should be cached but that gets into detecting file changes. Maybe later...
section_lines = []
for target in _targets:
if target.file == fn:
section_lines.append(target.line)
if len(section_lines) == 0:
sublime.message_dialog("No tracked sections.\nIs this file in the current notr project?")
return # --- early return
section_lines.sort()
if next: # Find the next section line.
new_line = section_lines[0] # default is first line
for sl in section_lines:
if sl > sel_line:
new_line = sl
break;
else:
# Find the previous section line.
new_line = section_lines[-1] # default is last line
for sl in reversed(section_lines):
if sl < sel_line:
new_line = sl
break;
view.run_command("goto_line", {"line": new_line})
def is_visible(self):
return _check_syntax(self.view)
#-----------------------------------------------------------------------------------
class NotrInsertHruleCommand(sublime_plugin.TextCommand):
''' Insert visuals. '''
def run(self, edit, fill_str, reps):
v = self.view
# Start of current line.
caret = sc.get_single_caret(v)
if caret is not None:
lst = v.line(caret)
s = fill_str * reps + '\n'
v.insert(edit, lst.a, s)
def is_visible(self):
return _check_syntax(self.view)
#-----------------------------------------------------------------------------------
class NotrInsertTargetFromClipCommand(sublime_plugin.TextCommand):
''' Insert target from clipboard. Assumes user clipped appropriate string. '''
def run(self, edit):
s = f'<give_me_a_ref_name>({sublime.get_clipboard()})'
caret = sc.get_single_caret(self.view)
if caret is not None:
self.view.insert(edit, caret, s)
def is_visible(self):
return _check_syntax(self.view) and sublime.get_clipboard() != ''
#-----------------------------------------------------------------------------------
class NotrInsertRefCommand(sublime_plugin.TextCommand):
''' Insert ref from list of known refs. '''
_targets_to_select = []
def run(self, edit):
# Show a quickpanel of all target names.
del edit
self._targets_to_select = _filter_order_targets(sort=False, mru_first=True, current_file=self.view.file_name())
panel_items = _build_selector(self._targets_to_select)
win = self.view.window()
if win is not None:
win.show_quick_panel(panel_items, on_select=self.on_sel_ref)
def on_sel_ref(self, *args, **kwargs):
del kwargs
if len(args) > 0 and args[0] >= 0:
s = f'<*{self._targets_to_select[args[0]].name}>'
self.view.run_command("insert", {"characters": s}) # Insert in view
def is_visible(self):
return _check_syntax(self.view)
#-----------------------------------------------------------------------------------
#-------------------------- Private Functions --------------------------------------
#-----------------------------------------------------------------------------------
#-----------------------------------------------------------------------------------
def _write_store():
''' Save everything. '''
store_fn = sc.get_store_fn()
with open(store_fn, 'w') as fp:
json.dump(_store, fp, indent=4)
#-----------------------------------------------------------------------------------
def _open_project(project_fn):
global _store, _current_project, _current_mru
try:
expfn = sc.expand_vars(project_fn)
if expfn is None:
sc.error(f'Invalid project file: {project_fn}')
return
if _store is None:
sc.error(f'Store not initialized.')
return
with open(expfn, 'r') as fp:
s = fp.read()
proj = json.loads(s)
# Check file integrity.
if "notr_paths" not in proj or "notr_index" not in proj:
# Broken file.
sc.error(f'Invalid notr project file {expfn} - needs to be edited')
return
# Non-fatal patchups.
if "fixed_hl" not in proj:
proj["fixed_hl"] = []
if 'sticky' not in proj:
proj['sticky'] = []
_current_project = proj
_current_project['_fn'] = expfn # for downstream
# Reset flags first.
for _, v in _store.items():
v['active'] = False
# Get dynamic stuff. Add if not included.
if expfn not in _store: # new, add
_store[expfn] = {'active': True, 'mru': []}
else:
_store[expfn]['active'] = True
_current_mru = _store[expfn]['mru']
s = f'Opened notr project file {project_fn}'
sc.info(s)
sublime.status_message(s)
# sublime.message_dialog(s)
except Exception as e:
# Assume bad project file.
sc.error(f'Error opening notr project file {project_fn}: {e}', e.__traceback__)
#-----------------------------------------------------------------------------------
def _process_notr_files(window):
''' Get all ntr files and grab their goodies. '''
_targets.clear()
_refs.clear()
_parse_errors.clear()
ntr_files = []
if _current_project is None:
return
# Open and process all notr files.
settings = sublime.load_settings(sc.get_settings_fn())
# Index first.
index_path = None
notr_index = _current_project['notr_index']
if notr_index is not None:
index_path = sc.expand_vars(notr_index)
if index_path is not None and os.path.exists(index_path):
ntr_files.append(index_path)
else:
_user_error(sc.get_settings_fn(), -1, f'Invalid path in project: {index_path}')
# Paths.
notr_paths = _current_project['notr_paths']
for npath in notr_paths:
expath = sc.expand_vars(npath)
if expath is not None and os.path.exists(expath):
for nfile in glob.glob(os.path.join(expath, '*.ntr')):
if index_path is None or not os.path.samefile(nfile, index_path): # don't do index twice
ntr_files.append(nfile)
else:
_user_error(sc.get_settings_fn(), -1, f'Invalid path in project: {npath}')
# Process the files. Targets are ordered by sections then files/uris.
sections = []
uris = []
for nfile in ntr_files:
fparts = _process_notr_file(nfile)
if fparts is not None:
sections.extend(fparts[0])
uris.extend(fparts[1])
_refs.extend(fparts[2])
_targets.extend(sections)
_targets.extend(uris)
# Check all user refs are valid.
valid_refs = []
for target in _targets:
if len(target.name) > 0:
valid_refs.append(target.name)
for ref in _refs:
if ref.name not in valid_refs:
_user_error(ref.file, ref.line, f'Invalid ref name:{ref.name}')
# Do output if errors.
if len(_parse_errors) > 0:
output_view = None
working_dir = ''
use_panel = True # prefs.get("show_panel_on_build", True)
# Create output to panel or view. Don't call get_output_panel until the regexes are assigned.
if use_panel:
output_view = window.create_output_panel("exec")
else:
output_view = sc.create_new_view(window, '')
# Enable result navigation.
settings = output_view.settings()
settings.set('result_file_regex', r'^([^\(]+)\(([0-9]+)\)(): (.*)$')
settings.set('result_base_dir', working_dir)
# Create a second time after assigning the above regex and settings.
if use_panel:
output_view = window.create_output_panel("exec")
window.run_command('show_panel', {'panel': 'output.exec'})
else:
window.focus_view(output_view)
# Fill with info.
output_view.run_command('append', {'characters': "Notr file errors:\n"})
for p in _parse_errors:
output_view.run_command('append', {'characters': f'{p[0]}({p[1]}): {p[2]}\n', 'force': True, 'scroll_to_end': True})
#-----------------------------------------------------------------------------------
def _process_notr_file(ntr_fn):
''' Process one notr file. Regex and process sections and links.
This collects the text and checks syntax. Validity will be checked when all files processed.
Returns (sections, links, refs)
'''
sections = []
links = []
refs = []
no_index = False
line_num = -1
try:
with open(ntr_fn, 'r', encoding='utf-8') as file: # need to explicitly set encoding because default windows is ascii
lines = file.read().splitlines()
line_num = 1
# Get the things of interest defined in the file. Must match the syntax.
re_directives = re.compile(r'^:(.*)')
re_links = re.compile(r'<([^>)]+)>\(([^\)]+)\)')
re_refs = re.compile(r'<\* *([^\>]+)>')
re_sections = re.compile(r'^(#+ +[^\[]+) *(?:\[(.*)\])?')
settings = sublime.load_settings(sc.get_settings_fn())
section_marker_size = int(str(settings.get('section_marker_size')))
for line in lines:
### Directives/aliases.
# :MY_PATH=some/where/my
# :NO_INDEX
matches = re_directives.findall(line)
for m in matches:
handled = False
parts = m.strip().split('=')
if len(parts) == 1:
directive = parts[0].strip()
if directive == 'NO_INDEX':
no_index = True
handled = True
elif len(parts) == 2:
alias = parts[0].strip()
value = parts[1].strip()
os.environ[alias] = value
handled = True # so far
if not handled:
_user_error(ntr_fn, line_num, 'Invalid directive')
### Links - also checks validity.
# [yer news](https://nytimes.com)
# [some felix]($NOTES_PATH/felix9.jpg)
matches = re_links.findall(line)
for m in matches:
if len(m) >= 2:
name = m[0].strip()
res = sc.expand_vars(m[1].strip())
if res is None:
# Bad env var.
_user_error(ntr_fn, line_num, f'Bad env var: {m[1]}')
else:
ttype = None
_, ext = os.path.splitext(res)
if ext in IMAGE_TYPES:
ttype = 'image'
elif res.startswith('http'):
ttype = 'uri'
elif os.path.exists(res):
ttype = 'path'
else:
_user_error(ntr_fn, line_num, f'Invalid target resource: {res}')
if ttype is not None:
links.append(Target(name, ttype, '', 0, [], res, ntr_fn, line_num))
else:
_user_error(ntr_fn, line_num, 'Invalid syntax')
### Refs - will be validated at end.
# [*some felix]
# [*yer news]
# [*ST executable dir]
# [* #section no tags].
# [*page2#P2 section 2]
matches = re_refs.findall(line)
for m in matches:
name = m.strip()
# If it's local section insert the froot.
if name.startswith('#'):
froot = _get_froot(ntr_fn)
name = froot + name
refs.append(Ref(name, ntr_fn, line_num))
### Sections
# # Some name[tag1 tag2]
matches = re_sections.findall(line)
for m in matches:
valid = True
hashes = ''
name=''
tags = []
if len(m) == 2:
content = m[0].strip().split(None, 1)
# print(content)
if len(content) == 2:
hashes = content[0].strip()
name = f'{_get_froot(ntr_fn)}{hashes}{content[1].strip()}'
else:
valid = False
if valid:
tags = m[1].strip().split()
else:
valid = False
if valid:
if len(hashes) <= section_marker_size:
sections.append(Target(name, 'section', '', len(hashes), tags, '', ntr_fn, line_num))
else:
_user_error(ntr_fn, line_num, 'Invalid syntax')
line_num += 1
except Exception as e:
_user_error(ntr_fn, line_num, f'Error processing file: {e}')
sc.error(f'Error processing file: {ntr_fn}:{line_num} {e}', e.__traceback__)
return None
return None if no_index else (sections, links, refs)
#-----------------------------------------------------------------------------------
def _build_selector(targets):
''' Populate the selector. '''
panel_items = []
for target in targets:
# ann = f'{target.ttype} ({target.category})'
ann = target.ttype
if target.ttype == 'section':
tt = "S"
# hide? ann = ''
elif target.ttype == 'uri':
tt = "R"
elif target.ttype == 'image':
tt = "R"
elif target.ttype == 'path':
tt = "R"
else:
tt = "?"
# COLOR_REDISH/_ORANGISH, _GREENISH/_YELLOWISH/_CYANISH, _BLUISH, _PURPLISH, _PINKISH (_DARK, _LIGHT)
if target.category == 'sticky':
clr = sublime.KindId.COLOR_PURPLISH
elif target.category == 'mru':
clr = sublime.KindId.COLOR_REDISH
else:
clr = sublime.KindId.COLOR_GREENISH
sty = (clr, tt, '')
lbl = target.name if len(target.name) > 0 else target.resource
panel_items.append(sublime.QuickPanelItem(trigger=f'{lbl}', annotation=ann, kind=sty))
return panel_items
#-----------------------------------------------------------------------------------
def _get_all_tags():
''' Return all tags found in all ntr files. Honors sort_tags_alpha setting. '''
settings = sublime.load_settings(sc.get_settings_fn())
sort_tags_alpha = settings.get('sort_tags_alpha')
tags = {} # k:tag text v:count
for target in _targets:
for tag in target.tags:
tags[tag] = tags[tag] + 1 if tag in tags else 1
sorted_tags = []
if sort_tags_alpha:
sorted_tags = sorted(tags.keys())
else: # Sort by frequency.
sorted_tags = [x[0] for x in sorted(tags.items(), key=lambda x:x[1], reverse=True)]
return sorted_tags
#-----------------------------------------------------------------------------------
def _get_all_ntr_files():
''' Return a sorted list of all processed .ntr file names. '''
fns = []
for target in _targets:
if target.file not in fns:
fns.append(target.file)
return sorted(fns)
#-----------------------------------------------------------------------------------
def _filter_order_targets(**kwargs):
''' Get filtered/ordered list of targets.
Options facilitate usage for UI presentation or internal consumption:
sort: T/F asc only
mru_first: Put the mru first
current_file: If provided put these after mru and before the rest
tags: filter by tags
returns a list of targets
'''
if _current_project is None:
return []
sticky = _current_project['sticky']
current_file_targets = []
other_targets = []
# Args.
sort = True if "sort" in kwargs and kwargs["sort"] else False
tags = kwargs["tags"] if "tags" in kwargs else []
mru_first = True if "mru_first" in kwargs and kwargs["mru_first"] else False
current_file = None
if ("current_file" in kwargs and
kwargs["current_file"] is not None and
os.path.exists(kwargs["current_file"])):
current_file = kwargs["current_file"]
# Cache some targets to maintain order.
sticky_cache = {}
mru_cache = {}
for target in _targets:
target.category = '' # default
# Sticky always wins.
if target.name in sticky:
target.category = 'sticky'
sticky_cache[target.name] = target
else: # The others.
# Check tags.
tag_ok = len(tags) == 0
for t in tags:
if t in target.tags:
tag_ok = True
if tag_ok:
if mru_first and target.name in _current_mru:
target.category = 'mru'
mru_cache[target.name] = target
elif current_file is not None:
if target.file is not None and os.path.exists(target.file) and os.path.samefile(target.file, current_file):
current_file_targets.append(target)
else:
other_targets.append(target)
else:
other_targets.append(target)
# Sort the rest?
if sort:
current_file_targets = sorted(current_file_targets)
other_targets = sorted(other_targets)
# Collect and return.
ret = []
# Order these.
for st in sticky: # by position in settings