bug-gnu-emacs
[Top][All Lists]
Advanced

[Date Prev][Date Next][Thread Prev][Thread Next][Date Index][Thread Index]

bug#41571: 27.0.91; "(elisp) Interpolated Strings" is under "(elisp) Tex


From: Basil L. Contovounesios
Subject: bug#41571: 27.0.91; "(elisp) Interpolated Strings" is under "(elisp) Text"
Date: Sun, 31 May 2020 10:24:46 +0100
User-agent: Gnus/5.13 (Gnus v5.13) Emacs/28.0.50 (gnu/linux)

Eli Zaretskii <eliz@gnu.org> writes:

>> From: "Basil L. Contovounesios" <contovob@tcd.ie>
>> Cc: 41571@debbugs.gnu.org
>> Date: Fri, 29 May 2020 19:35:31 +0100
>> 
>> How's the attached for emacs-27?
>
> Thanks, it's much better, but it still "needs work".

Thanks for the very helpful feedback.

> (Btw, why are you attachments appear before the text?  It makes
> responding harder, because I need to bring the citations to the
> front.)

Sorry, I got into the habit of doing that because I wasn't sure whether
attachments should go before or after the email signature.  I'm guessing
after?

>> +The functions described in this section accept a fixed set of
>> +specification characters.  The next section describes a function
>> +@code{format-spec} which accepts custom specification characters.
>
> This would benefit from making it clear what you mean by
> "specification characters".  An example would clarify that.
>
> (It is actually a general comment to your text: you frequently use
> terms that are left unexplained, which makes the reader stumble and
> try to understand what you mean.  Specific examples below.)

I agree that this sentence can be further clarified, but in general I
tried to reuse existing terminology and phrasing to the greatest extent
possible (maybe even too much).

E.g. the preceding paragraphs in "(elisp) Formatting Strings" define and
make extensive use of the terms "format string", "format specification",
"format specification character", as well as the shorter forms
"specification character" and "specification".

>> +It is, in some circumstances, useful to allow users to control how
>
> The beginning of this sentence is unnecessarily complex.  A much
> simpler variant would be
>
>   Sometimes it is useful to allow Lisp programs to control...
>
> Note that I replaced "users" with "Lisp  programs", since we are not
> talking about Emacs users here.

Done.

>> +A more convenient format string for such cases would be something like
>> +@code{"%f %l <%e>"}, where each specification character carries more
>> +semantic information and can easily be rearranged relative to other
>> +specification characters.  The function @code{format-spec} described
>> +in this section performs a similar function to @code{format}, except
>> +it operates on format control strings that comprise arbitrary
>> +specification characters.
>
> "comprise" => "include", or even just "use"

Done.

>> +@defun format-spec format specification &optional only-present
>> +This function returns a string equal to the format control string
>
> The "equal" here is confusing, because equality is not really
> important here, especially since the job of this function is to
> produce strings that are NOT equal to the original.

I agree; this is just a copy of the corresponding text from "(elisp)
Formatting Strings".

>> +@var{format}, replacing any format specifications it contains with
>> +values found in the alist @var{specification} (@pxref{Association
>> +Lists}).
>> +
>> +Each key in @var{specification} is a format specification character,
>> +and its associated value is the string to replace it with.  For
>> +example, an alist entry @code{(?a . "alpha")} means to replace any
>> +@samp{%a} specifications in @var{format} with @samp{alpha}.
>
> You say "key in SPECIFICATION", but SPECIFICATION is an alist, and a
> key in an alist has well-known meaning.  The "key" above should be
> "association".

"Each key in [the alist] is a [...] character" means each key in each
association in the alist is a character, so the wording is not wrong,
just unclear.

> And in general I'd rearrange the text to make the
> format of SPECIFICATION more explicit, something like:
>
>   @defun format-spec template spec-alist &optional only-present
>   This function returns a format string suitable for using in
>   @code{format} and similar functions.  The format string is produced
>   from @var{template} according to conversions specified in
>   @var{spec-alist}, which is an alist (@pxref{Association Lists}) of
>   the form @w{@code{(@var{letter} . @var{replacement})}}.  Each
>   specification @code{%@var{letter}} in @var{template} will be
>   replaced by @var{replacement} when producing the resulting format
>   string.

This wording is much clearer, but the description of the output is
wrong: 'format-spec' and 'format' both produce the same result - a
formatted string, not a format string.  'format-spec' is an alternative
to 'format', not a precursor.

>> +Some useful properties are gained as a result of @var{specification}
>> +being an alist.  The alist may contain more unique keys than there are
>> +unique specification characters in @var{format}; unused keys are
>> +simply ignored.  If the same key is contained more than once, the
>> +first one found is used.  If @var{format} contains the same format
>> +specification character more than once, then the same value found in
>> +@var{specification} is used as a basis for all of that character's
>> +substitutions.
>
> Here you use "key" without first explaining what it is.

I was relying on the preceding xref to the node on alists, which defines
the terms "alist", "key", and "associated value".

> Also, this paragraph describes several distinct features, so it is
> better to use an itemized list instead of just one sentence after
> another: it makes the description easier to grasp by dividing it into
> distinct smaller chunks.

Done, including mentioning that associations can appear in a different
order to their corresponding format specifications in the format string.

