home *** CD-ROM | disk | FTP | other *** search
/ InfoMagic Internet Tools 1993 July / Internet Tools.iso / RockRidge / mail / vm / vm-5.33beta / vm-page.el < prev    next >
Encoding:
Text File  |  1993-04-11  |  10.1 KB  |  297 lines

  1. ;;; Commands to move around within a VM message
  2. ;;; Copyright (C) 1989, 1990, 1991 Kyle E. Jones
  3. ;;;
  4. ;;; This program is free software; you can redistribute it and/or modify
  5. ;;; it under the terms of the GNU General Public License as published by
  6. ;;; the Free Software Foundation; either version 1, or (at your option)
  7. ;;; any later version.
  8. ;;;
  9. ;;; This program is distributed in the hope that it will be useful,
  10. ;;; but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. ;;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  12. ;;; GNU General Public License for more details.
  13. ;;;
  14. ;;; You should have received a copy of the GNU General Public License
  15. ;;; along with this program; if not, write to the Free Software
  16. ;;; Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
  17.  
  18. (defun vm-scroll-forward (&optional arg)
  19.   "Scroll forward a screenful of text.
  20. If the current message is being previewed, the message body is revealed.
  21. If at the end of the current message, moves to the next message iff the
  22. value of vm-auto-next-message is non-nil.
  23. Prefix N scrolls forward N lines."
  24.   (interactive "P")
  25.   (let ((mp-changed (vm-follow-summary-cursor)) was-invisible do-next-message)
  26.     (vm-select-folder-buffer)
  27.     (vm-sanity-check-modification-flag)
  28.     (vm-check-for-killed-summary)
  29.     (vm-error-if-folder-empty)
  30.     (if (vm-within-current-message-buffer
  31.      (null (get-buffer-window (current-buffer))))
  32.     (progn
  33.       (vm-display-current-message-buffer)
  34.       (setq was-invisible t)))
  35.     (if (eq vm-system-state 'previewing)
  36.     (vm-show-current-message)
  37.       (if (or mp-changed was-invisible)
  38.       (vm-howl-if-eom)
  39.     (let ((vmp vm-message-pointer)
  40.           (msg-buf (vm-current-message-buffer))
  41.           (h-diff 0)
  42.           w old-w result)
  43.     (setq vm-system-state 'reading)
  44.     (setq old-w (get-buffer-window msg-buf))
  45.     (if (not (eq vm-window-configuration 'paging-message))
  46.         (save-excursion (vm-set-window-configuration 'paging-message)))
  47.     (setq w (get-buffer-window msg-buf))
  48.     (if (null w)
  49.         (error "paging-message configuration hides the message buffer.")
  50.       (setq h-diff (- (window-height w) (window-height old-w))))
  51.     (setq old-w (selected-window))
  52.     (vm-within-current-message-buffer
  53.      (unwind-protect
  54.          (progn
  55.            (select-window w)
  56.            (let ((next-screen-context-lines
  57.               (+ next-screen-context-lines h-diff))
  58.              ;; restore
  59.              (vm-message-pointer vmp))
  60.          (while (eq (setq result (vm-scroll-forward-internal arg))
  61.                 'tryagain))
  62.          (cond ((and (not (eq result 'next-message))
  63.                  vm-honor-page-delimiters)
  64.             (vm-narrow-to-page)
  65.             ;; This voodoo is required!  For some
  66.             ;; reason the 18.52 emacs display
  67.             ;; doesn't immediately reflect the
  68.             ;; clip region change that occurs
  69.             ;; above without this mantra. 
  70.             (scroll-up 0)))))
  71.        (select-window old-w)))
  72.     (cond ((eq result 'next-message)
  73.            (vm-next-message)
  74.            (vm-set-window-configuration 'auto-next-message))
  75.           ((eq result 'end-of-message)
  76.            (vm-set-window-configuration 'end-of-message)
  77.            (let ((vm-message-pointer vmp))
  78.          (vm-emit-eom-blurb)))
  79.           (t
  80.            (and (> (prefix-numeric-value arg) 0)
  81.             (vm-howl-if-eom))))))))
  82.   (if (not (or vm-startup-message-displayed vm-inhibit-startup-message))
  83.       (vm-display-startup-message)))
  84.  
  85. (defun vm-scroll-forward-internal (arg)
  86.   (let ((direction (prefix-numeric-value arg))
  87.     (w (selected-window)))
  88.     (condition-case error-data
  89.     (progn (scroll-up arg) nil)
  90.       (error
  91.        (if (or (and (< direction 0)
  92.             (> (point-min) (vm-text-of (car vm-message-pointer))))
  93.            (and (>= direction 0)
  94.             (/= (point-max)
  95.             (vm-text-end-of (car vm-message-pointer)))))
  96.        (progn
  97.          (vm-widen-page)
  98.          (if (>= direction 0)
  99.          (progn
  100.            (forward-page 1)
  101.            (set-window-start w (point))
  102.            nil )
  103.            (if (or (bolp)
  104.                (not (save-excursion
  105.                   (beginning-of-line)
  106.                   (looking-at page-delimiter))))
  107.            (forward-page -1))
  108.            (beginning-of-line)
  109.            (set-window-start w (point))
  110.            'tryagain))
  111.      (if (eq (car error-data) 'end-of-buffer)
  112.          (if vm-auto-next-message
  113.          'next-message
  114.            (set-window-point w (point))
  115.            'end-of-message)))))))
  116.  
  117. (defun vm-howl-if-eom ()
  118.   (let ((vmp vm-message-pointer))
  119.     (vm-within-current-message-buffer
  120.      (let ((w (get-buffer-window (current-buffer)))
  121.        (vm-message-pointer vmp))
  122.        (and w
  123.         (save-excursion
  124.           (save-window-excursion
  125.         (condition-case ()
  126.             (let ((next-screen-context-lines 0))
  127.               (select-window w)
  128.               (save-excursion
  129.             (save-window-excursion
  130.               (scroll-up nil)))
  131.               nil)
  132.           (error t))))
  133.         (= (vm-text-end-of (car vm-message-pointer)) (point-max))
  134.         (vm-emit-eom-blurb))))))
  135.  
  136. (defun vm-emit-eom-blurb ()
  137.   (message "End of message %s from %s"
  138.        (vm-number-of (car vm-message-pointer))
  139.        (vm-su-full-name (car vm-message-pointer))))
  140.  
  141. (defun vm-scroll-backward (arg)
  142.   "Scroll backward a screenful of text.
  143. Prefix N scrolls backward N lines."
  144.   (interactive "P")
  145.   (vm-scroll-forward (cond ((null arg) '-)
  146.                ((symbolp arg) nil)
  147.                ((consp arg) (list (- (car arg))))
  148.                (t arg))))
  149.  
  150. (defun vm-preview-current-message ()
  151.   (setq vm-system-state 'previewing)
  152.   (if (and (null (vm-within-current-message-buffer
  153.           (get-buffer-window (current-buffer))))
  154.        (eq major-mode 'vm-virtual-mode)
  155.        (not (one-window-p t)))
  156.       (vm-display-current-message-buffer t))
  157.   (let ((vmp vm-message-pointer))
  158.     (vm-within-current-message-buffer
  159.      (let ((vm-message-pointer vmp))
  160.        (widen)
  161.        (narrow-to-region
  162.     (vm-vheaders-of (car vm-message-pointer))
  163.     (if vm-preview-lines
  164.         (min
  165.          (vm-text-end-of (car vm-message-pointer))
  166.          (save-excursion
  167.            (goto-char (vm-text-of (car vm-message-pointer)))
  168.            (forward-line (if (natnump vm-preview-lines)
  169.                  vm-preview-lines
  170.                    0))
  171.            (point)))
  172.       (vm-text-of (car vm-message-pointer))))
  173.        (if vm-honor-page-delimiters
  174.        (vm-narrow-to-page))
  175.        (goto-char (vm-text-of (car vm-message-pointer)))
  176.        ;; If we have a window, set window start appropriately.
  177.        ;; Highlight appropriate headers if current buffer is visible.
  178.        (let ((w (get-buffer-window (current-buffer))))
  179.      (if w (set-window-start w (point-min)))
  180.      (vm-highlight-headers (car vm-message-pointer) w)))))
  181.   ;; De Morgan's Theorems could clear away most of the following negations,
  182.   ;; but the resulting code would be horribly obfuscated.
  183.   (if (or (null vm-preview-lines)
  184.       (and (not vm-preview-read-messages)
  185.            (not (vm-new-flag (car vm-message-pointer)))
  186.            (not (vm-unread-flag (car vm-message-pointer)))))
  187.       (vm-show-current-message)
  188.     (vm-update-summary-and-mode-line)))
  189.  
  190. (defun vm-show-current-message ()
  191.   (setq vm-system-state 'showing)
  192.   (let ((vmp vm-message-pointer))
  193.     (vm-within-current-message-buffer
  194.      (let ((vm-message-pointer vmp))
  195.        (save-excursion
  196.      (goto-char (point-min))
  197.      (widen)
  198.      (narrow-to-region (point) (vm-text-end-of (car vm-message-pointer))))
  199.        (if vm-honor-page-delimiters
  200.        (progn
  201.          (if (looking-at page-delimiter)
  202.          (forward-page 1))
  203.          (vm-narrow-to-page))))))
  204.   (cond ((vm-new-flag (car vm-message-pointer))
  205.      (vm-set-new-flag (car vm-message-pointer) nil))
  206.     ((vm-unread-flag (car vm-message-pointer))
  207.      (vm-set-unread-flag (car vm-message-pointer) nil)))
  208.   (vm-update-summary-and-mode-line))
  209.  
  210. (defun vm-expose-hidden-headers ()
  211.   "Toggle exposing and hiding message headers that are normally not visible."
  212.   (interactive)
  213.   (vm-follow-summary-cursor)
  214.   (vm-select-folder-buffer)
  215.   (vm-check-for-killed-summary)
  216.   (vm-error-if-folder-empty)
  217.   (let ((vmp vm-message-pointer))
  218.     (vm-within-current-message-buffer
  219.      (let* ((vm-message-pointer vmp)
  220.         (exposed (= (point-min) (vm-start-of (car vm-message-pointer)))))
  221.        (vm-widen-page)
  222.        (goto-char (point-max))
  223.        (widen)
  224.        (if exposed
  225.        (narrow-to-region (point) (vm-vheaders-of (car vm-message-pointer)))
  226.      (narrow-to-region (point) (vm-start-of (car vm-message-pointer))))
  227.        (goto-char (point-min))
  228.        (let (w)
  229.      (setq w (get-buffer-window (current-buffer)))
  230.      (and w (set-window-point w (point-min)))
  231.      (and w
  232.           (= (window-start w) (vm-vheaders-of (car vm-message-pointer)))
  233.           (not exposed)
  234.           (set-window-start w (vm-start-of (car vm-message-pointer)))))
  235.        (if vm-honor-page-delimiters
  236.        (vm-narrow-to-page))))))
  237.  
  238. (defun vm-widen-page ()
  239.   (if (or (> (point-min) (vm-text-of (car vm-message-pointer)))
  240.       (/= (point-max) (vm-text-end-of (car vm-message-pointer))))
  241.       (narrow-to-region (vm-vheaders-of (car vm-message-pointer))
  242.             (if (or (vm-new-flag (car vm-message-pointer))
  243.                 (vm-unread-flag (car vm-message-pointer)))
  244.                 (vm-text-of (car vm-message-pointer))
  245.               (vm-text-end-of (car vm-message-pointer))))))
  246.  
  247. (defun vm-narrow-to-page ()
  248.   (save-excursion
  249.     (let (min max (omin (point-min)) (omax (point-max)))
  250.       (if (or (bolp) (not (save-excursion
  251.                 (beginning-of-line)
  252.                 (looking-at page-delimiter))))
  253.       (forward-page -1))
  254.       (setq min (point))
  255.       (forward-page 1)
  256.       (beginning-of-line)
  257.       (setq max (point))
  258.       (narrow-to-region min max))))
  259.  
  260. (defun vm-beginning-of-message ()
  261.   "Moves to the beginning of the current message."
  262.   (interactive)
  263.   (vm-follow-summary-cursor)
  264.   (vm-select-folder-buffer)
  265.   (vm-check-for-killed-summary)
  266.   (vm-error-if-folder-empty)
  267.   (let ((vmp vm-message-pointer))
  268.     (vm-within-current-message-buffer
  269.      (let ((vm-message-pointer vmp))
  270.        (vm-widen-page)
  271.        (push-mark)
  272.        (goto-char (point-min))
  273.        (if vm-honor-page-delimiters
  274.        (vm-narrow-to-page)))))
  275.   (vm-display-current-message-buffer))
  276.  
  277. (defun vm-end-of-message ()
  278.   "Moves to the end of the current message, exposing and flagging it read
  279. as necessary."
  280.   (interactive)
  281.   (vm-follow-summary-cursor)
  282.   (vm-select-folder-buffer)
  283.   (vm-check-for-killed-summary)
  284.   (vm-error-if-folder-empty)
  285.   (if (eq vm-system-state 'previewing)
  286.       (vm-show-current-message))
  287.   (setq vm-system-state 'reading)
  288.   (let ((vmp vm-message-pointer))
  289.     (vm-within-current-message-buffer
  290.      (let ((vmp vm-message-pointer))
  291.        (vm-widen-page)
  292.        (push-mark)
  293.        (goto-char (point-max))
  294.        (if vm-honor-page-delimiters
  295.        (vm-narrow-to-page)))))
  296.   (vm-display-current-message-buffer t))
  297.