123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710 |
- (defgroup time-stamp nil
- "Maintain last change time stamps in files edited by Emacs."
- :group 'data
- :group 'extensions)
- (defcustom time-stamp-format "%:y-%02m-%02d %02H:%02M:%02S %u"
- "Format of the string inserted by \\[time-stamp].
- The value may be a string or a list. Lists are supported only for
- backward compatibility; see variable `time-stamp-old-format-warn'.
- A string is used verbatim except for character sequences beginning
- with %, as follows. The values of non-numeric formatted items depend
- on the locale setting recorded in `system-time-locale' and
- `locale-coding-system'. The examples here are for the default
- \(`C') locale.
- %:a weekday name: `Monday'. %#A gives uppercase: `MONDAY'
- %3a abbreviated weekday: `Mon'. %3A gives uppercase: `MON'
- %:b month name: `January'. %#B gives uppercase: `JANUARY'
- %3b abbreviated month: `Jan'. %3B gives uppercase: `JAN'
- %02d day of month
- %02H 24-hour clock hour
- %02I 12-hour clock hour
- %02m month number
- %02M minute
- %#p `am' or `pm'. %P gives uppercase: `AM' or `PM'
- %02S seconds
- %w day number of week, Sunday is 0
- %02y 2-digit year: `03' %:y 4-digit year: `2003'
- %z time zone name: `est'. %Z gives uppercase: `EST'
- Non-date items:
- %% a literal percent character: `%'
- %f file name without directory %F gives absolute pathname
- %s system name
- %u user's login name %U user's full name
- %h mail host name
- Decimal digits between the % and the type character specify the
- field width. Strings are truncated on the right; years on the left.
- A leading zero in the field width zero-fills a number.
- For example, to get the format used by the `date' command,
- use \"%3a %3b %2d %02H:%02M:%02S %Z %:y\".
- In the future these formats will be aligned more with `format-time-string'.
- Because of this transition, the default padding for numeric formats will
- change in a future version. Therefore either a padding width should be
- specified, or the : modifier should be used to explicitly request the
- historical default."
- :type 'string
- :group 'time-stamp
- :version "20.1")
- (defcustom time-stamp-active t
- "Non-nil to enable time-stamping of buffers by \\[time-stamp].
- Can be toggled by \\[time-stamp-toggle-active].
- See also the variable `time-stamp-warn-inactive'."
- :type 'boolean
- :group 'time-stamp)
- (defcustom time-stamp-warn-inactive t
- "Have \\[time-stamp] warn if a buffer did not get time-stamped.
- If non-nil, a warning is displayed if `time-stamp-active' has
- deactivated time stamping and the buffer contains a template that
- otherwise would have been updated."
- :type 'boolean
- :group 'time-stamp
- :version "19.29")
- (defcustom time-stamp-old-format-warn 'ask
- "Action if `time-stamp-format' is an old-style list.
- If `error', the format is not used. If `ask', the user is queried about
- using the time-stamp-format. If `warn', a warning is displayed.
- If nil, no notification is given."
- :type '(choice (const :tag "Don't use the format" error)
- (const ask)
- (const warn)
- (const :tag "No notification" nil))
- :group 'time-stamp)
- (defcustom time-stamp-time-zone nil
- "If non-nil, a string naming the timezone to be used by \\[time-stamp].
- Format is the same as that used by the environment variable TZ on your system."
- :type '(choice (const nil) string)
- :group 'time-stamp
- :version "20.1")
- (defvar time-stamp-line-limit 8
- "Lines of a file searched; positive counts from start, negative from end.
- The patterns `time-stamp-start' and `time-stamp-end' must be found in
- the first (last) `time-stamp-line-limit' lines of the file for the
- file to be time-stamped by \\[time-stamp]. A value of 0 searches the
- entire buffer (use with care).
- This value can also be set with the variable `time-stamp-pattern'.
- Do not change `time-stamp-line-limit', `time-stamp-start',
- `time-stamp-end', or `time-stamp-pattern' for yourself or you will be
- incompatible with other people's files! If you must change them for some
- application, do so in the local variables section of the time-stamped file
- itself.")
- (defvar time-stamp-start "Time-stamp:[ \t]+\\\\?[\"<]+"
- "Regexp after which the time stamp is written by \\[time-stamp].
- See also the variables `time-stamp-end' and `time-stamp-line-limit'.
- This value can also be set with the variable `time-stamp-pattern'.
- Do not change `time-stamp-line-limit', `time-stamp-start',
- `time-stamp-end', or `time-stamp-pattern' for yourself or you will be
- incompatible with other people's files! If you must change them for some
- application, do so in the local variables section of the time-stamped file
- itself.")
- (defvar time-stamp-end "\\\\?[\">]"
- "Regexp marking the text after the time stamp.
- \\[time-stamp] deletes the text between the first match of `time-stamp-start'
- and the following match of `time-stamp-end', then writes the
- time stamp specified by `time-stamp-format' between them.
- This value can also be set with the variable `time-stamp-pattern'.
- The end text normally starts on the same line as the start text ends,
- but if there are any newlines in `time-stamp-format', the same number
- of newlines must separate the start and end. \\[time-stamp] tries
- to not change the number of lines in the buffer. `time-stamp-inserts-lines'
- controls this behavior.
- Do not change `time-stamp-start', `time-stamp-end', `time-stamp-pattern',
- or `time-stamp-inserts-lines' for yourself or you will be incompatible
- with other people's files! If you must change them for some application,
- do so in the local variables section of the time-stamped file itself.")
- (defvar time-stamp-inserts-lines nil
- "Whether \\[time-stamp] can change the number of lines in a file.
- If nil, \\[time-stamp] skips as many lines as there are newlines in
- `time-stamp-format' before looking for the `time-stamp-end' pattern,
- thus it tries not to change the number of lines in the buffer.
- If non-nil, \\[time-stamp] starts looking for the end pattern
- immediately after the start pattern. This behavior can cause
- unexpected changes in the buffer if used carelessly, but it is useful
- for generating repeated time stamps.
- Do not change `time-stamp-end' or `time-stamp-inserts-lines' for
- yourself or you will be incompatible with other people's files!
- If you must change them for some application, do so in the local
- variables section of the time-stamped file itself.")
- (defvar time-stamp-count 1
- "How many templates \\[time-stamp] will look for in a buffer.
- The same time stamp will be written in each case.
- Do not change `time-stamp-count' for yourself or you will be
- incompatible with other people's files! If you must change it for
- some application, do so in the local variables section of the
- time-stamped file itself.")
- (defvar time-stamp-pattern nil
- "Convenience variable setting all `time-stamp' location and format values.
- This string has four parts, each of which is optional.
- These four parts set `time-stamp-line-limit', `time-stamp-start',
- `time-stamp-format', and `time-stamp-end'. See the documentation
- for each of these variables for details.
- The first part is a number followed by a slash; the number sets the number
- of lines at the beginning (negative counts from end) of the file searched
- for the time stamp. The number and the slash may be omitted to use the
- normal value.
- The second part is a regexp identifying the pattern preceding the time stamp.
- This part may be omitted to use the normal pattern.
- The third part specifies the format of the time stamp inserted. See
- the documentation for `time-stamp-format' for details. Specify this
- part as \"%%\" to use the normal format.
- The fourth part is a regexp identifying the pattern following the time stamp.
- This part may be omitted to use the normal pattern.
- Examples:
- \"-10/\"
- \"-9/^Last modified: %%$\"
- \"@set Time-stamp: %:b %:d, %:y$\"
- \"newcommand{\\\\\\\\timestamp}{%%}\"
- Do not change `time-stamp-pattern' `time-stamp-line-limit',
- `time-stamp-start', or `time-stamp-end' for yourself or you will be
- incompatible with other people's files! If you must change them for
- some application, do so only in the local variables section of the
- time-stamped file itself.")
- (defun time-stamp ()
- "Update the time stamp string(s) in the buffer.
- A template in a file can be automatically updated with a new time stamp
- every time you save the file. Add this line to your .emacs file:
- (add-hook 'before-save-hook 'time-stamp)
- or customize `before-save-hook' through Custom.
- Normally the template must appear in the first 8 lines of a file and
- look like one of the following:
- Time-stamp: <>
- Time-stamp: \" \"
- The time stamp is written between the brackets or quotes:
- Time-stamp: <2001-02-18 10:20:51 gildea>
- The time stamp is updated only if the variable `time-stamp-active' is non-nil.
- The format of the time stamp is set by the variable `time-stamp-pattern' or
- `time-stamp-format'. The variables `time-stamp-pattern',
- `time-stamp-line-limit', `time-stamp-start', `time-stamp-end',
- `time-stamp-count', and `time-stamp-inserts-lines' control finding
- the template."
- (interactive)
- (let ((line-limit time-stamp-line-limit)
- (ts-start time-stamp-start)
- (ts-format time-stamp-format)
- (ts-end time-stamp-end)
- (ts-count time-stamp-count)
- (format-lines 0)
- (end-lines 1)
- (start nil)
- search-limit)
- (if (stringp time-stamp-pattern)
- (progn
- (string-match "\\`\\(\\(-?[0-9]+\\)/\\)?\\([^%]+\\)?\\(\\(%[-.,:@+_ #^()0-9]*[A-Za-z%][^%]*\\)*%[-.,:@+_ #^()0-9]*[A-Za-z%]\\)?\\([^%]+\\)?\\'" time-stamp-pattern)
- (and (match-beginning 2)
- (setq line-limit
- (string-to-number (match-string 2 time-stamp-pattern))))
- (and (match-beginning 3)
- (setq ts-start (match-string 3 time-stamp-pattern)))
- (and (match-beginning 4)
- (not (string-equal (match-string 4 time-stamp-pattern) "%%"))
- (setq ts-format (match-string 4 time-stamp-pattern)))
- (and (match-beginning 6)
- (setq ts-end (match-string 6 time-stamp-pattern)))))
- (cond ((not (integerp line-limit))
- (setq line-limit 8)
- (message "time-stamp-line-limit is not an integer")
- (sit-for 1)))
- (cond ((not (integerp ts-count))
- (setq ts-count 1)
- (message "time-stamp-count is not an integer")
- (sit-for 1))
- ((< ts-count 1)
-
-
- (setq ts-count 1)))
-
- (if (stringp ts-format)
- (let ((nl-start 0))
- (while (string-match "\n" ts-format nl-start)
- (setq format-lines (1+ format-lines) nl-start (match-end 0)))))
- (let ((nl-start 0))
- (while (string-match "\n" ts-end nl-start)
- (setq end-lines (1+ end-lines) nl-start (match-end 0))))
-
- (save-excursion
- (save-restriction
- (widen)
- (cond ((> line-limit 0)
- (goto-char (setq start (point-min)))
- (forward-line line-limit)
- (setq search-limit (point)))
- ((< line-limit 0)
- (goto-char (setq search-limit (point-max)))
- (forward-line line-limit)
- (setq start (point)))
- (t
- (setq start (point-min))
- (setq search-limit (point-max))))))
- (while (and start
- (< start search-limit)
- (> ts-count 0))
- (setq start (time-stamp-once start search-limit ts-start ts-end
- ts-format format-lines end-lines))
- (setq ts-count (1- ts-count))))
- nil)
- (defun time-stamp-once (start search-limit ts-start ts-end
- ts-format format-lines end-lines)
- "Update one time stamp. Internal routine called by \\[time-stamp].
- Returns the end point, which is where `time-stamp' begins the next search."
- (let ((case-fold-search nil)
- (end nil)
- end-search-start
- (end-length nil))
- (save-excursion
- (save-restriction
- (widen)
-
- (while (and (< (goto-char start) search-limit)
- (not end)
- (re-search-forward ts-start search-limit 'move))
- (setq start (point))
- (if (not time-stamp-inserts-lines)
- (forward-line format-lines))
- (setq end-search-start (max start (point)))
- (if (= (forward-line end-lines) 0)
- (progn
- (and (bolp) (backward-char))
- (let ((line-end (min (point) search-limit)))
- (if (>= line-end end-search-start)
- (progn
- (goto-char end-search-start)
- (if (re-search-forward ts-end line-end t)
- (progn
- (setq end (match-beginning 0))
- (setq end-length (- (match-end 0) end))))))))))))
- (if end
- (progn
-
- (cond
- ((not time-stamp-active)
- (if time-stamp-warn-inactive
-
- (progn
- (message "Warning: time-stamp-active is off; did not time-stamp buffer.")
- (sit-for 1))))
- ((not (and (stringp ts-start)
- (stringp ts-end)))
- (message "time-stamp-start or time-stamp-end is not a string")
- (sit-for 1))
- (t
- (let ((new-time-stamp (time-stamp-string ts-format)))
- (if (and (stringp new-time-stamp)
- (not (string-equal (buffer-substring start end)
- new-time-stamp)))
- (save-excursion
- (save-restriction
- (widen)
- (delete-region start end)
- (goto-char start)
- (insert-and-inherit new-time-stamp)
- (setq end (point))
-
- (if (search-backward "\t" start t)
- (progn
- (untabify start end)
- (setq end (point))))))))))))
-
- (and end end-length
- (+ end end-length))))
- (defun time-stamp-toggle-active (&optional arg)
- "Toggle `time-stamp-active', setting whether \\[time-stamp] updates a buffer.
- With ARG, turn time stamping on if and only if arg is positive."
- (interactive "P")
- (setq time-stamp-active
- (if (null arg)
- (not time-stamp-active)
- (> (prefix-numeric-value arg) 0)))
- (message "time-stamp is now %s." (if time-stamp-active "active" "off")))
- (defun time-stamp-string (&optional ts-format)
- "Generate the new string to be inserted by \\[time-stamp].
- Optionally use format TS-FORMAT instead of `time-stamp-format' to
- format the string."
- (or ts-format
- (setq ts-format time-stamp-format))
- (if (stringp ts-format)
- (if (stringp time-stamp-time-zone)
- (let ((ts-real-time-zone (getenv "TZ")))
- (unwind-protect
- (progn
- (setenv "TZ" time-stamp-time-zone)
- (format-time-string
- (time-stamp-string-preprocess ts-format)))
- (setenv "TZ" ts-real-time-zone)))
- (format-time-string
- (time-stamp-string-preprocess ts-format)))
-
- (cond ((or (eq time-stamp-old-format-warn 'error)
- (and (eq time-stamp-old-format-warn 'ask)
- (not (y-or-n-p "Use non-string time-stamp-format? "))))
- (message "Warning: no time-stamp: time-stamp-format not a string")
- (sit-for 1)
- nil)
- (t
- (cond ((eq time-stamp-old-format-warn 'warn)
- (message "Obsolescent time-stamp-format type; should be string")
- (sit-for 1)))
- (time-stamp-fconcat ts-format " ")))))
- (defconst time-stamp-no-file "(no file)"
- "String to use when the buffer is not associated with a file.")
- (defun time-stamp-string-preprocess (format &optional time)
- "Use a FORMAT to format date, time, file, and user information.
- Optional second argument TIME is only for testing.
- Implements non-time extensions to `format-time-string'
- and all `time-stamp-format' compatibility."
- (let ((fmt-len (length format))
- (ind 0)
- cur-char
- (prev-char nil)
- (result "")
- field-width
- field-result
- alt-form change-case
- (paren-level 0))
- (while (< ind fmt-len)
- (setq cur-char (aref format ind))
- (setq
- result
- (concat result
- (cond
- ((eq cur-char ?%)
-
- (setq alt-form nil change-case nil field-width "")
- (while (progn
- (setq ind (1+ ind))
- (setq cur-char (if (< ind fmt-len)
- (aref format ind)
- ?\0))
- (or (eq ?. cur-char)
- (eq ?, cur-char) (eq ?: cur-char) (eq ?@ cur-char)
- (eq ?- cur-char) (eq ?+ cur-char) (eq ?_ cur-char)
- (eq ?\s cur-char) (eq ?# cur-char) (eq ?^ cur-char)
- (and (eq ?\( cur-char)
- (not (eq prev-char ?\\))
- (setq paren-level (1+ paren-level)))
- (if (and (eq ?\) cur-char)
- (not (eq prev-char ?\\))
- (> paren-level 0))
- (setq paren-level (1- paren-level))
- (and (> paren-level 0)
- (< ind fmt-len)))
- (if (and (<= ?0 cur-char) (>= ?9 cur-char))
-
- (let ((field-index ind))
- (while (progn
- (setq ind (1+ ind))
- (setq cur-char (if (< ind fmt-len)
- (aref format ind)
- ?\0))
- (and (<= ?0 cur-char) (>= ?9 cur-char))))
- (setq field-width (substring format field-index ind))
- (setq ind (1- ind))
- t))))
- (setq prev-char cur-char)
-
- (cond ((eq cur-char ?:)
- (setq alt-form t))
- ((eq cur-char ?#)
- (setq change-case t))))
- (setq field-result
- (cond
- ((eq cur-char ?%)
- "%%")
- ((eq cur-char ?a)
- (if change-case
- (format-time-string "%#a" time)
- (or alt-form (not (string-equal field-width ""))
- (time-stamp-conv-warn "%a" "%:a"))
- (if (and alt-form (not (string-equal field-width "")))
- ""
- (format-time-string "%A" time))))
- ((eq cur-char ?A)
- (if alt-form
- (format-time-string "%A" time)
- (or change-case (not (string-equal field-width ""))
- (time-stamp-conv-warn "%A" "%#A"))
- (format-time-string "%#A" time)))
- ((eq cur-char ?b)
- (if change-case
- (format-time-string "%#b" time)
- (or alt-form (not (string-equal field-width ""))
- (time-stamp-conv-warn "%b" "%:b"))
- (if (and alt-form (not (string-equal field-width "")))
- ""
- (format-time-string "%B" time))))
- ((eq cur-char ?B)
- (if alt-form
- (format-time-string "%B" time)
- (or change-case (not (string-equal field-width ""))
- (time-stamp-conv-warn "%B" "%#B"))
- (format-time-string "%#B" time)))
- ((eq cur-char ?d)
- (time-stamp-do-number cur-char alt-form field-width time))
- ((eq cur-char ?H)
- (time-stamp-do-number cur-char alt-form field-width time))
- ((eq cur-char ?I)
- (time-stamp-do-number cur-char alt-form field-width time))
- ((eq cur-char ?m)
- (time-stamp-do-number cur-char alt-form field-width time))
- ((eq cur-char ?M)
- (time-stamp-do-number cur-char alt-form field-width time))
- ((eq cur-char ?p)
- (or change-case
- (time-stamp-conv-warn "%p" "%#p"))
- (format-time-string "%#p" time))
- ((eq cur-char ?P)
- (format-time-string "%p" time))
- ((eq cur-char ?S)
- (time-stamp-do-number cur-char alt-form field-width time))
- ((eq cur-char ?w)
- (format-time-string "%w" time))
- ((eq cur-char ?y)
- (or alt-form (not (string-equal field-width ""))
- (time-stamp-conv-warn "%y" "%:y"))
- (string-to-number (format-time-string "%Y" time)))
- ((eq cur-char ?Y)
- (string-to-number (format-time-string "%Y" time)))
- ((eq cur-char ?z)
- (if change-case
- ""
- (format-time-string "%#Z" time)))
- ((eq cur-char ?Z)
- (if change-case
- (format-time-string "%#Z" time)
- (format-time-string "%Z" time)))
- ((eq cur-char ?f)
- (if buffer-file-name
- (file-name-nondirectory buffer-file-name)
- time-stamp-no-file))
- ((eq cur-char ?F)
- (or buffer-file-name
- time-stamp-no-file))
- ((eq cur-char ?s)
- (system-name))
- ((eq cur-char ?u)
- (user-login-name))
- ((eq cur-char ?U)
- (user-full-name))
- ((eq cur-char ?l)
- (user-login-name))
- ((eq cur-char ?L)
- (user-full-name))
- ((eq cur-char ?h)
- (time-stamp-mail-host-name))
- ((eq cur-char ?q)
- (let ((qualname (system-name)))
- (if (string-match "\\." qualname)
- (substring qualname 0 (match-beginning 0))
- qualname)))
- ((eq cur-char ?Q)
- (system-name))
- ))
- (let ((padded-result
- (format (format "%%%s%c"
- field-width
- (if (numberp field-result) ?d ?s))
- (or field-result ""))))
- (let* ((initial-length (length padded-result))
- (desired-length (if (string-equal field-width "")
- initial-length
- (string-to-number field-width))))
- (if (> initial-length desired-length)
-
- (if (stringp field-result)
- (substring padded-result 0 desired-length)
- (if (eq cur-char ?y)
- (substring padded-result (- desired-length))
- padded-result))
- padded-result))))
- (t
- (char-to-string cur-char)))))
- (setq ind (1+ ind)))
- result))
- (defun time-stamp-do-number (format-char alt-form field-width time)
- "Handle compatible FORMAT-CHAR where only default width/padding will change.
- ALT-FORM is whether `#' specified. FIELD-WIDTH is the string
- width specification or \"\". TIME is the time to convert."
- (let ((format-string (concat "%" (char-to-string format-char))))
- (and (not alt-form) (string-equal field-width "")
- (time-stamp-conv-warn format-string
- (format "%%:%c" format-char)))
- (if (and alt-form (not (string-equal field-width "")))
- ""
- (string-to-number (format-time-string format-string time)))))
- (defvar time-stamp-conversion-warn t
- "Warn about soon-to-be-unsupported forms in `time-stamp-format'.
- If nil, these warnings are disabled, which would be a bad idea!
- You really need to update your files instead.
- The new formats will work with old versions of Emacs.
- New formats are being recommended now to allow `time-stamp-format'
- to change in the future to be compatible with `format-time-string'.
- The new forms being recommended now will continue to work then.")
- (defun time-stamp-conv-warn (old-form new-form)
- "Display a warning about a soon-to-be-obsolete format.
- Suggests replacing OLD-FORM with NEW-FORM."
- (cond
- (time-stamp-conversion-warn
- (with-current-buffer (get-buffer-create "*Time-stamp-compatibility*")
- (goto-char (point-max))
- (if (bobp)
- (progn
- (insert
- "The formats recognized in time-stamp-format will change in a future release\n"
- "to be compatible with the new, expanded format-time-string function.\n\n"
- "The following obsolescent time-stamp-format construct(s) were found:\n\n")))
- (insert "\"" old-form "\" -- use " new-form "\n"))
- (display-buffer "*Time-stamp-compatibility*"))))
- (defun time-stamp-mail-host-name ()
- "Return the name of the host where the user receives mail.
- This is the value of `mail-host-address' if bound and a string,
- otherwise the value of the function `system-name'."
- (or (and (boundp 'mail-host-address)
- (stringp mail-host-address)
- mail-host-address)
- (system-name)))
- (defun time-stamp-fconcat (list sep)
- "Similar to (mapconcat 'funcall LIST SEP) but LIST allows literals.
- If an element of LIST is a symbol, it is funcalled to get the string to use;
- the separator SEP is used between two strings obtained by funcalling a
- symbol. Otherwise the element itself is inserted; no separator is used
- around literals."
- (let ((return-string "")
- (insert-sep-p nil))
- (while list
- (cond ((symbolp (car list))
- (if insert-sep-p
- (setq return-string (concat return-string sep)))
- (setq return-string (concat return-string (funcall (car list))))
- (setq insert-sep-p t))
- (t
- (setq return-string (concat return-string (car list)))
- (setq insert-sep-p nil)))
- (setq list (cdr list)))
- return-string))
- (provide 'time-stamp)
|