elisp-vcs/emacs-w3m/w3m-lnum.el
2010-08-17 08:59:01 +02:00

567 lines
20 KiB
EmacsLisp

;;; w3m-lnum.el --- Operations using link numbers
;; Copyright (C) 2004, 2005, 2006, 2007, 2009, 2010
;; TSUCHIYA Masatoshi <tsuchiya@namazu.org>
;; Authors: TSUCHIYA Masatoshi <tsuchiya@namazu.org>
;; Modified by Andrey Kotlarski <m00naticus@gmail.com>
;; Keywords: w3m, WWW, hypermedia
;; This file is a part of emacs-w3m.
;; This program is free software; you can redistribute it and/or modify
;; it under the terms of the GNU General Public License as published by
;; the Free Software Foundation; either version 2, or (at your option)
;; any later version.
;; This program 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
;; GNU General Public License for more details.
;; You should have received a copy of the GNU General Public License
;; along with this program; see the file COPYING. If not, write to
;; the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
;; Boston, MA 02110-1301, USA.
;;; Commentary:
;; This file provides a minor mode to enable Conkeror style operations
;; using link numbers. Mostly point operations are extended beyond
;; current point but there are also new features like
;; `w3m-go-to-linknum' for quickly navigating to links, form fields
;; or buttons and `w3m-linknum-follow' for visiting links, activating
;; form fields or pushing buttons.
;;; Usage:
;; Install this file to an appropriate directory, and add this
;; expression to your ~/.emacs-w3m if you want automatically
;; activating this minor mode
;; (w3m-link-numbering-mode 1)
;; or alternatively this to your .emacs file before loading w3m
;; (add-hook 'w3m-mode-hook 'w3m-link-numbering-mode)
;; or just use interactive command `w3m-link-numbering-mode' to toggle
;; mode.
;;; Code:
(eval-when-compile
(require 'cl))
(require 'w3m)
(defface w3m-link-numbering
'((((class color) (background light)) (:foreground "gray60"))
(((class color) (background dark)) (:foreground "gray50")))
"Face used to highlight link numbers."
:group 'w3m-face)
;; backward-compatibility alias
(put 'w3m-link-numbering-face 'face-alias 'w3m-link-numbering)
(defface w3m-linknum-minibuffer-prompt
'((((class color) (background light) (type tty)) (:foreground "blue"))
(((class color) (background dark)) (:foreground "cyan"))
(t (:foreground "medium blue")))
"Face for w3m linknum minibuffer prompt."
:group 'w3m-face)
(defface w3m-linknum-match
'((((class color) (background light) (type tty))
(:background "yellow" :foreground "black"))
(((class color) (background dark) (type tty))
(:background "blue" :foreground "white"))
(((class color) (background light)) (:background "yellow1"))
(((class color) (background dark)) (:background "RoyalBlue3"))
(t (:background "gray")))
"Face used to highlight matches in `w3m-link-numbering-mode'."
:group 'w3m-face)
(defcustom w3m-link-numbering-mode-hook nil
"*Hook run after `w3m-link-numbering-mode' initialization."
:group 'w3m
:type 'hook)
(defmacro w3m-substitute-key-definitions (new-map old-map &rest keys)
"In NEW-MAP substitute cascade of OLD-MAP KEYS.
KEYS is alternating list of key-value."
(let ((n-map new-map)
(o-map old-map))
`(progn
,@(let ((res nil))
(while keys
(push `(substitute-key-definition
,(car keys) ,(cadr keys) ,n-map ,o-map)
res)
(setq keys (cddr keys)))
(nreverse res)))))
(defvar w3m-link-numbering-mode-map nil
"Keymap used when `w3m-link-numbering-mode' is active.")
(unless w3m-link-numbering-mode-map
(let ((map (make-sparse-keymap)))
(define-key map "f" 'w3m-linknum-follow)
(define-key map "F" 'w3m-go-to-linknum)
(w3m-substitute-key-definitions
map w3m-mode-map
'w3m-view-image 'w3m-linknum-view-image
'w3m-save-image 'w3m-linknum-save-image
'w3m-download-this-url 'w3m-linknum-download-this-url
'w3m-edit-this-url 'w3m-linknum-edit-this-url
'w3m-toggle-inline-image 'w3m-linknum-toggle-inline-image
'w3m-print-this-url 'w3m-linknum-print-this-url
'w3m-external-view-this-url 'w3m-linknum-external-view-this-url
'w3m-bookmark-add-this-url 'w3m-linknum-bookmark-add-this-url
'w3m-zoom-in-image 'w3m-linknum-zoom-in-image
'w3m-zoom-out-image 'w3m-linknum-zoom-out-image)
(setq w3m-link-numbering-mode-map map)))
(defvar w3m-link-numbering-mode nil
"Non-nil if w3m operations using link numbers are enabled.")
(make-variable-buffer-local 'w3m-link-numbering-mode)
(unless (assq 'w3m-link-numbering-mode minor-mode-alist)
(push (list 'w3m-link-numbering-mode "[ln]") minor-mode-alist))
(unless (assq 'w3m-link-numbering-mode minor-mode-map-alist)
(push (cons 'w3m-link-numbering-mode w3m-link-numbering-mode-map)
minor-mode-map-alist))
(defun w3m-linknum-remove-overlays (&optional arg)
"Remove numbering and match overlays.
With ARG remove only temporary match"
(if arg
(catch 'done
(dolist (overlay (overlays-in (point-min) (point-max)))
(when (overlay-get overlay 'w3m-linknum-match)
(delete-overlay overlay)
(throw 'done nil))))
(dolist (overlay (overlays-in (point-min) (point-max)))
(if (or (overlay-get overlay 'w3m-link-numbering-overlay)
(overlay-get overlay 'w3m-linknum-match))
(delete-overlay overlay)))))
;;;###autoload
(defun w3m-link-numbering-mode (&optional arg)
"Minor mode to extend point commands by using Conkeror style number selection.
With prefix ARG 0 disable battery included point functions, otherwise
enable them. With no prefix ARG - toggle."
(interactive "P")
(let ((w3m-linknum-status w3m-link-numbering-mode))
;; find current numbering status of w3m buffers
(unless (eq major-mode 'w3m-mode)
(save-current-buffer
(setq w3m-linknum-status
(catch 'found-w3m
(dolist (buf (buffer-list))
(set-buffer buf)
(if (eq major-mode 'w3m-mode)
(throw 'found-w3m w3m-link-numbering-mode)))))))
(setq arg (not (if arg
(zerop arg)
w3m-linknum-status)))
(unless (eq arg w3m-linknum-status) ; if change of mode status
(if arg
(progn (add-hook 'w3m-mode-hook 'w3m-link-numbering-mode)
(run-hooks 'w3m-link-numbering-mode-hook)
(w3m-message "Link numbering keys on"))
(remove-hook 'w3m-mode-hook 'w3m-link-numbering-mode)
(w3m-message "Link numbering keys off"))
;; change numbering status of all w3m buffers
(save-current-buffer
(dolist (buf (buffer-list))
(set-buffer buf)
(if (eq major-mode 'w3m-mode)
(setq w3m-link-numbering-mode arg)))))))
(defun w3m-link-numbering (arg)
"Make overlays that display link numbers.
With ARG 0 clear numbering overlay. With ARG 2 index only images.
With ARG 4 index form fields and buttons along links."
(if (zerop arg)
(w3m-linknum-remove-overlays)
(save-excursion
(goto-char (point-min))
(let ((i 0)
(next-func 'w3m-goto-next-anchor)
pos overlay num)
(if (= arg 2)
(setq next-func (lambda () (if (w3m-goto-next-image)
(point)))))
(catch 'already-numbered
(while (setq pos (funcall next-func))
(when (or (> arg 1)
(get-char-property pos 'w3m-href-anchor))
(if (get-char-property pos
'w3m-link-numbering-overlay)
(throw 'already-numbered nil))
(setq overlay (make-overlay pos (1+ pos))
num (format "[%d]" (incf i)))
(overlay-put overlay 'before-string num)
(w3m-static-if (featurep 'xemacs)
(set-glyph-face (extent-begin-glyph overlay)
'w3m-link-numbering)
(w3m-add-face-property 0 (length num)
'w3m-link-numbering num)
(overlay-put overlay 'evaporate t))
(overlay-put overlay
'w3m-link-numbering-overlay i))))))))
(defun w3m-read-int-interactive (prompt fun &optional default)
"Interactively read a valid integer from minubuffer with PROMPT.
Execute a one argument function FUN with every current valid integer.
Initial value is DEFAULT if specified or 0.
Use <return> to submit current value, <backspace> for correction
and <C-g> or <escape> to quit action."
(let ((prompt (propertize prompt 'face
'w3m-linknum-minibuffer-prompt))
(num (or default 0))
(min-len (length prompt))
ch)
(let ((temp-prompt (format "%s%d" prompt num)))
(while (not (memq
(setq ch (w3m-static-if (featurep 'xemacs)
(let (event key)
(display-message 'no-log temp-prompt)
(setq event (next-command-event))
(and (key-press-event-p event)
(or (event-to-character event)
(characterp
(setq key (event-key event)))
key)))
(read-event temp-prompt)))
'(return 10 13 ?\n ?\r ?\C-g escape 27 ?\e)))
(cond ((and (memq ch '(backspace 8 ?\C-h))
(> (length temp-prompt) min-len))
(setq num (/ num 10)
temp-prompt (format "%s%d" prompt num))
(funcall fun num))
((and (w3m-static-if (featurep 'xemacs)
(characterp ch)
(numberp ch))
(> ch 47) (< ch 58))
(setq num (+ (* num 10) (- ch 48))
temp-prompt (format "%s%d" prompt num))
(funcall fun num))))
(if (memq ch '(?\C-g escape 27 ?\e))
(keyboard-quit))
num)))
(defmacro w3m-with-linknum (type &rest body)
"Within TYPE anchor numbering execute BODY.
Types are: 0 no numbering, 1 links, 2 images,
4 links, form fields and buttons.
Then restore previous numbering condition."
`(progn (w3m-link-numbering ,type)
(unwind-protect (progn ,@body)
(w3m-linknum-remove-overlays))))
(defun w3m-highlight-numbered-anchor (arg)
"Highlight specified by ARG number anchor."
(catch 'done
(let (found-prev marked-new)
(dolist (overlay (overlays-in (point-min) (point-max)))
(cond
((overlay-get overlay 'w3m-linknum-match)
(delete-overlay overlay)
(setq found-prev t))
((eq arg (overlay-get overlay 'w3m-link-numbering-overlay))
(let* ((start (overlay-start overlay))
(match-overlay
(make-overlay
start
(next-single-property-change
start
(cond ((get-text-property start 'w3m-href-anchor)
'w3m-href-anchor)
((get-text-property start 'w3m-image)
'w3m-image)
(t 'w3m-action))))))
(overlay-put match-overlay 'w3m-linknum-match t)
(overlay-put match-overlay 'face 'w3m-linknum-match))
(setq marked-new t)))
(and found-prev marked-new (throw 'done nil))))))
(defun w3m-get-anchor-info (&optional num)
"Get info (url/action position [image image-alt]) of anchor numbered as NUM.
If NUM is not specified, use currently highlighted anchor."
(macrolet
((get-match-info
(condition)
`(dolist (overlay (overlays-in (point-min) (point-max)))
(if ,condition
(let* ((pos (overlay-start overlay))
(href (get-text-property pos 'w3m-href-anchor)))
(throw
'found
(if href (list href pos
(get-text-property pos 'w3m-image)
(get-text-property pos 'w3m-image-alt))
(list (get-text-property pos 'w3m-action)
pos))))))))
(catch 'found
(if num (get-match-info
(eq num (overlay-get
overlay 'w3m-link-numbering-overlay)))
(get-match-info (overlay-get overlay 'w3m-linknum-match))))))
;;;###autoload
(defun w3m-go-to-linknum (arg)
"Turn on link and form numbers and ask for one to go to.
With prefix ARG don't highlight current link.
0 corresponds to location url."
(interactive "P")
(w3m-with-linknum
4
(let ((info (if arg
(let ((num (w3m-read-number "Anchor number: ")))
(if (zerop num)
(list nil 16)
(w3m-get-anchor-info num)))
(if (zerop (w3m-read-int-interactive
"Anchor number: "
'w3m-highlight-numbered-anchor))
(list nil 16)
(w3m-get-anchor-info)))))
(if info
(progn
(push-mark (point))
(goto-char (cadr info)))
(w3m-message "No valid anchor selected")))))
(defun w3m-linknum-get-action (&optional prompt type)
"Turn on link numbers and return list of url or action, position
and image url if such of PROMPT selected anchor.
TYPE sets types of anchors to be numbered, if nil or 4, number urls,
form fields and buttons. 1 - only links, 2 - only images.
Highlight every intermediate result anchor.
Input 0 corresponds to current page url."
(w3m-with-linknum
(or type 4)
(if (and (zerop (w3m-read-int-interactive
(or prompt "Anchor number: ")
'w3m-highlight-numbered-anchor))
(not (eq type 2)))
(list w3m-current-url 16 nil nil)
(w3m-get-anchor-info))))
;;;###autoload
(defun w3m-linknum-follow (arg)
"Turn on link numbers, ask for one and execute appropriate action on it.
When link - visit it, when button - press, when input - activate it.
With prefix ARG visit link in new session or move over field/button
before activate/press."
(interactive "P")
(let ((info (w3m-linknum-get-action
(concat "Follow " (if arg "in new session ")
"(select anchor): "))))
(if info
(let ((action (car info)))
(cond ((stringp action) ; url
(if arg (w3m-goto-url-new-session action)
(push-mark (point))
(goto-char (cadr info))
(w3m-history-store-position)
(w3m-goto-url action)))
((eq (car action) 'w3m-form-submit) ; button
(when arg
(push-mark (point))
(goto-char (cadr info)))
(widget-button-press (cadr info) action))
(t (if arg ; form field
(progn (push-mark (point))
(goto-char (cadr info))
(let ((w3m-form-new-session t)
(w3m-form-download nil))
(eval action)))
(save-excursion
(goto-char (cadr info))
(let ((w3m-form-new-session nil)
(w3m-form-download nil))
(eval action)))))))
(w3m-message "No valid anchor selected"))))
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;; linknum alternatives to w3m user commands on point
;;;###autoload
(defun w3m-linknum-toggle-inline-image (&optional arg)
"If image at point, toggle it.
Otherwise turn on link numbers and toggle selected image.
With prefix ARG open url under image in new session.
If no such url, move over image and toggle it."
(interactive "P")
(if (w3m-image)
(let ((url (get-char-property (point) 'w3m-href-anchor)))
(if (and arg url)
(w3m-goto-url-new-session url)
(w3m-toggle-inline-image)))
(let ((im (w3m-linknum-get-action
(if arg
"Open image url in new session: "
"Toggle image: ")
2)))
(if im
(if arg
(if (car im)
(w3m-goto-url-new-session (car im))
(push-mark (point))
(goto-char (cadr im))
(w3m-toggle-inline-image))
(save-excursion (goto-char (cadr im))
(w3m-toggle-inline-image)))
(w3m-message "No image selected")))))
;;;###autoload
(defun w3m-linknum-view-image ()
"Display the image under point in the external viewer.
If no image at poing, turn on image numbers and display selected.
The viewer is defined in `w3m-content-type-alist' for every type of an
image."
(interactive)
(let ((im (w3m-url-valid (w3m-image))))
(cond (im (w3m-external-view im))
((setq im (w3m-linknum-get-action
"Open image url in external viewer: " 2))
(w3m-external-view (caddr im)))
(t (w3m-message "No image selected")))))
;;;###autoload
(defun w3m-linknum-save-image ()
"Save the image under point to a file.
If no image at poing, turn on image numbers and save selected.
The default name will be the original name of the image."
(interactive)
(let ((im (w3m-url-valid (w3m-image))))
(cond (im (w3m-download im))
((setq im (w3m-linknum-get-action "Save image: " 2))
(w3m-download (caddr im)))
(t (w3m-message "No image selected")))))
(defmacro w3m-linknum-zoom-image (rate &optional in)
"Zoom image under the point.
Numeric prefix RATE specifies how many percent the image is
changed by. Default is the value of the `w3m-resize-image-scale'
variable. If no image under point, activate numbering and ask
for one. When IN zoom in, otherwise zoom out."
`(progn
(or (w3m-display-graphic-p)
(error "Can't display images in this environment"))
(or (w3m-imagick-convert-program-available-p)
(error "ImageMagick's `convert' program is required"))
(let ((im (w3m-image)))
(cond
(im (w3m-resize-inline-image-internal
im
(,(if in '+ '-) 100 (or ,rate w3m-resize-image-scale))))
((setq im (w3m-linknum-get-action
,(concat "Zoom " (if in "in" "out") " image: ") 2))
(save-excursion
(goto-char (cadr im))
(w3m-resize-inline-image-internal
(car im)
(,(if in '+ '-) 100 (or ,rate w3m-resize-image-scale)))))
(t (w3m-message "No image at point"))))))
(defun w3m-linknum-zoom-in-image (&optional rate)
"Zoom in an image on the point.
Numeric prefix RATE specifies how many percent the image is
enlarged by \(30 means enlarging the image by 130%). The default
is the value of the `w3m-resize-image-scale' variable. If no
image under point, activate numbering and ask for one."
(interactive "P")
(w3m-linknum-zoom-image rate t))
(defun w3m-linknum-zoom-out-image (&optional rate)
"Zoom out an image on the point.
Numeric prefix RATE specifies how many percent the image is shrunk by
\(30 means shrinking the image by 70%). The default is the value of
the `w3m-resize-image-scale' variable.
If no image under point, activate numbering and ask for one."
(interactive "P")
(w3m-linknum-zoom-image rate))
;;;###autoload
(defun w3m-linknum-external-view-this-url ()
"Launch the external browser and display the link at point.
If no link at point, turn on link numbers and open selected externally."
(interactive)
(let ((url (w3m-url-valid (or (w3m-anchor) (w3m-image)
(car
(w3m-linknum-get-action
"Open in external browser: " 1))))))
(if url
(w3m-external-view url)
(w3m-message "No URL selected"))))
;;;###autoload
(defun w3m-linknum-edit-this-url ()
"Edit the page linked from the anchor under the cursor.
If no such, turn on link numbers and edit selected."
(interactive)
(let ((url (or (w3m-url-valid (w3m-anchor))
(car (w3m-linknum-get-action
"Select link to edit: " 1)))))
(if url
(w3m-edit-url url)
(w3m-message "No URL selected"))))
;;;###autoload
(defun w3m-linknum-print-this-url ()
"Display the url under point in the echo area and put it into `kill-ring'.
If no url under point, activate numbering and select one."
(interactive)
(if (or (w3m-anchor) (w3m-image))
(w3m-print-this-url t)
(let ((link (w3m-linknum-get-action "Select URL to copy: " 1)))
(if link
(let ((url (car link)))
(kill-new url)
(w3m-message "%s%s" (let ((im-alt (cadddr link)))
(if (zerop (length im-alt))
""
(concat im-alt ": ")))
url))
(w3m-message "No URL selected")))))
;;;###autoload
(defun w3m-linknum-download-this-url ()
"Download the file or the page pointed to by the link under point.
If no point, activate numbering and select andchor to download."
(interactive)
(if (or (w3m-anchor) (w3m-image) (w3m-action))
(w3m-download-this-url)
(let ((info (w3m-linknum-get-action
"Select anchor to download: ")))
(if info
(save-excursion
(goto-char (cadr info))
(w3m-download-this-url))
(w3m-message "No anchor selected")))))
;;;###autoload
(defun w3m-linknum-bookmark-add-this-url ()
"Add link under cursor to bookmark.
If no link under point, activate numbering and ask for one."
(interactive)
(let ((url (w3m-anchor)))
(cond
(url (w3m-bookmark-add
url
(buffer-substring-no-properties
(previous-single-property-change (+ 1 (point))
'w3m-href-anchor)
(next-single-property-change (point) 'w3m-href-anchor)))
(message "Added"))
((setq url (w3m-linknum-get-action "Select URL to bookmark: " 1))
(w3m-bookmark-add
(car url)
(buffer-substring-no-properties
(previous-single-property-change (+ 1 (cadr url))
'w3m-href-anchor)
(next-single-property-change (cadr url) 'w3m-href-anchor)))
(w3m-message "added"))
(t (w3m-message "No url selected")))))
(provide 'w3m-lnum)
;;; w3m-lnum.el ends here