-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathorg-zettel-ref-list.el
907 lines (797 loc) · 36.7 KB
/
org-zettel-ref-list.el
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
;;; org-zettel-ref-list.el --- Reading management interface for org-zettel-ref -*- lexical-binding: t; -*-
;;; Commentary:
;; This file provides a reading management interface for org-zettel-ref-mode.
;; It manages reading states and integrates with the database system
;; provided by org-zettel-ref-db.el.
;;; Code:
(require 'org-zettel-ref-list-filter)
(require 'org-zettel-ref-db)
(require 'transient)
;;;----------------------------------------------------------------------------
;;; Variables
;;;----------------------------------------------------------------------------
(defcustom org-zettel-ref-directory (expand-file-name "~/Documents/ref/")
"File path for the reference directory.")
(defvar org-zettel-ref-filter-presets-file
(expand-file-name "org-zettel-ref-filters.el" org-zettel-ref-directory)
"File path for storing filter presets.")
(defvar org-zettel-ref-list-mode-map (make-sparse-keymap)
"Keymap for org-zettel-ref reading mode.")
(defgroup org-zettel-ref-list nil
"Customization group for org-zettel-ref reading manager."
:group 'org-zettel-ref)
(defun org-zettel-ref-ensure-directory ()
"Ensure the reference directory exists."
(unless (file-exists-p org-zettel-ref-directory)
(make-directory org-zettel-ref-directory t))
org-zettel-ref-directory)
;;;----------------------------------------------------------------------------
;;; Mode Definition
;;;----------------------------------------------------------------------------
(define-derived-mode org-zettel-ref-list-mode tabulated-list-mode "Zettel-Ref"
"Major mode for displaying and managing Zettel references."
(setq tabulated-list-format
[("Title" 60 t)
("Author" 20 t)
("Modified" 20 t)
("Keywords" 30 t)])
(setq tabulated-list-padding 2)
(setq tabulated-list-sort-key (cons "Title" nil))
;; Add mouse support
(make-local-variable 'mouse-face-highlight-property)
(setq mouse-face-highlight-property 'highlight)
;; Start file monitoring
;;(org-zettel-ref-watch-directory)
;; Stop monitoring when buffer is killed
;;(add-hook 'kill-buffer-hook #'org-zettel-ref-unwatch-directory nil t)
(tabulated-list-init-header))
;;;----------------------------------------------------------------------------
;;; File Name Parsing and Formatting
;;;----------------------------------------------------------------------------
(defconst org-zettel-ref-author-regexp "^\\(.*?\\)__"
"Match the author part in the file name.")
(defconst org-zettel-ref-title-regexp "\\(?:^\\|__\\)\\(.*?\\)\\(?:==\\|$\\)"
"Match the title part in the file name.")
(defconst org-zettel-ref-keywords-regexp "==\\(.*?\\)\\(?:\\..*\\)?$"
"Match the keywords part in the file name.")
;; Parse file name
(defun org-zettel-ref-parse-filename (filename)
"Parse FILENAME into a list of (author title keywords)."
(let (author title keywords)
;; Remove .org extension
(setq filename (file-name-sans-extension filename))
;; Parse author using regexp
(when (string-match org-zettel-ref-author-regexp filename)
(setq author (match-string 1 filename)))
;; Parse title
(if author
(when (string-match "__\\([^=]+\\)" filename)
(setq title (match-string 1 filename)))
(when (string-match "^\\([^=]+\\)" filename)
(setq title (match-string 1 filename))))
;; Parse keywords
(when (string-match org-zettel-ref-keywords-regexp filename)
(setq keywords (split-string (match-string 1 filename) "_")))
;; Trim all parts
(setq author (when author (string-trim author))
title (when title (string-trim title))
keywords (when keywords (mapcar #'string-trim keywords)))
(list author title keywords)))
;; Format file name
(defun org-zettel-ref-format-filename (author title keywords)
"Generate a standard file name: AUTHOR__TITLE==KEYWORDS.org"
(concat
(when author (concat author "__"))
title
(when keywords (concat "==" (string-join keywords "_")))
".org"))
;; Get file modified time
(defun org-zettel-ref-get-modified-time (file)
"Get the modified time of FILE."
(format-time-string "%Y-%m-%d %H:%M:%S"
(nth 5 (file-attributes file))))
;; Parse file info
(defun org-zettel-ref-parse-file (file)
"Parse FILE to extract basic information."
(let* ((filename (file-name-nondirectory file))
(parsed (org-zettel-ref-parse-filename filename)))
(list :title (nth 1 parsed)
:author (nth 0 parsed)
:keywords (nth 2 parsed))))
;;;----------------------------------------------------------------------------
;;; Display Interface
;;;----------------------------------------------------------------------------
;; Refresh list
(defun org-zettel-ref-list-refresh ()
"Refresh the reference list display."
(interactive)
(when (eq major-mode 'org-zettel-ref-list-mode)
(let* ((marked-files org-zettel-ref-marked-files)
(current-pos (point))
(inhibit-read-only t))
;; Clear all overlays
(when (boundp 'org-zettel-ref-mark-overlays)
(dolist (ov org-zettel-ref-mark-overlays)
(delete-overlay ov))
(setq org-zettel-ref-mark-overlays nil))
;; Clear buffer
(erase-buffer)
;; Get and filter entries
(condition-case err
(progn
(let ((entries (org-zettel-ref-list--get-entries)))
(when (fboundp 'org-zettel-ref-apply-filters)
(setq entries (org-zettel-ref-apply-filters entries)))
(setq tabulated-list-entries entries))
;; Print list
(tabulated-list-print t))
(error
(message "Error refreshing list: %s" (error-message-string err))))
;; Restore marks if needed
(when (and (boundp 'org-zettel-ref-marked-files)
marked-files)
(save-excursion
(goto-char (point-min))
(while (not (eobp))
(let ((file (org-zettel-ref-list-get-file-at-point)))
(when (member file marked-files)
(let* ((beg (line-beginning-position))
(end (1+ (line-end-position)))
(ov (make-overlay beg end)))
(overlay-put ov 'face 'org-zettel-ref-marked-face)
(overlay-put ov 'org-zettel-ref-marked t)
(push ov org-zettel-ref-mark-overlays))))
(forward-line 1))))
;; Restore cursor position or move to beginning
(if (< current-pos (point-max))
(goto-char current-pos)
(goto-char (point-min))))))
;; Column Components
(defun org-zettel-ref-column-author ()
"Author column definition."
(list "Author" 15 t))
(defun org-zettel-ref-column-title ()
"Title column definition."
(list "Title" 50 t))
(defun org-zettel-ref-column-modified ()
"Modified time column definition."
(list "Modified" 20 t))
(defun org-zettel-ref-column-keywords ()
"Keywords column definition."
(list "Keywords" 30 t))
(defun org-zettel-ref-list-columns ()
"Return a list of column definitions."
(list (org-zettel-ref-column-title)
(org-zettel-ref-column-author)
(org-zettel-ref-column-modified)
(org-zettel-ref-column-keywords)))
;;;----------------------------------------------------------------------------
;;; Display: Sorting
;;;----------------------------------------------------------------------------
(defun org-zettel-ref-list-sort-by-column (&optional column)
"Sort the list by COLUMN."
(interactive)
(unless column
(setq column (tabulated-list-get-sort-key))))
;;;----------------------------------------------------------------------------
;;; Display Content: Entry Formatting
;;;----------------------------------------------------------------------------
(defun org-zettel-ref-list--format-entry (id entry)
"Format entry for tabulated list display.
ID is the entry identifier.
ENTRY is the org-zettel-ref-entry struct."
(if (null entry)
(progn
(message "Warning: Nil entry for id %s" id)
(make-vector (length (org-zettel-ref-list-columns)) " "))
(let* ((file-path (org-zettel-ref-ref-entry-file-path entry))
(title (or (org-zettel-ref-ref-entry-title entry)
(file-name-base file-path)))
(title-with-props ; Add file path property to title
(propertize title
'file-path file-path
'help-echo file-path)))
(vector
title-with-props ; Title (with file path property)
(or (org-zettel-ref-ref-entry-author entry) "")
(format-time-string "%Y-%m-%d %H:%M:%S"
(org-zettel-ref-ref-entry-modified entry))
(if-let* ((keywords (org-zettel-ref-ref-entry-keywords entry)))
(string-join keywords ", ")
"")))))
(defun org-zettel-ref-list--get-entries ()
"Get entries for tabulated list display."
(let ((entries '())
(db (org-zettel-ref-ensure-db)))
(maphash
(lambda (id entry)
(when-let* ((file-path (org-zettel-ref-ref-entry-file-path entry))
(exists (file-exists-p file-path)))
(let* ((file-name (file-name-nondirectory file-path))
(parsed-info (org-zettel-ref-parse-filename file-name))
(author (nth 0 parsed-info))
(title (nth 1 parsed-info))
(keywords (nth 2 parsed-info)))
(push (list file-path
(vector
;; Title column
(propertize
(or title "Untitled")
'file-path file-path
'help-echo file-path)
;; Author column
(propertize
(or author "")
'help-echo (format "Author: %s" (or author "Unknown")))
;; Modified time column
(format-time-string "%Y-%m-%d %H:%M:%S"
(org-zettel-ref-ref-entry-modified entry))
;; Keywords column
(string-join (or keywords '()) ", ")))
entries))))
(org-zettel-ref-db-refs db))
(nreverse entries)))
(defun org-zettel-ref-ref-entry-p (obj)
"Check if OBJ is a valid ref-entry struct."
(and (recordp obj)
(eq (type-of obj) 'org-zettel-ref-ref-entry)))
;;;----------------------------------------------------------------------------
;;; List Operation: Mark
;;;----------------------------------------------------------------------------
(defvar-local org-zettel-ref-marked-files nil
"List of files marked for deletion.")
(defvar-local org-zettel-ref-mark-overlays nil
"List of overlays for marked lines.")
(defface org-zettel-ref-marked-face
'((t :inherit warning :extend t))
"Highlight style for marked files.")
(defun org-zettel-ref-mark-file ()
"Mark the current file for deletion."
(interactive)
(when-let* ((file (org-zettel-ref-list-get-file-at-point))
(beg (line-beginning-position))
(end (1+ (line-end-position))))
;; Create highlight overlay
(let ((ov (make-overlay beg end)))
(overlay-put ov 'face 'org-zettel-ref-marked-face)
(overlay-put ov 'org-zettel-ref-marked t)
(push ov org-zettel-ref-mark-overlays)
(push file org-zettel-ref-marked-files))
(forward-line 1)
(message "Marked %s for deletion" file)))
(defun org-zettel-ref-unmark-file ()
"Unmark the current file."
(interactive)
(when-let* ((file (org-zettel-ref-list-get-file-at-point)))
;; Remove current line overlay
(let ((overlays (overlays-in (line-beginning-position)
(1+ (line-end-position)))))
(dolist (ov overlays)
(when (overlay-get ov 'org-zettel-ref-marked)
(setq org-zettel-ref-mark-overlays
(delq ov org-zettel-ref-mark-overlays))
(delete-overlay ov))))
;; Remove file from marked list
(setq org-zettel-ref-marked-files
(delete file org-zettel-ref-marked-files))
(forward-line 1)
(message "Unmarked %s" file)))
(defun org-zettel-ref-unmark-all ()
"Unmark all files."
(interactive)
;; Clear all overlays
(dolist (ov org-zettel-ref-mark-overlays)
(delete-overlay ov))
(setq org-zettel-ref-mark-overlays nil
org-zettel-ref-marked-files nil)
(message "Unmarked all files"))
;;;--------------------------------------------------------------------------
;;; File Operation: Rename
;;;----------------------------------------------------------------------------
(defun org-zettel-ref-get-existing-authors (db)
"Get a list of all existing authors from the database."
(let ((authors '()))
(maphash
(lambda (_id entry)
(when-let* ((author (org-zettel-ref-ref-entry-author entry)))
(push author authors)))
(org-zettel-ref-db-refs db))
(delete-dups authors)))
(defun org-zettel-ref-get-existing-titles (db)
"Get a list of all existing titles from the database."
(let ((titles '()))
(maphash
(lambda (_id entry)
(when-let* ((title (org-zettel-ref-ref-entry-title entry)))
(push title titles)))
(org-zettel-ref-db-refs db))
(delete-dups titles)))
(defun org-zettel-ref-get-existing-keywords (db)
"Get a list of all existing keywords from the database."
(let ((keywords '()))
(maphash
(lambda (_id entry)
(when-let* ((entry-keywords (org-zettel-ref-ref-entry-keywords entry)))
(setq keywords (append keywords entry-keywords))))
(org-zettel-ref-db-refs db))
(delete-dups keywords)))
(defun org-zettel-ref-rename--prompt-title (current-title)
"Prompt for title input with completion, CURRENT-TITLE is the current title."
(let* ((db (org-zettel-ref-ensure-db))
(existing-titles (org-zettel-ref-get-existing-titles db))
(prompt (format "Title%s: "
(if current-title
(format " (current: %s)" current-title)
""))))
(completing-read prompt existing-titles
nil nil nil nil current-title)))
(defun org-zettel-ref-rename--prompt-keywords (current-keywords)
"Prompt for keywords input with completion, CURRENT-KEYWORDS is the current keywords."
(let* ((db (org-zettel-ref-ensure-db))
(existing-keywords (org-zettel-ref-get-existing-keywords db))
(current (if current-keywords
(string-join current-keywords ", ")
""))
(input (completing-read-multiple
(format "Keywords%s (comma-separated): "
(if current
(format " (current: %s)" current)
""))
existing-keywords
nil nil current)))
(if (stringp input)
(split-string input "[,\s]+" t)
input)))?
(defun org-zettel-ref-rename--prompt-author (current-author)
"Prompt for author input with completion, CURRENT-AUTHOR is the current author."
(let* ((db (org-zettel-ref-ensure-db))
(existing-authors (org-zettel-ref-get-existing-authors db))
(prompt (format "Author%s: "
(if current-author
(format " (current: %s)" current-author)
""))))
(completing-read prompt existing-authors
nil nil nil nil current-author)))
(defun org-zettel-ref-rename--prompt-title (current-title)
"Prompt for title input, CURRENT-TITLE is the current title."
(read-string
(format "Title%s: "
(if current-title
(format " (current: %s)" current-title)
""))
current-title))
(defun org-zettel-ref-rename--prompt-keywords (current-keywords)
"Prompt for keywords input, CURRENT-KEYWORDS is the current keywords."
(let ((current (if current-keywords
(string-join current-keywords ", ")
"")))
(split-string
(read-string
(format "Keywords%s (comma-separated): "
(if current
(format " (current: %s)" current)
""))
current)
"[,\s]+" t)))
(defun org-zettel-ref-list-rename-file ()
"Rename the file at point."
(interactive)
(let* ((db (org-zettel-ref-ensure-db))
(old-file (org-zettel-ref-list-get-file-at-point)))
(if (not old-file)
(message "No file selected")
(let* ((dir (file-name-directory old-file))
(ref-id (when old-file (org-zettel-ref-db-get-ref-id-by-path db old-file))))
(if (not ref-id)
(message "Error: Cannot find database entry for file: %s" old-file)
(let* ((ref-entry (org-zettel-ref-db-get-ref-entry db ref-id))
(current-author (org-zettel-ref-ref-entry-author ref-entry))
(current-title (org-zettel-ref-ref-entry-title ref-entry))
(current-keywords (org-zettel-ref-ref-entry-keywords ref-entry))
(new-author (org-zettel-ref-rename--prompt-author current-author))
(new-title (org-zettel-ref-rename--prompt-title current-title))
(new-keywords (org-zettel-ref-rename--prompt-keywords current-keywords))
(new-file-name (org-zettel-ref-format-filename new-author new-title new-keywords))
(new-file-path (expand-file-name new-file-name dir)))
(when (and (not (equal old-file new-file-path))
(y-or-n-p (format "Rename %s to %s? "
(file-name-nondirectory old-file)
(file-name-nondirectory new-file-path))))
;; Suspend file monitoring
(org-zettel-ref-unwatch-directory)
(condition-case err
(progn
;; Rename file
(rename-file old-file new-file-path t)
;; Update database
(org-zettel-ref-db-update-ref-path db old-file new-file-path)
(setf (org-zettel-ref-ref-entry-file-path ref-entry) new-file-path
(org-zettel-ref-ref-entry-title ref-entry) new-title
(org-zettel-ref-ref-entry-author ref-entry) new-author
(org-zettel-ref-ref-entry-keywords ref-entry) new-keywords)
(org-zettel-ref-db-update-ref-entry db ref-entry)
(org-zettel-ref-db-save db)
;; Update opened buffer
(when-let ((buf (get-file-buffer old-file)))
(with-current-buffer buf
(set-visited-file-name new-file-path)
(set-buffer-modified-p nil)))
;; Refresh display
(org-zettel-ref-list-refresh)
(message "File renamed from %s to %s"
(file-name-nondirectory old-file)
(file-name-nondirectory new-file-path)))
(error
(message "Error during rename: %s" (error-message-string err))))
;; Restart file monitoring
(run-with-timer 0.5 nil #'org-zettel-ref-watch-directory))))))))
;;;----------------------------------------------------------------------------
;;; File Operation: Edit Keywords
;;;----------------------------------------------------------------------------
(defun org-zettel-ref-list-edit-keywords ()
"Add or edit keywords for the file at point or marked files."
(interactive)
(let* ((db (org-zettel-ref-ensure-db))
(marked-files org-zettel-ref-marked-files)
(files (if (not (null marked-files))
marked-files
(list (org-zettel-ref-list-get-file-at-point)))))
(dolist (file files)
(let* ((filename (file-name-nondirectory file))
(components (org-zettel-ref-parse-filename filename))
;; Keep the base part of the original file name (up to == or .org)
(base-with-title (substring filename 0
(or (string-match "==" filename)
(- (length filename) 4)))) ; 减去.org
(current-keywords (if (nth 2 components)
(string-join (nth 2 components) ", ")
""))
(prompt (if (string-empty-p current-keywords)
(format "Add keywords (comma-separated) for %s: " filename)
(format "Edit keywords (current: %s) for %s: " current-keywords filename)))
(new-keywords-input (read-string prompt current-keywords)))
(unless (string-empty-p new-keywords-input)
(let* ((new-keywords-list (split-string new-keywords-input "[,\s]+" t))
(new-keywords-str (string-join new-keywords-list "_"))
(new-filename (concat base-with-title
"==" new-keywords-str
".org"))
(new-filepath (expand-file-name new-filename (file-name-directory file))))
(when (and (not (equal file new-filepath))
(y-or-n-p (format "Rename file to %s?" new-filename)))
;; Rename file
(condition-case err
(rename-file file new-filepath t)
(error
(message "Error renaming file: %s" (error-message-string err))
(signal (car err) (cdr err))))
;; Update database
(when-let* ((ref-id (org-zettel-ref-db-get-ref-id-by-path db file))
(ref-entry (org-zettel-ref-db-get-ref-entry db ref-id)))
(remhash file (org-zettel-ref-db-ref-paths db))
(puthash new-filepath ref-id (org-zettel-ref-db-ref-paths db))
(setf (org-zettel-ref-ref-entry-file-path ref-entry) new-filepath
(org-zettel-ref-ref-entry-keywords ref-entry) new-keywords-list
(org-zettel-ref-ref-entry-modified ref-entry) (current-time)))
;; Update opened buffer
(when-let* ((buf (get-file-buffer file)))
(with-current-buffer buf
(set-visited-file-name new-filepath)
(set-buffer-modified-p nil))))))))
;; Save database and refresh display
(org-zettel-ref-db-save db)
(org-zettel-ref-list-refresh)
(org-zettel-ref-unmark-all)
(message "Keywords updated successfully")))
;;;-------------------------------------------------------------------------
;;; File Operation: Open
;;;--------------------------------------------------------------------------
(defun org-zettel-ref-list-open-file ()
"Open the file at point."
(interactive)
(let ((file (org-zettel-ref-list-get-file-at-point)))
(when file
(find-file file))))
;;;--------------------------------------------------------------------------
;;; File Operation: Delete
;;;--------------------------------------------------------------------------
(defun org-zettel-ref-list-delete-file ()
"Delete the currently selected file."
(interactive)
(let* ((file (org-zettel-ref-list-get-file-at-point))
(db (org-zettel-ref-ensure-db))
(ref-id (when file (org-zettel-ref-db-get-ref-id-by-path db file))))
(when (and file ref-id
(yes-or-no-p (format "Delete file %s? " file)))
;; Delete mapping relationship
(when-let* ((overview-id (org-zettel-ref-db-get-maps db ref-id)))
(remhash ref-id (org-zettel-ref-db-map db))
(remhash overview-id (org-zettel-ref-db-overviews db)))
;; Delete reference entry
(remhash ref-id (org-zettel-ref-db-refs db))
;; Delete path index
(remhash file (org-zettel-ref-db-ref-paths db))
;; Delete actual file
(condition-case err
(progn
(delete-file file)
(org-zettel-ref-db-save db)
(org-zettel-ref-list-refresh)
(message "File deleted successfully"))
(error
(message "Error deleting file %s: %s"
file (error-message-string err)))))))
(defun org-zettel-ref-list-delete-marked-files ()
"Delete all marked files."
(interactive)
(let* ((files org-zettel-ref-marked-files)
(file-count (length files))
(db (org-zettel-ref-ensure-db))
(deleted 0))
(when (and files
(yes-or-no-p
(format "Delete %d marked file%s? "
file-count
(if (= file-count 1) "" "s"))))
(dolist (file files)
(when-let* ((ref-id (org-zettel-ref-db-get-ref-id-by-path db file)))
;; Delete mapping relationship
(when-let* ((overview-id (org-zettel-ref-db-get-maps db ref-id)))
(remhash ref-id (org-zettel-ref-db-map db))
(remhash overview-id (org-zettel-ref-db-overviews db)))
;; Delete reference entry
(remhash ref-id (org-zettel-ref-db-refs db))
;; Delete path index
(remhash file (org-zettel-ref-db-ref-paths db))
;; Delete actual file
(condition-case err
(progn
(delete-file file)
(cl-incf deleted))
(error
(message "Error deleting file %s: %s"
file (error-message-string err))))))
;; Save database and refresh
(org-zettel-ref-db-save db)
(setq org-zettel-ref-marked-files nil)
(org-zettel-ref-list-refresh)
(message "Successfully deleted %d of %d files"
deleted file-count))))
;;;----------------------------------------------------------------------------
;;; Interactive Menu System
;;;----------------------------------------------------------------------------
(defvar org-zettel-ref-list-actions
'(("open file" . org-zettel-ref-list-open-file)
("rename file" . org-zettel-ref-list-rename-file)
("refresh list" . org-zettel-ref-list-refresh)
("delete file" . org-zettel-ref-list-delete-file)
("mark file" . org-zettel-ref-mark-file)
("unmark file" . org-zettel-ref-unmark-file)
("delete marked files" . org-zettel-ref-list-delete-marked-files)
("unmark all" . org-zettel-ref-unmark-all)
("filter by regexp" . org-zettel-ref-filter-by-regexp)
("clear filters" . org-zettel-ref-clear-all-filters)
("manage filter presets" . org-zettel-ref-filter-manage-presets))
"Available actions for reference list management.")
(defun org-zettel-ref-list-menu ()
"Display reference list management actions in minibuffer."
(interactive)
(let* ((choices (mapcar #'car org-zettel-ref-list-actions))
(choice (completing-read "Select action: " choices nil t))
(action (cdr (assoc choice org-zettel-ref-list-actions))))
(when action
(call-interactively action))))
;; Add menu key binding while keeping existing bindings
(define-key org-zettel-ref-list-mode-map (kbd "C-c C-m") #'org-zettel-ref-list-menu)
;; Keep existing key bindings
(define-key org-zettel-ref-list-mode-map (kbd "RET") #'org-zettel-ref-list-open-file)
(define-key org-zettel-ref-list-mode-map (kbd "o") #'org-zettel-ref-list-open-file)
(define-key org-zettel-ref-list-mode-map (kbd "r") #'org-zettel-ref-list-rename-file)
(define-key org-zettel-ref-list-mode-map (kbd "g") #'org-zettel-ref-list-refresh)
(define-key org-zettel-ref-list-mode-map (kbd "d") #'org-zettel-ref-list-delete-file)
(define-key org-zettel-ref-list-mode-map (kbd "k") #'org-zettel-ref-list-edit-keywords)
(define-key org-zettel-ref-list-mode-map (kbd "m") #'org-zettel-ref-mark-file)
(define-key org-zettel-ref-list-mode-map (kbd "u") #'org-zettel-ref-unmark-file)
(define-key org-zettel-ref-list-mode-map (kbd "D") #'org-zettel-ref-list-delete-marked-files)
(define-key org-zettel-ref-list-mode-map (kbd "U") #'org-zettel-ref-unmark-all)
(define-key org-zettel-ref-list-mode-map (kbd "/ r") #'org-zettel-ref-filter-by-regexp)
(define-key org-zettel-ref-list-mode-map (kbd "/ c") #'org-zettel-ref-clear-all-filters)
;(define-key org-zettel-ref-list-mode-map (kbd "/ p") #'org-zettel-ref-filter-manage-presets)
(define-key org-zettel-ref-list-mode-map (kbd "/ m") #'org-zettel-ref-filter-by-multiple-conditions)
;;;----------------------------------------------------------------------------
;;; org-zettel-ref-list
;;;----------------------------------------------------------------------------
(defun org-zettel-ref-list ()
"Display reference list."
(interactive)
(let ((db (org-zettel-ref-ensure-db)))
;; Always scan directory but only create entries for new files
(org-zettel-ref-scan-directory db)
(let ((buffer (get-buffer-create "*Org Zettel Ref List*")))
(with-current-buffer buffer
(org-zettel-ref-list-mode)
(let ((inhibit-read-only t))
(erase-buffer))
(let ((entries (org-zettel-ref-list--get-entries)))
(setq tabulated-list-entries entries)
(tabulated-list-print t))
(goto-char (point-min)))
(switch-to-buffer buffer)
buffer)))
;; Add helper function to get current item file path
(defun org-zettel-ref-list-get-file-at-point ()
"Get the file path associated with the current position."
(or (get-text-property (point) 'file-path)
(tabulated-list-get-id)
(error "No file associated with current position")))
;;;----------------------------------------------------------------------------
;;; File Operation: Scan Directory (add new ref files to db)
;;;----------------------------------------------------------------------------
(defun org-zettel-ref-scan-directory (db)
"Scan the reference directory and add files to database.
DB is the database object."
(let ((files (org-zettel-ref-find-ref-files))
(new-count 0)
(existing-count 0)
(added 0))
(message "Found %d files to process" (length files))
;; Reset ID counter at start of scan
(setq org-zettel-ref-id-counter 0)
(dolist (file files)
(let* ((file-path (expand-file-name file))
(ref-id (org-zettel-ref-db-get-ref-id-by-path db file-path)))
(if ref-id
(cl-incf existing-count)
(let* ((file-info (org-zettel-ref-parse-file file))
(title (plist-get file-info :title))
(author (plist-get file-info :author))
(keywords (plist-get file-info :keywords)))
;; 创建新条目
(org-zettel-ref-db-ensure-ref-entry db file-path title author keywords)
(cl-incf new-count)
(cl-incf added)
;; Save database every 100 entries
(when (zerop (mod added 100))
(message "Saving database after %d new entries..." added)
(org-zettel-ref-db-save db))))))
(message "Scan complete: %d new files, %d existing files"
new-count existing-count)
;; Final save if there are any new entries
(when (> new-count 0)
(org-zettel-ref-db-save db))))
;;;----------------------------------------------------------------------------
;;; File Operation: Remove Invalid Entries
;;;----------------------------------------------------------------------------
(defun org-zettel-ref-remove-entry (db id)
"Remove entry with specified ID from database.
Return t if deletion is successful, nil if entry does not exist."
(interactive
(let* ((db (org-zettel-ref-ensure-db))
(candidates
(let (items)
(maphash
(lambda (id entry)
(push (format "[%s] %s"
id
(or (org-zettel-ref-ref-entry-title entry)
(file-name-nondirectory
(org-zettel-ref-ref-entry-file-path entry))))
items))
(org-zettel-ref-db-refs db))
(sort items #'string>)))
(selection (completing-read "Select entry to delete: " candidates nil t)))
(list db (substring selection 1 (string-match "]" selection)))))
(when-let* ((entry (org-zettel-ref-db-get-ref-entry db id)))
;; Delete entry
(remhash id (org-zettel-ref-db-refs db))
;; Save database
(org-zettel-ref-db-save db)
;; Display result
(when (called-interactively-p 'any)
(message "Deleted entry: [%s] %s"
id
(or (org-zettel-ref-ref-entry-title entry)
(file-name-nondirectory
(org-zettel-ref-ref-entry-file-path entry)))))
t))
;; Batch delete function
(defun org-zettel-ref-remove-entries (db ids)
"Remove multiple entries from database.
IDS is a list of entry IDs. Return number of deleted entries."
(interactive
(let* ((db (org-zettel-ref-ensure-db))
(candidates
(let (items)
(maphash
(lambda (id entry)
(push (format "[%s] %s"
id
(or (org-zettel-ref-ref-entry-title entry)
(file-name-nondirectory
(org-zettel-ref-ref-entry-file-path entry))))
items))
(org-zettel-ref-db-refs db))
(sort items #'string>)))
(selections (completing-read-multiple
"Select entries to delete (comma-separated): "
candidates nil t)))
(list db
(mapcar (lambda (s)
(substring s 1 (string-match "]" s)))
selections))))
(let ((removed 0))
(dolist (id ids)
(when-let* ((entry (org-zettel-ref-db-get-ref-entry db id)))
;; Delete entry
(remhash id (org-zettel-ref-db-refs db))
(cl-incf removed)
;; Display each deleted entry
(when (called-interactively-p 'any)
(message "Deleted entry: [%s] %s"
id
(or (org-zettel-ref-ref-entry-title entry)
(file-name-nondirectory
(org-zettel-ref-ref-entry-file-path entry)))))
;; Save database
(org-zettel-ref-db-save db)
(when (called-interactively-p 'any)
(message "Deleted %d entries" removed)))
removed)))
;;;----------------------------------------------------------------------------
;;; Find ref files and parse file info
;;;----------------------------------------------------------------------------
(defun org-zettel-ref-find-ref-files ()
"Find all referenced files.
Return a list of file paths."
(when (file-exists-p org-zettel-ref-directory)
(directory-files org-zettel-ref-directory t "\\.org$")))
;;;----------------------------------------------------------------------------
;;; File Monitoring
;;;----------------------------------------------------------------------------
(defvar org-zettel-ref-file-watch-descriptor nil
"File monitoring descriptor.")
(defun org-zettel-ref-handle-file-change (event)
"Handle file change EVENT from file monitoring system."
(when (and (bound-and-true-p org-zettel-ref-file-watch-descriptor)
(buffer-live-p (get-buffer "*Org Zettel Ref List*")))
(let ((event-type (nth 1 event))
(file (nth 2 event)))
(when (and file
(stringp file)
(string-match-p "\\.org$" file)
(not (string-match-p "^\\." (file-name-nondirectory file))))
(run-with-timer
0.5 nil
(lambda ()
(when (buffer-live-p (get-buffer "*Org Zettel Ref List*"))
(with-current-buffer "*Org Zettel Ref List*"
(when (eq major-mode 'org-zettel-ref-list-mode)
(org-zettel-ref-list-refresh)
(message "Refreshed due to file change: %s"
(file-name-nondirectory file)))))))))))
(defun org-zettel-ref-watch-directory ()
"Start monitoring changes in the reference file directory."
(when (file-exists-p org-zettel-ref-directory)
;; Ensure no existing watch
(org-zettel-ref-unwatch-directory)
(condition-case err
(let ((descriptor (file-notify-add-watch
org-zettel-ref-directory
'(change attribute-change)
#'org-zettel-ref-handle-file-change)))
(setq-local org-zettel-ref-file-watch-descriptor descriptor)
(message "Started monitoring directory: %s" org-zettel-ref-directory))
(error
(message "Error setting up file watch: %s" (error-message-string err))))))
(defun org-zettel-ref-unwatch-directory ()
"Stop monitoring the reference file directory."
(when (bound-and-true-p org-zettel-ref-file-watch-descriptor)
(condition-case err
(progn
(file-notify-rm-watch org-zettel-ref-file-watch-descriptor)
(setq-local org-zettel-ref-file-watch-descriptor nil)
(message "Stopped monitoring directory"))
(error
(message "Error removing file watch: %s" (error-message-string err))
(setq-local org-zettel-ref-file-watch-descriptor nil)))))
(provide 'org-zettel-ref-list)