blob: 9386917445f89fb558c1ec74b240d0bd60de2e63 (
plain)
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
|
# plugin to allow searching all the documentation using a regexp
# check the Help menu for the Search item to use it
package require Tk 8.4
package require pd_bindings
package require pd_menucommands
namespace eval ::dialog_search:: {
variable searchtext {}
variable selected_file {}
variable selected_basedir {}
variable basedir_list {}
}
# TODO filter out x,y numbers and #X, highlight obj/text/array/etc
# TODO show the busy state somehow, grey out entry?
# TODO add implied '^' to searches that don't contain regexp [a-zA-Z0-9 _-]
# find_doc_files
# basedir - the directory to start looking in
proc ::dialog_search::find_doc_files { basedir } {
# Fix the directory name, this ensures the directory name is in the
# native format for the platform and contains a final directory seperator
set basedir [string trimright [file join $basedir { }]]
set fileList {}
# Look in the current directory for matching files, -type {f r}
# means ony readable normal files are looked at, -nocomplain stops
# an error being thrown if the returned list is empty
foreach fileName [glob -nocomplain -type {f r} -path $basedir *.txt *.pd] {
lappend fileList $fileName
}
# Now look for any sub direcories in the current directory
foreach dirName [glob -nocomplain -type {d r} -path $basedir *] {
# Recusively call the routine on the sub directory and append any
# new files to the results
set subDirList [find_doc_files $dirName]
if { [llength $subDirList] > 0 } {
foreach subDirFile $subDirList {
lappend fileList $subDirFile
}
}
}
return $fileList
}
proc ::dialog_search::selectline {listingwidget} {
variable selected_file
variable selected_basedir
variable basedir_list
set selection [$listingwidget curselection]
if {$selection eq ""} {
set selected_file ""
} else {
set line [$listingwidget get $selection]
set selected_file [string replace $line [string first ":" $line] end]
set selected_basedir [lindex $basedir_list $selection]
}
}
proc ::dialog_search::open_line {} {
variable selected_file
variable selected_basedir
if {$selected_file ne ""} {
menu_doc_open $selected_basedir $selected_file
}
}
proc ::dialog_search::readfile {filename} {
set fp [open $filename]
set file_contents [split [read $fp] \n]
close $fp
return $file_contents
}
proc ::dialog_search::search {searchtext} {
variable basedir_list {}
if {$searchtext eq ""} return
.search.searchtextentry selection clear
set widget .search.resultslistbox
$widget delete 0 end
foreach basedir [concat [file join $::sys_libdir doc] $::sys_searchpath $::sys_staticpath] {
# Fix the directory name, this ensures the directory name is in the
# native format for the platform and contains a final directory seperator
set basedir [file normalize $basedir]
foreach docfile [find_doc_files $basedir] {
searchfile $searchtext [readfile $docfile] $widget \
[string replace $docfile 0 [string length $basedir]] $basedir
}
}
}
proc ::dialog_search::searchfile {searchtext file_contents widget filename basedir} {
variable basedir_list
set n 0
foreach line $file_contents {
if {[regexp -nocase -- $searchtext $line]} {
$widget insert end "$filename: $line"
lappend basedir_list $basedir
incr n
}
}
}
proc ::dialog_search::ok {mytoplevel} {
# this is a placeholder for the standard dialog bindings
}
proc ::dialog_search::cancel {mytoplevel} {
wm withdraw .search
}
proc ::dialog_search::open_search_dialog {mytoplevel} {
if {[winfo exists $mytoplevel]} {
wm deiconify $mytoplevel
raise $mytoplevel
} else {
create_dialog $mytoplevel
}
}
proc ::dialog_search::create_dialog {mytoplevel} {
variable selected_file
toplevel $mytoplevel
wm title $mytoplevel [_ "Search"]
entry $mytoplevel.searchtextentry -bg white -textvar ::dialog_search::searchtext \
-highlightcolor blue -font 18 -borderwidth 1
bind $mytoplevel.searchtextentry <Return> "$mytoplevel.searchbutton invoke"
# TODO add history like in the find box
bind $mytoplevel.searchtextentry <Up> {set ::dialog_search::searchtext ""}
bind $mytoplevel.searchtextentry <$::modifier-Key-a> \
"$mytoplevel.searchtextentry selection range 0 end"
listbox $mytoplevel.resultslistbox -yscrollcommand "$mytoplevel.yscrollbar set" \
-bg white -highlightcolor blue -height 30 -width 80
scrollbar $mytoplevel.yscrollbar -command "$mytoplevel.resultslistbox yview" \
-takefocus 0
button $mytoplevel.searchbutton -text [_ "Search"] -takefocus 0 \
-command {::dialog_search::search $::dialog_search::searchtext}
grid $mytoplevel.searchtextentry $mytoplevel.searchbutton - -sticky ew
grid $mytoplevel.resultslistbox - $mytoplevel.yscrollbar -sticky news
grid columnconfigure $mytoplevel 0 -weight 1
grid rowconfigure $mytoplevel 1 -weight 1
bind $mytoplevel.resultslistbox <<ListboxSelect>> \
"::dialog_search::selectline $mytoplevel.resultslistbox"
bind $mytoplevel.resultslistbox <Key-Return> ::dialog_search::open_line
bind $mytoplevel.resultslistbox <Double-ButtonRelease-1> ::dialog_search::open_line
::pd_bindings::dialog_bindings $mytoplevel "search"
focus $mytoplevel.searchtextentry
}
# create the menu item on load
set mymenu .menubar.help
set inserthere [$mymenu index [_ "Report a bug"]]
$mymenu insert $inserthere separator
$mymenu insert $inserthere command -label [_ "Search"] \
-command {::dialog_search::open_search_dialog .search}
::dialog_search::open_search_dialog .search
|