mirror of
https://git.savannah.gnu.org/git/emacs.git
synced 2024-12-13 09:32:47 +00:00
426 lines
15 KiB
EmacsLisp
426 lines
15 KiB
EmacsLisp
;;; erc-stamp.el --- Timestamping for ERC messages
|
|
|
|
;; Copyright (C) 2002-2004, 2006-2014 Free Software Foundation, Inc.
|
|
|
|
;; Author: Mario Lang <mlang@delysid.org>
|
|
;; Maintainer: emacs-devel@gnu.org
|
|
;; Keywords: comm, processes, timestamp
|
|
;; URL: http://www.emacswiki.org/cgi-bin/wiki.pl?ErcStamp
|
|
|
|
;; This file is part of GNU Emacs.
|
|
|
|
;; GNU Emacs 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 3 of the License, or
|
|
;; (at your option) any later version.
|
|
|
|
;; GNU Emacs 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 GNU Emacs. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
;;; Commentary:
|
|
|
|
;; The code contained in this module is responsible for inserting
|
|
;; timestamps into ERC buffers. In order to actually activate this,
|
|
;; you must call `erc-timestamp-mode'.
|
|
|
|
;; You can choose between two different ways of inserting timestamps.
|
|
;; Customize `erc-insert-timestamp-function' and
|
|
;; `erc-insert-away-timestamp-function'.
|
|
|
|
;;; Code:
|
|
|
|
(require 'erc)
|
|
(require 'erc-compat)
|
|
|
|
(defgroup erc-stamp nil
|
|
"For long conversation on IRC it is sometimes quite
|
|
useful to have individual messages timestamp. This
|
|
group provides settings related to the format and display
|
|
of timestamp information in `erc-mode' buffer.
|
|
|
|
For timestamping to be activated, you just need to load `erc-stamp'
|
|
in your init file or interactively using `load-library'."
|
|
:group 'erc)
|
|
|
|
(defcustom erc-timestamp-format "[%H:%M]"
|
|
"If set to a string, messages will be timestamped.
|
|
This string is processed using `format-time-string'.
|
|
Good examples are \"%T\" and \"%H:%M\".
|
|
|
|
If nil, timestamping is turned off."
|
|
:group 'erc-stamp
|
|
:type '(choice (const nil)
|
|
(string)))
|
|
|
|
(defcustom erc-timestamp-format-left "\n[%a %b %e %Y]\n"
|
|
"If set to a string, messages will be timestamped.
|
|
This string is processed using `format-time-string'.
|
|
Good examples are \"%T\" and \"%H:%M\".
|
|
|
|
This timestamp is used for timestamps on the left side of the
|
|
screen when `erc-insert-timestamp-function' is set to
|
|
`erc-insert-timestamp-left-and-right'.
|
|
|
|
If nil, timestamping is turned off."
|
|
:group 'erc-stamp
|
|
:type '(choice (const nil)
|
|
(string)))
|
|
|
|
(defcustom erc-timestamp-format-right " [%H:%M]"
|
|
"If set to a string, messages will be timestamped.
|
|
This string is processed using `format-time-string'.
|
|
Good examples are \"%T\" and \"%H:%M\".
|
|
|
|
This timestamp is used for timestamps on the right side of the
|
|
screen when `erc-insert-timestamp-function' is set to
|
|
`erc-insert-timestamp-left-and-right'.
|
|
|
|
If nil, timestamping is turned off."
|
|
:group 'erc-stamp
|
|
:type '(choice (const nil)
|
|
(string)))
|
|
|
|
(defcustom erc-insert-timestamp-function 'erc-insert-timestamp-left-and-right
|
|
"Function to use to insert timestamps.
|
|
|
|
It takes a single argument STRING which is the final string
|
|
which all text-properties already appended. This function only cares about
|
|
inserting this string at the right position. Narrowing is in effect
|
|
while it is called, so (point-min) and (point-max) determine the region to
|
|
operate on.
|
|
|
|
You will probably want to set
|
|
`erc-insert-away-timestamp-function' to the same value."
|
|
:group 'erc-stamp
|
|
:type '(choice (const :tag "Both sides" erc-insert-timestamp-left-and-right)
|
|
(const :tag "Right" erc-insert-timestamp-right)
|
|
(const :tag "Left" erc-insert-timestamp-left)
|
|
function))
|
|
|
|
(defcustom erc-away-timestamp-format "<%H:%M>"
|
|
"Timestamp format used when marked as being away.
|
|
|
|
If nil, timestamping is turned off when away unless `erc-timestamp-format'
|
|
is set.
|
|
|
|
If `erc-timestamp-format' is set, this will not be used."
|
|
:group 'erc-stamp
|
|
:type '(choice (const nil)
|
|
(string)))
|
|
|
|
(defcustom erc-insert-away-timestamp-function
|
|
'erc-insert-timestamp-left-and-right
|
|
"Function to use to insert the away timestamp.
|
|
|
|
See `erc-insert-timestamp-function' for details."
|
|
:group 'erc-stamp
|
|
:type '(choice (const :tag "Both sides" erc-insert-timestamp-left-and-right)
|
|
(const :tag "Right" erc-insert-timestamp-right)
|
|
(const :tag "Left" erc-insert-timestamp-left)
|
|
function))
|
|
|
|
(defcustom erc-hide-timestamps nil
|
|
"If non-nil, timestamps will be invisible.
|
|
|
|
This is useful for logging, because, although timestamps will be
|
|
hidden, they will still be present in the logs."
|
|
:group 'erc-stamp
|
|
:type 'boolean)
|
|
|
|
(defcustom erc-echo-timestamps nil
|
|
"If non-nil, print timestamp in the minibuffer when point is moved.
|
|
Using this variable, you can turn off normal timestamping,
|
|
and simply move point to an irc message to see its timestamp
|
|
printed in the minibuffer."
|
|
:group 'erc-stamp
|
|
:type 'boolean)
|
|
|
|
(defcustom erc-echo-timestamp-format "Timestamped %A, %H:%M:%S"
|
|
"Format string to be used when `erc-echo-timestamps' is non-nil.
|
|
This string specifies the format of the timestamp being echoed in
|
|
the minibuffer."
|
|
:group 'erc-stamp
|
|
:type 'string)
|
|
|
|
(defcustom erc-timestamp-intangible t
|
|
"Whether the timestamps should be intangible, i.e. prevent the point
|
|
from entering them and instead jump over them."
|
|
:group 'erc-stamp
|
|
:type 'boolean)
|
|
|
|
(defface erc-timestamp-face '((t :weight bold :foreground "green"))
|
|
"ERC timestamp face."
|
|
:group 'erc-faces)
|
|
|
|
;;;###autoload (autoload 'erc-timestamp-mode "erc-stamp" nil t)
|
|
(define-erc-module stamp timestamp
|
|
"This mode timestamps messages in the channel buffers."
|
|
((add-hook 'erc-mode-hook 'erc-munge-invisibility-spec)
|
|
(add-hook 'erc-insert-modify-hook 'erc-add-timestamp t)
|
|
(add-hook 'erc-send-modify-hook 'erc-add-timestamp t))
|
|
((remove-hook 'erc-mode-hook 'erc-munge-invisibility-spec)
|
|
(remove-hook 'erc-insert-modify-hook 'erc-add-timestamp)
|
|
(remove-hook 'erc-send-modify-hook 'erc-add-timestamp)))
|
|
|
|
(defun erc-add-timestamp ()
|
|
"Add timestamp and text-properties to message.
|
|
|
|
This function is meant to be called from `erc-insert-modify-hook'
|
|
or `erc-send-modify-hook'."
|
|
(unless (get-text-property (point) 'invisible)
|
|
(let ((ct (current-time)))
|
|
(if (fboundp erc-insert-timestamp-function)
|
|
(funcall erc-insert-timestamp-function
|
|
(erc-format-timestamp ct erc-timestamp-format))
|
|
(error "Timestamp function unbound"))
|
|
(when (and (fboundp erc-insert-away-timestamp-function)
|
|
erc-away-timestamp-format
|
|
(erc-away-time)
|
|
(not erc-timestamp-format))
|
|
(funcall erc-insert-away-timestamp-function
|
|
(erc-format-timestamp ct erc-away-timestamp-format)))
|
|
(add-text-properties (point-min) (point-max)
|
|
(list 'timestamp ct))
|
|
(add-text-properties (point-min) (point-max)
|
|
(list 'point-entered 'erc-echo-timestamp)))))
|
|
|
|
(defvar erc-timestamp-last-inserted nil
|
|
"Last timestamp inserted into the buffer.")
|
|
(make-variable-buffer-local 'erc-timestamp-last-inserted)
|
|
|
|
(defvar erc-timestamp-last-inserted-left nil
|
|
"Last timestamp inserted into the left side of the buffer.
|
|
This is used when `erc-insert-timestamp-function' is set to
|
|
`erc-timestamp-left-and-right'")
|
|
(make-variable-buffer-local 'erc-timestamp-last-inserted-left)
|
|
|
|
(defvar erc-timestamp-last-inserted-right nil
|
|
"Last timestamp inserted into the right side of the buffer.
|
|
This is used when `erc-insert-timestamp-function' is set to
|
|
`erc-timestamp-left-and-right'")
|
|
(make-variable-buffer-local 'erc-timestamp-last-inserted-right)
|
|
|
|
(defcustom erc-timestamp-only-if-changed-flag t
|
|
"Insert timestamp only if its value changed since last insertion.
|
|
If `erc-insert-timestamp-function' is `erc-insert-timestamp-left', a
|
|
string of spaces which is the same size as the timestamp is added to
|
|
the beginning of the line in its place. If you use
|
|
`erc-insert-timestamp-right', nothing gets inserted in place of the
|
|
timestamp."
|
|
:group 'erc-stamp
|
|
:type 'boolean)
|
|
|
|
(defcustom erc-timestamp-right-column nil
|
|
"If non-nil, the column at which the timestamp is inserted,
|
|
if the timestamp is to be printed to the right. If nil,
|
|
`erc-insert-timestamp-right' will use other means to determine
|
|
the correct column."
|
|
:group 'erc-stamp
|
|
:type '(choice
|
|
(integer :tag "Column number")
|
|
(const :tag "Unspecified" nil)))
|
|
|
|
(defcustom erc-timestamp-use-align-to (and (not (featurep 'xemacs))
|
|
(>= emacs-major-version 22)
|
|
(eq window-system 'x))
|
|
"If non-nil, use the :align-to display property to align the stamp.
|
|
This gives better results when variable-width characters (like
|
|
Asian language characters and math symbols) precede a timestamp.
|
|
Unfortunately, it only works in Emacs 22 and when using the X
|
|
Window System.
|
|
|
|
A side effect of enabling this is that there will only be one
|
|
space before a right timestamp in any saved logs."
|
|
:group 'erc-stamp
|
|
:type 'boolean)
|
|
|
|
(defun erc-insert-timestamp-left (string)
|
|
"Insert timestamps at the beginning of the line."
|
|
(goto-char (point-min))
|
|
(let* ((ignore-p (and erc-timestamp-only-if-changed-flag
|
|
(string-equal string erc-timestamp-last-inserted)))
|
|
(len (length string))
|
|
(s (if ignore-p (make-string len ? ) string)))
|
|
(unless ignore-p (setq erc-timestamp-last-inserted string))
|
|
(erc-put-text-property 0 len 'field 'erc-timestamp s)
|
|
(erc-put-text-property 0 len 'invisible 'timestamp s)
|
|
(insert s)))
|
|
|
|
(defun erc-insert-aligned (string pos)
|
|
"Insert STRING at the POSth column.
|
|
|
|
If `erc-timestamp-use-align-to' is t, use the :align-to display
|
|
property to get to the POSth column."
|
|
(if (not erc-timestamp-use-align-to)
|
|
(indent-to pos)
|
|
(insert " ")
|
|
(put-text-property (1- (point)) (point) 'display
|
|
(list 'space ':align-to pos)))
|
|
(insert string))
|
|
|
|
;; Silence byte-compiler
|
|
(defvar erc-fill-column)
|
|
|
|
(defun erc-insert-timestamp-right (string)
|
|
"Insert timestamp on the right side of the screen.
|
|
STRING is the timestamp to insert. The function is a possible value
|
|
for `erc-insert-timestamp-function'.
|
|
|
|
If `erc-timestamp-only-if-changed-flag' is nil, a timestamp is always
|
|
printed. If this variable is non-nil, a timestamp is only printed if
|
|
it is different from the last.
|
|
|
|
If `erc-timestamp-right-column' is set, its value will be used as the
|
|
column at which the timestamp is to be printed. If it is nil, and
|
|
`erc-fill-mode' is active, then the timestamp will be printed just
|
|
before `erc-fill-column'. Otherwise, if the current buffer is
|
|
shown in a window, that window's width is used. If the buffer is
|
|
not shown, and `fill-column' is set, then the timestamp will be
|
|
printed just `fill-column'. As a last resort, the timestamp will
|
|
be printed just before the window-width."
|
|
(unless (and erc-timestamp-only-if-changed-flag
|
|
(string-equal string erc-timestamp-last-inserted))
|
|
(setq erc-timestamp-last-inserted string)
|
|
(goto-char (point-max))
|
|
(forward-char -1);; before the last newline
|
|
(let* ((current-window (get-buffer-window (current-buffer)))
|
|
(str-width (string-width string))
|
|
(pos (cond
|
|
(erc-timestamp-right-column erc-timestamp-right-column)
|
|
((and (boundp 'erc-fill-mode)
|
|
erc-fill-mode
|
|
(boundp 'erc-fill-column)
|
|
erc-fill-column)
|
|
(1+ (- erc-fill-column str-width)))
|
|
(fill-column
|
|
(1+ (- fill-column str-width)))
|
|
(t
|
|
(- (window-width) str-width 1))))
|
|
(from (point))
|
|
(col (current-column))
|
|
indent)
|
|
;; The following is a kludge used to calculate whether to move
|
|
;; to the next line before inserting a stamp. It allows for
|
|
;; some margin of error if what is displayed on the line differs
|
|
;; from the number of characters on the line.
|
|
(setq col (+ col (ceiling (/ (- col (- (point) (point-at-bol))) 1.6))))
|
|
(if (< col pos)
|
|
(erc-insert-aligned string pos)
|
|
(newline)
|
|
(indent-to pos)
|
|
(setq from (point))
|
|
(insert string))
|
|
(erc-put-text-property from (point) 'field 'erc-timestamp)
|
|
(erc-put-text-property from (point) 'rear-nonsticky t)
|
|
(when erc-timestamp-intangible
|
|
(erc-put-text-property from (1+ (point)) 'intangible t)))))
|
|
|
|
(defun erc-insert-timestamp-left-and-right (string)
|
|
"This is another function that can be assigned to
|
|
`erc-insert-timestamp-function'. If the date is changed, it will
|
|
print a blank line, the date, and another blank line. If the time is
|
|
changed, it will then print it off to the right."
|
|
(let* ((ct (current-time))
|
|
(ts-left (erc-format-timestamp ct erc-timestamp-format-left))
|
|
(ts-right (erc-format-timestamp ct erc-timestamp-format-right)))
|
|
;; insert left timestamp
|
|
(unless (string-equal ts-left erc-timestamp-last-inserted-left)
|
|
(goto-char (point-min))
|
|
(erc-put-text-property 0 (length ts-left) 'field 'erc-timestamp ts-left)
|
|
(insert ts-left)
|
|
(setq erc-timestamp-last-inserted-left ts-left))
|
|
;; insert right timestamp
|
|
(let ((erc-timestamp-only-if-changed-flag t)
|
|
(erc-timestamp-last-inserted erc-timestamp-last-inserted-right))
|
|
(erc-insert-timestamp-right ts-right)
|
|
(setq erc-timestamp-last-inserted-right ts-right))))
|
|
|
|
;; for testing: (setq erc-timestamp-only-if-changed-flag nil)
|
|
|
|
(defun erc-format-timestamp (time format)
|
|
"Return TIME formatted as string according to FORMAT.
|
|
Return the empty string if FORMAT is nil."
|
|
(if format
|
|
(let ((ts (format-time-string format time)))
|
|
(erc-put-text-property 0 (length ts) 'face 'erc-timestamp-face ts)
|
|
(erc-put-text-property 0 (length ts) 'invisible 'timestamp ts)
|
|
(erc-put-text-property 0 (length ts)
|
|
'isearch-open-invisible 'timestamp ts)
|
|
;; N.B. Later use categories instead of this harmless, but
|
|
;; inelegant, hack. -- BPT
|
|
(and erc-timestamp-intangible
|
|
(not erc-hide-timestamps) ; bug#11706
|
|
(erc-put-text-property 0 (length ts) 'intangible t ts))
|
|
ts)
|
|
""))
|
|
|
|
;; This function is used to munge `buffer-invisibility-spec to an
|
|
;; appropriate value. Currently, it only handles timestamps, thus its
|
|
;; location. If you add other features which affect invisibility,
|
|
;; please modify this function and move it to a more appropriate
|
|
;; location.
|
|
(defun erc-munge-invisibility-spec ()
|
|
(if erc-hide-timestamps
|
|
(setq buffer-invisibility-spec
|
|
(if (listp buffer-invisibility-spec)
|
|
(cons 'timestamp buffer-invisibility-spec)
|
|
(list 't 'timestamp)))
|
|
(setq buffer-invisibility-spec
|
|
(if (listp buffer-invisibility-spec)
|
|
(remove 'timestamp buffer-invisibility-spec)
|
|
(list 't)))))
|
|
|
|
(defun erc-hide-timestamps ()
|
|
"Hide timestamp information from display."
|
|
(interactive)
|
|
(setq erc-hide-timestamps t)
|
|
(erc-munge-invisibility-spec))
|
|
|
|
(defun erc-show-timestamps ()
|
|
"Show timestamp information on display.
|
|
This function only works if `erc-timestamp-format' was previously
|
|
set, and timestamping is already active."
|
|
(interactive)
|
|
(setq erc-hide-timestamps nil)
|
|
(erc-munge-invisibility-spec))
|
|
|
|
(defun erc-toggle-timestamps ()
|
|
"Hide or show timestamps in ERC buffers.
|
|
|
|
Note that timestamps can only be shown for a message using this
|
|
function if `erc-timestamp-format' was set and timestamping was
|
|
enabled when the message was inserted."
|
|
(interactive)
|
|
(if erc-hide-timestamps
|
|
(setq erc-hide-timestamps nil)
|
|
(setq erc-hide-timestamps t))
|
|
(mapc (lambda (buffer)
|
|
(with-current-buffer buffer
|
|
(erc-munge-invisibility-spec)))
|
|
(erc-buffer-list)))
|
|
|
|
(defun erc-echo-timestamp (before now)
|
|
"Print timestamp text-property of an IRC message.
|
|
Argument BEFORE is where point was before it got moved and
|
|
NOW is position of point currently."
|
|
(when erc-echo-timestamps
|
|
(let ((stamp (get-text-property now 'timestamp)))
|
|
(when stamp
|
|
(message "%s" (format-time-string erc-echo-timestamp-format
|
|
stamp))))))
|
|
|
|
(provide 'erc-stamp)
|
|
|
|
;;; erc-stamp.el ends here
|
|
;;
|
|
;; Local Variables:
|
|
;; indent-tabs-mode: t
|
|
;; tab-width: 8
|
|
;; End:
|
|
|