>> +The optional argument @var{only-present} indicates how to handle
>> +format specification characters in @var{format} that are not found in
>> +@var{specification}.  If it is @code{nil} or omitted, an error is
>> +emitted.
>
> Passive tense alert!  Suggest to rephrase
>
>   If it is @code{nil} or omitted, the function signals an error.

Fire extinguished.

>> +The syntax of format specifications accepted by @code{format-spec} is
>> +similar, but not identical, to that accepted by @code{format}.  In
>> +both cases, a format specification is a sequence of characters
>> +beginning with @samp{%} and ending with an alphabetic letter such as
>> +@samp{s}.  The only exception to this is the specification @samp{%%},
>> +which is replaced with a single @samp{%}.
>
> How is what's described in the last sentence "an exception"?  Format
> strings used by 'format' also behave like that, right?

It's an exception to "beginning with % and ending with a letter".

Would it be clearer if I said "the only specification that does not end
in a letter is %%, which is replaced with a single % in the output"?

>> +Unlike @code{format}, which assigns specific meanings to a fixed set
>> +of specification characters, @code{format-spec} accepts arbitrary
>> +specification characters and treats them all equally.  For example:
>> +
>> +@example
>> +(format-spec "su - %u %l"
>> +             `((?u . ,(user-login-name))
>> +               (?l . "ls")))
>> +     @result{} "su - foo ls"
>> +@end example
>
> This example stops short of explaining why this function is useful:
> making the replacements fixed strings, as in "ls", is not the reason.
> OTOH, the use of user-login-name is obfuscated by the backtick
> notation, which seems to say that some magic is needed here.
>
> I think the reason for having this function should be explained
> better, with more meaningful examples.

Hm, I'm not sure how to give a better existential justification; this
example just serves as a usage example.

The main use case for format-spec I've seen is where one part of the
program produces an alist with all the information that could ever be
needed, and another part of the program formats an often
user-customisable format string using this data.

An example of such a use case is in battery.el, where the alist produced
by battery-status-function is used to format battery-echo-area-format
and battery-mode-line-format (battery.el doesn't currently use
format-spec, but it could and my WIP patch for master changes that).

Would replicating such a use case make a better example?  E.g.:

  (setq my-site-info
        (list (cons ?s system-name)
              (cons ?t (symbol-name system-type))
              (cons ?c system-configuration)
              (cons ?v emacs-version)
              (cons ?e invocation-name)
              (cons ?p (number-to-string (emacs-pid)))
              (cons ?a user-mail-address)
              (cons ?n user-full-name)))

  (format-spec "%e %v (%c)" my-site-info)
    => "emacs 28.0.50 (x86_64-pc-linux-gnu)"

  (format-spec "%n <%a>" my-site-info)
    => "Emacs Developers <emacs-devel@gnu.org>"

>> +@item 0
>> +This flag causes any padding inserted by the width, if specified, to
>                                 ^^^^^^^^^^^^^^^^^^^^^
> Width cannot insert anything, so this should be reworded.  Same in a
> few other items.

Most of this phrasing is taken from "(elisp) Formatting Strings".

Is it clear enough to say "...causes any padding specified by the width
to..."?

>> +@item <
>> +This flag causes the substitution to be truncated to the given width,
>> +if specified, by removing characters from the left.
>
> "truncated ... by removing characters" is unnecessarily complicated.
> Why not say simply "truncated on the left"?

Done.

>> +@item >
>> +This flag causes the substitution to be truncated to the given width,
>> +if specified, by removing characters from the right.
>
> Same here.

Done.

>> +As is the case with @code{format}, a format specification can include
>> +a width, which is a decimal number that appears after any flags.  If a
>> +substitution contains fewer characters than its specified width, it is
>> +extended with padding, normally comprising spaces inserted on the^^^^^
>>  ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
>> +left:
>>  ^^^^
>
> "it is padded on the left" is simpler and more clear.

Done.

>> +Here is a more complicated example that combines several
>> +aforementioned features:
>> +
>> +@example
>> +(format-spec "%<06e %<06b"
>> +             '((?b . "beta")
>> +               (?e . "epsilon")))
>> +     @result{} "psilon 00beta"
>> +@end example
>
> Can we make this example be less trivial?  This use case doesn't
> justify using format-spec at all.

It's hard coming up with a simple enough example that both justifies
format-spec and showcases modifier combinations.  Would replicating a
subset of the output of battery-status-function be any good?  E.g.:

  (setq my-battery-info
        (list (cons ?p "73")      ; Percentage
              (cons ?L "Battery") ; Status
              (cons ?t "2:23")    ; Remaining time
              (cons ?c "24330")   ; Capacity
              (cons ?r "10.6")))  ; Rate of discharge

  (format-spec "%>^-3L : %3p%% (%05t left)" my-battery-info)
    => "BAT :  73% (02:23 left)"

  (format-spec "%>^-3L : %3p%% (%05t left)"
               (cons (cons ?L "AC")
                     my-battery-info))
    => "AC  :  73% (02:23 left)"

> Even the subtle point of having the format specs in the order
> different from the alist is not evident unless you make a point of
> mentioning it (something that IMO should have been done earlier in the
> description).

Done.

How's the new attached version?

-- 
Basil

Attachment: 0001-Improve-format-spec-documentation-bug-41571.patch
Description: Text Data


reply via email to

[Prev in Thread] Current Thread [Next in Thread]