adam-gui/vendor/fyne.io/fyne/v2/layout/boxlayout.go
2024-04-29 19:13:50 +02:00

149 lines
3.6 KiB
Go

package layout
import (
"fyne.io/fyne/v2"
"fyne.io/fyne/v2/theme"
)
// Declare conformity with Layout interface
var _ fyne.Layout = (*boxLayout)(nil)
type boxLayout struct {
horizontal bool
}
// NewHBoxLayout returns a horizontal box layout for stacking a number of child
// canvas objects or widgets left to right. The objects are always displayed
// at their horizontal MinSize. Use a different layout if the objects are intended
// to be larger then their horizontal MinSize.
func NewHBoxLayout() fyne.Layout {
return &boxLayout{true}
}
// NewVBoxLayout returns a vertical box layout for stacking a number of child
// canvas objects or widgets top to bottom. The objects are always displayed
// at their vertical MinSize. Use a different layout if the objects are intended
// to be larger then their vertical MinSize.
func NewVBoxLayout() fyne.Layout {
return &boxLayout{false}
}
func (g *boxLayout) isSpacer(obj fyne.CanvasObject) bool {
if !obj.Visible() {
return false // invisible spacers don't impact layout
}
spacer, ok := obj.(SpacerObject)
if !ok {
return false
}
if g.horizontal {
return spacer.ExpandHorizontal()
}
return spacer.ExpandVertical()
}
// Layout is called to pack all child objects into a specified size.
// For a VBoxLayout this will pack objects into a single column where each item
// is full width but the height is the minimum required.
// Any spacers added will pad the view, sharing the space if there are two or more.
func (g *boxLayout) Layout(objects []fyne.CanvasObject, size fyne.Size) {
spacers := 0
visibleObjects := 0
// Size taken up by visible objects
total := float32(0)
for _, child := range objects {
if !child.Visible() {
continue
}
if g.isSpacer(child) {
spacers++
continue
}
visibleObjects++
if g.horizontal {
total += child.MinSize().Width
} else {
total += child.MinSize().Height
}
}
padding := theme.Padding()
// Amount of space not taken up by visible objects and inter-object padding
var extra float32
if g.horizontal {
extra = size.Width - total - (padding * float32(visibleObjects-1))
} else {
extra = size.Height - total - (padding * float32(visibleObjects-1))
}
// Spacers split extra space equally
spacerSize := float32(0)
if spacers > 0 {
spacerSize = extra / float32(spacers)
}
x, y := float32(0), float32(0)
for _, child := range objects {
if !child.Visible() {
continue
}
if g.isSpacer(child) {
if g.horizontal {
x += spacerSize
} else {
y += spacerSize
}
continue
}
child.Move(fyne.NewPos(x, y))
if g.horizontal {
width := child.MinSize().Width
x += padding + width
child.Resize(fyne.NewSize(width, size.Height))
} else {
height := child.MinSize().Height
y += padding + height
child.Resize(fyne.NewSize(size.Width, height))
}
}
}
// MinSize finds the smallest size that satisfies all the child objects.
// For a BoxLayout this is the width of the widest item and the height is
// the sum of of all children combined with padding between each.
func (g *boxLayout) MinSize(objects []fyne.CanvasObject) fyne.Size {
minSize := fyne.NewSize(0, 0)
addPadding := false
padding := theme.Padding()
for _, child := range objects {
if !child.Visible() || g.isSpacer(child) {
continue
}
childMin := child.MinSize()
if g.horizontal {
minSize.Height = fyne.Max(childMin.Height, minSize.Height)
minSize.Width += childMin.Width
if addPadding {
minSize.Width += padding
}
} else {
minSize.Width = fyne.Max(childMin.Width, minSize.Width)
minSize.Height += childMin.Height
if addPadding {
minSize.Height += padding
}
}
addPadding = true
}
return minSize
}