-
-
Notifications
You must be signed in to change notification settings - Fork 1.5k
Description
Checklist
- I have searched the issue tracker for open issues that relate to the same problem, before opening a new one.
- This issue only relates to a single bug. I will open new issues for any other problems.
Describe the bug
If a multiline entry has enough text for a scrollbar to appear, whenever this entry gains or loses focus the widget automatically scrolls the text so that the cursor position is visible. Typically this behavior is very far from user's expectation. The entry should automatically scroll to the cursor position only when something happens at the cursor (e.g., user inputs something or changes the cursor position with arrow keys).
Example scenario: user has edited some part of the text in the multiline entry and wants to edit something below. User scrolls to see the location of the next planned edit. However, before clicking there, user does something with other widgets in the window (imagine clicking on the save icon, for example). At this moment, the entry scrolls back to the location of the first edit, to user's annoyance.
I have provided a toy example of this scenario below. However, the same issue can appear in other scenarios. For example, the scrolling to the cursor position also happens when the entry gains focus. This leads to very inconvenient behavior when user scrolls a currently unfocused multiline entry to find the place for a next edit, then clicks there expecting the cursor to jump to that position, but at this moment the entry scrolls back to the previous cursor position and the click lands at an unexpected position.
How to reproduce
- Run the code fragment below. If necessary, shrink the window so that the text in the multiline entry on the right can be scrolled.
- Put cursor somewhere in the first paragraph in the multiline entry.
- Without changing the cursor position, scroll the text downwards (via mouse wheel or equivalent). It's easiest to see the bug effect when scrolled to the very end.
- Click on the left-side entry.
- Observe that the multiline entry has jumped back to the cursor position.
Screenshots
fyne-focus-bug-2025-12-03_16.07.15.webm
Example code
(This example is AI generated, but works OK)
package main
import (
"fyne.io/fyne/v2"
"fyne.io/fyne/v2/app"
"fyne.io/fyne/v2/container"
"fyne.io/fyne/v2/widget"
)
func main() {
myApp := app.New()
myWindow := myApp.NewWindow("Fyne Bug Demo")
myWindow.Resize(fyne.NewSize(800, 600))
// Left side: single line text entry
leftEntry := widget.NewEntry()
leftEntry.SetPlaceHolder("Enter text here...")
// Right side: multiline text entry
rightEntry := widget.NewMultiLineEntry()
rightEntry.SetText(generateFixedText())
rightEntry.Wrapping = fyne.TextWrapWord
// Create horizontal split container
split := container.NewHSplit(
leftEntry,
rightEntry,
)
split.Offset = 0.3 // 30% left, 70% right
myWindow.SetContent(split)
myWindow.ShowAndRun()
}
func rotText(s string, n int) string {
runes := []rune(s)
for i, r := range runes {
switch {
case r >= 'A' && r <= 'Z':
runes[i] = 'A' + (r-'A'+rune(n))%26
case r >= 'a' && r <= 'z':
runes[i] = 'a' + (r-'a'+rune(n))%26
}
}
return string(runes)
}
func generateFixedText() string {
fragment := "Lorem ipsum dolor sit amet, consectetur adipiscing elit. Sed do eiusmod tempor incididunt ut labore et dolore magna aliqua.\n" +
"Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris nisi ut aliquip ex ea commodo consequat. Duis aute irure " +
"dolor in reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla pariatur.\n\n"
text := fragment
text += rotText(fragment, 4)
text += rotText(fragment, 8)
text += rotText(fragment, 12)
text += rotText(fragment, 16)
return text
}Fyne version
2.7.1
Go compiler version
1.19
Operating system and version
Pop OS 22.04
Additional Information
No response