Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Make sixel output flicker free #1943

Draft
wants to merge 2 commits into
base: master
Choose a base branch
from
Draft

Conversation

Kuchteq
Copy link

@Kuchteq Kuchteq commented Apr 2, 2025

Majorly overhaul sixel preview. Basically fixes #1738
Largely inspired by how it is done in tcell's sixel demo

I upload the comparison between current master, this pr and the last known version when there was no flicker. Terminal emulator: foot.

flicker-small.mp4

It might miss some edge cases, so please suggest any changes.

@Kuchteq Kuchteq marked this pull request as draft April 2, 2025 14:10
Copy link
Collaborator

@joelim-work joelim-work left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Thanks for working on this - I have tested this myself on wezterm and found that the overall experience was smoother.

I should also mention that sixel support was originally implemented in #1211, and it predates the LockRegion/TTY support added in tcell 2.7.0, so thank you for taking the time to update the code accordingly.

I do have a few comments that I would like to see addressed before merging this, and even then I think I will leave it here for a while in case other users wish to test the changes themselves.

sxs.altFill = !sxs.altFill
func (sxs *sixelScreen) clearSixel(win *win, screen tcell.Screen, filePath string) {
if filePath != sxs.lastFile || win.w != sxs.lastWinW || win.h != sxs.lastWinH {
screen.LockRegion(win.x, win.y, win.w, win.h, false)
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Does Screen.LockRegion (with lock = false) actually clear an existing sixel image? I was under the impression that it merely unlocks the region so that changes can be made from tcell.

The reason I mention this is that when I was testing I found that images of different sizes would just be overlaid on top of each other, instead of clearing the existing image first and then drawing a new one. This only occurs when the sixel data is cached like below (note that there is no exit 1):

#!/bin/sh
case "$(file -Lb --mime-type -- "$1")" in
  image/*)
    chafa -f sixel -s "$2x$3" --animate off --polite on "$1"
    ;;
  text/*)
    cat "$1"
    ;;
esac

I'm guessing what happens is that when the sixel data is not cached, then there is no issue:

  1. The region is unlocked in clearSixel
  2. A new blank reg object is created as a temporary placeholder while the sixel data is being reloaded
  3. When the sixel data has finished loading, the screen is cleared at the start of the ui.draw function, thereby clearing the existing sixel image
  4. The new sixel image is drawn

And if the sixel data is cached, then steps 2 and 3 are skipped. But this is just a guess on my part and I could be wrong.

Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is a very insightful comment. My config included an exit 1 and I was quite frankly oblivious to the whole caching mechanism while trying to implement this. I'll keep this in mind. I got a bit overexcited about this implementation so I'm keeping it a draft until I make it solid.

@Kuchteq
Copy link
Author

Kuchteq commented Apr 3, 2025

A prominent issue that I've encountered is how when I open up some file that spawns another window and the open function is a regular shell (marked with $), it will cause a rerender after the command is executed and then if the window gets resized in the meantime (say due to running a tiling wm) there will be another rerender caused by the dimension change. This frequently corrupts the screen but it's not a guarantee, so there might be some race condition as to which event gets launched first. I'll have a go at it to try to address that.

// Get the terminfo for our current terminal
ti, err := tcell.LookupTerminfo(os.Getenv("TERM"))
if err != nil {
log.Printf("terminal lookup failed during sixel render %s", err)
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think you are missing a return here.

if sxs.lastFile != filePath {
sxs.altFill = !sxs.altFill
func (sxs *sixelScreen) clearSixel(win *win, screen tcell.Screen, filePath string) {
if filePath != sxs.lastFile || win.w != sxs.lastWinW || win.h != sxs.lastWinH {
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

So I tried lf using the default settings without the sixel option enabled, and found that this was still being called. I'm not sure how expensive calling Screen.LockRegion is. Maybe it is better to only call this when a sixel image is currently being shown, i.e. add the condition sxs.lastFile != "".

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

lf image preview flickering
2 participants