diff --git a/internal/widget/base.go b/internal/widget/base.go index 36d2c9aaef..236f35e045 100644 --- a/internal/widget/base.go +++ b/internal/widget/base.go @@ -14,6 +14,7 @@ type Base struct { hidden atomic.Bool position async.Position size async.Size + minCache async.Size impl atomic.Pointer[fyne.Widget] } @@ -63,6 +64,11 @@ func (w *Base) Move(pos fyne.Position) { // MinSize for the widget - it should never be resized below this value. func (w *Base) MinSize() fyne.Size { + minCache := w.minCache.Load() + if !minCache.IsZero() { + return minCache + } + impl := w.super() r := cache.Renderer(impl) @@ -70,7 +76,9 @@ func (w *Base) MinSize() fyne.Size { return fyne.NewSize(0, 0) } - return r.MinSize() + min := r.MinSize() + w.minCache.Store(min) + return min } // Visible returns whether or not this widget should be visible. @@ -112,6 +120,8 @@ func (w *Base) Refresh() { return } + w.minCache.Store(fyne.Size{}) + render := cache.Renderer(impl) render.Refresh() } diff --git a/widget/activity.go b/widget/activity.go index 07b62fb431..1bfa3eb584 100644 --- a/widget/activity.go +++ b/widget/activity.go @@ -34,8 +34,7 @@ func NewActivity() *Activity { func (a *Activity) MinSize() fyne.Size { a.ExtendBaseWidget(a) - - return fyne.NewSquareSize(a.Theme().Size(theme.SizeNameInlineIcon)) + return a.BaseWidget.MinSize() } // Start the activity indicator animation diff --git a/widget/check.go b/widget/check.go index b03bfeedae..96368f41b0 100644 --- a/widget/check.go +++ b/widget/check.go @@ -24,8 +24,6 @@ type Check struct { hovered bool binder basicBinder - - minSize fyne.Size // cached for hover/tap position calculations } // NewCheck creates a new check widget with the set label and change handler @@ -119,8 +117,9 @@ func (c *Check) MouseMoved(me *desktop.MouseEvent) { // only hovered if cached minSize has not been initialized (test code) // or the pointer is within the "active" area of the widget (its minSize) - c.hovered = c.minSize.IsZero() || - (me.Position.X <= c.minSize.Width && me.Position.Y <= c.minSize.Height) + minSize := c.MinSize() + c.hovered = minSize.IsZero() || + (me.Position.X <= minSize.Width && me.Position.Y <= minSize.Height) if oldHovered != c.hovered { c.Refresh() @@ -132,8 +131,9 @@ func (c *Check) Tapped(pe *fyne.PointEvent) { if c.Disabled() { return } - if !c.minSize.IsZero() && - (pe.Position.X > c.minSize.Width || pe.Position.Y > c.minSize.Height) { + minSize := c.MinSize() + if !minSize.IsZero() && + (pe.Position.X > minSize.Width || pe.Position.Y > minSize.Height) { // tapped outside the active area of the widget return } @@ -151,8 +151,7 @@ func (c *Check) Tapped(pe *fyne.PointEvent) { // MinSize returns the size that this widget should not shrink below func (c *Check) MinSize() fyne.Size { c.ExtendBaseWidget(c) - c.minSize = c.BaseWidget.MinSize() - return c.minSize + return c.BaseWidget.MinSize() } // CreateRenderer is a private method to Fyne which links this widget to its renderer diff --git a/widget/entry.go b/widget/entry.go index 1398341d2c..56a18cc922 100644 --- a/widget/entry.go +++ b/widget/entry.go @@ -95,7 +95,6 @@ type Entry struct { ActionItem fyne.CanvasObject `json:"-"` binder basicBinder conversionError error - minCache fyne.Size multiLineRows int // override global default number of visible lines // undoStack stores the data necessary for undo/redo functionality @@ -401,29 +400,8 @@ func (e *Entry) KeyUp(key *fyne.KeyEvent) { // // Implements: fyne.Widget func (e *Entry) MinSize() fyne.Size { - e.propertyLock.RLock() - cached := e.minCache - e.propertyLock.RUnlock() - if !cached.IsZero() { - return cached - } - e.ExtendBaseWidget(e) - - th := e.Theme() - iconSpace := th.Size(theme.SizeNameInlineIcon) + th.Size(theme.SizeNameLineSpacing) - min := e.BaseWidget.MinSize() - if e.ActionItem != nil { - min = min.Add(fyne.NewSize(iconSpace, 0)) - } - if e.Validator != nil { - min = min.Add(fyne.NewSize(iconSpace, 0)) - } - - e.propertyLock.Lock() - e.minCache = min - e.propertyLock.Unlock() - return min + return e.BaseWidget.MinSize() } // MouseDown called on mouse click, this triggers a mouse click which can move the cursor, @@ -488,14 +466,6 @@ func (e *Entry) Redo() { e.Refresh() } -func (e *Entry) Refresh() { - e.propertyLock.Lock() - e.minCache = fyne.Size{} - e.propertyLock.Unlock() - - e.BaseWidget.Refresh() -} - // SelectedText returns the text currently selected in this Entry. // If there is no selection it will return the empty string. func (e *Entry) SelectedText() string { @@ -1679,25 +1649,39 @@ func (r *entryRenderer) MinSize() fyne.Size { if rend := cache.Renderer(r.entry.content); rend != nil { rend.(*entryContentRenderer).updateScrollDirections() } - if r.scroll.Direction == widget.ScrollNone { - return r.entry.content.MinSize().Add(fyne.NewSize(0, r.entry.Theme().Size(theme.SizeNameInputBorder)*2)) - } - innerPadding := r.entry.Theme().Size(theme.SizeNameInnerPadding) - textSize := r.entry.Theme().Size(theme.SizeNameText) - charMin := r.entry.placeholderProvider().charMinSize(r.entry.Password, r.entry.TextStyle, textSize) - minSize := charMin.Add(fyne.NewSquareSize(innerPadding)) + th := r.entry.Theme() + minSize := fyne.Size{} + + if r.scroll.Direction == widget.ScrollNone { + minSize = r.entry.content.MinSize().AddWidthHeight(0, th.Size(theme.SizeNameInputBorder)*2) + } else { + innerPadding := th.Size(theme.SizeNameInnerPadding) + textSize := th.Size(theme.SizeNameText) + charMin := r.entry.placeholderProvider().charMinSize(r.entry.Password, r.entry.TextStyle, textSize) + minSize = charMin.Add(fyne.NewSquareSize(innerPadding)) + + if r.entry.MultiLine { + count := r.entry.multiLineRows + if count <= 0 { + count = multiLineRows + } - if r.entry.MultiLine { - count := r.entry.multiLineRows - if count <= 0 { - count = multiLineRows + minSize.Height = charMin.Height*float32(count) + innerPadding } - minSize.Height = charMin.Height*float32(count) + innerPadding + minSize = minSize.AddWidthHeight(innerPadding*2, innerPadding) + } + + iconSpace := th.Size(theme.SizeNameInlineIcon) + th.Size(theme.SizeNameLineSpacing) + if r.entry.ActionItem != nil { + minSize.Width += iconSpace + } + if r.entry.Validator != nil { + minSize.Width += iconSpace } - return minSize.Add(fyne.NewSize(innerPadding*2, innerPadding)) + return minSize } func (r *entryRenderer) Objects() []fyne.CanvasObject { diff --git a/widget/entry_test.go b/widget/entry_test.go index 9a00103ce5..b2bab74d27 100644 --- a/widget/entry_test.go +++ b/widget/entry_test.go @@ -454,7 +454,7 @@ func TestEntry_MinSize(t *testing.T) { min = entry.MinSize() entry.ActionItem = canvas.NewCircle(color.Black) entry.Refresh() - assert.Equal(t, min.Add(fyne.NewSize(theme.IconInlineSize()+theme.Padding(), 0)), entry.MinSize()) + assert.Equal(t, min.Add(fyne.NewSize(theme.IconInlineSize()+theme.LineSpacing(), 0)), entry.MinSize()) } func TestEntryMultiline_MinSize(t *testing.T) { diff --git a/widget/gridwrap.go b/widget/gridwrap.go index 6be54a6a92..b6245396fc 100644 --- a/widget/gridwrap.go +++ b/widget/gridwrap.go @@ -112,7 +112,6 @@ func (l *GridWrap) FocusLost() { // MinSize returns the size that this widget should not shrink below. func (l *GridWrap) MinSize() fyne.Size { l.ExtendBaseWidget(l) - return l.BaseWidget.MinSize() } diff --git a/widget/hyperlink.go b/widget/hyperlink.go index e07f3156e9..91d00a2fd4 100644 --- a/widget/hyperlink.go +++ b/widget/hyperlink.go @@ -163,11 +163,8 @@ func (hl *Hyperlink) Refresh() { // MinSize returns the smallest size this widget can shrink to func (hl *Hyperlink) MinSize() fyne.Size { - if len(hl.provider.Segments) == 0 { - hl.syncSegments() - } - - return hl.provider.MinSize() + hl.ExtendBaseWidget(hl) + return hl.BaseWidget.MinSize() } // Resize sets a new size for the hyperlink. @@ -191,6 +188,7 @@ func (hl *Hyperlink) SetText(text string) { return // Not initialized yet. } hl.syncSegments() + hl.ResetMinSizeCache() hl.provider.Refresh() } diff --git a/widget/label.go b/widget/label.go index e3bab90a57..a587137d36 100644 --- a/widget/label.go +++ b/widget/label.go @@ -3,7 +3,6 @@ package widget import ( "fyne.io/fyne/v2" "fyne.io/fyne/v2/data/binding" - "fyne.io/fyne/v2/internal/cache" "fyne.io/fyne/v2/theme" ) @@ -77,12 +76,8 @@ func (l *Label) CreateRenderer() fyne.WidgetRenderer { // // Implements: fyne.Widget func (l *Label) MinSize() fyne.Size { - if l.provider == nil { - l.ExtendBaseWidget(l) - cache.Renderer(l.super()) - } - - return l.provider.MinSize() + l.ExtendBaseWidget(l) + return l.BaseWidget.MinSize() } // Refresh triggers a redraw of the label. diff --git a/widget/list.go b/widget/list.go index 53bca499b1..dab4f25b91 100644 --- a/widget/list.go +++ b/widget/list.go @@ -110,7 +110,6 @@ func (l *List) FocusLost() { // MinSize returns the size that this widget should not shrink below. func (l *List) MinSize() fyne.Size { l.ExtendBaseWidget(l) - return l.BaseWidget.MinSize() } diff --git a/widget/richtext.go b/widget/richtext.go index 007b8e3c67..d237c0ec56 100644 --- a/widget/richtext.go +++ b/widget/richtext.go @@ -45,7 +45,6 @@ type RichText struct { visualCache map[RichTextSegment][]fyne.CanvasObject cacheLock sync.Mutex - minCache fyne.Size } // NewRichText returns a new RichText widget that renders the given text and segments. @@ -89,19 +88,14 @@ func (t *RichText) CreateRenderer() fyne.WidgetRenderer { // MinSize calculates the minimum size of a rich text widget. // This is based on the contained text with a standard amount of padding added. func (t *RichText) MinSize() fyne.Size { - // we don't return the minCache here, as any internal segments could have caused it to change... t.ExtendBaseWidget(t) - - min := t.BaseWidget.MinSize() - t.minCache = min - return min + return t.BaseWidget.MinSize() } // Refresh triggers a redraw of the rich text. // // Implements: fyne.Widget func (t *RichText) Refresh() { - t.minCache = fyne.Size{} t.updateRowBounds() for _, s := range t.Segments { @@ -123,10 +117,11 @@ func (t *RichText) Resize(size fyne.Size) { } t.size.Store(size) + minSize := t.MinSize() t.propertyLock.RLock() segments := t.Segments - skipResize := !t.minCache.IsZero() && size.Width >= t.minCache.Width && size.Height >= t.minCache.Height && t.Wrapping == fyne.TextWrapOff && t.Truncation == fyne.TextTruncateOff + skipResize := !minSize.IsZero() && size.Width >= minSize.Width && size.Height >= minSize.Height && t.Wrapping == fyne.TextWrapOff && t.Truncation == fyne.TextTruncateOff t.propertyLock.RUnlock() if skipResize { diff --git a/widget/select_entry.go b/widget/select_entry.go index 806d17c8c0..d6161252f9 100644 --- a/widget/select_entry.go +++ b/widget/select_entry.go @@ -55,7 +55,7 @@ func (e *SelectEntry) Disable() { // Implements: fyne.Widget func (e *SelectEntry) MinSize() fyne.Size { e.ExtendBaseWidget(e) - return e.Entry.MinSize() + return e.BaseWidget.MinSize() } // Move changes the relative position of the select entry. diff --git a/widget/separator.go b/widget/separator.go index 8f61de80ef..30d9433156 100644 --- a/widget/separator.go +++ b/widget/separator.go @@ -55,8 +55,7 @@ func (s *Separator) CreateRenderer() fyne.WidgetRenderer { // Implements: fyne.Widget func (s *Separator) MinSize() fyne.Size { s.ExtendBaseWidget(s) - t := s.Theme().Size(theme.SizeNameSeparatorThickness) - return fyne.NewSize(t, t) + return s.BaseWidget.MinSize() } var _ fyne.WidgetRenderer = (*separatorRenderer)(nil) @@ -68,8 +67,7 @@ type separatorRenderer struct { } func (r *separatorRenderer) MinSize() fyne.Size { - t := r.d.Theme().Size(theme.SizeNameSeparatorThickness) - return fyne.NewSize(t, t) + return fyne.NewSquareSize(r.d.Theme().Size(theme.SizeNameSeparatorThickness)) } func (r *separatorRenderer) Refresh() { diff --git a/widget/widget.go b/widget/widget.go index c5335485d8..43e7a21a7b 100644 --- a/widget/widget.go +++ b/widget/widget.go @@ -16,6 +16,7 @@ import ( // BaseWidget provides a helper that handles basic widget behaviours. type BaseWidget struct { size async.Size + minCache async.Size position async.Position Hidden bool @@ -69,6 +70,11 @@ func (w *BaseWidget) Move(pos fyne.Position) { // MinSize for the widget - it should never be resized below this value. func (w *BaseWidget) MinSize() fyne.Size { + minCache := w.minCache.Load() + if !minCache.IsZero() { + return minCache + } + impl := w.super() r := cache.Renderer(impl) @@ -76,7 +82,9 @@ func (w *BaseWidget) MinSize() fyne.Size { return fyne.Size{} } - return r.MinSize() + min := r.MinSize() + w.minCache.Store(min) + return min } // Visible returns whether or not this widget should be visible. @@ -123,6 +131,8 @@ func (w *BaseWidget) Refresh() { return } + w.minCache.Store(fyne.Size{}) + w.propertyLock.Lock() w.themeCache = nil w.propertyLock.Unlock() @@ -131,6 +141,11 @@ func (w *BaseWidget) Refresh() { render.Refresh() } +// ResetMinSizeCache resets the cached MinSize for this widget. +func (w *BaseWidget) ResetMinSizeCache() { + w.minCache.Store(fyne.Size{}) +} + // Theme returns a cached Theme instance for this widget (or its extending widget). // This will be the app theme in most cases, or a widget specific theme if it is inside a ThemeOverride container. // @@ -171,6 +186,7 @@ func (w *BaseWidget) SetFieldsAndRefresh(f func()) { return } impl.Refresh() + w.ResetMinSizeCache() } // super will return the actual object that this represents.