no message
original commit: 2711ca7727891e9e99871a9fe0267ab1e8138d6b
This commit is contained in:
parent
b88a38cca0
commit
15505d40e4
|
@ -8,27 +8,22 @@
|
|||
* manuals as `doc' sub-collections?
|
||||
|#
|
||||
|
||||
|
||||
(module help mzscheme
|
||||
(require "startup-url.ss"
|
||||
(lib "framework.ss" "framework")
|
||||
"help-unit.ss"
|
||||
"help-sig.ss")
|
||||
"help-sig.ss"
|
||||
(lib "plt-installer.ss" "setup")
|
||||
(lib "getinfo.ss" "setup")
|
||||
(lib "mred.ss"))
|
||||
|
||||
(define-values/invoke-unit/sig
|
||||
help:get-info^
|
||||
(unit/sig help:get-info^
|
||||
(import)
|
||||
|
||||
(define (get-language-level)
|
||||
'unknown)
|
||||
(define (get-teachpack-names)
|
||||
'unknown))
|
||||
drscheme:export:help-info)
|
||||
(provide-signature-elements help^)
|
||||
|
||||
(define frame-mixin values)
|
||||
(define (user-defined-doc-position x) #f)
|
||||
|
||||
;; just in case drscheme hasn't been run before, we
|
||||
;; need a default for this preference.
|
||||
(preferences:set-default
|
||||
'drscheme:font-size
|
||||
(send (send (send (make-object text%)
|
||||
|
@ -37,10 +32,11 @@
|
|||
get-size)
|
||||
(lambda (x) (and (number? x) (exact? x) (= x (floor x)))))
|
||||
|
||||
(define-values/invoke-unit/sig help:help^
|
||||
(define-values/invoke-unit/sig help^
|
||||
help-unit@
|
||||
#f
|
||||
setup:plt-installer^
|
||||
setup:get-info^
|
||||
mred^
|
||||
framework^
|
||||
(frame-mixin)
|
||||
|
|
433
collects/help/private/search.ss
Normal file
433
collects/help/private/search.ss
Normal file
|
@ -0,0 +1,433 @@
|
|||
(unit/sig help:search^
|
||||
(import help:doc-position^
|
||||
mzlib:function^)
|
||||
|
||||
; Define an order for the documentation:
|
||||
; and the names of the standard documentation
|
||||
(define-values (standard-html-doc-position known-manuals)
|
||||
(let ([pr (require-library "docpos.ss" "help")])
|
||||
(values (car pr) (cdr pr))))
|
||||
|
||||
(define (html-doc-position x)
|
||||
(or (user-defined-doc-position x)
|
||||
(standard-html-doc-position x)))
|
||||
|
||||
; These are set by reset-doc-lists:
|
||||
;; docs, doc-names and doc-kinds are parallel lists. doc-kinds
|
||||
;; distinguishes between the two variants of docs.
|
||||
;; docs : (list-of (union string (list string string)))
|
||||
(define docs null)
|
||||
;; doc-names : (list-of string)
|
||||
(define doc-names null)
|
||||
;; doc-kinds : (list-of symbol)
|
||||
(define doc-kinds null)
|
||||
;; doc-collection-date : ??
|
||||
(define doc-collection-date #f)
|
||||
|
||||
(define colldocs (require-library "colldocs.ss" "help"))
|
||||
|
||||
(define re:title (regexp "<[tT][iI][tT][lL][eE]>(.*)</[tT][iI][tT][lL][eE]>"))
|
||||
|
||||
;; get-std-doc-title : string -> string
|
||||
;; gets the standard title of the documentation, from the
|
||||
;; known docs list.
|
||||
(define (get-std-doc-title path doc)
|
||||
(let ([a (assoc doc known-manuals)])
|
||||
(if a
|
||||
(cdr a)
|
||||
(let ([index-file (build-path path doc "index.htm")])
|
||||
(if (file-exists? index-file)
|
||||
(call-with-input-file index-file
|
||||
(lambda (port)
|
||||
(let loop ()
|
||||
(let ([l (read-line port)])
|
||||
(cond
|
||||
[(eof-object? l)
|
||||
doc]
|
||||
[(regexp-match re:title l)
|
||||
=>
|
||||
(lambda (m)
|
||||
(apply
|
||||
string
|
||||
(map (lambda (x) (if (char-whitespace? x) #\space x))
|
||||
(string->list (cadr m)))))]
|
||||
[else (loop)])))))
|
||||
doc)))))
|
||||
|
||||
|
||||
(define (reset-doc-lists)
|
||||
; Locate standard HTML documentation
|
||||
(define-values (std-docs std-doc-names)
|
||||
(let* ([path (with-handlers ([void (lambda (x) #f)])
|
||||
(collection-path "doc"))])
|
||||
(if path
|
||||
(let* ([doc-collections (directory-list path)]
|
||||
[docs (map (lambda (x) (build-path path x)) doc-collections)]
|
||||
[doc-names (map (lambda (x) (get-std-doc-title path x)) doc-collections)])
|
||||
;; Order the standard docs:
|
||||
(let ([ordered (quicksort
|
||||
(map cons docs doc-names)
|
||||
(lambda (a b)
|
||||
(< (html-doc-position (cdr a))
|
||||
(html-doc-position (cdr b)))))])
|
||||
(values (map car ordered) (map cdr ordered))))
|
||||
(values null null))))
|
||||
|
||||
; Check collections for doc.txt files:
|
||||
(define-values (txt-docs txt-doc-names)
|
||||
(colldocs quicksort))
|
||||
|
||||
(set! docs (append std-docs txt-docs))
|
||||
(set! doc-names (append
|
||||
std-doc-names
|
||||
(map (lambda (s) (format "the ~a collection" s))
|
||||
txt-doc-names)))
|
||||
(set! doc-kinds (append (map (lambda (x) 'html) std-docs) (map (lambda (x) 'text) txt-docs)))
|
||||
|
||||
(with-handlers ([void (lambda (x)
|
||||
(set! doc-collection-date 'none))])
|
||||
(set! doc-collection-date
|
||||
(file-or-directory-modify-seconds
|
||||
(collection-path "doc")))))
|
||||
|
||||
(define MAX-HIT-COUNT 300)
|
||||
|
||||
(define (clean-html s)
|
||||
(regexp-replace*
|
||||
"&[^;]*;"
|
||||
(regexp-replace*
|
||||
"<[^>]*>"
|
||||
(regexp-replace*
|
||||
"&"
|
||||
(regexp-replace*
|
||||
">"
|
||||
(regexp-replace*
|
||||
"<"
|
||||
s
|
||||
"<")
|
||||
">")
|
||||
"\\&")
|
||||
"")
|
||||
""))
|
||||
|
||||
(define not-break? (lambda (x) (not (exn:misc:user-break? x))))
|
||||
|
||||
; One lock for all hash table operations is good enough
|
||||
(define ht-lock (make-semaphore 1))
|
||||
|
||||
(define (with-hash-table ht key compute)
|
||||
(dynamic-wind
|
||||
(lambda () (semaphore-wait ht-lock))
|
||||
(lambda ()
|
||||
(let ([sym (string->symbol key)])
|
||||
(hash-table-get
|
||||
ht
|
||||
sym
|
||||
(lambda ()
|
||||
(let ([v (compute)])
|
||||
(hash-table-put! ht sym v)
|
||||
v)))))
|
||||
(lambda () (semaphore-post ht-lock))))
|
||||
|
||||
(define html-keywords (make-hash-table))
|
||||
(define (load-html-keywords doc)
|
||||
(with-hash-table
|
||||
html-keywords
|
||||
doc
|
||||
(lambda ()
|
||||
(with-handlers ([not-break? (lambda (x) null)])
|
||||
(with-input-from-file (build-path doc "keywords")
|
||||
read)))))
|
||||
|
||||
(define html-indices (make-hash-table))
|
||||
(define (load-html-index doc)
|
||||
(with-hash-table
|
||||
html-indices
|
||||
doc
|
||||
(lambda ()
|
||||
(with-handlers ([not-break? (lambda (x) null)])
|
||||
(with-input-from-file (build-path doc "hdindex")
|
||||
read)))))
|
||||
|
||||
(define (parse-txt-file doc ht handle-one)
|
||||
(with-hash-table
|
||||
ht
|
||||
doc
|
||||
(lambda ()
|
||||
(with-handlers ([not-break? (lambda (x) null)])
|
||||
(with-input-from-file doc
|
||||
(lambda ()
|
||||
(let loop ([start 0])
|
||||
(let* ([r (read-line (current-input-port) 'any)]
|
||||
[next (if (eof-object? r)
|
||||
start
|
||||
(+ start (string-length r) 1))])
|
||||
(cond
|
||||
[(eof-object? r) null]
|
||||
[(handle-one r start) => (lambda (vs) (append vs (loop next)))]
|
||||
[else (loop next)])))))))))
|
||||
|
||||
(define re:keyword-line (regexp "^>"))
|
||||
(define text-keywords (make-hash-table))
|
||||
(define (load-txt-keywords doc)
|
||||
(parse-txt-file
|
||||
(apply build-path doc)
|
||||
text-keywords
|
||||
(lambda (r start)
|
||||
(cond
|
||||
[(regexp-match re:keyword-line r)
|
||||
(let* ([p (open-input-string (substring r 1 (string-length r)))]
|
||||
[entry (parameterize ([read-accept-bar-quote #f])
|
||||
(read p))]
|
||||
[key (let loop ([entry entry])
|
||||
(cond
|
||||
[(symbol? entry) entry]
|
||||
[(pair? entry) (if (eq? (car entry) 'quote)
|
||||
(loop (cadr entry))
|
||||
(loop (car entry)))]
|
||||
[else (error "bad entry")]))]
|
||||
[content (if (symbol? entry)
|
||||
(with-handlers ([not-break? (lambda (x) #f)])
|
||||
(let ([s (read p)])
|
||||
(if (eq? s '::)
|
||||
(read p)
|
||||
#f)))
|
||||
#f)])
|
||||
(list
|
||||
; Make the keyword entry:
|
||||
(list (symbol->string key) ; the keyword name
|
||||
(let ([p (open-output-string)])
|
||||
(if content
|
||||
(display content p)
|
||||
(if (and (pair? entry)
|
||||
(eq? (car entry) 'quote))
|
||||
(fprintf p "'~s" (cadr entry))
|
||||
(display entry p)))
|
||||
(get-output-string p)) ; the text to display
|
||||
(cadr doc) ; file
|
||||
start ; label (a position in this case)
|
||||
"doc.txt")))] ; title
|
||||
[else #f]))))
|
||||
|
||||
(define re:index-line (regexp "_([^_]*)_(.*)"))
|
||||
(define text-indices (make-hash-table))
|
||||
(define (load-txt-index doc)
|
||||
(parse-txt-file
|
||||
(apply build-path doc)
|
||||
text-indices
|
||||
(lambda (r start)
|
||||
(cond
|
||||
[(regexp-match re:index-line r)
|
||||
=> (lambda (m)
|
||||
(let loop ([m m])
|
||||
(let ([s (cadr m)])
|
||||
(cons
|
||||
; Make an index entry:
|
||||
(cons s start)
|
||||
(let ([m (regexp-match re:index-line (caddr m))])
|
||||
(if m
|
||||
(loop m)
|
||||
null))))))]
|
||||
[else #f]))))
|
||||
|
||||
(define re:splitter (regexp "^ *([^ ]+)(.*)"))
|
||||
(define (split-words s)
|
||||
(let ([m (regexp-match re:splitter s)])
|
||||
(if m
|
||||
(cons (cadr m)
|
||||
(split-words (caddr m)))
|
||||
null)))
|
||||
|
||||
(define (non-regexp s)
|
||||
(list->string
|
||||
(apply
|
||||
append
|
||||
(map
|
||||
(lambda (c)
|
||||
(cond
|
||||
[(memq c '(#\$ #\| #\\ #\[ #\] #\. #\* #\? #\+ #\( #\) #\^))
|
||||
(list #\\ c)]
|
||||
[(char-alphabetic? c)
|
||||
(list #\[ (char-upcase c) (char-downcase c) #\])]
|
||||
[else (list c)]))
|
||||
(string->list s)))))
|
||||
|
||||
(define (doc-collections-changed)
|
||||
(set! doc-collection-date #f))
|
||||
|
||||
(define re:url-dir (regexp "^([^/]*)/(.*)$"))
|
||||
(define (combine-path/url-path path url-path)
|
||||
(let loop ([path path]
|
||||
[url-path url-path])
|
||||
(cond
|
||||
[(regexp-match re:url-dir url-path)
|
||||
=>
|
||||
(lambda (m)
|
||||
(let* ([url-dir (cadr m)]
|
||||
[rest (caddr m)]
|
||||
[dir
|
||||
(cond
|
||||
[(string=? ".." url-dir) 'up]
|
||||
[(string=? "." url-dir) 'same]
|
||||
[(string=? "" url-dir) 'same]
|
||||
[else url-dir])])
|
||||
(loop (build-path path dir)
|
||||
rest)))]
|
||||
[else (build-path path url-path)])))
|
||||
|
||||
;; do-search : ((? -> ?)
|
||||
;; ??
|
||||
;; boolean
|
||||
;; boolean
|
||||
;; ??
|
||||
;; (-> A) ;; doesn't return
|
||||
;; (?? -> ??)
|
||||
;; (?? -> ??)
|
||||
;; (?? ?? ?? ?? ?? ?? -> ??)
|
||||
;; ->
|
||||
;; (union string #f))
|
||||
(define (do-search given-find search-level regexp? exact? ckey maxxed-out
|
||||
add-doc-section add-kind-section add-choice)
|
||||
; When new docs are installed, the directory's modification date changes:
|
||||
(unless (eq? doc-collection-date 'none)
|
||||
(when (or (not doc-collection-date)
|
||||
(> (file-or-directory-modify-seconds (collection-path "doc"))
|
||||
doc-collection-date))
|
||||
(reset-doc-lists)))
|
||||
(let* ([hit-count 0]
|
||||
[string-finds (list given-find)]
|
||||
[finds (cond
|
||||
[exact? (list given-find)]
|
||||
[regexp? (list (regexp given-find))]
|
||||
[else (let ([wl (split-words given-find)])
|
||||
(set! string-finds wl)
|
||||
(map regexp (map non-regexp wl)))])])
|
||||
(for-each
|
||||
(lambda (doc doc-name doc-kind)
|
||||
(define found-one #f)
|
||||
(define (found kind)
|
||||
(unless found-one
|
||||
(add-doc-section doc-name ckey))
|
||||
(unless (equal? found-one kind)
|
||||
(set! found-one kind)
|
||||
(add-kind-section kind ckey))
|
||||
(set! hit-count (add1 hit-count))
|
||||
(unless (< hit-count MAX-HIT-COUNT)
|
||||
(maxxed-out)))
|
||||
|
||||
;; Keyword search
|
||||
(let ([keys (case doc-kind
|
||||
[(html) (load-html-keywords doc)]
|
||||
[(text) (load-txt-keywords doc)]
|
||||
[else null])]
|
||||
[add-key-choice (lambda (v)
|
||||
(found "keyword entries")
|
||||
(add-choice
|
||||
(car v) ; key
|
||||
(cadr v) ; display
|
||||
(list-ref v 4) ; title
|
||||
(if (eq? 'text doc-kind)
|
||||
(apply build-path doc)
|
||||
(build-path doc (list-ref v 2))) ; file
|
||||
(list-ref v 3) ; label
|
||||
ckey))])
|
||||
|
||||
(unless regexp?
|
||||
(for-each
|
||||
(lambda (v)
|
||||
(when (string=? given-find (car v))
|
||||
(add-key-choice v)))
|
||||
keys))
|
||||
(unless (or exact? (null? finds))
|
||||
(for-each
|
||||
(lambda (v)
|
||||
(when (andmap (lambda (find) (regexp-match find (car v))) finds)
|
||||
(unless (and (not regexp?) (string=? given-find (car v)))
|
||||
(add-key-choice v))))
|
||||
keys)))
|
||||
;; Index search
|
||||
(unless (< search-level 1)
|
||||
(let ([index (case doc-kind
|
||||
[(html) (load-html-index doc)]
|
||||
[(text) (load-txt-index doc)]
|
||||
[else null])]
|
||||
[add-index-choice (lambda (name desc)
|
||||
(case doc-kind
|
||||
[(html)
|
||||
(found "index entries")
|
||||
(add-choice "" name
|
||||
(list-ref desc 2)
|
||||
(combine-path/url-path doc (list-ref desc 0))
|
||||
(list-ref desc 1)
|
||||
ckey)]
|
||||
[(text)
|
||||
(found "index entries")
|
||||
(add-choice "" name
|
||||
"indexed content"
|
||||
(apply build-path doc)
|
||||
desc
|
||||
ckey)]))])
|
||||
(when index
|
||||
(unless regexp?
|
||||
(for-each
|
||||
(lambda (v)
|
||||
(when (string=? given-find (car v))
|
||||
(add-index-choice (car v) (cdr v))))
|
||||
index))
|
||||
(unless (or exact? (null? finds))
|
||||
(for-each
|
||||
(lambda (v)
|
||||
(when (andmap (lambda (find) (regexp-match find (car v))) finds)
|
||||
(unless (and (not regexp?) (string=? given-find (car v)))
|
||||
(add-index-choice (car v) (cdr v)))))
|
||||
index)))))
|
||||
;; Content Search
|
||||
(unless (or (< search-level 2) exact? (null? finds))
|
||||
(let ([files (case doc-kind
|
||||
[(html) (with-handlers ([not-break? (lambda (x) null)])
|
||||
(map (lambda (x) (build-path doc x))
|
||||
(filter
|
||||
(lambda (x) (file-exists? (build-path doc x)))
|
||||
(directory-list doc))))]
|
||||
[(text) (list (apply build-path doc))]
|
||||
[else null])])
|
||||
(for-each
|
||||
(lambda (f)
|
||||
(with-handlers ([not-break? (lambda (x) #f)])
|
||||
(with-input-from-file f
|
||||
(lambda ()
|
||||
(let loop ()
|
||||
(let ([pos (file-position (current-input-port))]
|
||||
[r (read-line)])
|
||||
(unless (eof-object? r)
|
||||
(let ([m (andmap (lambda (find) (regexp-match find r)) finds)])
|
||||
(when m
|
||||
(found "text")
|
||||
(add-choice (car m)
|
||||
; Strip leading space and clean HTML
|
||||
(regexp-replace
|
||||
"^ [ ]*"
|
||||
(if (eq? doc-kind 'html)
|
||||
(clean-html r)
|
||||
r)
|
||||
"")
|
||||
"content"
|
||||
f
|
||||
(if (eq? doc-kind 'text) pos "NO TAG")
|
||||
ckey)))
|
||||
(loop))))))))
|
||||
files))))
|
||||
docs doc-names doc-kinds)
|
||||
(if (= 0 hit-count)
|
||||
(apply
|
||||
string-append
|
||||
"Nothing found for "
|
||||
(cond
|
||||
[(null? string-finds) (list "the empty search.")]
|
||||
[else
|
||||
(append
|
||||
(cons (format "\"~a\"" (car string-finds))
|
||||
(map (lambda (i) (format " and \"~a\"" i))
|
||||
(cdr string-finds)))
|
||||
(list "."))]))
|
||||
#f))))
|
Loading…
Reference in New Issue
Block a user