-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathna.el
1131 lines (1087 loc) · 50.8 KB
/
na.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
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
;;; na.el -- N-Angulator: Persistent N-Dimensional Sparse Array, Editor, and Browser
;; Copyright (C) 2000
;; Kevin Haddock and N-Angulator.org -- All Rights Reserved
;; Author: Kevin Haddock <support@n-angulator.org>
;; Maintainer: Kevin Haddock <support@n-angulator.org>
;; Keywords: database, hypermedia, persistent, object oriented, file tool, search engine
;; Version: 0.1
;; X-URL:
;; This program runs under XEmacs
;; N-Angulator is free software; you can redistribute it and/or modify it under the terms of
;; the N-Angulator.org Public License as published by N-Angulator.org
;; either version 1, or (at your option) any later version (see the file COPYING distributed
;; with this file or online at:
;;
;; http://www.n-angulator.org/license
;; N-Angulator is distributed in the hope that it will be useful,
;; but WITHOUT ANY WARRANTY; without even the implied warranty of
;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
;; N-Angulator.org Public License for more details.
;; You should have received a copy of the N-Angulator.org
;; Public License along with N-Angulator; see the file COPYING.
;; If not, write to:
;;
;; N-Angulator.org
;; Attn: KEVIN HADDOCK
;; P.O. Box 70404
;; Richmond, CA 94807-0404
;;
;; or email: support@n-angulator.org
;;; Commentary:
;;
;; This program has a companion shell script `na.member.sh'.
;; This utility can optionally work in conjunction with an
;; automated document generation system (mrg.el) by the same author.
;; N-Angulator makes use of Unix-like file systems that allow hard links to files
;; (the same file showing up in different places in the file-system hierarchy under
;; possibly different names). It turns this feature into an automated, multi-indexed,
;; multimedia file cabinet where items and groups of items can be incrementally searched
;; by doing a series of union and intersection 'set' operations. In N-Angulator speak, files
;; are called 'leaves,' directories are called 'branches,' and the incremental searches called
;; 'angles.' It is called 'N-angulator' because locating the desired information resembles
;; the process of 'triangulating' the source of a radio signal except that N different 'angles'
;; may be (and usually are) employed rather than just two.
;; N-Angulator also allows one to easily add, delete, and modify the elements in the tree, hence
;; it's role as file tool, or 'editor.' It is 'persistent' because very rapidly (usually no
;; more than 30 seconds, depending on the operating system's cache) all changes are committed
;; to disk and more or less permanent. It is somewhat object oriented because it uses the mime
;; library to deal with each file type independently.
;; N-Angangulator's possible (future) uses could be: Internet bookmark file, search engine,
;; link farm, sales contact management, multi-media archive, etc... or literally any extensive
;; multi-indexed filing application.
;; NOTE: N-Angulator uses a modified GPL license but construcively for contributors other
;; those specifically designated by its owner/originator, constibutions are considered governed
;; by GPL version 2. Contact the owner/originator at support@n-angulator.org for further details.
;; To do:
;; Get an invalid function nil menu when pulling off a branch/leaf menu.
;; Na-pool-delete does not rebuild the pools correctly. The subsequent branches
;; don't have everything as members as if the deleted leaf had not been selected
;; in the first instance.
;; Try dynamic scoping of default-directory with creation as local then destruction reveals orig
;; Add "display leaf type" and perhaps "display leaf info" to leaf menu.
;; Add "display branch info" to branch menu.
;; Somewhere along the line :pool gets a \n at its beginning which can cause excessive
;; na-branch-arguments rebuilds
;; Update screen widgets that have changed names or been deleted without disturbing anything else
;; Stop Na from messing up parens display highlighting in elisp source
;; Stop Na error clicking branch after creating leaf/link there
;; Make Na use eshell functions wherever possible
;; Fix pop up menu's solid lines (in widget package?)
;; Make inactive widgets in popup menus gray and unselectable (in widget package?)
;; Replicate this branch in new angle below.
;; eradicate branch and it's contents
;; Replicate this search in a new buffer.
;; Think about ways that newly created buffers could interact (drag n drop, cut and paste?)
;; Pseudo branches/leaves representing inode permissions, dates, #links, etc... information
;; Make a log of changes (so other machines could reflect updates)
;; Build an na.members.sh server (hashes for inode tables rather than 'find' shell cmd)
;; Test/implement gnuserv/gnuclient to windows machines over the internet
;; And/or/not radio buttons at the begining of each angle
;; Implement sugrp as xemacs primitive
;; Considering keeping directory inodes to improve screen updates
;; Make leaf and branch modifications more object oriented
;; Make Na look and behave somewhat like Nautilus (or integrate Na search features there)
;; link to selection to already existing link pops bogus error
;;
;;; Code:
;(load "eshell-auto")
(require 'dired)
(require 'widget)
(eval-when-compile
(require 'wid-edit))
(eval-when-compile (require 'cl-lib)) ; for cl-case function
(require 'mailcap)
;(require 'mm)
(require 'mm-view)
;(require 'mail-parse)
(require 'gnus)
(require 'edebug)
;(require 'mrg) ; comment out unless doing na-autolink
(unless (featurep 'xemacs)
(fset 'yes-or-no-p-dialog-box 'y-or-n-p)
(fset 'first 'cl-first)
(fset 'second 'cl-second)
(fset 'third 'cl-third)
(fset 'fourth 'cl-fourth)
(defalias 'exec-to-string 'shell-command-to-string))
;;
;; might need to add this back in for fsf port
;;
;; (dolist (event-type '(mouse-1 mouse-2 mouse-3
;; M-down-mouse-1 M-down-mouse-2 M-down-mouse-3))
;; (put event-type 'event-kind 'mouse-click)))
(mailcap-add-mailcap-entry
"application" "x-empty"
'((viewer . fundamental-mode)
(test . (fboundp 'fundamental-mode))
(type . "application/x-empty")))
(mailcap-add-mailcap-entry
"application" "dvi"
'((viewer . "kdvi %s")
(test . (eq (mm-device-type) 'x))
("needsx11")
(type . "application/dvi")))
(setq na-leaves '((dummy :pool ""))) ; where the leaf widgets get pushed so we know if one has been
; selected (because menu behavior changes)
(setq na-angles nil) ; the base of the widget angles tree
(setq na-need-br-rebuild nil) ; where widgets that need branches rebuilt get's pushed
(defgroup na-documentation nil "")
(defcustom na-edit-mode t
"True if desirous to edit leaf contents where-ever possible"
:group 'na
:type 'boolean)
(defcustom na-base-directory "/home/nadb/"
"Root dir of na sparse array objects"
:group 'na
:type '(directory))
(defcustom na-control-directory (concat na-base-directory "/.control")
"Root dir of na sparse array objects"
:group 'na
:type '(directory))
(defcustom na-root-tag " /"
"The default tag for each angle's root"
; Might want this to reflect which angle number later (e.g. a format?)
:group 'na
:type '(text))
(defcustom na-new-angle-tag "New Angle"
"The tag that get's displayed in the menu for the root"
:group 'na
:type '(text))
(fset 'na-message 'message);'yes-or-no-p-dialog-box)
(defun na-date (base)
"Return string representing link for format year/month/day"
(let ((list (split-string value "[, ]+")))
(concat base
(mapconcat 'identity
(list (third list) (first list) (second list)) "/"))))
(defun na-csz (base)
"Return string representing link for formats '(zip state/city)"
)
(defun na-rebuild-brmenus (&optional candidate)
"Rebuild the given widget's branch menu or all the menus flagged as
helneeding rebuilding by na-branch-delete if none given. In either case,
remove the rebuilt menu(s) from the list of ones flagged.
This is factored out because when moving links, first you start with a delete,
then a create and sometimes those may effect the same branch, however that branch
menu only needs to be rebuilt once. Not a common occurrance but possible."
(let (widget)
(if candidate
(progn
(when (memq candidate na-need-br-rebuild)
(setq na-need-br-rebuild
(delq candidate na-need-br-rebuild)))
(widget-put candidate :args (na-branch-arguments candidate)))
(while (setq widget (pop na-need-br-rebuild))
(widget-put widget :args (na-branch-arguments widget))))))
; Should probably do widget-setup after this and na-rebuild-brmenus
(defun na-pool-delete (inode &optional nextwidget inodeorig)
"remove inode(s) from nextwidget and its parents if that file does not
exist in their directories.
If nextwidget is not given, the deletion starts from the last widget
in the angles stack: (na-lastnode)"
(let* ((nextwidget (or nextwidget (na-lastnode))) ;if arg not given, start at end of stack
(inodeorig (or inodeorig inode)) ; preserve the original inode list if first entering
(parent (widget-get nextwidget :parent))
(child (car (widget-get nextwidget :children)))
(cname (and child (widget-get child :tag)))
default-directory wpoolorg wpool regexp apool children buttons leaves)
; moving the following statment up nearer the top of this function would make it quicker
(if (eq (car nextwidget) 'leaf) ;skip leaves
(na-pool-delete inode parent inodeorig)
(setq default-directory (widget-get nextwidget :path) ; set it to nextwidget's
wpoolorg (widget-get nextwidget :pool) ; get the widget's original pool
apool
(na-shell-read ; pool of inodes actually in this directory
"echo -n \\\";find * -maxdepth 0 -type f -printf '%i\n' 2>/dev/null |sort|grep -v '^$'|uniq; echo \\\"")
inode (na-isubtract apool inode))
;;; handle deletion before flagging others for branch menu rebuilding
;;; so leaves don't get flagged for menu rebuilding. -- still doesn't work
(when (and cname
(not (equal cname na-root-tag))
(not (file-exists-p cname))) ; delete widget from screen
(setq children (widget-get child :children)
buttons (widget-get child :buttons)) ; preserve grandchildren
(widget-put child :children nil) ; blast link to them
(widget-put child :buttons nil)
(widget-apply nextwidget :value-delete) ; delete child from screen
(widget-put nextwidget :children children) ; link around child
(widget-put nextwidget :buttons buttons)
(widget-put (car children) :parent nextwidget)
(cl-pushnew nextwidget na-need-br-rebuild)) ; flag this one for menu rebuild
(unless (eq 0 ; any branch containing this inode needs menu rebuilding
(length ; because it could have existed more than once here
(na-iinboth apool inodeorig)))
(cl-pushnew nextwidget na-need-br-rebuild))
(when (string-match ".+" inode) ; if there are any left to delete
(setq wpool (na-subtract wpoolorg inode)); eliminate from widget's pool ones actually found in this directory
(widget-put nextwidget :pool wpool)
(unless (eq (length wpool) (length wpoolorg)); if change, flag parent rebuid branch menu
(cl-pushnew nextwidget na-need-br-rebuild)))
(when parent; not past top of 1st angle
(na-pool-delete inode parent inodeorig)))))
(defun na-pool-add (inode &optional nextwidget)
"add inode(s) to nextwidget and its children if that/those file
exists (i.e. have been added) in their directories. Also update
branch menus for any branch wherein inode(s) is(are) found.
If nextwidget is not given, the addition starts from the first widget
in the angles stack: na-angles"
; this is a convoluted mess and needs to be refactored, but hey, it works, mostly :-)
(catch 'done
(let* ((nextwidget (or nextwidget na-angles)) ;if arg not given, start at beginning of stack
; pool of inodes actually in widget's directory
args ; steps through args to find child's new pool and set it
default-directory ;preserve current directory
(child (car (widget-get nextwidget :children))) ; set directory to this widgets
cname wpool apool dpool tmp)
(when (eq (car nextwidget) 'leaf)
(na-pool-add inode child)
(throw 'done nil))
(setq cname (widget-get child :tag)
wpool (widget-get nextwidget :pool)
default-directory (widget-get nextwidget :path)
apool (na-shell-read ; pool of inodes actually in this directory
"echo -n \\\";find * -maxdepth 0 -type f -printf '%i\n' 2>/dev/null|sort|grep -v '^$'|uniq; echo \\\"")
dpool (na-shell-read ; pool of inodes representing directories
"echo -n \\\";find * -maxdepth 0 -type d -printf '%i\n' 2>/dev/null|sort|grep -v '^$'|uniq; echo \\\""))
(when (eq nextwidget na-angles) ; no nextwidget argument was given (i.e. first time through)
(setq tmp ;tmp=inode w/directory ones removed
(na-isubtract inode dpool))
(widget-put nextwidget :pool
(setq wpool (na-iadd inode wpool))))
; (edebug)
(when (string-match ; if some of the inodes still in nextwidget's pool
".+" (na-iinboth inode wpool)) ; means they have not been made "others" by elimination yet
(na-rebuild-brmenus nextwidget) ;rebuild it's branch menu (and remove from rebuilds list)
(setq args (widget-get nextwidget :args))
(if (widget-get child :path) ;nextwidget has children and they are "real"
;figure out which argument is child and set child's pool to that
; argument's pool (thats just been updated with na-branch-arguments)
(while args ; some day need to see if setting child's pool to :choice's pool would work
(when (or (equal (widget-get (car args) :tag) cname) ; to speed this up (eliminate looping)
(and (equal cname na-root-tag)
(equal (widget-get (car args) :tag) na-new-angle-tag)))
(widget-put child :pool (widget-get (car args) :pool))
(setq args nil))
(setq args (cdr args)))
(throw 'done nil))) ; otherwise bail out
; (edebug)
(when (or
(string-match ; some inodes exist in current directory
".+" (na-iinboth apool inode))
; this widget on list of ones needing rebuild
(memq nextwidget na-need-br-rebuild))
(na-rebuild-brmenus nextwidget))
(unless (widget-get child :path) ; bail if no actual child
(throw 'done nil))
; (edebug)
(na-pool-add inode child)))) ; recurse in on 'child' (next node in stack)
(defsubst na-default (type keyword)
"Get the value of the keyword from the type this widget is derived from, e.g. the 'default'
value if this widget did not override the parent"
(widget-get (get (widget-type (get type 'widget-type)) 'widget-type) keyword))
(defun na-mime-method (name)
(mailcap-mime-info
(car
(mail-header-parse-content-type
(na-shell-string-read
(format "%s; name=\"%s\""
(format "file -i \"%s\" 2> /dev/null | cut -d ':' -f2- | tr -d ' \n'" name) name))))))
(defun na-mime-handle (name)
(mail-header-parse-content-type
(format "%s; name=\"%s\""
(na-shell-string-read
(format "file -i \"%s\" 2> /dev/null | cut -d ':' -f2- | tr -d ' \n'" name))
name)))
(define-widget 'leaf-edit 'choice-item ; was 'leaf
"The widget that actually edits the file"
)
(define-widget 'leaf 'menu-choice
"the widget representing the actual file; the 'terminal node'"
:format "%[%t%]%v"
:void '(item :format "")
:mouse-down-action 'na-leaf-press-action
:value-create 'na-leaf-create
:value-delete 'na-leaf-delete
:create 'na-node-create
:value-set (lambda (widget value)
(if (eq (widget-type (widget-get widget :explicit-choice))
'leaf-edit)
(let ((name (widget-get widget :tag)))
(setq foo (start-process "view" "*scratch*" "xdg-open" name )))
; (eshell-command (concat "xdg-open " "\"" name "\" &") "*scratch*"))
;There appears to be junk left in ~/tmp -- needs to be del'd
(funcall (na-default 'leaf :value-set) widget value))))
(defun na-leaf-delete (widget)
"Remove widget from the na-leaves before completing the delete"
(message "in na-leaf-delete")
(funcall (na-default 'leaf :value-delete) widget)
;; now remove all deleted inodes from na-leaves
(let ((leaves na-leaves))
(while leaves
(when ; widget's name and path is same as head of na-leaves
(and (equal (widget-get widget :tag) (widget-get (car leaves) :tag))
(equal (widget-get widget :path) (widget-get (car leaves) :path)))
(setq na-leaves (delq (car leaves) na-leaves))) ; remove it from na-leaves
(setq leaves (cdr leaves))))) ; step through
(defun na-leaf-create (widget)
"Needed to be modified slightly to make sure the path is set correctly"
(let* ((tag (widget-get widget :tag))
(pool (na-inodes tag)))
(widget-put widget :pool pool) ; this has to be intelligent. If the pool
(widget-put widget :path default-directory) ; buggy line?
(push widget na-leaves)
(widget-put widget :args
(list
(widget-convert `(angle
:tag "New Angle"
:pool ,pool
:args ,(na-branch-arguments widget)))
(widget-convert `(leaf-edit
:tag ,tag))))
(funcall (na-default 'leaf :value-create) widget))) ; during value-create
;;
;; added for the fsf port of the following function
;;
(if (not (fboundp 'event-button))
(defun event-button (event)
(let ((x (symbol-name (event-basic-type event))))
(if (not (string-match "^mouse-\\([0-9]+\\)" x))
(error "Not a button event: %S" event))
(string-to-number (substring x (match-beginning 1) (match-end 1))))))
(defun na-leaf-press-action (widget &optional event)
"The function that gets called when you press on a leaf"
(setq na-location widget)
(let ((button (event-button event))
(selected (widget-value na-selection)))
(setq default-directory (widget-get widget :path)
na-location widget)
(cl-case button
(2
(funcall (widget-choose
"Leaf Command Menu"
(append
'(("Remove this Link" . na-remove-link))
'(("Remove all Occurances" . na-purge-leaf))
'(("Set as Selection" . na-set-selection))
'(("Rename Leaf" . na-rename-leaf))
'(("Edit All Angles" . na-all-angles))
'(("Display all Angles" . na-all-angles-display))
'(("Demote Node" . na-demote-node))
(unless (equal "" selected)
(if (string-match "/$" selected)
(list
'("Link to Selection" . na-link-select)
'("Copy to Selection" . na-copy-select)
'("Move to Selection" . na-move-select))
'(("Duplicate Links to Selected" . na-dup-links)))))
event)))
(3
(funcall (na-default 'leaf :mouse-down-action) widget event)))))
(defun na-dup-links ()
"Link current leaf to all places selected leaf is linked."
; first get a list of all the branchnames selected has
(let ((default-directory "/") ; relying on dynamic scoping -- is this OK?
(selected (widget-value na-selection))
(existing (concat (widget-get widget :path)
(widget-get widget :tag))))
(na-error
(na-shell-string-read
(mapconcat ; map a link of the existing file to all the directories that selected exists in
(lambda (x)
(format
"ln \"%s\" \"%s\""
existing (concat na-base-directory
(substring x 0 (string-match "[^/]+$" x)))))
(let ((default-directory na-base-directory)) ; create a local default-directory.
; depend on dynamic scoping so na-shell-read
; will get it rather than the global version
(na-shell-read
(format
"echo -n \\(; find . -inum %i -printf '\"%%p\" '; echo \\)"
(string-to-number
(na-inodes (substring selected 1)))))) ";" )))))
(defun na-demote-node ()
"Turn this leaf into a branch with itself as its first contents"
(let* ((name (widget-get widget :tag))
(newname (read-from-minibuffer "Enter new link name: "
(widget-get widget :tag)))
(split (split-string newname "/"))
(node (nth (1- (length split)) split))) ; what was I thinking here?
(unless (eq "" newname)
(na-error
(na-shell-string-read
(concat
"tmpfile=na.demoted.$$;"
"mv \""
name
"\" $tmpfile &&"
"mkdir \"" name
"\" 2>/dev/null&&"
"mv $tmpfile \"" name "\"/\"" newname "\"")))
(na-refresh-screen widget))))
(defun na-rename-leaf ()
"Rename the selected leaf"
(let* ((name (widget-get widget :tag))
(newname (read-from-minibuffer "Enter new link name: "
(widget-get widget :tag)))
(split (split-string newname "/"))
(node (nth (1- (length split)) split))) ;; what was I thinking here?
(unless (eq "" newname)
(na-error
(na-shell-string-read
(concat "mv \"" name "\" \""
(when (eq 0 (string-match "/" newname))
na-base-directory) newname
"\"")))
(na-refresh-screen widget))))
(defun na-link-select ()
"Link the selected leaf to the selection"
(let* ((name (widget-get widget :tag))
(branchname (widget-value na-selection))
(branchsplit (split-string branchname "/"))
(branchnode (nth (1- (length branchsplit)) branchsplit))
(newname (read-from-minibuffer "Enter new link name: " name)))
(unless (eq "" branchname)
(na-error
(na-shell-string-read
(concat "ln \"" name "\" \""
(concat na-base-directory branchname
newname "\""))))
(na-pool-add branchnode)
(na-rebuild-brmenus)
(na-update-all-named branchnode)
(na-update-all-named branchname))))
(defun na-copy-select ()
"Copy the selected leaf to the selection"
(let* ((name (widget-get widget :tag))
(branchname (widget-value na-selection))
(branchsplit (split-string branchname "/"))
(branchnode (nth (1- (length branchsplit)) branchsplit))
(newname (read-from-minibuffer "Enter new leaf name: "
(widget-get widget :tag))))
(unless (eq "" branchname)
(na-error
(na-shell-string-read
(concat "cp \"" name "\" \""
na-base-directory branchname
newname "\"")))
(na-pool-add branchnode)
(na-rebuild-brmenus)
(na-update-all-named branchnode))))
(defun na-move-select ()
"move the selected leaf to the selection"
(let* ((name (widget-get widget :tag))
(branchname (widget-value na-selection))
(branchsplit (split-string branchname "/"))
(branchnode (nth (1- (length branchsplit)) branchsplit))
(newname (read-from-minibuffer "Enter new leaf name: " name)))
(unless (eq "" branchname)
(na-error
(na-shell-string-read
(concat "mv \"" name "\" \""
na-base-directory branchname
newname "\"")))
(na-pool-add branchnode)
(na-rebuild-brmenus)
(na-update-all-named branchnode))))
(defun na-purge-leaf ()
"Remove the selected leaf entirely"
(let* (temp
(name (widget-get widget :tag))
(inode (widget-get widget :pool))
(parent (widget-get widget :parent))
(parents (mapcar (lambda (x) ;get list of parents tag names
(file-name-as-directory
(file-name-nondirectory
(substring x 1 (1- (length x))))))
(mapcar 'file-name-directory (na-allnames-unsplit inode)))))
(when (yes-or-no-p-dialog-box
(format "Purge %s entirely?" name))
(setq temp default-directory ;(delete-file) works relative to default-directory
default-directory na-base-directory) ;however we are handing files relative to
(mapcar 'delete-file (na-allnames-unsplit inode)) ; na-base-directory
(setq default-directory temp) ;so we preserve it in temp
(funcall (widget-get parent :value-delete) parent)
(na-pool-delete inode)
(na-rebuild-brmenus)
(na-update-all-named parents))))
(defun na-remove-link ()
"Delete the selected link"
(let* ((name (widget-get widget :tag))
(inode (widget-get widget :pool))
(parent (widget-get widget :parent))
(parent-name (widget-get parent :tag)))
(when (yes-or-no-p-dialog-box (format "Delete link %s?" name))
(na-error (na-shell-string-read
(concat "rm -f \"" name "\"")))
; (funcall (widget-get parent :value-delete) parent)
(na-pool-delete inode)
(na-rebuild-brmenus)
(widget-setup))))
(defun na-leaf-release-action (widget &optional event)
"What get's called when you release the button on a leaf"
(setq default-directory (widget-get widget :path))
(funcall (na-default 'leaf :action) widget event))
(defun na-all-angles ()
"Edit all angles for this widget's inode"
(let ((inode (na-inodes (widget-get widget :tag)))
(pool (widget-get na-angles :pool))
(leaves na-leaves) ; save a copy of leaves stack
newangles leaves2)
(goto-char (point-max))
(setq na-leaves '((dummy :pool ""))) ; reset stack to empty
(setq newangles
(widget-create
(na-all-angles-list
(na-allnames inode) na-base-directory
(na-inodes na-base-directory)))
leaves2 na-leaves ; save the new leaves stack
na-leaves leaves) ; restore the old one for the upcoming delete
(widget-delete na-angles)
(setq na-angles newangles
na-leaves leaves2)
(widget-setup)))
(defun na-all-angles-display ()
"Show a read only popup dialog displaying all angles for this widget's inode"
(let ((inode (widget-get (car na-leaves) :pool)))
(make-dialog-box
'question :modal nil :title "Display All Angles"
:question
(na-shell-string-read
(format "(cd \"%s\";find * -inum %s -printf \"/%%p\n\n\" 2>/dev/null)"
na-base-directory (substring inode 0 (1- (length inode)))))
:buttons '(["Dismiss" '() t]))))
;; ; a directory is just a node (choice-item) when it is a part
;; ; of a menu (of a parent)
(define-widget 'branch 'menu-choice
"A widget representing basically, a directory"
:format "%[%t%]%v"
; :validate (lambda (x) t)
:validate (lambda (x) ; invalid to select an 'other' leaf
; (edebug)
(if (eq (widget-type x) 'node)
(let (inode)
(or
(equal ""
(setq inode ; a leaf inode exists?
(widget-get
(car na-leaves) :pool)))
(equal inode (widget-get x :pool)))) ; pool = inode
t))
:mouse-down-action 'na-branch-press-action
:action 'na-branch-action
:void '(item :format ""))
;; truth table for following function (xor)
;; eventp button!=3 action
;; -----------------------
;; 0 0 1
;; 0 1 1
;; 1 0 1
;; 1 1 0
(defun na-branch-action (widget &optional event)
"The function that get's called when you click on a branch"
(when (or (not (eventp event)) (eq (event-button event) 3))
(funcall (na-default 'branch :action) widget event)))
(defun na-branch-press-action (widget &optional event)
"What get's called when you press on a branch"
(let ((button (event-button event))
(path (widget-get widget :path)))
(setq default-directory (or path default-directory)
na-location widget)
(cl-case button
(2
(funcall (widget-choose
"Branch Command Menu"
(append
(if (cdr na-leaves) ; any actively selected leaves?
'(("Add a Link" . na-add-link))
'(("Create New Leaf" . na-new-leaf)
("Create BookMark" . na-bookmark)))
'(("Rename Branch" . na-rename-branch)) ; unless a root?
'(("Set as Selection" . na-set-selection))
'(("Add New Subbranch" . na-add-branch))
'(("Prune Branch" . na-prune-branch))
(unless (equal "" (widget-value na-selection))
(list
'("Link Contents to Selection" . na-branch-link)
'("Copy Contents to Selection" . na-branch-copy)
'("Move Contents to Selection" . na-branch-move))))
event)))
(3
(funcall (na-default 'branch :mouse-down-action) widget event)))))
(defun na-branch-move ()
"Move this branch and all the contents into the selection"
(let* ((name (widget-get widget :tag))
(branchname (widget-value na-selection))
(branchsplit (split-string branchname "/"))
(branchnode (nth (1- (length branchsplit)) branchsplit)) ; variable not used?
(newname (read-from-minibuffer "Enter new branch name: " name)))
(unless (eq "" branchname)
(na-error
(na-shell-string-read
(concat "mv . \"" na-base-directory branchname newname
"\""))))
(na-refresh-screen widget)))
(defun na-branch-copy ()
"Link all the contents of this branch into the selection"
(let* ((name (widget-get widget :tag))
(branchname (widget-value na-selection))
(branchsplit (split-string branchname "/"))
; (branchnode (nth (1- (length branchsplit)) branchsplit))
(newname (read-from-minibuffer "Enter new branch name: " name)))
(unless (eq "" branchname)
(na-error
(na-shell-string-read
(concat "find . -print | cpio -padm --quiet \""
na-base-directory branchname newname
"\"")))
(na-refresh-screen widget))))
(defun na-branch-link ()
"Link all the contents of this branch into the selection"
(let* ((name (widget-get widget :tag))
(branchname (widget-value na-selection))
(branchsplit (split-string branchname "/"))
(branchnode (nth (- (length branchsplit) 2) branchsplit))
(newname (read-from-minibuffer "Enter new branch name: " name)))
(unless (eq "" branchname)
(na-error
(na-shell-string-read
(concat "find . -print | cpio -padlm --quiet \""
na-base-directory branchname newname
"\"")))
(na-update-all-named (concat branchnode "/")))))
(defun na-rename-branch ()
"This gets called from a branch menu to rename the branch"
(let* ((tag (widget-get widget :tag))
(newname (read-from-minibuffer (format "Rename %s to: " tag tag)))
(result (when newname
(na-shell-string-read
(concat "cd ..;mv \"" tag "\" \""
newname "\"")))))
(na-error result)
(na-refresh-screen widget)))
(defun na-set-selection ()
"Set this branch as the selection for future copys or moves"
(widget-value-set
na-selection
(na-path na-location))
(widget-setup))
(defun na-path (widget)
"Return the string that represents the path to the given widget"
(if (eq 'root (widget-type widget))
"/"
(concat
(na-path
(widget-get widget :parent))
(widget-get widget :tag))))
(defun na-add-branch ()
"Create a new sub-branch under the current branch"
(let* ((name (read-from-minibuffer "Enter new sub-branch name: "))
(result
(na-shell-string-read
(concat "mkdir \"" name "\";"))))
(na-error result)
(na-update-all-named (widget-get widget :tag))
(widget-put widget :value (concat name "/"))))
(defun na-prune-branch ()
"Purge this branch and its subnodes"
(let ((name (widget-get widget :tag))
(parent (widget-get widget :parent))
(inode (widget-get widget :pool))
result)
(when (yes-or-no-p-dialog-box
(format "Delete branch %s and its contents?" name))
(na-error (na-shell-string-read
(concat "(cd ..; rm -fr \"" name
"\")")))
(funcall (widget-get parent :value-delete) parent)
(setq default-directory (widget-get parent :path))
(na-pool-delete (widget-get widget :pool))
(na-rebuild-brmenus)
(na-update-all-named
(widget-get parent :tag)))))
;; ; this function originally had the following line right after 'echo' below
;; ;<!DOCTYPE HTML PUBLIC \"-//W3C//DTD HTML 4.0 Transitional//EN\">
(defun na-bookmark ()
"Create a new bookmark under the current branch"
(let* ((newname (read-from-minibuffer "Enter bookmark name: "))
(newurl (read-from-minibuffer "Enter URL: " "http://www."))
(result (when newname
(na-shell-string-read
(concat "test ! -e \"" newname
"\" && echo '<html>
<meta http-equiv=\"REFRESH\" content=\"0;url="
newurl "\"><html>' > \""
newname
"\" || echo \"Error: File \'\"" newname
"\"\' exists\! \""))))
(tmp widget)
(inode (na-inodes newname)))
(na-error result)
(na-pool-add inode)
(na-rebuild-brmenus)
(widget-setup)))
(defun na-new-leaf ()
"Create a new leaf under the current branch"
(let* ((newname (read-from-minibuffer "Enter new leaf name: "))
(result (when newname
(na-shell-string-read
(concat "test ! -e \"" newname "\" && > \""
newname
"\" || echo \"Error: File '\"" newname
"\"' exists\! \""))))
(tmp widget)
(inode (na-inodes newname)))
(na-error result)
(na-pool-add inode)
(na-rebuild-brmenus)
(widget-setup)))
(defun na-add-link ()
"Create a new link for the given widget leaves (should all be same inode)"
(let* ((leaf (car na-leaves))
(path (widget-get leaf :path))
(name (widget-get leaf :tag))
(branchname (widget-get widget :tag))
(newname (read-from-minibuffer "Enter new link name: "
(widget-get leaf :tag)))
(result
(na-shell-string-read
(concat "ln \"" (concat path name) "\" \""
newname "\"")))
(tmp widget)
(inode (na-inodes newname)))
(na-error result)
(na-pool-add inode)
(na-rebuild-brmenus)
(widget-setup))) ; new link will only show up on branch menus so
; this is the proper refresh
(defun na-refresh-screen (widget &optional event)
"update all node's menus stopping at any change"
(goto-char (point-max)) ; put new tree at end
(let* ((source na-angles)
(newangles
(apply 'widget-create 'angle :path na-base-directory
:pool (na-inodes na-base-directory)
'((item ""))))
(select newangles)
(sourcechild (car (widget-get source :children)))
(leaves na-leaves) ; save original na-leaves
leaves2 path type pool tag choice args)
(setq na-leaves '((dummy :pool ""))) ; reinitialize na-leaves
(catch 'changed
(while (car (widget-get sourcechild :children))
(setq default-directory (setq path (widget-get select :path))
pool (widget-get select :pool); Isn't this auto-generated somewhere?
type (widget-type sourcechild)
tag (widget-get sourcechild :tag)
tag (if (equal tag na-root-tag)
na-new-angle-tag
tag))
(unless (equal tag na-new-angle-tag)
(unless
(directory-files ; file/directory still exists
path
nil ; not full names
(concat "^"
(car (split-string tag "/"))
"$") ; match regexp
t ; nosort
nil)
(throw 'changed t)))
(setq choice
(let ((args (widget-get source :args))
current found)
(while (and args (not found))
(setq current (car args)
args (cdr args)
found
(equal tag
(widget-get
current :tag))))
current))
(widget-put select :explicit-choice choice)
(widget-put select :value tag)
(widget-value-set select tag)
(setq source (car (widget-get source :children))
sourcechild (car (widget-get source :children))
select (car (widget-get select :children)))))
(setq leaves2 na-leaves
na-leaves leaves)
(widget-delete na-angles)
(setq na-leaves leaves2)
(setq na-angles newangles)
(widget-setup)))
(defun na-update-all-named (name)
"update all the widget's menus where the widgets are tagged with 'name'"
(let ((child na-angles) ;handles either a string naming the parent directory, or a list of strings
(names (if (stringp name) name
(apply 'concat name)))
cname)
(while (setq cname (widget-get child :tag))
(when (string-match cname names) ; when a value is getting deleted, it should be deleted here
; as well to update the display properly.
(widget-put child :args (na-branch-arguments child)))
(setq child (car (widget-get child :children))))))
(defsubst na-do-args (dirs files)
"Return a list of the node and leaf arguments from the given list of names
and inode strings"
(let (inode)
(append
(mapcar
(lambda (x)
`(node :tag ,(concat (first x) "/")
:path ,(concat default-directory (first x) "/")
:value ,(first x)
:pool ,(second x)
:args ((choice-item ""))))
dirs)
(mapcar (lambda (x)
`(leaf :tag ,(first x)
:path ,default-directory
:value ,(first x)
:pool ,(second x)
:inactive ,(equal (second x) "")
:args ((angle
:tag "New Angle"
:format "\n\n %[%t%]%v" ; optional?
:path ,na-base-directory
:pool ,(second x)
:value "New Angle"
:args ((choice-item ""))))
:args ((choice-item ""))))
files))))
(defsubst na-branch-arguments (widget)
"Return a properly formatted :arg list for the given widget"
(let* ((pool (widget-get widget :pool))
(foo (setq default-directory (widget-get widget :path)))
(data (na-shell-read
(concat "echo -n \"" pool "\" | na.members.sh"))))
(append
`((angle :tag "New Angle"
:format "\n\n %[%t%]%v"
:path ,na-base-directory
:pool ,pool
:value "New Angle"))
; (choice-item "")))
'((choice-item :value "-----------------"))
; '((choice-item :value "--single-line"))
(na-do-args (first data) (third data)) ;'member' branches and leaves
'((choice-item :value "-----------------"))
; '((choice-item :value "--single-line"))
(na-do-args (second data) (fourth data))))) ;'other' branches and leaves
(define-widget 'node 'branch ; the inherited widget has to do with how
;to interpret the args when creating this widget
"a meta-type representing either a leaf or a branch"
:create 'na-node-create)
(defun na-node-create (widget)
"Create a node by turning it into a leaf or a branch w/appropriate changes"
(let ((path (widget-get widget :path))
(orig widget)
; when working normal browsing, the following is not necessary, only when
; showing-all-angles! Why? How do I correct for it?
(pool (or (widget-get widget :pool)
(widget-get (widget-get widget :parent) :pool)))
(type (widget-type widget)))
(cond ((or (eq type 'node) (eq type 'angle))
; the documentation is not true when it says that the :create function returns
; the widget. it returns nil! See how (widget-create) works.
(setq default-directory
(or path (concat default-directory (widget-get widget :tag))))
(when (eq type 'angle)
(widget-put widget :tag na-root-tag)) ; necessary?
; need to add this widget to its own pool and set it's parent
; pool accordingly; Actually, need to add it to all the parent
; pools.
(when pool ; probably the wrong stratgey for setting the pool
(widget-put widget :pool pool))
(widget-put widget :args (na-branch-arguments widget))
(setcdr orig (cdr (widget-convert widget)))
(setcar orig
(or
(or
(and (eq type 'angle) 'root)
(and (eq type 'node) 'branch))
type))))
(funcall (na-default (widget-type widget) :create) orig)))
(define-widget 'angle 'branch
"This is the meta-widget that create is called on. It get's turned into a 'root
when it is initialized"
:tag na-root-tag ; would like to change this into "angle # 1:" etc...
:format "\n\n %[%t%]"
:create 'na-node-create ;this may change
:path na-base-directory
)
(define-widget 'root 'branch
"What an 'angle gets turned into once it is initialized. The corallary to 'branch
in the node/branch situation"
:path na-base-directory
:format "\n\n %[%t%]%v"
)
(define-widget 'either-or 'radio-button
"Similar to radio buttons but both buttons cannot be 'off'"
:action (lambda (widget)
(if (or (widget-get)))))
(defgroup na nil
"N-Angulator editor and support library"
:link '(custom-manual "(nd)Top")
:link '(url-link :tag "Development Page"
"http://www.N-Angulator.org")
:prefix "widget-"
:group 'extensions
:group 'hypermedia)
(defun na-anglepath (widget) ;untested
"Return the string representing the path to this widget"
(let* ((parent (widget-get widget :parent)))
(if parent
(mapconcat 'na-anglepath (list parent
(widget-value widget)) "/")
na-base-directory)))
(defun na-shell-read (command)
"Execute shell 'command' in default-directory then 'read' in and
return the results"
(read
(setq after (exec-to-string
(setq before command)))))
(defun na-shell-string-read (command)
"Execute shell 'command' in default-directory then 'read' in and
return the results"
(read (concat "\""
(setq after (exec-to-string
(setq before command))) "\"")))
(defun na-inodes (file)
"Return a quote delimited multi-line string of the inodes under or representing
file/directory or if file, return its inode"
(setq after (exec-to-string
(setq before (format
"find \"%s\" -type f -printf \"%%i\n\" 2>/dev/null|sort -u"
; "(cd \"%s\" ;find \"%s\" -printf \"%%i\n\" 2>/dev/null|sort -u )" ; make it include directories
file)))))
(defun na-isubtract (a b)
"subtract the inodes in string b from the inodes in string b"
(let* ((regexp (dired-string-replace-match ; build "or" regular expressions out of inodes
"..$"
(concat "^" (dired-string-replace-match "\n" b "$|^" nil t))
"" nil t)))
(na-shell-read ; strip out inodes
(format "echo -n \\\";echo -n \"%s\"|grep -v \"%s\"|echo \\\""
a regexp))))