unofficial mirror of bug-gnu-emacs@gnu.org 
 help / color / mirror / code / Atom feed
* bug#75291: Redisplay not updating fringe when face filter changes
@ 2025-01-02 17:30 Daniel Colascione
  2025-01-02 17:50 ` Eli Zaretskii
  0 siblings, 1 reply; 15+ messages in thread
From: Daniel Colascione @ 2025-01-02 17:30 UTC (permalink / raw)
  To: 75291

Sometimes we don't redraw when its effective face changes indirectly due
to a face filter condition evaluating differently.

An explicit redraw-display works around the problem, but it shouldn't be
necessary and is an efficiency hit.

See the code below:

;; -*- lexical-binding: t -*-

;; emacs -Q

(setf my-window (selected-window))

(dolist (face '(default fringe))
  (face-remap-add-relative
   face `(:filtered (:window foo t) (:background "green"))))

(set-window-parameter my-window 'foo t)

;; During this sit-for, background and fringe should be green
(sit-for 1)
(set-window-parameter my-window 'foo nil)

;; Enable to work around bug
(when nil
  (redraw-display))

;; During this sit-for, background and fringe should be their original
;; color (probably white), but only the background changes.
;; Without redraw-display, the fringe remains green.
(sit-for 1)





^ permalink raw reply	[flat|nested] 15+ messages in thread

* bug#75291: Redisplay not updating fringe when face filter changes
  2025-01-02 17:30 bug#75291: Redisplay not updating fringe when face filter changes Daniel Colascione
@ 2025-01-02 17:50 ` Eli Zaretskii
  2025-01-02 18:36   ` Daniel Colascione
  0 siblings, 1 reply; 15+ messages in thread
From: Eli Zaretskii @ 2025-01-02 17:50 UTC (permalink / raw)
  To: Daniel Colascione; +Cc: 75291

> From: Daniel Colascione <dancol@dancol.org>
> Date: Thu, 02 Jan 2025 12:30:16 -0500
> 
> Sometimes we don't redraw when its effective face changes indirectly due
> to a face filter condition evaluating differently.
> 
> An explicit redraw-display works around the problem, but it shouldn't be
> necessary and is an efficiency hit.
> 
> See the code below:
> 
> ;; -*- lexical-binding: t -*-
> 
> ;; emacs -Q
> 
> (setf my-window (selected-window))
> 
> (dolist (face '(default fringe))
>   (face-remap-add-relative
>    face `(:filtered (:window foo t) (:background "green"))))
> 
> (set-window-parameter my-window 'foo t)
> 
> ;; During this sit-for, background and fringe should be green
> (sit-for 1)
> (set-window-parameter my-window 'foo nil)
> 
> ;; Enable to work around bug
> (when nil
>   (redraw-display))
> 
> ;; During this sit-for, background and fringe should be their original
> ;; color (probably white), but only the background changes.
> ;; Without redraw-display, the fringe remains green.
> (sit-for 1)

I think this is bug#74876 again.  I tried to explain there why the way
the fringe drawing is implemented doesn't work well with
face-remapping (or with changes in the fringe face in general).

Maybe I'm missing something, but supporting what this and that bug
want would need a rewrite of update_window_fringes (and possibly also
the way we record fringes' attributes in the glyph row).





^ permalink raw reply	[flat|nested] 15+ messages in thread

* bug#75291: Redisplay not updating fringe when face filter changes
  2025-01-02 17:50 ` Eli Zaretskii
@ 2025-01-02 18:36   ` Daniel Colascione
  2025-01-02 19:24     ` Michal Nazarewicz
  2025-01-02 19:26     ` Eli Zaretskii
  0 siblings, 2 replies; 15+ messages in thread
From: Daniel Colascione @ 2025-01-02 18:36 UTC (permalink / raw)
  To: Eli Zaretskii; +Cc: 75291, Michal Nazarewicz

Eli Zaretskii <eliz@gnu.org> writes:

>> From: Daniel Colascione <dancol@dancol.org>
>> Date: Thu, 02 Jan 2025 12:30:16 -0500
>> 
>> Sometimes we don't redraw when its effective face changes indirectly due
>> to a face filter condition evaluating differently.
>> 
>> An explicit redraw-display works around the problem, but it shouldn't be
>> necessary and is an efficiency hit.
>> 
>> See the code below:
>> 
>> ;; -*- lexical-binding: t -*-
>> 
>> ;; emacs -Q
>> 
>> (setf my-window (selected-window))
>> 
>> (dolist (face '(default fringe))
>>   (face-remap-add-relative
>>    face `(:filtered (:window foo t) (:background "green"))))
>> 
>> (set-window-parameter my-window 'foo t)
>> 
>> ;; During this sit-for, background and fringe should be green
>> (sit-for 1)
>> (set-window-parameter my-window 'foo nil)
>> 
>> ;; Enable to work around bug
>> (when nil
>>   (redraw-display))
>> 
>> ;; During this sit-for, background and fringe should be their original
>> ;; color (probably white), but only the background changes.
>> ;; Without redraw-display, the fringe remains green.
>> (sit-for 1)
>
> I think this is bug#74876 again.  I tried to explain there why the way
> the fringe drawing is implemented doesn't work well with
> face-remapping (or with changes in the fringe face in general).
>
> Maybe I'm missing something, but supporting what this and that bug
> want would need a rewrite of update_window_fringes (and possibly also
> the way we record fringes' attributes in the glyph row).

That's amazing. Seven years ago, I implemented
https://github.com/dcolascione/emacs-window-highlight/blob/master/window-highlight.el.
I worked around the bug we're discussing using redraw-display. I see
that Michal (++) implemented the same feature independently and reported
the same bug just a few weeks before I got around to reporting the same
bug from seven years ago!  Pure serendipity.

(BTW: my code uses pre-redisplay-function, not a command hook. I've
tried it both ways and found the redisplay hook to be more reliable.)

Anyway, given that the feature has been implemented twice now, maybe we
can find some way to make it work efficiently?  I haven't looked into
how exactly we do fringe invalidation, but isn't there a way we can
limit the blast radius of the redraw by providing a lisp-level function
to invalidate extra GUI parts of specific windows rather than forcing a
redraw-frame?  It's not clear to me why we couldn't skip redrawing every
row's content but redraw every row's fringe anyway.

Ideally, a change to a face in which the change couldn't possibly affect
layout (e.g. changing a background color) would be pretty efficient from
a redisplay POV, since we wouldn't have to even try to reflow any text.





^ permalink raw reply	[flat|nested] 15+ messages in thread

* bug#75291: Redisplay not updating fringe when face filter changes
  2025-01-02 18:36   ` Daniel Colascione
@ 2025-01-02 19:24     ` Michal Nazarewicz
  2025-01-02 19:26     ` Eli Zaretskii
  1 sibling, 0 replies; 15+ messages in thread
From: Michal Nazarewicz @ 2025-01-02 19:24 UTC (permalink / raw)
  To: Daniel Colascione, Eli Zaretskii; +Cc: 75291

> Eli Zaretskii <eliz@gnu.org> writes:
>> I think this is bug#74876 again.  I tried to explain there why the way
>> the fringe drawing is implemented doesn't work well with
>> face-remapping (or with changes in the fringe face in general).
>>
>> Maybe I'm missing something, but supporting what this and that bug
>> want would need a rewrite of update_window_fringes (and possibly also
>> the way we record fringes' attributes in the glyph row).

On Thu, Jan 02 2025, Daniel Colascione wrote:
> (BTW: my code uses pre-redisplay-function, not a command hook. I've
> tried it both ways and found the redisplay hook to be more reliable.)

FYI, the actual auto-dim-other-buffers code doesn’t use
pre-command-hook.  It uses hooks for window, buffer or focus changes.

> Anyway, given that the feature has been implemented twice now, maybe we
> can find some way to make it work efficiently?

Since I managed to work-around the issue, I haven’t looked closely at
the low-level details of the redrawing code.  From afar it certainly
looks like it should be easy to force redrawing of fringes only, but
I also trust Eli’s opinion on complexity of this task more than my
impressions.

While I’d like to have a function indicating that particular window’s
fringes need to be updated (maybe something like ‘(force-window-update
window t)’, I won’t have time to work on this any time soon.

> Ideally, a change to a face in which the change couldn't possibly affect
> layout (e.g. changing a background color) would be pretty efficient from
> a redisplay POV, since we wouldn't have to even try to reflow any
> text.

I don’t think it’s that simple.  Even if all you’re changing is colour,
you still need to redraw the text.  At best, without anti-aliasing there
may be some way to optimise it but I doubt complicating the code to
support that would be worth it.

-- 
Best regards
ミハウ “𝓶𝓲𝓷𝓪86” ナザレヴイツ
«If at first you don’t succeed, give up skydiving»





^ permalink raw reply	[flat|nested] 15+ messages in thread

* bug#75291: Redisplay not updating fringe when face filter changes
  2025-01-02 18:36   ` Daniel Colascione
  2025-01-02 19:24     ` Michal Nazarewicz
@ 2025-01-02 19:26     ` Eli Zaretskii
  2025-01-02 19:50       ` Daniel Colascione
  1 sibling, 1 reply; 15+ messages in thread
From: Eli Zaretskii @ 2025-01-02 19:26 UTC (permalink / raw)
  To: Daniel Colascione; +Cc: 75291, mina86

merge 75291 74876
thanks

> From: Daniel Colascione <dancol@dancol.org>
> Cc: 75291@debbugs.gnu.org, Michal Nazarewicz <mina86@mina86.com>
> Date: Thu, 02 Jan 2025 13:36:34 -0500
> 
> Eli Zaretskii <eliz@gnu.org> writes:
> 
> > I think this is bug#74876 again.  I tried to explain there why the way
> > the fringe drawing is implemented doesn't work well with
> > face-remapping (or with changes in the fringe face in general).
> >
> > Maybe I'm missing something, but supporting what this and that bug
> > want would need a rewrite of update_window_fringes (and possibly also
> > the way we record fringes' attributes in the glyph row).
> 
> That's amazing. Seven years ago, I implemented
> https://github.com/dcolascione/emacs-window-highlight/blob/master/window-highlight.el.
> I worked around the bug we're discussing using redraw-display. I see
> that Michal (++) implemented the same feature independently and reported
> the same bug just a few weeks before I got around to reporting the same
> bug from seven years ago!  Pure serendipity.

Yes, redrawing everything will work, but will also cause flicker, and
is generally expensive, thus undesirable.

> Anyway, given that the feature has been implemented twice now, maybe we
> can find some way to make it work efficiently?  I haven't looked into
> how exactly we do fringe invalidation, but isn't there a way we can
> limit the blast radius of the redraw by providing a lisp-level function
> to invalidate extra GUI parts of specific windows rather than forcing a
> redraw-frame?  It's not clear to me why we couldn't skip redrawing every
> row's content but redraw every row's fringe anyway.
> 
> Ideally, a change to a face in which the change couldn't possibly affect
> layout (e.g. changing a background color) would be pretty efficient from
> a redisplay POV, since we wouldn't have to even try to reflow any text.

AFAIR, the current implementation is the other way around: when some
glyph row changes, we consider redrawing its fringes.  E.g.,
draw_window_fringes only considers glyph rows that are enabled in the
glyph matrix, which means redisplay found that glyph row has changed.
Clearly, someone didn't think the fringe face will change during a
session!

Regarding your idea about Lisp function that would invalidate GUI
parts: it is not very easy, since a Lisp program cannot easily know
where on the screen a given region of buffer positions will be.
There is posn-at-point, of course, but (a) it is quite expensive, and
(b) when Lisp runs, display could be outdated, so what posn-at-point
returns could be inaccurate.

In any case, I don't think this will be needed in the case in point,
because when the fringe face changes, we'd want to redraw the fringes
of the entire window, right?  So redisplay_window would need to notice
the change in the face, and invoke update_window_fringes in a special
way, such that update_window_fringes marks fringes to be redrawn not
only when the glyph row is enabled.  Maybe that would work.

The way to "notice the change in the face" is not a simple problem to
solve, btw.  We currently don't know which faces change when some face
is modified.  So we have a frame-global flag that is set when any face
changes its attributes.  If we use that flag for detecting potential
changes in the fringe face, we'd start redrawing fringes
unnecessarily.  We could add a special flag for the fringe face, but
then we'd need to add a special mechanism in xfaces.c which would
analyze each change in some face's attribute and determine whether it
could possibly affect the fringe face.  Or maybe I'm missing some more
elegant/easier solution.





^ permalink raw reply	[flat|nested] 15+ messages in thread

* bug#75291: Redisplay not updating fringe when face filter changes
  2025-01-02 19:26     ` Eli Zaretskii
@ 2025-01-02 19:50       ` Daniel Colascione
  2025-01-02 20:56         ` Eli Zaretskii
  0 siblings, 1 reply; 15+ messages in thread
From: Daniel Colascione @ 2025-01-02 19:50 UTC (permalink / raw)
  To: Eli Zaretskii; +Cc: 75291, mina86

Eli Zaretskii <eliz@gnu.org> writes:

> merge 75291 74876
> thanks
>
>> From: Daniel Colascione <dancol@dancol.org>
>> Cc: 75291@debbugs.gnu.org, Michal Nazarewicz <mina86@mina86.com>
>> Date: Thu, 02 Jan 2025 13:36:34 -0500
>> 
>> Eli Zaretskii <eliz@gnu.org> writes:
>> 
>> > I think this is bug#74876 again.  I tried to explain there why the way
>> > the fringe drawing is implemented doesn't work well with
>> > face-remapping (or with changes in the fringe face in general).
>> >
>> > Maybe I'm missing something, but supporting what this and that bug
>> > want would need a rewrite of update_window_fringes (and possibly also
>> > the way we record fringes' attributes in the glyph row).
>> 
>> That's amazing. Seven years ago, I implemented
>> https://github.com/dcolascione/emacs-window-highlight/blob/master/window-highlight.el.
>> I worked around the bug we're discussing using redraw-display. I see
>> that Michal (++) implemented the same feature independently and reported
>> the same bug just a few weeks before I got around to reporting the same
>> bug from seven years ago!  Pure serendipity.
>
> Yes, redrawing everything will work, but will also cause flicker, and
> is generally expensive, thus undesirable.

FWIW, it doesn't seem to cause flicker in practice.  I see flicker only
when walking through messages in mu4e --- we do redisplay and draw only
the background, and I haven't figured out why yet.  But in general, on a
modern window system, turning a given redisplay into a full redisplay
shouldn't cause flicker, even if it's inefficient.

>> Anyway, given that the feature has been implemented twice now, maybe we
>> can find some way to make it work efficiently?  I haven't looked into
>> how exactly we do fringe invalidation, but isn't there a way we can
>> limit the blast radius of the redraw by providing a lisp-level function
>> to invalidate extra GUI parts of specific windows rather than forcing a
>> redraw-frame?  It's not clear to me why we couldn't skip redrawing every
>> row's content but redraw every row's fringe anyway.
>> 
>> Ideally, a change to a face in which the change couldn't possibly affect
>> layout (e.g. changing a background color) would be pretty efficient from
>> a redisplay POV, since we wouldn't have to even try to reflow any text.
>
> AFAIR, the current implementation is the other way around: when some
> glyph row changes, we consider redrawing its fringes.  E.g.,
> draw_window_fringes only considers glyph rows that are enabled in the
> glyph matrix, which means redisplay found that glyph row has changed.
> Clearly, someone didn't think the fringe face will change during a
> session!

I came across overlay_arrows_changed_p --- isn't this function trying to
deal with exactly the case of something in the fringe changing outside
the changed text region?

> Regarding your idea about Lisp function that would invalidate GUI
> parts: it is not very easy, since a Lisp program cannot easily know
> where on the screen a given region of buffer positions will be.
> There is posn-at-point, of course, but (a) it is quite expensive, and
> (b) when Lisp runs, display could be outdated, so what posn-at-point
> returns could be inaccurate.

I was imagining a lisp function that would make the next redisplay of a
window do what you suggest in the next paragraph.

I'm not sure we'd even need an explicit Lisp function though.
Face filters with :window are defined to compare window parameter values
with eq, so couldn't we keep track of all the :filtered face
specifications we encounter during face resolution and have
set-window-parameter check whether the parameter it's setting is on the
list of possible face filters and, if so, force next redisplay to
evaluate faces? set-window-parameter wouldn't even have to do a deep
comparison, because it's just eq.

> In any case, I don't think this will be needed in the case in point,
> because when the fringe face changes, we'd want to redraw the fringes
> of the entire window, right?  So redisplay_window would need to notice
> the change in the face, and invoke update_window_fringes in a special
> way, such that update_window_fringes marks fringes to be redrawn not
> only when the glyph row is enabled.  Maybe that would work.

> The way to "notice the change in the face" is not a simple problem to
> solve, btw.  We currently don't know which faces change when some face
> is modified.  So we have a frame-global flag that is set when any face
> changes its attributes.  If we use that flag for detecting potential
> changes in the fringe face, we'd start redrawing fringes
> unnecessarily.

How many unnecessary face recalculations would we do? ISTM we could make
the invalidation pretty precise as long as we're just looking at
window parameters.

> We could add a special flag for the fringe face, but
> then we'd need to add a special mechanism in xfaces.c which would
> analyze each change in some face's attribute and determine whether it
> could possibly affect the fringe face.  Or maybe I'm missing some more
> elegant/easier solution.






^ permalink raw reply	[flat|nested] 15+ messages in thread

* bug#75291: Redisplay not updating fringe when face filter changes
  2025-01-02 19:50       ` Daniel Colascione
@ 2025-01-02 20:56         ` Eli Zaretskii
  2025-01-03 17:25           ` Daniel Colascione
  0 siblings, 1 reply; 15+ messages in thread
From: Eli Zaretskii @ 2025-01-02 20:56 UTC (permalink / raw)
  To: Daniel Colascione; +Cc: 75291, mina86

> From: Daniel Colascione <dancol@dancol.org>
> Cc: 75291@debbugs.gnu.org,  mina86@mina86.com
> Date: Thu, 02 Jan 2025 14:50:54 -0500
> 
> Eli Zaretskii <eliz@gnu.org> writes:
> 
> > Yes, redrawing everything will work, but will also cause flicker, and
> > is generally expensive, thus undesirable.
> 
> FWIW, it doesn't seem to cause flicker in practice.  I see flicker only
> when walking through messages in mu4e --- we do redisplay and draw only
> the background, and I haven't figured out why yet.  But in general, on a
> modern window system, turning a given redisplay into a full redisplay
> shouldn't cause flicker, even if it's inefficient.

I think it depends on whether you use double-buffering (some people
don't or cannot) and whether you have the mouse pointer over an Emacs
frame.  Also, depending on the GUI toolkit, the decorations might
flicker.

> I came across overlay_arrows_changed_p --- isn't this function trying to
> deal with exactly the case of something in the fringe changing outside
> the changed text region?

So you want to add to display_line code that sets each glyph_row's
redraw_fringe_bitmaps_p flag when the fringe face changes?  That could
probably work, provided that we disable redisplay optimizations which
might avoid calling display_line (you will see that we already disable
such optimizations when overlay_arrows_changed_p returns non-zero).
We might actually need to disable more of the optimizations, because
the overlay-arrow thing doesn't contradict the optimizations that
scroll the pixels, something that reaction to changes in the fringe
face cannot tolerate.

> > Regarding your idea about Lisp function that would invalidate GUI
> > parts: it is not very easy, since a Lisp program cannot easily know
> > where on the screen a given region of buffer positions will be.
> > There is posn-at-point, of course, but (a) it is quite expensive, and
> > (b) when Lisp runs, display could be outdated, so what posn-at-point
> > returns could be inaccurate.
> 
> I was imagining a lisp function that would make the next redisplay of a
> window do what you suggest in the next paragraph.

What does Lisp know about the fringe face that the display engine
doesn't?

> I'm not sure we'd even need an explicit Lisp function though.
> Face filters with :window are defined to compare window parameter values
> with eq, so couldn't we keep track of all the :filtered face
> specifications we encounter during face resolution and have
> set-window-parameter check whether the parameter it's setting is on the
> list of possible face filters and, if so, force next redisplay to
> evaluate faces? set-window-parameter wouldn't even have to do a deep
> comparison, because it's just eq.

First, the fringe face might not be window-specific (by default, it
isn't).  So I'm not sure how a window parameter will help.
face-remapping-alist is per-buffer, not per-window.

Next, the main problem with faces is face inheritance and face merging
(the latter is not relevant to fringe, I think).  Given that some
attribute of some face changes, how do you know whether such a change
causes the fringe face to change?  We'd probably need to realize it
anew, and then compare to the cached one?  And we'd need to do that
for each window, because of face-remapping-alist?

> > The way to "notice the change in the face" is not a simple problem to
> > solve, btw.  We currently don't know which faces change when some face
> > is modified.  So we have a frame-global flag that is set when any face
> > changes its attributes.  If we use that flag for detecting potential
> > changes in the fringe face, we'd start redrawing fringes
> > unnecessarily.
> 
> How many unnecessary face recalculations would we do? ISTM we could make
> the invalidation pretty precise as long as we're just looking at
> window parameters.

See above.

I think the proof of the proverbial pudding is in eating: maybe doing
the above will produce reasonable performance.





^ permalink raw reply	[flat|nested] 15+ messages in thread

* bug#75291: Redisplay not updating fringe when face filter changes
  2025-01-02 20:56         ` Eli Zaretskii
@ 2025-01-03 17:25           ` Daniel Colascione
  2025-01-03 19:31             ` Eli Zaretskii
  0 siblings, 1 reply; 15+ messages in thread
From: Daniel Colascione @ 2025-01-03 17:25 UTC (permalink / raw)
  To: Eli Zaretskii; +Cc: 75291, mina86

Eli Zaretskii <eliz@gnu.org> writes:

>> From: Daniel Colascione <dancol@dancol.org>
>> Cc: 75291@debbugs.gnu.org,  mina86@mina86.com
>> Date: Thu, 02 Jan 2025 14:50:54 -0500
>> 
>> Eli Zaretskii <eliz@gnu.org> writes:
>> 
>> > Yes, redrawing everything will work, but will also cause flicker, and
>> > is generally expensive, thus undesirable.
>> 
>> FWIW, it doesn't seem to cause flicker in practice.  I see flicker only
>> when walking through messages in mu4e --- we do redisplay and draw only
>> the background, and I haven't figured out why yet.  But in general, on a
>> modern window system, turning a given redisplay into a full redisplay
>> shouldn't cause flicker, even if it's inefficient.
>
> I think it depends on whether you use double-buffering (some people
> don't or cannot) and whether you have the mouse pointer over an Emacs
> frame.  Also, depending on the GUI toolkit, the decorations might
> flicker.

TTY windows don't have fringes, and the most commonly-used window
systems all do atomic updates nowadays.

>> I came across overlay_arrows_changed_p --- isn't this function trying to
>> deal with exactly the case of something in the fringe changing outside
>> the changed text region?
>
> So you want to add to display_line code that sets each glyph_row's
> redraw_fringe_bitmaps_p flag when the fringe face changes?  That could
> probably work, provided that we disable redisplay optimizations which
> might avoid calling display_line (you will see that we already disable
> such optimizations when overlay_arrows_changed_p returns non-zero).
> We might actually need to disable more of the optimizations, because
> the overlay-arrow thing doesn't contradict the optimizations that
> scroll the pixels, something that reaction to changes in the fringe
> face cannot tolerate.

That might work, but I don't think we even need anything that
complicated or low-level.  Not many are using :filtered now, and those
that do big redraws anyway.  How about this simpler code that gets us
correctness, albeit more conservatively?

diff --git a/src/window.c b/src/window.c
index 5a10c381eaf..6d135a67a66 100644
--- a/src/window.c
+++ b/src/window.c
@@ -2396,11 +2396,29 @@ DEFUN ("set-window-parameter", Fset_window_parameter,
   Lisp_Object old_alist_elt;
 
   old_alist_elt = Fassq (parameter, w->window_parameters);
+
+  /* If this window parameter has been used in a face remapping filter
+     expression and we changed its value, force a from-scratch redisplay
+     to make sure that everything that depends on the window parameter
+     value is up-to-date.  FIXME: instead of taking a sledgehammer to
+     redisplay, we could be more precise in tracking what display bits
+     depend on which remapped faces.  Skip redrawing TTY frames: they
+     don't have fringes or other graphical bits we might have to redraw
+     here.  */
+  if (SYMBOLP (parameter) &&
+      WINDOW_LIVE_P (window) &&
+      FRAME_WINDOW_P (WINDOW_XFRAME (w)) &&
+      !NILP (Fget (parameter, Qface_filter)) &&
+      !EQ (CDR_SAFE (old_alist_elt), value))
+    redraw_frame (WINDOW_XFRAME (w));
+
   if (NILP (old_alist_elt))
     wset_window_parameters
       (w, Fcons (Fcons (parameter, value), w->window_parameters));
   else
     Fsetcdr (old_alist_elt, value);
+
+
   return value;
 }
 
diff --git a/src/xfaces.c b/src/xfaces.c
index d1ca2e0d5d4..1f58bdbf6ae 100644
--- a/src/xfaces.c
+++ b/src/xfaces.c
@@ -2512,6 +2512,9 @@ evaluate_face_filter (Lisp_Object filter, struct window *w,
     if (!NILP (filter))
       goto err;
 
+    if (NILP (Fget (parameter, Qface_filter)))
+      Fput (parameter, Qface_filter, Qt);
+
     bool match = false;
     if (w)
       {
@@ -7623,6 +7626,8 @@ syms_of_xfaces (void)
   Vface_remapping_alist = Qnil;
   DEFSYM (Qface_remapping_alist,"face-remapping-alist");
 
+  DEFSYM (Qface_filter, "face-filter");
+
   DEFVAR_LISP ("face-font-rescale-alist", Vface_font_rescale_alist,
 	       doc: /* Alist of fonts vs the rescaling factors.
 Each element is a cons (FONT-PATTERN . RESCALE-RATIO), where





^ permalink raw reply related	[flat|nested] 15+ messages in thread

* bug#75291: Redisplay not updating fringe when face filter changes
  2025-01-03 17:25           ` Daniel Colascione
@ 2025-01-03 19:31             ` Eli Zaretskii
  2025-01-03 19:46               ` Daniel Colascione
  0 siblings, 1 reply; 15+ messages in thread
From: Eli Zaretskii @ 2025-01-03 19:31 UTC (permalink / raw)
  To: Daniel Colascione; +Cc: 75291, mina86

> From: Daniel Colascione <dancol@dancol.org>
> Cc: 75291@debbugs.gnu.org,  mina86@mina86.com
> Date: Fri, 03 Jan 2025 12:25:05 -0500
> 
> > I think it depends on whether you use double-buffering (some people
> > don't or cannot) and whether you have the mouse pointer over an Emacs
> > frame.  Also, depending on the GUI toolkit, the decorations might
> > flicker.
> 
> TTY windows don't have fringes, and the most commonly-used window
> systems all do atomic updates nowadays.

People still report flickering from time to time, so I don't think
this never happens.

> > So you want to add to display_line code that sets each glyph_row's
> > redraw_fringe_bitmaps_p flag when the fringe face changes?  That could
> > probably work, provided that we disable redisplay optimizations which
> > might avoid calling display_line (you will see that we already disable
> > such optimizations when overlay_arrows_changed_p returns non-zero).
> > We might actually need to disable more of the optimizations, because
> > the overlay-arrow thing doesn't contradict the optimizations that
> > scroll the pixels, something that reaction to changes in the fringe
> > face cannot tolerate.
> 
> That might work, but I don't think we even need anything that
> complicated or low-level.  Not many are using :filtered now, and those
> that do big redraws anyway.  How about this simpler code that gets us
> correctness, albeit more conservatively?

Doesn't that only support face remapping with :filtered attribute?
What about the more general case where the fringe face is remapped in
a way that's independent of the windows?





^ permalink raw reply	[flat|nested] 15+ messages in thread

* bug#75291: Redisplay not updating fringe when face filter changes
  2025-01-03 19:31             ` Eli Zaretskii
@ 2025-01-03 19:46               ` Daniel Colascione
  2025-01-03 20:10                 ` Eli Zaretskii
  0 siblings, 1 reply; 15+ messages in thread
From: Daniel Colascione @ 2025-01-03 19:46 UTC (permalink / raw)
  To: Eli Zaretskii; +Cc: 75291, mina86

Eli Zaretskii <eliz@gnu.org> writes:

>> From: Daniel Colascione <dancol@dancol.org>
>> Cc: 75291@debbugs.gnu.org,  mina86@mina86.com
>> Date: Fri, 03 Jan 2025 12:25:05 -0500
>> 
>> > I think it depends on whether you use double-buffering (some people
>> > don't or cannot) and whether you have the mouse pointer over an Emacs
>> > frame.  Also, depending on the GUI toolkit, the decorations might
>> > flicker.
>> 
>> TTY windows don't have fringes, and the most commonly-used window
>> systems all do atomic updates nowadays.
>
> People still report flickering from time to time, so I don't think
> this never happens.
>
>> > So you want to add to display_line code that sets each glyph_row's
>> > redraw_fringe_bitmaps_p flag when the fringe face changes?  That could
>> > probably work, provided that we disable redisplay optimizations which
>> > might avoid calling display_line (you will see that we already disable
>> > such optimizations when overlay_arrows_changed_p returns non-zero).
>> > We might actually need to disable more of the optimizations, because
>> > the overlay-arrow thing doesn't contradict the optimizations that
>> > scroll the pixels, something that reaction to changes in the fringe
>> > face cannot tolerate.
>> 
>> That might work, but I don't think we even need anything that
>> complicated or low-level.  Not many are using :filtered now, and those
>> that do big redraws anyway.  How about this simpler code that gets us
>> correctness, albeit more conservatively?
>
> Doesn't that only support face remapping with :filtered attribute?
> What about the more general case where the fringe face is remapped in
> a way that's independent of the windows?

That seems to work already. It's only in the fringe that I see problems
--- it just doesn't seem worth it to limit the redraw to the fringe.





^ permalink raw reply	[flat|nested] 15+ messages in thread

* bug#75291: Redisplay not updating fringe when face filter changes
  2025-01-03 19:46               ` Daniel Colascione
@ 2025-01-03 20:10                 ` Eli Zaretskii
  2025-01-03 20:27                   ` Eli Zaretskii
  2025-01-03 20:57                   ` Daniel Colascione
  0 siblings, 2 replies; 15+ messages in thread
From: Eli Zaretskii @ 2025-01-03 20:10 UTC (permalink / raw)
  To: Daniel Colascione; +Cc: 75291, mina86

> From: Daniel Colascione <dancol@dancol.org>
> Cc: 75291@debbugs.gnu.org,  mina86@mina86.com
> Date: Fri, 03 Jan 2025 14:46:20 -0500
> 
> Eli Zaretskii <eliz@gnu.org> writes:
> 
> >> From: Daniel Colascione <dancol@dancol.org>
> >> Cc: 75291@debbugs.gnu.org,  mina86@mina86.com
> >> Date: Fri, 03 Jan 2025 12:25:05 -0500
> >> 
> >> > I think it depends on whether you use double-buffering (some people
> >> > don't or cannot) and whether you have the mouse pointer over an Emacs
> >> > frame.  Also, depending on the GUI toolkit, the decorations might
> >> > flicker.
> >> 
> >> TTY windows don't have fringes, and the most commonly-used window
> >> systems all do atomic updates nowadays.
> >
> > People still report flickering from time to time, so I don't think
> > this never happens.
> >
> >> > So you want to add to display_line code that sets each glyph_row's
> >> > redraw_fringe_bitmaps_p flag when the fringe face changes?  That could
> >> > probably work, provided that we disable redisplay optimizations which
> >> > might avoid calling display_line (you will see that we already disable
> >> > such optimizations when overlay_arrows_changed_p returns non-zero).
> >> > We might actually need to disable more of the optimizations, because
> >> > the overlay-arrow thing doesn't contradict the optimizations that
> >> > scroll the pixels, something that reaction to changes in the fringe
> >> > face cannot tolerate.
> >> 
> >> That might work, but I don't think we even need anything that
> >> complicated or low-level.  Not many are using :filtered now, and those
> >> that do big redraws anyway.  How about this simpler code that gets us
> >> correctness, albeit more conservatively?
> >
> > Doesn't that only support face remapping with :filtered attribute?
> > What about the more general case where the fringe face is remapped in
> > a way that's independent of the windows?
> 
> That seems to work already. It's only in the fringe that I see problems
> --- it just doesn't seem worth it to limit the redraw to the fringe.

Sorry, I don't understand.  I _was_ talking about the fringe face.

But if redraw_frame solves the issue, and doesn't cause unpleasant or
expensive redraws, feel free to install on the master branch.  But
please change this:

+  if (SYMBOLP (parameter) &&
+      WINDOW_LIVE_P (window) &&
+      FRAME_WINDOW_P (WINDOW_XFRAME (w)) &&
+      !NILP (Fget (parameter, Qface_filter)) && <<<<<<<<<<<<<<<<<
+      !EQ (CDR_SAFE (old_alist_elt), value))
+    redraw_frame (WINDOW_XFRAME (w));

to say this instead:

+  if (SYMBOLP (parameter) &&
+      WINDOW_LIVE_P (window) &&
+      FRAME_WINDOW_P (WINDOW_XFRAME (w)) &&
+      EQ (Fget (parameter, Qface_filter), Qt) && <<<<<<<<<<<<<<<<<
+      !EQ (CDR_SAFE (old_alist_elt), value))
+    redraw_frame (WINDOW_XFRAME (w));

(A stylistic comment: our conventions is to put the && operator at the
beginning of a line, not at the end of a line.)





^ permalink raw reply	[flat|nested] 15+ messages in thread

* bug#75291: Redisplay not updating fringe when face filter changes
  2025-01-03 20:10                 ` Eli Zaretskii
@ 2025-01-03 20:27                   ` Eli Zaretskii
  2025-01-03 20:57                   ` Daniel Colascione
  1 sibling, 0 replies; 15+ messages in thread
From: Eli Zaretskii @ 2025-01-03 20:27 UTC (permalink / raw)
  To: dancol; +Cc: 75291, mina86

> Cc: 75291@debbugs.gnu.org, mina86@mina86.com
> Date: Fri, 03 Jan 2025 22:10:45 +0200
> From: Eli Zaretskii <eliz@gnu.org>
> 
> > From: Daniel Colascione <dancol@dancol.org>
> > Cc: 75291@debbugs.gnu.org,  mina86@mina86.com
> > Date: Fri, 03 Jan 2025 14:46:20 -0500
> > 
> > Eli Zaretskii <eliz@gnu.org> writes:
> > 
> > >> From: Daniel Colascione <dancol@dancol.org>
> > >> Cc: 75291@debbugs.gnu.org,  mina86@mina86.com
> > >> Date: Fri, 03 Jan 2025 12:25:05 -0500
> > >> 
> > >> > I think it depends on whether you use double-buffering (some people
> > >> > don't or cannot) and whether you have the mouse pointer over an Emacs
> > >> > frame.  Also, depending on the GUI toolkit, the decorations might
> > >> > flicker.
> > >> 
> > >> TTY windows don't have fringes, and the most commonly-used window
> > >> systems all do atomic updates nowadays.
> > >
> > > People still report flickering from time to time, so I don't think
> > > this never happens.
> > >
> > >> > So you want to add to display_line code that sets each glyph_row's
> > >> > redraw_fringe_bitmaps_p flag when the fringe face changes?  That could
> > >> > probably work, provided that we disable redisplay optimizations which
> > >> > might avoid calling display_line (you will see that we already disable
> > >> > such optimizations when overlay_arrows_changed_p returns non-zero).
> > >> > We might actually need to disable more of the optimizations, because
> > >> > the overlay-arrow thing doesn't contradict the optimizations that
> > >> > scroll the pixels, something that reaction to changes in the fringe
> > >> > face cannot tolerate.
> > >> 
> > >> That might work, but I don't think we even need anything that
> > >> complicated or low-level.  Not many are using :filtered now, and those
> > >> that do big redraws anyway.  How about this simpler code that gets us
> > >> correctness, albeit more conservatively?
> > >
> > > Doesn't that only support face remapping with :filtered attribute?
> > > What about the more general case where the fringe face is remapped in
> > > a way that's independent of the windows?
> > 
> > That seems to work already. It's only in the fringe that I see problems
> > --- it just doesn't seem worth it to limit the redraw to the fringe.
> 
> Sorry, I don't understand.  I _was_ talking about the fringe face.
> 
> But if redraw_frame solves the issue, and doesn't cause unpleasant or
> expensive redraws, feel free to install on the master branch.  But
> please change this:

Btw: this is _really_ a sledgehammer solution, and it will affect
similar changes in any face, not just the fringe face.  I wonder how
long will it take for complaints to come in.

How about leaving behind a variable exposed to Lisp that could be used
to disable this sledgehammer?  Then we at least could tell people who
don't want this how to disable it.





^ permalink raw reply	[flat|nested] 15+ messages in thread

* bug#75291: Redisplay not updating fringe when face filter changes
  2025-01-03 20:10                 ` Eli Zaretskii
  2025-01-03 20:27                   ` Eli Zaretskii
@ 2025-01-03 20:57                   ` Daniel Colascione
  2025-01-04  7:12                     ` Eli Zaretskii
  1 sibling, 1 reply; 15+ messages in thread
From: Daniel Colascione @ 2025-01-03 20:57 UTC (permalink / raw)
  To: Eli Zaretskii; +Cc: 75291, mina86

Eli Zaretskii <eliz@gnu.org> writes:

>> From: Daniel Colascione <dancol@dancol.org>
>> Cc: 75291@debbugs.gnu.org,  mina86@mina86.com
>> Date: Fri, 03 Jan 2025 14:46:20 -0500
>> 
>> Eli Zaretskii <eliz@gnu.org> writes:
>> 
>> >> From: Daniel Colascione <dancol@dancol.org>
>> >> Cc: 75291@debbugs.gnu.org,  mina86@mina86.com
>> >> Date: Fri, 03 Jan 2025 12:25:05 -0500
>> >> 
>> >> > I think it depends on whether you use double-buffering (some people
>> >> > don't or cannot) and whether you have the mouse pointer over an Emacs
>> >> > frame.  Also, depending on the GUI toolkit, the decorations might
>> >> > flicker.
>> >> 
>> >> TTY windows don't have fringes, and the most commonly-used window
>> >> systems all do atomic updates nowadays.
>> >
>> > People still report flickering from time to time, so I don't think
>> > this never happens.
>> >
>> >> > So you want to add to display_line code that sets each glyph_row's
>> >> > redraw_fringe_bitmaps_p flag when the fringe face changes?  That could
>> >> > probably work, provided that we disable redisplay optimizations which
>> >> > might avoid calling display_line (you will see that we already disable
>> >> > such optimizations when overlay_arrows_changed_p returns non-zero).
>> >> > We might actually need to disable more of the optimizations, because
>> >> > the overlay-arrow thing doesn't contradict the optimizations that
>> >> > scroll the pixels, something that reaction to changes in the fringe
>> >> > face cannot tolerate.
>> >> 
>> >> That might work, but I don't think we even need anything that
>> >> complicated or low-level.  Not many are using :filtered now, and those
>> >> that do big redraws anyway.  How about this simpler code that gets us
>> >> correctness, albeit more conservatively?
>> >
>> > Doesn't that only support face remapping with :filtered attribute?
>> > What about the more general case where the fringe face is remapped in
>> > a way that's independent of the windows?
>> 
>> That seems to work already. It's only in the fringe that I see problems
>> --- it just doesn't seem worth it to limit the redraw to the fringe.
>
> Sorry, I don't understand.  I _was_ talking about the fringe face.

I misread your question. To answer it: what are the circumstances in
which the effective fringe face can change behind our backs?  Any change
to a face attribute proper will cause a redraw anyway.
The face-remap.el functions call force-mode-line-update to make sure
face remapping lists take effect, so we should have non-window-specific
remapping changes covered.  What's left is the window parameters.
Any other cases?

> But if redraw_frame solves the issue, and doesn't cause unpleasant or
> expensive redraws, feel free to install on the master branch.  But
> please change this:
>
> +  if (SYMBOLP (parameter) &&
> +      WINDOW_LIVE_P (window) &&
> +      FRAME_WINDOW_P (WINDOW_XFRAME (w)) &&
> +      !NILP (Fget (parameter, Qface_filter)) && <<<<<<<<<<<<<<<<<
> +      !EQ (CDR_SAFE (old_alist_elt), value))
> +    redraw_frame (WINDOW_XFRAME (w));
>
> to say this instead:
>
> +  if (SYMBOLP (parameter) &&
> +      WINDOW_LIVE_P (window) &&
> +      FRAME_WINDOW_P (WINDOW_XFRAME (w)) &&
> +      EQ (Fget (parameter, Qface_filter), Qt) && <<<<<<<<<<<<<<<<<
> +      !EQ (CDR_SAFE (old_alist_elt), value))
> +    redraw_frame (WINDOW_XFRAME (w));
>
> (A stylistic comment: our conventions is to put the && operator at the
> beginning of a line, not at the end of a line.)

Thanks for the reminder!





^ permalink raw reply	[flat|nested] 15+ messages in thread

* bug#75291: Redisplay not updating fringe when face filter changes
  2025-01-03 20:57                   ` Daniel Colascione
@ 2025-01-04  7:12                     ` Eli Zaretskii
       [not found]                       ` <C3F80FE6-077F-4A74-9C48-DD9EAE4E3266@dancol.org>
  0 siblings, 1 reply; 15+ messages in thread
From: Eli Zaretskii @ 2025-01-04  7:12 UTC (permalink / raw)
  To: Daniel Colascione; +Cc: 75291, mina86

> From: Daniel Colascione <dancol@dancol.org>
> Cc: 75291@debbugs.gnu.org,  mina86@mina86.com
> Date: Fri, 03 Jan 2025 15:57:37 -0500
> 
> Eli Zaretskii <eliz@gnu.org> writes:
> 
> >> > Doesn't that only support face remapping with :filtered attribute?
> >> > What about the more general case where the fringe face is remapped in
> >> > a way that's independent of the windows?
> >> 
> >> That seems to work already. It's only in the fringe that I see problems
> >> --- it just doesn't seem worth it to limit the redraw to the fringe.
> >
> > Sorry, I don't understand.  I _was_ talking about the fringe face.
> 
> I misread your question. To answer it: what are the circumstances in
> which the effective fringe face can change behind our backs?

For example, face remapping via face-remapping-alist.

> Any change to a face attribute proper will cause a redraw anyway.

Changes in face attributes set the frame's redisplay flag, but that
just tells the display engine to disable some radical optimizations,
like considering only the selected window and only the line where the
cursor is.  AFAIU, this will not necessarily cause all the fringes of
all the frame's windows to be redrawn, because we only redraw the
fringes of the glyph rows that we consider changed.

> The face-remap.el functions call force-mode-line-update to make sure
> face remapping lists take effect

force-mode-line-update sets a buffer-specific flag that disables some
redisplay optimizations, but I don't think that guarantees redrawing
of all the fringes of all the windows which show that buffer.

So I'm not sure we have that covered.  But if you have tested that and
this actually work, fine; I don't pretend to know for sure this won't
work.





^ permalink raw reply	[flat|nested] 15+ messages in thread

* bug#75291: Redisplay not updating fringe when face filter changes
       [not found]                       ` <C3F80FE6-077F-4A74-9C48-DD9EAE4E3266@dancol.org>
@ 2025-01-18  9:29                         ` Eli Zaretskii
  0 siblings, 0 replies; 15+ messages in thread
From: Eli Zaretskii @ 2025-01-18  9:29 UTC (permalink / raw)
  To: Daniel Colascione; +Cc: 75291-done

> Date: Sat, 04 Jan 2025 15:24:43 -0500
> From: Daniel Colascione <dancol@dancol.org>
> 
> On January 4, 2025 2:12:05 AM EST, Eli Zaretskii <eliz@gnu.org> wrote:
> >> From: Daniel Colascione <dancol@dancol.org>
> >> Cc: 75291@debbugs.gnu.org,  mina86@mina86.com
> >> Date: Fri, 03 Jan 2025 15:57:37 -0500
> >> 
> >> Eli Zaretskii <eliz@gnu.org> writes:
> >> 
> >> >> > Doesn't that only support face remapping with :filtered attribute?
> >> >> > What about the more general case where the fringe face is remapped in
> >> >> > a way that's independent of the windows?
> >> >> 
> >> >> That seems to work already. It's only in the fringe that I see problems
> >> >> --- it just doesn't seem worth it to limit the redraw to the fringe.
> >> >
> >> > Sorry, I don't understand.  I _was_ talking about the fringe face.
> >> 
> >> I misread your question. To answer it: what are the circumstances in
> >> which the effective fringe face can change behind our backs?
> >
> >For example, face remapping via face-remapping-alist.
> 
> Directly, without going through face-remap? We already caution against that and should do so more
> strongly. I don't think it's worth supporting direct modification of the alist.
> 
> >
> >> Any change to a face attribute proper will cause a redraw anyway.
> >
> >Changes in face attributes set the frame's redisplay flag, but that
> >just tells the display engine to disable some radical optimizations,
> >like considering only the selected window and only the line where the
> >cursor is.  AFAIU, this will not necessarily cause all the fringes of
> >all the frame's windows to be redrawn, because we only redraw the
> >fringes of the glyph rows that we consider changed.
> >
> >> The face-remap.el functions call force-mode-line-update to make sure
> >> face remapping lists take effect
> >
> >force-mode-line-update sets a buffer-specific flag that disables some
> >redisplay optimizations, but I don't think that guarantees redrawing
> >of all the fringes of all the windows which show that buffer.
> 
> Somehow, the basic case we have in face-remap seems to work already. I've put a change in master to
> handle the window parameter filter case. Seems to work for me now without observable downsides. No extra
> flickering. There's also a way to turn it off.

No further comments, so I presume the bug is fixed, and I'm closing
it.





^ permalink raw reply	[flat|nested] 15+ messages in thread

end of thread, other threads:[~2025-01-18  9:29 UTC | newest]

Thread overview: 15+ messages (download: mbox.gz follow: Atom feed
-- links below jump to the message on this page --
2025-01-02 17:30 bug#75291: Redisplay not updating fringe when face filter changes Daniel Colascione
2025-01-02 17:50 ` Eli Zaretskii
2025-01-02 18:36   ` Daniel Colascione
2025-01-02 19:24     ` Michal Nazarewicz
2025-01-02 19:26     ` Eli Zaretskii
2025-01-02 19:50       ` Daniel Colascione
2025-01-02 20:56         ` Eli Zaretskii
2025-01-03 17:25           ` Daniel Colascione
2025-01-03 19:31             ` Eli Zaretskii
2025-01-03 19:46               ` Daniel Colascione
2025-01-03 20:10                 ` Eli Zaretskii
2025-01-03 20:27                   ` Eli Zaretskii
2025-01-03 20:57                   ` Daniel Colascione
2025-01-04  7:12                     ` Eli Zaretskii
     [not found]                       ` <C3F80FE6-077F-4A74-9C48-DD9EAE4E3266@dancol.org>
2025-01-18  9:29                         ` Eli Zaretskii

Code repositories for project(s) associated with this public inbox

	https://git.savannah.gnu.org/cgit/emacs.git

This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for read-only IMAP folder(s) and NNTP newsgroup(s).