* bug#56197: 28.1; lisp-fill-paragraph result regressed with Emacs 28
2022-06-24 16:17 bug#56197: 28.1; lisp-fill-paragraph result regressed with Emacs 28 Maxim Cournoyer
@ 2022-06-25 11:53 ` Lars Ingebrigtsen
2022-06-25 12:42 ` Eli Zaretskii
2022-06-27 1:53 ` bug#56197: 28.1; lisp-fill-paragraph result regressed with " Maxim Cournoyer
2025-01-04 13:03 ` bug#56197: [PATCH] lisp: Introduce lisp-fill-paragraph-as-displayed option Maxim Cournoyer
` (2 subsequent siblings)
3 siblings, 2 replies; 36+ messages in thread
From: Lars Ingebrigtsen @ 2022-06-25 11:53 UTC (permalink / raw)
To: Maxim Cournoyer; +Cc: 56197
Maxim Cournoyer <maxim.cournoyer@gmail.com> writes:
> ;; Emacs 28
> (description "IBus-Anthy is an engine for the input bus \"IBus\"). It adds the Anthy
> Japanese language input method to IBus. Because most graphical applications
> allow text input via IBus, installing this package will enable Japanese
> language input in most graphical applications.")
[...]
> Simply commenting out the newly added block, evaluating the defun and
> running it on my example reverts to the previous correct behavior.
I'm not sure the previous behaviour was any more correct. It's now
filling that string as if it, well, is a string, so that if you insert
it somewhere, the lines have similar lengths. The previous behaviour
was to fill "what you see in the buffer", which is wrong in most
contexts.
So I don't know. Anybody have an opinion?
--
(domestic pets only, the antidote for overdose, milk.)
bloggy blog: http://lars.ingebrigtsen.no
^ permalink raw reply [flat|nested] 36+ messages in thread
* bug#56197: 28.1; lisp-fill-paragraph result regressed with Emacs 28
2022-06-25 11:53 ` Lars Ingebrigtsen
@ 2022-06-25 12:42 ` Eli Zaretskii
2022-06-25 12:45 ` Lars Ingebrigtsen
2022-06-27 1:53 ` bug#56197: 28.1; lisp-fill-paragraph result regressed with " Maxim Cournoyer
1 sibling, 1 reply; 36+ messages in thread
From: Eli Zaretskii @ 2022-06-25 12:42 UTC (permalink / raw)
To: Lars Ingebrigtsen; +Cc: 56197, maxim.cournoyer
> Cc: 56197@debbugs.gnu.org
> From: Lars Ingebrigtsen <larsi@gnus.org>
> Date: Sat, 25 Jun 2022 13:53:44 +0200
>
> Maxim Cournoyer <maxim.cournoyer@gmail.com> writes:
>
> > ;; Emacs 28
> > (description "IBus-Anthy is an engine for the input bus \"IBus\"). It adds the Anthy
> > Japanese language input method to IBus. Because most graphical applications
> > allow text input via IBus, installing this package will enable Japanese
> > language input in most graphical applications.")
>
> [...]
>
> > Simply commenting out the newly added block, evaluating the defun and
> > running it on my example reverts to the previous correct behavior.
>
> I'm not sure the previous behaviour was any more correct. It's now
> filling that string as if it, well, is a string, so that if you insert
> it somewhere, the lines have similar lengths. The previous behaviour
> was to fill "what you see in the buffer", which is wrong in most
> contexts.
>
> So I don't know. Anybody have an opinion?
I actually don't understand what kind of "Lisp code" is the above
snippet. It doesn't look to me as valid Lisp code. So there's no
criteria for judging the correctness here, it seems.
^ permalink raw reply [flat|nested] 36+ messages in thread
* bug#56197: 28.1; lisp-fill-paragraph result regressed with Emacs 28
2022-06-25 12:42 ` Eli Zaretskii
@ 2022-06-25 12:45 ` Lars Ingebrigtsen
2022-06-25 12:48 ` Lars Ingebrigtsen
0 siblings, 1 reply; 36+ messages in thread
From: Lars Ingebrigtsen @ 2022-06-25 12:45 UTC (permalink / raw)
To: Eli Zaretskii; +Cc: 56197, maxim.cournoyer
Eli Zaretskii <eliz@gnu.org> writes:
> I actually don't understand what kind of "Lisp code" is the above
> snippet. It doesn't look to me as valid Lisp code. So there's no
> criteria for judging the correctness here, it seems.
It's just (foo "... very long multiline string"). I was also confused,
because the string looked very odd -- it has a ) inside the string, but
no (, but that's not really relevant.
--
(domestic pets only, the antidote for overdose, milk.)
bloggy blog: http://lars.ingebrigtsen.no
^ permalink raw reply [flat|nested] 36+ messages in thread
* bug#56197: 28.1; lisp-fill-paragraph result regressed with Emacs 28
2022-06-25 12:45 ` Lars Ingebrigtsen
@ 2022-06-25 12:48 ` Lars Ingebrigtsen
2022-06-25 13:00 ` Lars Ingebrigtsen
0 siblings, 1 reply; 36+ messages in thread
From: Lars Ingebrigtsen @ 2022-06-25 12:48 UTC (permalink / raw)
To: Eli Zaretskii; +Cc: 56197, maxim.cournoyer
Lars Ingebrigtsen <larsi@gnus.org> writes:
> It's just (foo "... very long multiline string"). I was also confused,
> because the string looked very odd -- it has a ) inside the string, but
> no (, but that's not really relevant.
Oh, hang on -- there definitely is a bug here. Here's a better test case:
(foo-bar-zot-foo-bar-zot-foo-bar-zot-foo-bar-zot-foo-bar-zot "This is a very long line This is a very long line This is a very long line This is a very long line This is a very long line.
And another long line.")
`M-q' in the first line does nothing, and it should. I'll have a look.
--
(domestic pets only, the antidote for overdose, milk.)
bloggy blog: http://lars.ingebrigtsen.no
^ permalink raw reply [flat|nested] 36+ messages in thread
* bug#56197: 28.1; lisp-fill-paragraph result regressed with Emacs 28
2022-06-25 12:48 ` Lars Ingebrigtsen
@ 2022-06-25 13:00 ` Lars Ingebrigtsen
2022-06-29 18:03 ` Stefan Kangas
0 siblings, 1 reply; 36+ messages in thread
From: Lars Ingebrigtsen @ 2022-06-25 13:00 UTC (permalink / raw)
To: Eli Zaretskii; +Cc: 56197, maxim.cournoyer
Lars Ingebrigtsen <larsi@gnus.org> writes:
> `M-q' in the first line does nothing, and it should. I'll have a look.
I've now fixed that, but the question still remains -- should
(foo-bar-zot-foo-bar-zot-foo-bar-zot-foo-bar-zot-foo-bar-zot "This is a very long line This is a very long line This is a very long line This")
be filled as
(foo-bar-zot-foo-bar-zot-foo-bar-zot-foo-bar-zot-foo-bar-zot "This is a very long line This is a very long line This is a very
long line This")
(i.e., fill the string using fill-column, or
(foo-bar-zot-foo-bar-zot-foo-bar-zot-foo-bar-zot-foo-bar-zot "This is a
very long line This is a very long line This is a very long line This")
(i.e., fill the text as is in the buffer).
--
(domestic pets only, the antidote for overdose, milk.)
bloggy blog: http://lars.ingebrigtsen.no
^ permalink raw reply [flat|nested] 36+ messages in thread
* bug#56197: 28.1; lisp-fill-paragraph result regressed with Emacs 28
2022-06-25 13:00 ` Lars Ingebrigtsen
@ 2022-06-29 18:03 ` Stefan Kangas
2022-06-30 9:32 ` Lars Ingebrigtsen
2024-12-25 20:15 ` Felix Lechner via Bug reports for GNU Emacs, the Swiss army knife of text editors
0 siblings, 2 replies; 36+ messages in thread
From: Stefan Kangas @ 2022-06-29 18:03 UTC (permalink / raw)
To: Lars Ingebrigtsen; +Cc: Eli Zaretskii, 56197, maxim.cournoyer
Lars Ingebrigtsen <larsi@gnus.org> writes:
> I've now fixed that, but the question still remains -- should
>
> (foo-bar-zot-foo-bar-zot-foo-bar-zot-foo-bar-zot-foo-bar-zot "This is a very long line This is a very long line This is a very long line This")
>
> be filled as
>
> (foo-bar-zot-foo-bar-zot-foo-bar-zot-foo-bar-zot-foo-bar-zot "This is a very long line This is a very long line This is a very
> long line This")
>
> (i.e., fill the string using fill-column, or
>
> (foo-bar-zot-foo-bar-zot-foo-bar-zot-foo-bar-zot-foo-bar-zot "This is a
> very long line This is a very long line This is a very long line This")
>
> (i.e., fill the text as is in the buffer).
The former sounds more useful, IMO. I don't want to mess up my strings
just to have pretty source code; I can make such adjustments manually
when I need to.
^ permalink raw reply [flat|nested] 36+ messages in thread
* bug#56197: 28.1; lisp-fill-paragraph result regressed with Emacs 28
2022-06-29 18:03 ` Stefan Kangas
@ 2022-06-30 9:32 ` Lars Ingebrigtsen
2022-06-30 9:33 ` Lars Ingebrigtsen
2022-06-30 11:31 ` Maxim Cournoyer
2024-12-25 20:15 ` Felix Lechner via Bug reports for GNU Emacs, the Swiss army knife of text editors
1 sibling, 2 replies; 36+ messages in thread
From: Lars Ingebrigtsen @ 2022-06-30 9:32 UTC (permalink / raw)
To: Stefan Kangas; +Cc: Eli Zaretskii, 56197, maxim.cournoyer
Stefan Kangas <stefan@marxist.se> writes:
> The former sounds more useful, IMO. I don't want to mess up my strings
> just to have pretty source code; I can make such adjustments manually
> when I need to.
So the votes are in, and I guess we'll keep the current behaviour in
Emacs 29, and I'm therefore closing this bug report.
--
(domestic pets only, the antidote for overdose, milk.)
bloggy blog: http://lars.ingebrigtsen.no
^ permalink raw reply [flat|nested] 36+ messages in thread
* bug#56197: 28.1; lisp-fill-paragraph result regressed with Emacs 28
2022-06-30 9:32 ` Lars Ingebrigtsen
@ 2022-06-30 9:33 ` Lars Ingebrigtsen
2024-12-26 15:16 ` Stefan Kangas
2022-06-30 11:31 ` Maxim Cournoyer
1 sibling, 1 reply; 36+ messages in thread
From: Lars Ingebrigtsen @ 2022-06-30 9:33 UTC (permalink / raw)
To: Stefan Kangas; +Cc: Eli Zaretskii, 56197, maxim.cournoyer
Lars Ingebrigtsen <larsi@gnus.org> writes:
> So the votes are in, and I guess we'll keep the current behaviour in
> Emacs 29, and I'm therefore closing this bug report.
Or -- perhaps it should be backported to Emacs 28 -- but not right now,
since the release is impending. So I'll keep this open until after
Emacs 28.2, I think.
--
(domestic pets only, the antidote for overdose, milk.)
bloggy blog: http://lars.ingebrigtsen.no
^ permalink raw reply [flat|nested] 36+ messages in thread
* bug#56197: 28.1; lisp-fill-paragraph result regressed with Emacs 28
2022-06-30 9:33 ` Lars Ingebrigtsen
@ 2024-12-26 15:16 ` Stefan Kangas
2024-12-28 5:52 ` Maxim Cournoyer
0 siblings, 1 reply; 36+ messages in thread
From: Stefan Kangas @ 2024-12-26 15:16 UTC (permalink / raw)
To: Lars Ingebrigtsen; +Cc: Eli Zaretskii, maxim.cournoyer, 56197-done
Lars Ingebrigtsen <larsi@gnus.org> writes:
> Lars Ingebrigtsen <larsi@gnus.org> writes:
>
>> So the votes are in, and I guess we'll keep the current behaviour in
>> Emacs 29, and I'm therefore closing this bug report.
>
> Or -- perhaps it should be backported to Emacs 28 -- but not right now,
> since the release is impending. So I'll keep this open until after
> Emacs 28.2, I think.
Emacs 28 was released on 2022-04-04, so I'm closing this bug now.
^ permalink raw reply [flat|nested] 36+ messages in thread
* bug#56197: 28.1; lisp-fill-paragraph result regressed with Emacs 28
2024-12-26 15:16 ` Stefan Kangas
@ 2024-12-28 5:52 ` Maxim Cournoyer
2024-12-28 8:47 ` Eli Zaretskii
2024-12-28 8:52 ` Stefan Kangas
0 siblings, 2 replies; 36+ messages in thread
From: Maxim Cournoyer @ 2024-12-28 5:52 UTC (permalink / raw)
To: Stefan Kangas; +Cc: Lars Ingebrigtsen, Eli Zaretskii, 56197-done
Hi Stefan,
Stefan Kangas <stefankangas@gmail.com> writes:
> Lars Ingebrigtsen <larsi@gnus.org> writes:
>
>> Lars Ingebrigtsen <larsi@gnus.org> writes:
>>
>>> So the votes are in, and I guess we'll keep the current behaviour in
>>> Emacs 29, and I'm therefore closing this bug report.
>>
>> Or -- perhaps it should be backported to Emacs 28 -- but not right now,
>> since the release is impending. So I'll keep this open until after
>> Emacs 28.2, I think.
>
> Emacs 28 was released on 2022-04-04, so I'm closing this bug now.
The issue says "regressed in", not "affects only", for what it's worth.
The regression/change in behavior still applies to the current release
of Emacs.
--
Thanks,
Maxim
^ permalink raw reply [flat|nested] 36+ messages in thread
* bug#56197: 28.1; lisp-fill-paragraph result regressed with Emacs 28
2024-12-28 5:52 ` Maxim Cournoyer
@ 2024-12-28 8:47 ` Eli Zaretskii
2024-12-28 14:51 ` Maxim Cournoyer
2024-12-28 8:52 ` Stefan Kangas
1 sibling, 1 reply; 36+ messages in thread
From: Eli Zaretskii @ 2024-12-28 8:47 UTC (permalink / raw)
To: Maxim Cournoyer; +Cc: larsi, stefankangas, 56197
> From: Maxim Cournoyer <maxim.cournoyer@gmail.com>
> Cc: Lars Ingebrigtsen <larsi@gnus.org>, Eli Zaretskii <eliz@gnu.org>,
> 56197-done@debbugs.gnu.org
> Date: Sat, 28 Dec 2024 14:52:52 +0900
>
> Hi Stefan,
>
> Stefan Kangas <stefankangas@gmail.com> writes:
>
> > Lars Ingebrigtsen <larsi@gnus.org> writes:
> >
> >> Lars Ingebrigtsen <larsi@gnus.org> writes:
> >>
> >>> So the votes are in, and I guess we'll keep the current behaviour in
> >>> Emacs 29, and I'm therefore closing this bug report.
> >>
> >> Or -- perhaps it should be backported to Emacs 28 -- but not right now,
> >> since the release is impending. So I'll keep this open until after
> >> Emacs 28.2, I think.
> >
> > Emacs 28 was released on 2022-04-04, so I'm closing this bug now.
>
> The issue says "regressed in", not "affects only", for what it's worth.
> The regression/change in behavior still applies to the current release
> of Emacs.
We don't agree it's a regression, as shown in the discussion of this
bug.
^ permalink raw reply [flat|nested] 36+ messages in thread
* bug#56197: 28.1; lisp-fill-paragraph result regressed with Emacs 28
2024-12-28 8:47 ` Eli Zaretskii
@ 2024-12-28 14:51 ` Maxim Cournoyer
0 siblings, 0 replies; 36+ messages in thread
From: Maxim Cournoyer @ 2024-12-28 14:51 UTC (permalink / raw)
To: Eli Zaretskii; +Cc: larsi, stefankangas, 56197
Hello Eli,
Eli Zaretskii <eliz@gnu.org> writes:
[...]
>> > Emacs 28 was released on 2022-04-04, so I'm closing this bug now.
>>
>> The issue says "regressed in", not "affects only", for what it's worth.
>> The regression/change in behavior still applies to the current release
>> of Emacs.
>
> We don't agree it's a regression, as shown in the discussion of this
> bug.
We can call it 'change in behavior'; I agree in most cases the new
behavior is probably good. I don't see us being in disagreement; I
think Felix and I are pointing that the change has impacted negatively
at least one use case, and we're exploring ideas for a better solution
to that than pasting the previous copy in a project's .dir-locals.el
file (I'll explain in your other question why I see this as suboptimal).
--
Thanks,
Maxim
^ permalink raw reply [flat|nested] 36+ messages in thread
* bug#56197: 28.1; lisp-fill-paragraph result regressed with Emacs 28
2024-12-28 5:52 ` Maxim Cournoyer
2024-12-28 8:47 ` Eli Zaretskii
@ 2024-12-28 8:52 ` Stefan Kangas
1 sibling, 0 replies; 36+ messages in thread
From: Stefan Kangas @ 2024-12-28 8:52 UTC (permalink / raw)
To: Maxim Cournoyer; +Cc: Lars Ingebrigtsen, Eli Zaretskii, 56197
reopen 56197
thanks
Maxim Cournoyer <maxim.cournoyer@gmail.com> writes:
> Hi Stefan,
>
> Stefan Kangas <stefankangas@gmail.com> writes:
>
>> Emacs 28 was released on 2022-04-04, so I'm closing this bug now.
>
> The issue says "regressed in", not "affects only", for what it's worth.
> The regression/change in behavior still applies to the current release
> of Emacs.
Sorry for misunderstanding. I'm reopening the bug with this message.
^ permalink raw reply [flat|nested] 36+ messages in thread
* bug#56197: 28.1; lisp-fill-paragraph result regressed with Emacs 28
2022-06-30 9:32 ` Lars Ingebrigtsen
2022-06-30 9:33 ` Lars Ingebrigtsen
@ 2022-06-30 11:31 ` Maxim Cournoyer
2022-07-01 9:05 ` Lars Ingebrigtsen
1 sibling, 1 reply; 36+ messages in thread
From: Maxim Cournoyer @ 2022-06-30 11:31 UTC (permalink / raw)
To: Lars Ingebrigtsen, Stefan Kangas; +Cc: Eli Zaretskii, 56197
On June 30, 2022 5:32:08 AM EDT, Lars Ingebrigtsen <larsi@gnus.org> wrote:
>Stefan Kangas <stefan@marxist.se> writes:
>
>> The former sounds more useful, IMO. I don't want to mess up my strings
>> just to have pretty source code; I can make such adjustments manually
>> when I need to.
>
>So the votes are in, and I guess we'll keep the current behaviour in
>Emacs 29, and I'm therefore closing this bug report.
>
Just to make sure, this means the paragraph filling behavior seen in the examples I shared would be unchanged, and different from that in Emacs 27 and earlier, correct?
To me, the previous behavior was useful; is there a way to configure Emacs to behave that way with Emacs 28?
Thank you!
Maxim
Hi,
^ permalink raw reply [flat|nested] 36+ messages in thread
* bug#56197: 28.1; lisp-fill-paragraph result regressed with Emacs 28
2022-06-30 11:31 ` Maxim Cournoyer
@ 2022-07-01 9:05 ` Lars Ingebrigtsen
0 siblings, 0 replies; 36+ messages in thread
From: Lars Ingebrigtsen @ 2022-07-01 9:05 UTC (permalink / raw)
To: Maxim Cournoyer; +Cc: Eli Zaretskii, Stefan Kangas, 56197
Maxim Cournoyer <maxim.cournoyer@gmail.com> writes:
> Just to make sure, this means the paragraph filling behavior seen in
> the examples I shared would be unchanged, and different from that in
> Emacs 27 and earlier, correct?
Yup.
> To me, the previous behavior was useful; is there a way to configure
> Emacs to behave that way with Emacs 28?
I guess you can set fill-paragraph-function to a function that does what
you want?
--
(domestic pets only, the antidote for overdose, milk.)
bloggy blog: http://lars.ingebrigtsen.no
^ permalink raw reply [flat|nested] 36+ messages in thread
* bug#56197: 28.1; lisp-fill-paragraph result regressed with Emacs 28
2022-06-29 18:03 ` Stefan Kangas
2022-06-30 9:32 ` Lars Ingebrigtsen
@ 2024-12-25 20:15 ` Felix Lechner via Bug reports for GNU Emacs, the Swiss army knife of text editors
2024-12-26 6:21 ` Eli Zaretskii
1 sibling, 1 reply; 36+ messages in thread
From: Felix Lechner via Bug reports for GNU Emacs, the Swiss army knife of text editors @ 2024-12-25 20:15 UTC (permalink / raw)
To: 56197; +Cc: Lars Ingebrigtsen, Eli Zaretskii, Stefan Kangas, Maxim Cournoyer
Hi everyone,
> fill the string using fill-column, or [...] fill the text as is in the
> buffer
> It's now filling that string as if it, well, is a string, so that if
> you insert it somewhere, the lines have similar lengths. The previous
> behaviour was to fill "what you see in the buffer"
> The former sounds more useful, IMO. I don't want to mess up my strings
> just to have pretty source code
Filling strings in code would be useful, but isn't that a separate,
don't-break-my-strings feature?
Historically, the point of text justification is to make text fit on a
screen. For example, the documentation for fill-region refers to
columns, which are features of buffers:
Column beyond which automatic line-wrapping should happen.
Auto-fill-mode is consistent:
inserting a space at a column beyond current-fill-column
automatically breaks the line
In a grand sweep, the manual explains what needs to fit where:
“Filling” text means breaking it up into lines that fit a specified
width.
Section 26.6.2 ("Explicit Fill Commands") is even more, well, explicit:
The command ‘M-q’ (‘fill-paragraph’) “fills” the current paragraph.
It redistributes the line breaks within the paragraph, and deletes
any excess space and tab characters occurring within the paragraph,
in such a way that the lines end up fitting within a certain maximum
width.
How text shows on a screen is clearly a central feature. The manual
continues:
The maximum line width for filling is specified by the buffer-local
variable ‘fill-column’. The default value (*note Locals::) is 70.
The easiest way to set ‘fill-column’ in the current buffer is to use
the command ‘C-x f’ (‘set-fill-column’). [...] Note that, by its
very nature, ‘fill-column’ is measured in column units; the actual
position of that column on a graphical display depends on the font
being used. In particular, using variable-pitch fonts will cause
the ‘fill-column’ occupy different horizontal positions on display
in different lines.
In my view, the string interpretation calls for a different, though
related feature.
Maybe there could be (setq fill-strings-instead-of-text t) ? Thanks!
Kind regards
Felix
^ permalink raw reply [flat|nested] 36+ messages in thread
* bug#56197: 28.1; lisp-fill-paragraph result regressed with Emacs 28
2024-12-25 20:15 ` Felix Lechner via Bug reports for GNU Emacs, the Swiss army knife of text editors
@ 2024-12-26 6:21 ` Eli Zaretskii
2024-12-28 5:26 ` Maxim Cournoyer
0 siblings, 1 reply; 36+ messages in thread
From: Eli Zaretskii @ 2024-12-26 6:21 UTC (permalink / raw)
To: Felix Lechner; +Cc: larsi, 56197, maxim.cournoyer, stefankangas
> From: Felix Lechner <felix.lechner@lease-up.com>
> Cc: Maxim Cournoyer <maxim.cournoyer@gmail.com>, Stefan Kangas
> <stefankangas@gmail.com>, Eli Zaretskii <eliz@gnu.org>, Lars Ingebrigtsen
> <larsi@gnus.org>
> Date: Wed, 25 Dec 2024 12:15:44 -0800
>
> > fill the string using fill-column, or [...] fill the text as is in the
> > buffer
>
> > It's now filling that string as if it, well, is a string, so that if
> > you insert it somewhere, the lines have similar lengths. The previous
> > behaviour was to fill "what you see in the buffer"
>
> > The former sounds more useful, IMO. I don't want to mess up my strings
> > just to have pretty source code
>
> Filling strings in code would be useful, but isn't that a separate,
> don't-break-my-strings feature?
Not necessarily. I frequently fill stuff in my code, and don't want
to use a separate command if the region I fill includes strings (or
comments, or something other that needs special filling behavior).
> Historically, the point of text justification is to make text fit on a
> screen. For example, the documentation for fill-region refers to
> columns, which are features of buffers:
>
> Column beyond which automatic line-wrapping should happen.
>
> Auto-fill-mode is consistent:
>
> inserting a space at a column beyond current-fill-column
> automatically breaks the line
>
> In a grand sweep, the manual explains what needs to fit where:
>
> “Filling” text means breaking it up into lines that fit a specified
> width.
>
> Section 26.6.2 ("Explicit Fill Commands") is even more, well, explicit:
>
> The command ‘M-q’ (‘fill-paragraph’) “fills” the current paragraph.
> It redistributes the line breaks within the paragraph, and deletes
> any excess space and tab characters occurring within the paragraph,
> in such a way that the lines end up fitting within a certain maximum
> width.
>
> How text shows on a screen is clearly a central feature. The manual
> continues:
>
> The maximum line width for filling is specified by the buffer-local
> variable ‘fill-column’. The default value (*note Locals::) is 70.
> The easiest way to set ‘fill-column’ in the current buffer is to use
> the command ‘C-x f’ (‘set-fill-column’). [...] Note that, by its
> very nature, ‘fill-column’ is measured in column units; the actual
> position of that column on a graphical display depends on the font
> being used. In particular, using variable-pitch fonts will cause
> the ‘fill-column’ occupy different horizontal positions on display
> in different lines.
>
> In my view, the string interpretation calls for a different, though
> related feature.
You are quoting text which talks about the _default_ filling. The
default filling is tailored to "uniform" text, i.e. really to Text
mode and its descendants.
However, Emacs lets major modes customize filling as appropriate for
the mode, by defining mode-specific filling functions. Which is what
happens in this case: lisp-mode.el defines a fill-paragraph function
that is specific to Lisp modes. It is completely legitimate for such
mode-specific functions to special-case strings inside code and do
something special about that.
So I don't see how what we do now is against the spirit of filling.
(Btw, I think it's high time we closed that bug, since Emacs 28.2 was
released long ago.)
^ permalink raw reply [flat|nested] 36+ messages in thread
* bug#56197: 28.1; lisp-fill-paragraph result regressed with Emacs 28
2024-12-26 6:21 ` Eli Zaretskii
@ 2024-12-28 5:26 ` Maxim Cournoyer
2024-12-28 8:45 ` Eli Zaretskii
0 siblings, 1 reply; 36+ messages in thread
From: Maxim Cournoyer @ 2024-12-28 5:26 UTC (permalink / raw)
To: Eli Zaretskii; +Cc: larsi, 56197, Felix Lechner, stefankangas
Hello,
Eli Zaretskii <eliz@gnu.org> writes:
[...]
>> Filling strings in code would be useful, but isn't that a separate,
>> don't-break-my-strings feature?
>
> Not necessarily. I frequently fill stuff in my code, and don't want
> to use a separate command if the region I fill includes strings (or
> comments, or something other that needs special filling behavior).
I agree that having a single way to 'fill text/code/whatever' is nice.
[...]
> You are quoting text which talks about the _default_ filling. The
> default filling is tailored to "uniform" text, i.e. really to Text
> mode and its descendants.
>
> However, Emacs lets major modes customize filling as appropriate for
> the mode, by defining mode-specific filling functions. Which is what
> happens in this case: lisp-mode.el defines a fill-paragraph function
> that is specific to Lisp modes. It is completely legitimate for such
> mode-specific functions to special-case strings inside code and do
> something special about that.
>
> So I don't see how what we do now is against the spirit of filling.
Maybe it doesn't in general, but in the case of Guix packages, it's
strikingly odd compared to the past behavior which was respecting the
fill-column in a long multi-line string (refer to the original issue I
had detailed in this thread for an example).
> (Btw, I think it's high time we closed that bug, since Emacs 28.2 was
> released long ago.)
It's a change of behavior introduced in version 28 that apparently
doesn't make unanimity (though perhaps noticed mostly by Guix
developers, where the use of the multi-line strings for package
descriptions makes this issue very visible and annoying); the GNU Guix
project has been carrying this in its .dir-locals.el file, which simply
reverts to the old behavior before commit 9bf367e1848:
--8<---------------cut here---------------start------------->8---
;; Emacs 28 changed the behavior of 'lisp-fill-paragraph', which causes the
;; first line of package descriptions to extrude past 'fill-column', and
;; somehow that is deemed more correct upstream (see:
;; https://issues.guix.gnu.org/56197).
(eval . (progn
(require 'lisp-mode)
(defun emacs27-lisp-fill-paragraph (&optional justify)
(interactive "P")
(or (fill-comment-paragraph justify)
(let ((paragraph-start
(concat paragraph-start
"\\|\\s-*\\([(;\"]\\|\\s-:\\|`(\\|#'(\\)"))
(paragraph-separate
(concat paragraph-separate "\\|\\s-*\".*[,\\.]$"))
(fill-column (if (and (integerp emacs-lisp-docstring-fill-column)
(derived-mode-p 'emacs-lisp-mode))
emacs-lisp-docstring-fill-column
fill-column)))
(fill-paragraph justify))
;; Never return nil.
t))
(setq-local fill-paragraph-function #'emacs27-lisp-fill-paragraph)))
--8<---------------cut here---------------end--------------->8---
I can't say it feels very satisfactory; a switch like one imagined by
Felix could be a step in the right direction; it'd be at least more
concise in the project .dir-locals. Would a patch implementing that be
welcome?
Happy New Year,
--
Maxim
^ permalink raw reply [flat|nested] 36+ messages in thread
* bug#56197: 28.1; lisp-fill-paragraph result regressed with Emacs 28
2024-12-28 5:26 ` Maxim Cournoyer
@ 2024-12-28 8:45 ` Eli Zaretskii
2024-12-28 15:14 ` Maxim Cournoyer
0 siblings, 1 reply; 36+ messages in thread
From: Eli Zaretskii @ 2024-12-28 8:45 UTC (permalink / raw)
To: Maxim Cournoyer; +Cc: larsi, 56197, felix.lechner, stefankangas
> From: Maxim Cournoyer <maxim.cournoyer@gmail.com>
> Cc: Felix Lechner <felix.lechner@lease-up.com>, 56197@debbugs.gnu.org,
> stefankangas@gmail.com, larsi@gnus.org
> Date: Sat, 28 Dec 2024 14:26:32 +0900
>
> I can't say it feels very satisfactory; a switch like one imagined by
> Felix could be a step in the right direction; it'd be at least more
> concise in the project .dir-locals. Would a patch implementing that be
> welcome?
I don't see how a user option to control this could be useful, since
the preferred behavior is not only buffer-local, but also specific to
certain syntactic constructs. But I won't object to having such an
option.
I also don't see what's wrong with the solution of having a special
function in .dir-locals.el. We don't pretend that the default Emacs
behavior should necessarily fit all the use cases, and provide ample
opportunities for customizing Emacs behavior for that reason.
Defining a custom fill-paragraph function is a perfectly valid
solution, not very different from having a user option for the same
purpose. So I'm not sure I understand why you prefer adding an
option, when the Guix project has already solved the problem in a
perfectly legitimate and clean way.
^ permalink raw reply [flat|nested] 36+ messages in thread
* bug#56197: 28.1; lisp-fill-paragraph result regressed with Emacs 28
2024-12-28 8:45 ` Eli Zaretskii
@ 2024-12-28 15:14 ` Maxim Cournoyer
2025-01-04 10:09 ` bug#56197: lisp-fill-paragraph behavior changed in " Maxim Cournoyer
0 siblings, 1 reply; 36+ messages in thread
From: Maxim Cournoyer @ 2024-12-28 15:14 UTC (permalink / raw)
To: Eli Zaretskii; +Cc: larsi, 56197, felix.lechner, stefankangas
Hello,
Eli Zaretskii <eliz@gnu.org> writes:
>> From: Maxim Cournoyer <maxim.cournoyer@gmail.com>
>> Cc: Felix Lechner <felix.lechner@lease-up.com>, 56197@debbugs.gnu.org,
>> stefankangas@gmail.com, larsi@gnus.org
>> Date: Sat, 28 Dec 2024 14:26:32 +0900
>>
>> I can't say it feels very satisfactory; a switch like one imagined by
>> Felix could be a step in the right direction; it'd be at least more
>> concise in the project .dir-locals. Would a patch implementing that be
>> welcome?
>
> I don't see how a user option to control this could be useful, since
> the preferred behavior is not only buffer-local, but also specific to
> certain syntactic constructs. But I won't object to having such an
> option.
Having the behavior defined per-project or even globally (reverting to
the the pre-Emacs 28 behavior) via a simple option seems like it'd
simplify things, and make them discoverable.
> I also don't see what's wrong with the solution of having a special
> function in .dir-locals.el. We don't pretend that the default Emacs
> behavior should necessarily fit all the use cases, and provide ample
> opportunities for customizing Emacs behavior for that reason.
> Defining a custom fill-paragraph function is a perfectly valid
> solution, not very different from having a user option for the same
> purpose. So I'm not sure I understand why you prefer adding an
> option, when the Guix project has already solved the problem in a
> perfectly legitimate and clean way.
For one, having to put that largish piece of evaluated code in the
.dir-locals.el of the project means each new developer is prompted to
trust it, and it makes it more intimidating/pollutes their user conf
file (being added to the 'safe-local-variable-values' value).
It's also not discoverable; a customizable option would help in this
regard.
--
Thanks,
Maxim
^ permalink raw reply [flat|nested] 36+ messages in thread
* bug#56197: lisp-fill-paragraph behavior changed in Emacs 28
2024-12-28 15:14 ` Maxim Cournoyer
@ 2025-01-04 10:09 ` Maxim Cournoyer
2025-01-04 10:14 ` Maxim Cournoyer
2025-01-04 14:04 ` Eli Zaretskii
0 siblings, 2 replies; 36+ messages in thread
From: Maxim Cournoyer @ 2025-01-04 10:09 UTC (permalink / raw)
To: Eli Zaretskii; +Cc: larsi, 56197, felix.lechner, stefankangas
[-- Attachment #1: Type: text/plain, Size: 1176 bytes --]
Hi Eli et al.,
Maxim Cournoyer <maxim.cournoyer@gmail.com> writes:
[...]
>> I don't see how a user option to control this could be useful, since
>> the preferred behavior is not only buffer-local, but also specific to
>> certain syntactic constructs. But I won't object to having such an
>> option.
>
> Having the behavior defined per-project or even globally (reverting to
> the the pre-Emacs 28 behavior) via a simple option seems like it'd
> simplify things, and make them discoverable.
I tried fixing this generally, as it seems to me that something in
lisp-mode should be meet the needs of all lisp-derived languages such as
Scheme and not just Elisp. I first added two tests, one of which
ensures no regression to the original bug that lead to this current
behavioral change (bug#28937) and the other one that should pass once
the issue reported here (bug#56197) is resolved.
The last patch is a WIP that didn't work; I was hoping that inserting
spaces corresponding to the width of the indent in the narrowed string
would cause the indent to be preserved only for the first line. I don't
have other ideas at the moment; I'd appreciate if someone could tip in.
[-- Warning: decoded text below may be mangled, UTF-8 assumed --]
[-- Attachment #2: 0001-test-lisp-Add-new-test-for-filling-behavior-change.patch --]
[-- Type: text/x-patch, Size: 2115 bytes --]
From aacf65440070df2ba356af1d118a50993fc8f865 Mon Sep 17 00:00:00 2001
From: Maxim Cournoyer <maxim.cournoyer@gmail.com>
Date: Mon, 30 Dec 2024 12:02:36 +0900
Subject: [PATCH 1/3] test/lisp: Add new test for filling behavior change.
Starting with Emacs 28, filling strings now happens in a narrowed scope,
and looses the leading indentation and can cause the string to extend
past the fill-column value (see bug#56197).
* test/lisp/emacs-lisp/lisp-mode-tests.el
(lisp-fill-paragraph-respects-fill-column): New test.
---
test/lisp/emacs-lisp/lisp-mode-tests.el | 27 +++++++++++++++++++++++++
1 file changed, 27 insertions(+)
diff --git a/test/lisp/emacs-lisp/lisp-mode-tests.el b/test/lisp/emacs-lisp/lisp-mode-tests.el
index da02be65d03..9a2b1ea4654 100644
--- a/test/lisp/emacs-lisp/lisp-mode-tests.el
+++ b/test/lisp/emacs-lisp/lisp-mode-tests.el
@@ -308,6 +308,33 @@ lisp-indent-defun
(indent-region (point-min) (point-max))
(should (equal (buffer-string) orig)))))
+\f
+;;; Filling
+
+(ert-deftest lisp-fill-paragraph-respects-fill-column ()
+ "Test bug#56197 -- more specifically, validate that a leading indentation
+for a string is preserved in the filled string."
+ (with-temp-buffer
+ ;; The following is a contrived example that demonstrates the
+ ;; fill-column problem when the string to fill is indented.
+ (insert "\
+\(defun test ()
+ \"This is a long string which is indented by a considerable value,
+causing it to protrude from the configured `fill-column' since
+lisp-fill-paragraph was refactored in version 28.\"
+ (message \"Hi!\"))")
+ (emacs-lisp-mode)
+ (search-backward "This is a long string")
+ (fill-paragraph) ;function under test
+ (goto-char (point-min))
+ (let ((i 1)
+ (lines-count (count-lines (point-min) (point-max))))
+ (while (< i lines-count)
+ (beginning-of-line i)
+ (end-of-line)
+ (should (<= (current-column) fill-column))
+ (setq i (1+ i))))))
+
\f
;;; Fontification
base-commit: e32484547d0813665334bfd34b741492dde0d374
--
2.46.0
[-- Warning: decoded text below may be mangled, UTF-8 assumed --]
[-- Attachment #3: 0002-test-lisp-Add-a-test-checking-docstring-boundaries-w.patch --]
[-- Type: text/x-patch, Size: 1436 bytes --]
From e5685497111ef71e57948f023f8d2a03647c3d69 Mon Sep 17 00:00:00 2001
From: Maxim Cournoyer <maxim.cournoyer@gmail.com>
Date: Mon, 30 Dec 2024 14:26:46 +0900
Subject: [PATCH 2/3] test/lisp: Add a test checking docstring boundaries when
filling.
* test/lisp/emacs-lisp/lisp-mode-tests.el
(lisp-fill-paragraph-docstring-boundaries): New test.
---
test/lisp/emacs-lisp/lisp-mode-tests.el | 19 +++++++++++++++++++
1 file changed, 19 insertions(+)
diff --git a/test/lisp/emacs-lisp/lisp-mode-tests.el b/test/lisp/emacs-lisp/lisp-mode-tests.el
index 9a2b1ea4654..eac2763c595 100644
--- a/test/lisp/emacs-lisp/lisp-mode-tests.el
+++ b/test/lisp/emacs-lisp/lisp-mode-tests.el
@@ -311,6 +311,25 @@ lisp-indent-defun
\f
;;; Filling
+(ert-deftest lisp-fill-paragraph-docstring-boundaries ()
+ "Test bug#28937, ensuring filling the docstring filled is properly
+bounded."
+ (with-temp-buffer
+ (insert "\
+(defun test ()
+ \"This is a test docstring.
+Here is some more text.\"
+ 1
+ 2
+ 3
+ 4
+ 5)")
+ (let ((correct (buffer-string)))
+ (emacs-lisp-mode)
+ (search-backward "This is a test docstring")
+ (fill-paragraph) ;function under test
+ (should (equal (buffer-string) correct)))))
+
(ert-deftest lisp-fill-paragraph-respects-fill-column ()
"Test bug#56197 -- more specifically, validate that a leading indentation
for a string is preserved in the filled string."
--
2.46.0
[-- Warning: decoded text below may be mangled, UTF-8 assumed --]
[-- Attachment #4: 0003-wip-work-on-solution.patch --]
[-- Type: text/x-patch, Size: 3778 bytes --]
From 5bc5de2fc6f7783b0cd71c5945755fc98431aa60 Mon Sep 17 00:00:00 2001
From: Maxim Cournoyer <maxim.cournoyer@gmail.com>
Date: Tue, 31 Dec 2024 10:18:30 +0900
Subject: [PATCH 3/3] wip: work on solution
---
lisp/emacs-lisp/lisp-mode.el | 33 +++++++++++++++++++++++----------
1 file changed, 23 insertions(+), 10 deletions(-)
diff --git a/lisp/emacs-lisp/lisp-mode.el b/lisp/emacs-lisp/lisp-mode.el
index d0c32d238bc..6c6d88f73a5 100644
--- a/lisp/emacs-lisp/lisp-mode.el
+++ b/lisp/emacs-lisp/lisp-mode.el
@@ -1,6 +1,6 @@
;;; lisp-mode.el --- Lisp mode, and its idiosyncratic commands -*- lexical-binding:t -*-
-;; Copyright (C) 1985-1986, 1999-2024 Free Software Foundation, Inc.
+;; Copyright (C) 1985-1986, 1999-2025 Free Software Foundation, Inc.
;; Maintainer: emacs-devel@gnu.org
;; Keywords: lisp, languages
@@ -1480,30 +1480,34 @@ lisp-fill-paragraph
(derived-mode-p 'emacs-lisp-mode))
emacs-lisp-docstring-fill-column
fill-column)))
- (let ((ppss (syntax-ppss))
- (start (point))
- ;; Avoid recursion if we're being called directly with
- ;; `M-x lisp-fill-paragraph' in an `emacs-lisp-mode' buffer.
- (fill-paragraph-function t))
+ (let* ((ppss (syntax-ppss))
+ (start (point))
+ ;; Avoid recursion if we're being called directly with
+ ;; `M-x lisp-fill-paragraph' in an `emacs-lisp-mode' buffer.
+ (fill-paragraph-function t)
+ (string-start (ppss-comment-or-string-start ppss))
+ (indent (save-excursion
+ (goto-char string-start)
+ (current-column))))
(save-excursion
(save-restriction
;; If we're not inside a string, then do very basic
;; filling. This avoids corrupting embedded strings in
;; code.
- (if (not (ppss-comment-or-string-start ppss))
+ (if (not string-start)
(lisp--fill-line-simple)
;; If we're in a string, then narrow (roughly) to that
;; string before filling. This avoids filling Lisp
;; statements that follow the string.
(when (ppss-string-terminator ppss)
- (goto-char (ppss-comment-or-string-start ppss))
+ (goto-char string-start)
;; The string may be unterminated -- in that case, don't
;; narrow.
(when (ignore-errors
(progn
(forward-sexp 1)
t))
- (narrow-to-region (1+ (ppss-comment-or-string-start ppss))
+ (narrow-to-region (1+ string-start)
(1- (point)))))
;; Move back to where we were.
(goto-char start)
@@ -1516,7 +1520,16 @@ lisp-fill-paragraph
(goto-char (point-min))
(forward-line 1)
(narrow-to-region (point) (point-max))))
- (fill-paragraph justify)))))))
+ ;; Insert spaces to reproduce the same leading indent
+ ;; for the string inside the narrowed region, to avoid
+ ;; bug#56197.
+ (save-excursion
+ (goto-char (point-min))
+ (insert (make-string indent ?\s)))
+ (fill-paragraph justify)
+ (save-excursion ;clean up
+ (goto-char (point-min))
+ (delete-char indent))))))))
;; Never return nil.
t)
--
2.46.0
[-- Attachment #5: Type: text/plain, Size: 443 bytes --]
If that's not easily fixable, perhaps I'll submit a patch to the paredit
project so that they stop relying on lisp-paragraph-fill [0] and instead
use the default (which is fill-paragraph). That's how I came to notice
this behavior change in GNU Guix, which is written in Scheme; using
scheme-mode doesn't exhibit the issue since it's just using
fill-paragraph.
[0] https://paredit.org/cgit/paredit/tree/paredit.el#n1066
--
Thanks,
Maxim
^ permalink raw reply related [flat|nested] 36+ messages in thread
* bug#56197: lisp-fill-paragraph behavior changed in Emacs 28
2025-01-04 10:09 ` bug#56197: lisp-fill-paragraph behavior changed in " Maxim Cournoyer
@ 2025-01-04 10:14 ` Maxim Cournoyer
2025-01-04 14:04 ` Eli Zaretskii
1 sibling, 0 replies; 36+ messages in thread
From: Maxim Cournoyer @ 2025-01-04 10:14 UTC (permalink / raw)
To: Eli Zaretskii; +Cc: larsi, 56197, felix.lechner, stefankangas
Hi again,
[...]
> If that's not easily fixable, perhaps I'll submit a patch to the paredit
> project so that they stop relying on lisp-paragraph-fill [0] and instead
> use the default (which is fill-paragraph). That's how I came to notice
> this behavior change in GNU Guix, which is written in Scheme; using
> scheme-mode doesn't exhibit the issue since it's just using
> fill-paragraph.
Actually, it seems scheme-mode at least in Emacs 29.4 does make use of
lisp-fill-paragraph, so should exhibit the same behavior/problem:
c.f. circa line 178 of scheme.el:
--8<---------------cut here---------------start------------->8---
(setq-local fill-paragraph-function 'lisp-fill-paragraph)
--8<---------------cut here---------------end--------------->8---
--
Thanks,
Maxim
^ permalink raw reply [flat|nested] 36+ messages in thread
* bug#56197: lisp-fill-paragraph behavior changed in Emacs 28
2025-01-04 10:09 ` bug#56197: lisp-fill-paragraph behavior changed in " Maxim Cournoyer
2025-01-04 10:14 ` Maxim Cournoyer
@ 2025-01-04 14:04 ` Eli Zaretskii
2025-01-04 14:19 ` Eli Zaretskii
1 sibling, 1 reply; 36+ messages in thread
From: Eli Zaretskii @ 2025-01-04 14:04 UTC (permalink / raw)
To: Maxim Cournoyer; +Cc: larsi, 56197, felix.lechner, stefankangas
> From: Maxim Cournoyer <maxim.cournoyer@gmail.com>
> Cc: larsi@gnus.org, 56197@debbugs.gnu.org, felix.lechner@lease-up.com,
> stefankangas@gmail.com
> Date: Sat, 04 Jan 2025 19:09:42 +0900
>
> >> I don't see how a user option to control this could be useful, since
> >> the preferred behavior is not only buffer-local, but also specific to
> >> certain syntactic constructs. But I won't object to having such an
> >> option.
> >
> > Having the behavior defined per-project or even globally (reverting to
> > the the pre-Emacs 28 behavior) via a simple option seems like it'd
> > simplify things, and make them discoverable.
>
> I tried fixing this generally, as it seems to me that something in
> lisp-mode should be meet the needs of all lisp-derived languages such as
> Scheme and not just Elisp. I first added two tests, one of which
> ensures no regression to the original bug that lead to this current
> behavioral change (bug#28937) and the other one that should pass once
> the issue reported here (bug#56197) is resolved.
>
> The last patch is a WIP that didn't work; I was hoping that inserting
> spaces corresponding to the width of the indent in the narrowed string
> would cause the indent to be preserved only for the first line. I don't
> have other ideas at the moment; I'd appreciate if someone could tip in.
Since you submitted a new bug report about this issue, does that mean
these comments and the patches are no longer pertinent?
^ permalink raw reply [flat|nested] 36+ messages in thread
* bug#56197: lisp-fill-paragraph behavior changed in Emacs 28
2025-01-04 14:04 ` Eli Zaretskii
@ 2025-01-04 14:19 ` Eli Zaretskii
2025-01-14 4:51 ` Maxim Cournoyer
0 siblings, 1 reply; 36+ messages in thread
From: Eli Zaretskii @ 2025-01-04 14:19 UTC (permalink / raw)
To: maxim.cournoyer; +Cc: larsi, 56197, felix.lechner, stefankangas
> Cc: larsi@gnus.org, 56197@debbugs.gnu.org, felix.lechner@lease-up.com,
> stefankangas@gmail.com
> Date: Sat, 04 Jan 2025 16:04:36 +0200
> From: Eli Zaretskii <eliz@gnu.org>
>
> > From: Maxim Cournoyer <maxim.cournoyer@gmail.com>
> > Cc: larsi@gnus.org, 56197@debbugs.gnu.org, felix.lechner@lease-up.com,
> > stefankangas@gmail.com
> > Date: Sat, 04 Jan 2025 19:09:42 +0900
> >
> > >> I don't see how a user option to control this could be useful, since
> > >> the preferred behavior is not only buffer-local, but also specific to
> > >> certain syntactic constructs. But I won't object to having such an
> > >> option.
> > >
> > > Having the behavior defined per-project or even globally (reverting to
> > > the the pre-Emacs 28 behavior) via a simple option seems like it'd
> > > simplify things, and make them discoverable.
> >
> > I tried fixing this generally, as it seems to me that something in
> > lisp-mode should be meet the needs of all lisp-derived languages such as
> > Scheme and not just Elisp. I first added two tests, one of which
> > ensures no regression to the original bug that lead to this current
> > behavioral change (bug#28937) and the other one that should pass once
> > the issue reported here (bug#56197) is resolved.
> >
> > The last patch is a WIP that didn't work; I was hoping that inserting
> > spaces corresponding to the width of the indent in the narrowed string
> > would cause the indent to be preserved only for the first line. I don't
> > have other ideas at the moment; I'd appreciate if someone could tip in.
>
> Since you submitted a new bug report about this issue, does that mean
> these comments and the patches are no longer pertinent?
Sorry, I should have written "new patch", not "new bug report".
^ permalink raw reply [flat|nested] 36+ messages in thread
* bug#56197: lisp-fill-paragraph behavior changed in Emacs 28
2025-01-04 14:19 ` Eli Zaretskii
@ 2025-01-14 4:51 ` Maxim Cournoyer
0 siblings, 0 replies; 36+ messages in thread
From: Maxim Cournoyer @ 2025-01-14 4:51 UTC (permalink / raw)
To: Eli Zaretskii; +Cc: larsi, 56197, felix.lechner, stefankangas
Hi Eli,
Eli Zaretskii <eliz@gnu.org> writes:
[...]
>> Since you submitted a new bug report about this issue, does that mean
>> these comments and the patches are no longer pertinent?
>
> Sorry, I should have written "new patch", not "new bug report".
Yes, the last patch I've sent is no longer a WIP and addresses the issue
for me (while improving the coverage of the test suite). Please have a
look when time allows!
--
Thanks,
Maxim
^ permalink raw reply [flat|nested] 36+ messages in thread
* bug#56197: 28.1; lisp-fill-paragraph result regressed with Emacs 28
2022-06-25 11:53 ` Lars Ingebrigtsen
2022-06-25 12:42 ` Eli Zaretskii
@ 2022-06-27 1:53 ` Maxim Cournoyer
1 sibling, 0 replies; 36+ messages in thread
From: Maxim Cournoyer @ 2022-06-27 1:53 UTC (permalink / raw)
To: Lars Ingebrigtsen; +Cc: 56197
Hello,
Lars Ingebrigtsen <larsi@gnus.org> writes:
> Maxim Cournoyer <maxim.cournoyer@gmail.com> writes:
>
>> ;; Emacs 28
>> (description "IBus-Anthy is an engine for the input bus \"IBus\"). It adds the Anthy
>> Japanese language input method to IBus. Because most graphical applications
>> allow text input via IBus, installing this package will enable Japanese
>> language input in most graphical applications.")
>
> [...]
>
>> Simply commenting out the newly added block, evaluating the defun and
>> running it on my example reverts to the previous correct behavior.
>
> I'm not sure the previous behaviour was any more correct. It's now
> filling that string as if it, well, is a string, so that if you insert
> it somewhere, the lines have similar lengths. The previous behaviour
> was to fill "what you see in the buffer", which is wrong in most
> contexts.
>
> So I don't know. Anybody have an opinion?
Apologies if my previous example lacked too much context and confused
more than helped. Here's another example, where I just experienced the
problem after revamping the GNU Guix 'font-abattis-cantarrel' package
definition:
--8<---------------cut here---------------start------------->8---
(define-public font-abattis-cantarell
(package
(name "font-abattis-cantarell")
(version "0.303")
(source
(origin
(method git-fetch)
(uri (git-reference
(url "https://gitlab.gnome.org/GNOME/cantarell-fonts")
(commit (string-append "v" version))))
(file-name (git-file-name name version))
(sha256
(base32
"1d1ay0fdqchk0wa5yqxis2c98imvzsbbd2kjv0x8sk4fm419847b"))))
(build-system meson-build-system)
(arguments
(list #:configure-flags #~(list "-Dbuildstatics=true")))
(native-inputs
(list gettext-minimal
psautohint
python
python-cffsubr
python-fontmath
python-statmake
python-ufo2ft))
(home-page "https://wiki.gnome.org/Projects/CantarellFonts")
(synopsis "Cantarell sans-serif typeface")
(description "The Cantarell font family is a contemporary Humanist sans-serif designed for
on-screen reading. It is used by GNOME@tie{}3. This package contains both
the non-variable as well as the variable versions of the font.")
(license license:silofl1.1)))
--8<---------------cut here---------------end--------------->8---
This is a Scheme sexp extracted from the (gnu packages fonts) Guile
module. Hitting `lisp-fill-paragraph' (M-q) causes the above
indentation, where the first line of the description extends past the
`fill-column' value.
I hope that helps,
Thanks!
Maxim
^ permalink raw reply [flat|nested] 36+ messages in thread
* bug#56197: [PATCH] lisp: Introduce lisp-fill-paragraph-as-displayed option.
2022-06-24 16:17 bug#56197: 28.1; lisp-fill-paragraph result regressed with Emacs 28 Maxim Cournoyer
2022-06-25 11:53 ` Lars Ingebrigtsen
@ 2025-01-04 13:03 ` Maxim Cournoyer
2025-01-04 14:02 ` Eli Zaretskii
2025-01-19 12:51 ` bug#56197: [PATCH v2] lisp: Introduce a `lisp-fill-paragraph-as-displayed' variable Maxim Cournoyer
2025-01-21 2:50 ` bug#56197: [PATCH v3] " Maxim Cournoyer
3 siblings, 1 reply; 36+ messages in thread
From: Maxim Cournoyer @ 2025-01-04 13:03 UTC (permalink / raw)
To: 56197; +Cc: Maxim Cournoyer, eliz, larsi, felix.lechner, stefankangas
Starting with Emacs 28, filling strings now happens in a narrowed scope,
and looses the leading indentation and can cause the string to extend
past the fill-column value. Introduce lisp-fill-paragraph-as-displayed
as a new option allowing users to easily opt out of this new behavior.
* lisp/emacs-lisp/lisp-mode.el (lisp-fill-paragraph-as-displayed): New
variable.
(lisp-fill-paragraph): Honor it, by avoiding the logic narrow to strings
before applying fill-paragraph.
* test/lisp/emacs-lisp/lisp-mode-tests.el
(lisp-fill-paragraph-respects-fill-column): Test it.
(lisp-fill-paragraph-docstring-boundaries): New test, as a safeguard to
avoid regressions.
Fixes: bug#56197
---
lisp/emacs-lisp/lisp-mode.el | 74 ++++++++++++++-----------
test/lisp/emacs-lisp/lisp-mode-tests.el | 47 ++++++++++++++++
2 files changed, 90 insertions(+), 31 deletions(-)
diff --git a/lisp/emacs-lisp/lisp-mode.el b/lisp/emacs-lisp/lisp-mode.el
index d0c32d238bc..cc9f6f51cb5 100644
--- a/lisp/emacs-lisp/lisp-mode.el
+++ b/lisp/emacs-lisp/lisp-mode.el
@@ -1,6 +1,6 @@
;;; lisp-mode.el --- Lisp mode, and its idiosyncratic commands -*- lexical-binding:t -*-
-;; Copyright (C) 1985-1986, 1999-2024 Free Software Foundation, Inc.
+;; Copyright (C) 1985-1986, 1999-2025 Free Software Foundation, Inc.
;; Maintainer: emacs-devel@gnu.org
;; Keywords: lisp, languages
@@ -1431,6 +1431,16 @@ emacs-lisp-docstring-fill-column
:group 'lisp
:version "30.1")
+(defcustom lisp-fill-paragraph-as-displayed nil
+ "Fill paragraphs as displayed in the buffer, preserving surrounding
+context such as the leading indentation. This is useful if respecting
+`fill-column' is more important than avoiding surrounding code from
+being filled, and makes `lisp-fill-paragraph' behave as it used to in
+Emacs 27 and prior versions."
+ :type 'boolean
+ :group 'lisp
+ :version "31.0")
+
(defun lisp-fill-paragraph (&optional justify)
"Like \\[fill-paragraph], but handle Emacs Lisp comments and docstrings.
If any of the current line is a comment, fill the comment or the
@@ -1480,42 +1490,44 @@ lisp-fill-paragraph
(derived-mode-p 'emacs-lisp-mode))
emacs-lisp-docstring-fill-column
fill-column)))
- (let ((ppss (syntax-ppss))
- (start (point))
- ;; Avoid recursion if we're being called directly with
- ;; `M-x lisp-fill-paragraph' in an `emacs-lisp-mode' buffer.
- (fill-paragraph-function t))
+ (let* ((ppss (syntax-ppss))
+ (start (point))
+ ;; Avoid recursion if we're being called directly with
+ ;; `M-x lisp-fill-paragraph' in an `emacs-lisp-mode' buffer.
+ (fill-paragraph-function t)
+ (string-start (ppss-comment-or-string-start ppss)))
(save-excursion
(save-restriction
;; If we're not inside a string, then do very basic
;; filling. This avoids corrupting embedded strings in
;; code.
- (if (not (ppss-comment-or-string-start ppss))
+ (if (not string-start)
(lisp--fill-line-simple)
- ;; If we're in a string, then narrow (roughly) to that
- ;; string before filling. This avoids filling Lisp
- ;; statements that follow the string.
- (when (ppss-string-terminator ppss)
- (goto-char (ppss-comment-or-string-start ppss))
- ;; The string may be unterminated -- in that case, don't
- ;; narrow.
- (when (ignore-errors
- (progn
- (forward-sexp 1)
- t))
- (narrow-to-region (1+ (ppss-comment-or-string-start ppss))
- (1- (point)))))
- ;; Move back to where we were.
- (goto-char start)
- ;; We should fill the first line of a string
- ;; separately (since it's usually a doc string).
- (if (= (line-number-at-pos) 1)
- (narrow-to-region (line-beginning-position)
- (line-beginning-position 2))
- (save-excursion
- (goto-char (point-min))
- (forward-line 1)
- (narrow-to-region (point) (point-max))))
+ (unless lisp-fill-paragraph-as-displayed
+ ;; If we're in a string, then narrow (roughly) to that
+ ;; string before filling. This avoids filling Lisp
+ ;; statements that follow the string.
+ (when (ppss-string-terminator ppss)
+ (goto-char string-start)
+ ;; The string may be unterminated -- in that case, don't
+ ;; narrow.
+ (when (ignore-errors
+ (progn
+ (forward-sexp 1)
+ t))
+ (narrow-to-region (1+ string-start)
+ (1- (point)))))
+ ;; Move back to where we were.
+ (goto-char start)
+ ;; We should fill the first line of a string
+ ;; separately (since it's usually a doc string).
+ (if (= (line-number-at-pos) 1)
+ (narrow-to-region (line-beginning-position)
+ (line-beginning-position 2))
+ (save-excursion
+ (goto-char (point-min))
+ (forward-line 1)
+ (narrow-to-region (point) (point-max)))))
(fill-paragraph justify)))))))
;; Never return nil.
t)
diff --git a/test/lisp/emacs-lisp/lisp-mode-tests.el b/test/lisp/emacs-lisp/lisp-mode-tests.el
index da02be65d03..347b3d5b642 100644
--- a/test/lisp/emacs-lisp/lisp-mode-tests.el
+++ b/test/lisp/emacs-lisp/lisp-mode-tests.el
@@ -308,6 +308,53 @@ lisp-indent-defun
(indent-region (point-min) (point-max))
(should (equal (buffer-string) orig)))))
+\f
+;;; Filling
+
+(ert-deftest lisp-fill-paragraph-docstring-boundaries ()
+ "Test bug#28937, ensuring filling the docstring filled is properly
+bounded."
+ (with-temp-buffer
+ (insert "\
+(defun test ()
+ \"This is a test docstring.
+Here is some more text.\"
+ 1
+ 2
+ 3
+ 4
+ 5)")
+ (let ((correct (buffer-string)))
+ (emacs-lisp-mode)
+ (search-backward "This is a test docstring")
+ (fill-paragraph) ;function under test
+ (should (equal (buffer-string) correct)))))
+
+(ert-deftest lisp-fill-paragraph-as-displayed ()
+ "Test bug#56197 -- more specifically, validate that a leading indentation
+for a string is preserved in the filled string."
+ (let ((lisp-fill-paragraph-as-displayed t) ;option under test
+ (source "\
+'(description \"This is a very long string which is indented by a considerable value, causing it to
+protrude from the configured `fill-column' since
+lisp-fill-paragraph was refactored in version 28.\")"))
+ (with-temp-buffer
+ ;; The following is a contrived example that demonstrates the
+ ;; fill-column problem when the string to fill is indented.
+ (insert source)
+ (emacs-lisp-mode)
+ (search-backward "This is a very long string")
+ (fill-paragraph) ;function under test
+ (goto-char (point-min))
+ (message "%s" (buffer-substring-no-properties (point-min) (point-max)))
+ (let ((i 1)
+ (lines-count (count-lines (point-min) (point-max))))
+ (while (< i lines-count)
+ (beginning-of-line i)
+ (end-of-line)
+ (should (<= (current-column) fill-column))
+ (setq i (1+ i)))))))
+
\f
;;; Fontification
--
2.46.0
^ permalink raw reply related [flat|nested] 36+ messages in thread
* bug#56197: [PATCH] lisp: Introduce lisp-fill-paragraph-as-displayed option.
2025-01-04 13:03 ` bug#56197: [PATCH] lisp: Introduce lisp-fill-paragraph-as-displayed option Maxim Cournoyer
@ 2025-01-04 14:02 ` Eli Zaretskii
2025-01-16 5:04 ` bug#56197: lisp-fill-paragraph behavior changed in Emacs 28 Maxim Cournoyer
0 siblings, 1 reply; 36+ messages in thread
From: Eli Zaretskii @ 2025-01-04 14:02 UTC (permalink / raw)
To: Maxim Cournoyer
Cc: larsi, felix.lechner, 56197, maxim.cournoyer, stefankangas
> Cc: Maxim Cournoyer <maxim.cournoyer@gmail.com>, eliz@gnu.org, larsi@gnus.org, felix.lechner@lease-up.com, stefankangas@gmail.com
> From: Maxim Cournoyer <maxim.cournoyer@gmail.com>
> Date: Sat, 4 Jan 2025 22:03:43 +0900
>
> Starting with Emacs 28, filling strings now happens in a narrowed scope,
> and looses the leading indentation and can cause the string to extend
> past the fill-column value. Introduce lisp-fill-paragraph-as-displayed
> as a new option allowing users to easily opt out of this new behavior.
Thanks. But this is not a user-level problem, so the variable to
control this should IMO be a defvar, not a defcustom. Then Lisp
programs which need to get back old behavior for some reason could
bind the variable around the call.
P.S. I don't see your copyright assignment on file, so if you want to
contribute such large changes, let's please start your legal paperwork
of assigning the copyright to the FSF. OK?
^ permalink raw reply [flat|nested] 36+ messages in thread
* bug#56197: lisp-fill-paragraph behavior changed in Emacs 28
2025-01-04 14:02 ` Eli Zaretskii
@ 2025-01-16 5:04 ` Maxim Cournoyer
2025-01-16 8:22 ` Eli Zaretskii
0 siblings, 1 reply; 36+ messages in thread
From: Maxim Cournoyer @ 2025-01-16 5:04 UTC (permalink / raw)
To: Eli Zaretskii; +Cc: larsi, 56197, felix.lechner, stefankangas
Hi Eli,
Eli Zaretskii <eliz@gnu.org> writes:
>> Cc: Maxim Cournoyer <maxim.cournoyer@gmail.com>, eliz@gnu.org,
>> larsi@gnus.org, felix.lechner@lease-up.com, stefankangas@gmail.com
>> From: Maxim Cournoyer <maxim.cournoyer@gmail.com>
>> Date: Sat, 4 Jan 2025 22:03:43 +0900
>>
>> Starting with Emacs 28, filling strings now happens in a narrowed scope,
>> and looses the leading indentation and can cause the string to extend
>> past the fill-column value. Introduce lisp-fill-paragraph-as-displayed
>> as a new option allowing users to easily opt out of this new behavior.
>
> Thanks. But this is not a user-level problem, so the variable to
> control this should IMO be a defvar, not a defcustom. Then Lisp
> programs which need to get back old behavior for some reason could
> bind the variable around the call.
I'm not sure. A user (such as myself) may prefer the old behavior, and
customize lisp-fill-paragraph-as-displayed (setting it to t) so that
this behavior is now the default everywhere. It also makes it
more easily discoverable. So unless you see a strong reason against
using defcustom, it seems preferable to me than defvar.
> P.S. I don't see your copyright assignment on file, so if you want to
> contribute such large changes, let's please start your legal paperwork
> of assigning the copyright to the FSF. OK?
That's fine; I'm happy to assign my copyright w.r.t. Emacs changes to
the FSF; please send the paperwork over!
--
Thanks,
Maxim
^ permalink raw reply [flat|nested] 36+ messages in thread
* bug#56197: lisp-fill-paragraph behavior changed in Emacs 28
2025-01-16 5:04 ` bug#56197: lisp-fill-paragraph behavior changed in Emacs 28 Maxim Cournoyer
@ 2025-01-16 8:22 ` Eli Zaretskii
2025-01-19 12:35 ` Maxim Cournoyer
0 siblings, 1 reply; 36+ messages in thread
From: Eli Zaretskii @ 2025-01-16 8:22 UTC (permalink / raw)
To: Maxim Cournoyer; +Cc: larsi, 56197, felix.lechner, stefankangas
> From: Maxim Cournoyer <maxim.cournoyer@gmail.com>
> Cc: larsi@gnus.org, felix.lechner@lease-up.com, 56197@debbugs.gnu.org,
> stefankangas@gmail.com
> Date: Thu, 16 Jan 2025 14:04:55 +0900
>
> Eli Zaretskii <eliz@gnu.org> writes:
>
> >> Cc: Maxim Cournoyer <maxim.cournoyer@gmail.com>, eliz@gnu.org,
> >> larsi@gnus.org, felix.lechner@lease-up.com, stefankangas@gmail.com
> >> From: Maxim Cournoyer <maxim.cournoyer@gmail.com>
> >> Date: Sat, 4 Jan 2025 22:03:43 +0900
> >>
> >> Starting with Emacs 28, filling strings now happens in a narrowed scope,
> >> and looses the leading indentation and can cause the string to extend
> >> past the fill-column value. Introduce lisp-fill-paragraph-as-displayed
> >> as a new option allowing users to easily opt out of this new behavior.
> >
> > Thanks. But this is not a user-level problem, so the variable to
> > control this should IMO be a defvar, not a defcustom. Then Lisp
> > programs which need to get back old behavior for some reason could
> > bind the variable around the call.
>
> I'm not sure. A user (such as myself) may prefer the old behavior, and
> customize lisp-fill-paragraph-as-displayed (setting it to t) so that
> this behavior is now the default everywhere. It also makes it
> more easily discoverable. So unless you see a strong reason against
> using defcustom, it seems preferable to me than defvar.
Users can also set a defvar, if they want this globally. However, the
original problem is not a global one, it is specific to some
situations in some particular major mode.
The important question here is: how common is the situation where a
user will prefer to set that globally? I think this could only happen
if the user uses a major mode of just one variant of Lisp-like
languages, and especially if that one variant is not Emacs Lisp.
In addition, making it a defcustom means Lisp programs cannot easily
bind it to specific values when they need it (overriding user options
is considered unclean in Emacs).
So my preference is to introduce a defvar, and only promote it to a
user option if we have enough demand in the future.
> > P.S. I don't see your copyright assignment on file, so if you want to
> > contribute such large changes, let's please start your legal paperwork
> > of assigning the copyright to the FSF. OK?
>
> That's fine; I'm happy to assign my copyright w.r.t. Emacs changes to
> the FSF; please send the paperwork over!
Will do, thanks.
^ permalink raw reply [flat|nested] 36+ messages in thread
* bug#56197: lisp-fill-paragraph behavior changed in Emacs 28
2025-01-16 8:22 ` Eli Zaretskii
@ 2025-01-19 12:35 ` Maxim Cournoyer
0 siblings, 0 replies; 36+ messages in thread
From: Maxim Cournoyer @ 2025-01-19 12:35 UTC (permalink / raw)
To: Eli Zaretskii; +Cc: larsi, 56197, felix.lechner, stefankangas
Hello Eli,
Eli Zaretskii <eliz@gnu.org> writes:
[...]
>> > Thanks. But this is not a user-level problem, so the variable to
>> > control this should IMO be a defvar, not a defcustom. Then Lisp
>> > programs which need to get back old behavior for some reason could
>> > bind the variable around the call.
>>
>> I'm not sure. A user (such as myself) may prefer the old behavior, and
>> customize lisp-fill-paragraph-as-displayed (setting it to t) so that
>> this behavior is now the default everywhere. It also makes it
>> more easily discoverable. So unless you see a strong reason against
>> using defcustom, it seems preferable to me than defvar.
>
> Users can also set a defvar, if they want this globally. However, the
> original problem is not a global one, it is specific to some
> situations in some particular major mode.
>
> The important question here is: how common is the situation where a
> user will prefer to set that globally? I think this could only happen
> if the user uses a major mode of just one variant of Lisp-like
> languages, and especially if that one variant is not Emacs Lisp.
Good point. The current behavior is probably useful when editing Emacs
Lisp, so I agree that setting this precisely for specific modes (such as
scheme-mode) instead probably makes more sense.
> In addition, making it a defcustom means Lisp programs cannot easily
> bind it to specific values when they need it (overriding user options
> is considered unclean in Emacs).
>
> So my preference is to introduce a defvar, and only promote it to a
> user option if we have enough demand in the future.
OK! I'll send a reworked version using defvar shortly. Thanks for the
thoughtful discussion!
--
Maxim
^ permalink raw reply [flat|nested] 36+ messages in thread
* bug#56197: [PATCH v2] lisp: Introduce a `lisp-fill-paragraph-as-displayed' variable.
2022-06-24 16:17 bug#56197: 28.1; lisp-fill-paragraph result regressed with Emacs 28 Maxim Cournoyer
2022-06-25 11:53 ` Lars Ingebrigtsen
2025-01-04 13:03 ` bug#56197: [PATCH] lisp: Introduce lisp-fill-paragraph-as-displayed option Maxim Cournoyer
@ 2025-01-19 12:51 ` Maxim Cournoyer
2025-01-19 13:13 ` Eli Zaretskii
2025-01-21 2:50 ` bug#56197: [PATCH v3] " Maxim Cournoyer
3 siblings, 1 reply; 36+ messages in thread
From: Maxim Cournoyer @ 2025-01-19 12:51 UTC (permalink / raw)
To: 56197; +Cc: eliz, Maxim Cournoyer, larsi, felix.lechner, stefankangas
Starting with Emacs 28, filling strings now happens in a narrowed scope,
and looses the leading indentation and can cause the string to extend
past the fill-column value. Introduce `lisp-fill-paragraph-as-displayed'
as a new variable allowing opting out of this new behavior in specific
scenarios (such as when using the Scheme major mode, say).
* lisp/emacs-lisp/lisp-mode.el (lisp-fill-paragraph-as-displayed): New
variable.
(lisp-fill-paragraph): Honor it, by avoiding the logic narrow to strings
before applying fill-paragraph.
* test/lisp/emacs-lisp/lisp-mode-tests.el
(lisp-fill-paragraph-respects-fill-column): Test it.
(lisp-fill-paragraph-docstring-boundaries): New test, as a safeguard to
avoid regressions.
Fixes: bug#56197
---
Changes since v1: Use defvar, not defcustom + light rewordings of some
doc/comments
lisp/emacs-lisp/lisp-mode.el | 72 ++++++++++++++-----------
test/lisp/emacs-lisp/lisp-mode-tests.el | 47 ++++++++++++++++
2 files changed, 88 insertions(+), 31 deletions(-)
diff --git a/lisp/emacs-lisp/lisp-mode.el b/lisp/emacs-lisp/lisp-mode.el
index d0c32d238bc..c55dd3c6528 100644
--- a/lisp/emacs-lisp/lisp-mode.el
+++ b/lisp/emacs-lisp/lisp-mode.el
@@ -1,6 +1,6 @@
;;; lisp-mode.el --- Lisp mode, and its idiosyncratic commands -*- lexical-binding:t -*-
-;; Copyright (C) 1985-1986, 1999-2024 Free Software Foundation, Inc.
+;; Copyright (C) 1985-1986, 1999-2025 Free Software Foundation, Inc.
;; Maintainer: emacs-devel@gnu.org
;; Keywords: lisp, languages
@@ -1431,6 +1431,14 @@ emacs-lisp-docstring-fill-column
:group 'lisp
:version "30.1")
+(defvar lisp-fill-paragraph-as-displayed nil
+ "This variable can be set to true to fill paragraphs as displayed in the
+buffer, preserving surrounding context such as the leading indentation.
+This is useful if respecting `fill-column' is more important than
+preventing surrounding code from being filled, and makes
+`lisp-fill-paragraph' behave as it used to in Emacs 27 and prior
+versions.")
+
(defun lisp-fill-paragraph (&optional justify)
"Like \\[fill-paragraph], but handle Emacs Lisp comments and docstrings.
If any of the current line is a comment, fill the comment or the
@@ -1480,42 +1488,44 @@ lisp-fill-paragraph
(derived-mode-p 'emacs-lisp-mode))
emacs-lisp-docstring-fill-column
fill-column)))
- (let ((ppss (syntax-ppss))
- (start (point))
- ;; Avoid recursion if we're being called directly with
- ;; `M-x lisp-fill-paragraph' in an `emacs-lisp-mode' buffer.
- (fill-paragraph-function t))
+ (let* ((ppss (syntax-ppss))
+ (start (point))
+ ;; Avoid recursion if we're being called directly with
+ ;; `M-x lisp-fill-paragraph' in an `emacs-lisp-mode' buffer.
+ (fill-paragraph-function t)
+ (string-start (ppss-comment-or-string-start ppss)))
(save-excursion
(save-restriction
;; If we're not inside a string, then do very basic
;; filling. This avoids corrupting embedded strings in
;; code.
- (if (not (ppss-comment-or-string-start ppss))
+ (if (not string-start)
(lisp--fill-line-simple)
- ;; If we're in a string, then narrow (roughly) to that
- ;; string before filling. This avoids filling Lisp
- ;; statements that follow the string.
- (when (ppss-string-terminator ppss)
- (goto-char (ppss-comment-or-string-start ppss))
- ;; The string may be unterminated -- in that case, don't
- ;; narrow.
- (when (ignore-errors
- (progn
- (forward-sexp 1)
- t))
- (narrow-to-region (1+ (ppss-comment-or-string-start ppss))
- (1- (point)))))
- ;; Move back to where we were.
- (goto-char start)
- ;; We should fill the first line of a string
- ;; separately (since it's usually a doc string).
- (if (= (line-number-at-pos) 1)
- (narrow-to-region (line-beginning-position)
- (line-beginning-position 2))
- (save-excursion
- (goto-char (point-min))
- (forward-line 1)
- (narrow-to-region (point) (point-max))))
+ (unless lisp-fill-paragraph-as-displayed
+ ;; If we're in a string, then narrow (roughly) to that
+ ;; string before filling. This avoids filling Lisp
+ ;; statements that follow the string.
+ (when (ppss-string-terminator ppss)
+ (goto-char string-start)
+ ;; The string may be unterminated -- in that case, don't
+ ;; narrow.
+ (when (ignore-errors
+ (progn
+ (forward-sexp 1)
+ t))
+ (narrow-to-region (1+ string-start)
+ (1- (point)))))
+ ;; Move back to where we were.
+ (goto-char start)
+ ;; We should fill the first line of a string
+ ;; separately (since it's usually a doc string).
+ (if (= (line-number-at-pos) 1)
+ (narrow-to-region (line-beginning-position)
+ (line-beginning-position 2))
+ (save-excursion
+ (goto-char (point-min))
+ (forward-line 1)
+ (narrow-to-region (point) (point-max)))))
(fill-paragraph justify)))))))
;; Never return nil.
t)
diff --git a/test/lisp/emacs-lisp/lisp-mode-tests.el b/test/lisp/emacs-lisp/lisp-mode-tests.el
index da02be65d03..7f5c97ace4d 100644
--- a/test/lisp/emacs-lisp/lisp-mode-tests.el
+++ b/test/lisp/emacs-lisp/lisp-mode-tests.el
@@ -308,6 +308,53 @@ lisp-indent-defun
(indent-region (point-min) (point-max))
(should (equal (buffer-string) orig)))))
+\f
+;;; Filling
+
+(ert-deftest lisp-fill-paragraph-docstring-boundaries ()
+ "Test bug#28937, ensuring filling the docstring filled is properly
+bounded."
+ (with-temp-buffer
+ (insert "\
+(defun test ()
+ \"This is a test docstring.
+Here is some more text.\"
+ 1
+ 2
+ 3
+ 4
+ 5)")
+ (let ((correct (buffer-string)))
+ (emacs-lisp-mode)
+ (search-backward "This is a test docstring")
+ (fill-paragraph) ;function under test
+ (should (equal (buffer-string) correct)))))
+
+(ert-deftest lisp-fill-paragraph-as-displayed ()
+ "Test bug#56197 -- more specifically, validate that a leading indentation
+for a string is preserved in the filled string."
+ (let ((lisp-fill-paragraph-as-displayed t) ;variable under test
+ ;; The following is a contrived example that demonstrates the
+ ;; fill-column problem when the string to fill is indented.
+ (source "\
+'(description \"This is a very long string which is indented by a considerable value, causing it to
+protrude from the configured `fill-column' since
+lisp-fill-paragraph was refactored in version 28.\")"))
+ (with-temp-buffer
+ (insert source)
+ (emacs-lisp-mode)
+ (search-backward "This is a very long string")
+ (fill-paragraph) ;function under test
+ (goto-char (point-min))
+ (message "%s" (buffer-substring-no-properties (point-min) (point-max)))
+ (let ((i 1)
+ (lines-count (count-lines (point-min) (point-max))))
+ (while (< i lines-count)
+ (beginning-of-line i)
+ (end-of-line)
+ (should (<= (current-column) fill-column))
+ (setq i (1+ i)))))))
+
\f
;;; Fontification
--
2.46.0
^ permalink raw reply related [flat|nested] 36+ messages in thread
* bug#56197: [PATCH v2] lisp: Introduce a `lisp-fill-paragraph-as-displayed' variable.
2025-01-19 12:51 ` bug#56197: [PATCH v2] lisp: Introduce a `lisp-fill-paragraph-as-displayed' variable Maxim Cournoyer
@ 2025-01-19 13:13 ` Eli Zaretskii
2025-01-21 2:43 ` Maxim Cournoyer
0 siblings, 1 reply; 36+ messages in thread
From: Eli Zaretskii @ 2025-01-19 13:13 UTC (permalink / raw)
To: Maxim Cournoyer
Cc: larsi, maxim.cournoyer, 56197, felix.lechner, stefankangas
> From: Maxim Cournoyer <maxim.cournoyer@gmail.com>
> Cc: eliz@gnu.org,
> larsi@gnus.org,
> felix.lechner@lease-up.com,
> stefankangas@gmail.com,
> Maxim Cournoyer <maxim.cournoyer@gmail.com>
> Date: Sun, 19 Jan 2025 21:51:56 +0900
>
> Starting with Emacs 28, filling strings now happens in a narrowed scope,
> and looses the leading indentation and can cause the string to extend
> past the fill-column value. Introduce `lisp-fill-paragraph-as-displayed'
> as a new variable allowing opting out of this new behavior in specific
> scenarios (such as when using the Scheme major mode, say).
Thanks, a few minor comments below.
> * lisp/emacs-lisp/lisp-mode.el (lisp-fill-paragraph-as-displayed): New
> variable.
> (lisp-fill-paragraph): Honor it, by avoiding the logic narrow to strings
> before applying fill-paragraph.
> * test/lisp/emacs-lisp/lisp-mode-tests.el
> (lisp-fill-paragraph-respects-fill-column): Test it.
> (lisp-fill-paragraph-docstring-boundaries): New test, as a safeguard to
> avoid regressions.
Please format these entries according to our conventions, mostly
regarding the line length (it should be at most 69 columns, preferably
no more than 64.
> -;; Copyright (C) 1985-1986, 1999-2024 Free Software Foundation, Inc.
> +;; Copyright (C) 1985-1986, 1999-2025 Free Software Foundation, Inc.
Please rebase on the current master branch, where the copyright years
were already updated.
> +(defvar lisp-fill-paragraph-as-displayed nil
> + "This variable can be set to true to fill paragraphs as displayed in the
> +buffer, preserving surrounding context such as the leading indentation.
> +This is useful if respecting `fill-column' is more important than
> +preventing surrounding code from being filled, and makes
> +`lisp-fill-paragraph' behave as it used to in Emacs 27 and prior
> +versions.")
The first line of a doc string should be a single full sentence.
(This is because the various apropos commands show only the first
line.)
More importantly, I think this doc string should describe the main use
case for which the default behavior is tuned: filling Emacs Lisp doc
strings, with their special treatment of the first line.
^ permalink raw reply [flat|nested] 36+ messages in thread
* bug#56197: [PATCH v2] lisp: Introduce a `lisp-fill-paragraph-as-displayed' variable.
2025-01-19 13:13 ` Eli Zaretskii
@ 2025-01-21 2:43 ` Maxim Cournoyer
0 siblings, 0 replies; 36+ messages in thread
From: Maxim Cournoyer @ 2025-01-21 2:43 UTC (permalink / raw)
To: Eli Zaretskii; +Cc: larsi, 56197, felix.lechner, stefankangas
Hi,
Eli Zaretskii <eliz@gnu.org> writes:
[...]
>> * lisp/emacs-lisp/lisp-mode.el (lisp-fill-paragraph-as-displayed): New
>> variable.
>> (lisp-fill-paragraph): Honor it, by avoiding the logic narrow to strings
>> before applying fill-paragraph.
>> * test/lisp/emacs-lisp/lisp-mode-tests.el
>> (lisp-fill-paragraph-respects-fill-column): Test it.
>> (lisp-fill-paragraph-docstring-boundaries): New test, as a safeguard to
>> avoid regressions.
>
> Please format these entries according to our conventions, mostly
> regarding the line length (it should be at most 69 columns, preferably
> no more than 64.
Done. It's odd that when using magit there's no easy way to have this
this set by default (it defaults to use the text-mode major mode). I've
M-x log-edit-mode in the edit buffer to get the .dir-locals.el
preferences applied.
>> -;; Copyright (C) 1985-1986, 1999-2024 Free Software Foundation, Inc.
>> +;; Copyright (C) 1985-1986, 1999-2025 Free Software Foundation, Inc.
>
> Please rebase on the current master branch, where the copyright years
> were already updated.
Done.
>> +(defvar lisp-fill-paragraph-as-displayed nil
>> + "This variable can be set to true to fill paragraphs as displayed in the
>> +buffer, preserving surrounding context such as the leading indentation.
>> +This is useful if respecting `fill-column' is more important than
>> +preventing surrounding code from being filled, and makes
>> +`lisp-fill-paragraph' behave as it used to in Emacs 27 and prior
>> +versions.")
>
> The first line of a doc string should be a single full sentence.
> (This is because the various apropos commands show only the first
> line.)
>
> More importantly, I think this doc string should describe the main use
> case for which the default behavior is tuned: filling Emacs Lisp doc
> strings, with their special treatment of the first line.
I expound the docstring to add more information w.r.t. to the default
behavior of `lisp-fill-paragraph'. I hope it addresses your comment.
I'll send v3 shortly.
--
Thanks,
Maxim
^ permalink raw reply [flat|nested] 36+ messages in thread
* bug#56197: [PATCH v3] lisp: Introduce a `lisp-fill-paragraph-as-displayed' variable.
2022-06-24 16:17 bug#56197: 28.1; lisp-fill-paragraph result regressed with Emacs 28 Maxim Cournoyer
` (2 preceding siblings ...)
2025-01-19 12:51 ` bug#56197: [PATCH v2] lisp: Introduce a `lisp-fill-paragraph-as-displayed' variable Maxim Cournoyer
@ 2025-01-21 2:50 ` Maxim Cournoyer
3 siblings, 0 replies; 36+ messages in thread
From: Maxim Cournoyer @ 2025-01-21 2:50 UTC (permalink / raw)
To: 56197; +Cc: eliz, Maxim Cournoyer, larsi, felix.lechner, stefankangas
Starting with Emacs 28, filling strings now happens in a
narrowed scope, and looses the leading indentation and can cause
the string to extend past the fill-column value. Introduce
`lisp-fill-paragraph-as-displayed' as a new variable allowing
opting out of this new behavior in specific scenarios (such as
when using the Scheme major mode, say).
* lisp/emacs-lisp/lisp-mode.el
(lisp-fill-paragraph-as-displayed): New variable.
(lisp-fill-paragraph): Honor it, by avoiding the logic narrow to
strings before applying fill-paragraph.
* test/lisp/emacs-lisp/lisp-mode-tests.el
(lisp-fill-paragraph-respects-fill-column): Test it.
(lisp-fill-paragraph-docstring-boundaries): New test, as a
safeguard to avoid regressions.
Fixes: bug#56197
---
lisp/emacs-lisp/lisp-mode.el | 75 +++++++++++++++----------
test/lisp/emacs-lisp/lisp-mode-tests.el | 47 ++++++++++++++++
2 files changed, 92 insertions(+), 30 deletions(-)
diff --git a/lisp/emacs-lisp/lisp-mode.el b/lisp/emacs-lisp/lisp-mode.el
index 2b75a5fd038..9e8292b992a 100644
--- a/lisp/emacs-lisp/lisp-mode.el
+++ b/lisp/emacs-lisp/lisp-mode.el
@@ -1431,6 +1431,19 @@ emacs-lisp-docstring-fill-column
:group 'lisp
:version "30.1")
+(defvar lisp-fill-paragraph-as-displayed nil
+ "Modify the behavior of `lisp-fill-paragraph'.
+The default behavior of `lisp-fill-paragraph' is tuned for filling Emacs
+Lisp doc strings, with their special treatment for the first line.
+Particularly, strings are filled in a narrowed context to avoid filling
+surrounding code, which means any leading indent is disregarded, which
+can cause the filled string to extend passed the configured
+`fill-column' variable value. If you would rather fill the string in
+its original context and ensure the `fill-column' value is more strictly
+respected, set this variable to true. Doing so makes
+`lisp-fill-paragraph' behave as it used to in Emacs 27 and prior
+versions.")
+
(defun lisp-fill-paragraph (&optional justify)
"Like \\[fill-paragraph], but handle Emacs Lisp comments and docstrings.
If any of the current line is a comment, fill the comment or the
@@ -1480,42 +1493,44 @@ lisp-fill-paragraph
(derived-mode-p 'emacs-lisp-mode))
emacs-lisp-docstring-fill-column
fill-column)))
- (let ((ppss (syntax-ppss))
- (start (point))
- ;; Avoid recursion if we're being called directly with
- ;; `M-x lisp-fill-paragraph' in an `emacs-lisp-mode' buffer.
- (fill-paragraph-function t))
+ (let* ((ppss (syntax-ppss))
+ (start (point))
+ ;; Avoid recursion if we're being called directly with
+ ;; `M-x lisp-fill-paragraph' in an `emacs-lisp-mode' buffer.
+ (fill-paragraph-function t)
+ (string-start (ppss-comment-or-string-start ppss)))
(save-excursion
(save-restriction
;; If we're not inside a string, then do very basic
;; filling. This avoids corrupting embedded strings in
;; code.
- (if (not (ppss-comment-or-string-start ppss))
+ (if (not string-start)
(lisp--fill-line-simple)
- ;; If we're in a string, then narrow (roughly) to that
- ;; string before filling. This avoids filling Lisp
- ;; statements that follow the string.
- (when (ppss-string-terminator ppss)
- (goto-char (ppss-comment-or-string-start ppss))
- ;; The string may be unterminated -- in that case, don't
- ;; narrow.
- (when (ignore-errors
- (progn
- (forward-sexp 1)
- t))
- (narrow-to-region (1+ (ppss-comment-or-string-start ppss))
- (1- (point)))))
- ;; Move back to where we were.
- (goto-char start)
- ;; We should fill the first line of a string
- ;; separately (since it's usually a doc string).
- (if (= (line-number-at-pos) 1)
- (narrow-to-region (line-beginning-position)
- (line-beginning-position 2))
- (save-excursion
- (goto-char (point-min))
- (forward-line 1)
- (narrow-to-region (point) (point-max))))
+ (unless lisp-fill-paragraph-as-displayed
+ ;; If we're in a string, then narrow (roughly) to that
+ ;; string before filling. This avoids filling Lisp
+ ;; statements that follow the string.
+ (when (ppss-string-terminator ppss)
+ (goto-char string-start)
+ ;; The string may be unterminated -- in that case, don't
+ ;; narrow.
+ (when (ignore-errors
+ (progn
+ (forward-sexp 1)
+ t))
+ (narrow-to-region (1+ string-start)
+ (1- (point)))))
+ ;; Move back to where we were.
+ (goto-char start)
+ ;; We should fill the first line of a string
+ ;; separately (since it's usually a doc string).
+ (if (= (line-number-at-pos) 1)
+ (narrow-to-region (line-beginning-position)
+ (line-beginning-position 2))
+ (save-excursion
+ (goto-char (point-min))
+ (forward-line 1)
+ (narrow-to-region (point) (point-max)))))
(fill-paragraph justify)))))))
;; Never return nil.
t)
diff --git a/test/lisp/emacs-lisp/lisp-mode-tests.el b/test/lisp/emacs-lisp/lisp-mode-tests.el
index 3a765eab625..96e37114276 100644
--- a/test/lisp/emacs-lisp/lisp-mode-tests.el
+++ b/test/lisp/emacs-lisp/lisp-mode-tests.el
@@ -308,6 +308,53 @@ lisp-indent-defun
(indent-region (point-min) (point-max))
(should (equal (buffer-string) orig)))))
+\f
+;;; Filling
+
+(ert-deftest lisp-fill-paragraph-docstring-boundaries ()
+ "Test bug#28937, ensuring filling the docstring filled is properly
+bounded."
+ (with-temp-buffer
+ (insert "\
+(defun test ()
+ \"This is a test docstring.
+Here is some more text.\"
+ 1
+ 2
+ 3
+ 4
+ 5)")
+ (let ((correct (buffer-string)))
+ (emacs-lisp-mode)
+ (search-backward "This is a test docstring")
+ (fill-paragraph) ;function under test
+ (should (equal (buffer-string) correct)))))
+
+(ert-deftest lisp-fill-paragraph-as-displayed ()
+ "Test bug#56197 -- more specifically, validate that a leading indentation
+for a string is preserved in the filled string."
+ (let ((lisp-fill-paragraph-as-displayed t) ;variable under test
+ ;; The following is a contrived example that demonstrates the
+ ;; fill-column problem when the string to fill is indented.
+ (source "\
+'(description \"This is a very long string which is indented by a considerable value, causing it to
+protrude from the configured `fill-column' since
+lisp-fill-paragraph was refactored in version 28.\")"))
+ (with-temp-buffer
+ (insert source)
+ (emacs-lisp-mode)
+ (search-backward "This is a very long string")
+ (fill-paragraph) ;function under test
+ (goto-char (point-min))
+ (message "%s" (buffer-substring-no-properties (point-min) (point-max)))
+ (let ((i 1)
+ (lines-count (count-lines (point-min) (point-max))))
+ (while (< i lines-count)
+ (beginning-of-line i)
+ (end-of-line)
+ (should (<= (current-column) fill-column))
+ (setq i (1+ i)))))))
+
\f
;;; Fontification
--
2.46.0
^ permalink raw reply related [flat|nested] 36+ messages in thread