objects/slider.go

303 lines
7.6 KiB
Go
Raw Normal View History

2023-08-10 15:52:01 -06:00
package objects
import "math"
2023-08-11 23:02:48 -06:00
import "image"
2023-08-10 15:52:01 -06:00
import "git.tebibyte.media/tomo/tomo"
2023-08-11 23:02:48 -06:00
import "git.tebibyte.media/tomo/tomo/input"
import "git.tebibyte.media/tomo/tomo/event"
2023-08-10 15:52:01 -06:00
var _ tomo.Object = new(Slider)
2023-09-14 15:03:19 -06:00
// Slider is a control that selects a numeric value between 0 and 1.
//
// Sub-components:
// - SliderHandle is the grabbable handle of the slider.
//
// Tags:
// - [vertical] The slider is oriented vertically.
// - [horizontall] The slider is oriented horizontally.
//
// SliderHandle tags:
// - [vertical] The handle is oriented vertically.
// - [horizontall] The handle is oriented horizontally.
2023-08-10 15:52:01 -06:00
type Slider struct {
box tomo.ContainerBox
2024-08-16 16:35:19 -06:00
handle *sliderHandle
2023-08-11 23:02:48 -06:00
layout sliderLayout
dragging bool
dragOffset image.Point
step float64
2023-08-24 22:09:34 -06:00
2023-08-11 23:02:48 -06:00
on struct {
valueChange event.FuncBroadcaster
confirm event.FuncBroadcaster
2023-08-11 23:02:48 -06:00
}
2023-08-10 15:52:01 -06:00
}
2024-08-16 16:35:19 -06:00
type sliderHandle struct {
2023-08-10 15:52:01 -06:00
tomo.Box
}
2023-08-11 23:02:48 -06:00
func newSlider (orient string, value float64) *Slider {
slider := &Slider {
box: tomo.NewContainerBox(),
2024-08-16 16:35:19 -06:00
handle: &sliderHandle {
2023-08-10 15:52:01 -06:00
Box: tomo.NewBox(),
},
2023-08-11 23:02:48 -06:00
layout: sliderLayout {
vertical: orient == "vertical",
value: math.NaN(),
2023-08-11 23:02:48 -06:00
},
step: 0.05,
2023-08-10 15:52:01 -06:00
}
2023-08-24 22:09:34 -06:00
slider.handle.SetRole(tomo.R("objects", "SliderHandle"))
slider.handle.SetTag(orient, true)
slider.box.SetRole(tomo.R("objects", "Slider"))
slider.box.SetTag(orient, true)
2024-06-03 19:13:18 -06:00
slider.box.Add(slider.handle)
slider.box.SetFocusable(true)
slider.SetValue(value)
slider.box.SetInputMask(true)
slider.box.OnKeyUp(slider.handleKeyUp)
slider.box.OnKeyDown(slider.handleKeyDown)
slider.box.OnButtonDown(slider.handleButtonDown)
slider.box.OnButtonUp(slider.handleButtonUp)
slider.box.OnMouseMove(slider.handleMouseMove)
slider.box.OnScroll(slider.handleScroll)
return slider
2023-08-10 15:52:01 -06:00
}
2023-09-14 15:03:19 -06:00
// NewVerticalSlider creates a new vertical slider with the specified value.
2023-08-11 23:02:48 -06:00
func NewVerticalSlider (value float64) *Slider {
return newSlider("vertical", value)
}
2023-09-14 15:03:19 -06:00
// NewHorizontalSlider creates a new horizontal slider with the specified value.
2023-08-11 23:02:48 -06:00
func NewHorizontalSlider (value float64) *Slider {
return newSlider("horizontal", value)
}
// GetBox returns the underlying box.
func (this *Slider) GetBox () tomo.Box {
return this.box
}
// SetFocused sets whether or not this slider has keyboard focus. If set to
// true, this method will steal focus away from whichever object currently has
// focus.
func (this *Slider) SetFocused (focused bool) {
this.box.SetFocused(focused)
}
2023-09-14 15:03:19 -06:00
// SetValue sets the value of the slider between 0 and 1.
2023-08-11 23:02:48 -06:00
func (this *Slider) SetValue (value float64) {
if value < 0 { value = 0 }
if value > 1 { value = 1 }
if value == this.layout.value { return }
this.layout.value = value
this.box.SetAttr(tomo.ALayout(this.layout))
2023-08-11 23:02:48 -06:00
}
2023-09-14 15:03:19 -06:00
// Value returns the value of the slider between 0 and 1.
2023-08-11 23:02:48 -06:00
func (this *Slider) Value () float64 {
return this.layout.value
2023-08-10 15:52:01 -06:00
}
// OnValueChange specifies a function to be called when the user moves the
// slider.
func (this *Slider) OnValueChange (callback func ()) event.Cookie {
return this.on.valueChange.Connect(callback)
2023-08-11 23:02:48 -06:00
}
// OnConfirm specifies a function to be called when the user stops moving the
2024-06-25 00:33:13 -06:00
// slider.
func (this *Slider) OnConfirm (callback func ()) event.Cookie {
return this.on.confirm.Connect(callback)
2024-06-25 00:33:13 -06:00
}
2024-07-25 10:58:38 -06:00
func (this *Slider) handleKeyDown (key input.Key, numpad bool) bool {
2023-08-24 22:09:34 -06:00
var increment float64; if this.layout.vertical {
increment = -0.05
} else {
increment = 0.05
}
2023-08-11 23:02:48 -06:00
switch key {
case input.KeyUp, input.KeyLeft:
2024-09-12 00:34:28 -06:00
if this.box.Window().Modifiers().Alt() {
2023-08-11 23:02:48 -06:00
this.SetValue(0)
} else {
2023-08-24 22:09:34 -06:00
this.SetValue(this.Value() - increment)
2023-08-11 23:02:48 -06:00
}
this.on.valueChange.Broadcast()
2024-07-25 10:58:38 -06:00
return true
2023-08-11 23:02:48 -06:00
case input.KeyDown, input.KeyRight:
2024-09-12 00:34:28 -06:00
if this.box.Window().Modifiers().Alt() {
2023-08-11 23:02:48 -06:00
this.SetValue(1)
} else {
2023-08-24 22:09:34 -06:00
this.SetValue(this.Value() + increment)
2023-08-11 23:02:48 -06:00
}
this.on.valueChange.Broadcast()
2024-07-25 10:58:38 -06:00
return true
2023-08-11 23:02:48 -06:00
case input.KeyHome:
this.SetValue(0)
this.on.valueChange.Broadcast()
2024-07-25 10:58:38 -06:00
return true
2023-08-11 23:02:48 -06:00
case input.KeyEnd:
this.SetValue(1)
this.on.valueChange.Broadcast()
2024-07-25 10:58:38 -06:00
return true
2023-08-11 23:02:48 -06:00
}
2024-07-25 10:58:38 -06:00
return false
2023-08-11 23:02:48 -06:00
}
2024-07-25 10:58:38 -06:00
func (this *Slider) handleKeyUp (key input.Key, numpad bool) bool {
switch key {
case input.KeyUp, input.KeyLeft: return true
case input.KeyDown, input.KeyRight: return true
case input.KeyHome: return true
case input.KeyEnd: return true
}
return false
2024-07-21 09:48:28 -06:00
}
2024-07-25 10:58:38 -06:00
func (this *Slider) handleButtonDown (button input.Button) bool {
pointer := this.box.Window().MousePosition()
2023-08-11 23:02:48 -06:00
handle := this.handle.Bounds()
2023-09-14 19:40:06 -06:00
within := pointer.In(handle)
var above bool; if this.layout.vertical {
above = pointer.Y < handle.Min.Y + handle.Dy() / 2
2023-08-11 23:02:48 -06:00
} else {
2023-09-14 19:40:06 -06:00
above = pointer.X < handle.Min.X + handle.Dx() / 2
2023-08-11 23:02:48 -06:00
}
2023-08-24 22:09:34 -06:00
2023-08-11 23:02:48 -06:00
switch button {
case input.ButtonLeft:
if within {
this.dragging = true
this.dragOffset =
pointer.Sub(this.handle.Bounds().Min).
Add(this.box.InnerBounds().Min)
2023-08-11 23:02:48 -06:00
this.drag()
} else {
this.dragOffset = this.fallbackDragOffset()
this.dragging = true
this.drag()
}
case input.ButtonMiddle:
if above {
this.SetValue(0)
this.on.valueChange.Broadcast()
this.on.confirm.Broadcast()
2023-08-11 23:02:48 -06:00
} else {
this.SetValue(1)
this.on.valueChange.Broadcast()
this.on.confirm.Broadcast()
2023-08-11 23:02:48 -06:00
}
case input.ButtonRight:
if above {
this.SetValue(this.Value() - this.step)
this.on.valueChange.Broadcast()
this.on.confirm.Broadcast()
2023-08-11 23:02:48 -06:00
} else {
this.SetValue(this.Value() + this.step)
this.on.valueChange.Broadcast()
this.on.confirm.Broadcast()
2023-08-11 23:02:48 -06:00
}
}
2024-07-25 10:58:38 -06:00
return true
2023-08-11 23:02:48 -06:00
}
2024-07-25 10:58:38 -06:00
func (this *Slider) handleButtonUp (button input.Button) bool {
if button != input.ButtonLeft || !this.dragging { return true }
2023-08-11 23:02:48 -06:00
this.dragging = false
this.on.confirm.Broadcast()
2024-07-25 10:58:38 -06:00
return true
2023-08-11 23:02:48 -06:00
}
2024-07-25 10:58:38 -06:00
func (this *Slider) handleMouseMove () bool {
if !this.dragging { return false }
2023-08-11 23:02:48 -06:00
this.drag()
2024-07-25 10:58:38 -06:00
return true
2023-08-11 23:02:48 -06:00
}
2024-07-25 10:58:38 -06:00
func (this *Slider) handleScroll (x, y float64) bool {
delta := (x + y) * 0.005
this.SetValue(this.Value() + delta)
this.on.valueChange.Broadcast()
this.on.confirm.Broadcast()
2024-07-25 10:58:38 -06:00
return true
}
2023-08-11 23:02:48 -06:00
func (this *Slider) drag () {
pointer := this.box.Window().MousePosition().Sub(this.dragOffset)
gutter := this.box.InnerBounds()
2023-08-11 23:02:48 -06:00
handle := this.handle.Bounds()
if this.layout.vertical {
this.SetValue (
2023-08-24 22:09:34 -06:00
1 -
2023-08-11 23:02:48 -06:00
float64(pointer.Y) /
float64(gutter.Dy() - handle.Dy()))
} else {
this.SetValue (
float64(pointer.X) /
float64(gutter.Dx() - handle.Dx()))
}
this.on.valueChange.Broadcast()
2023-08-11 23:02:48 -06:00
}
func (this *Slider) fallbackDragOffset () image.Point {
if this.layout.vertical {
return this.box.InnerBounds().Min.
2023-08-11 23:02:48 -06:00
Add(image.Pt(0, this.handle.Bounds().Dy() / 2))
} else {
return this.box.InnerBounds().Min.
2023-08-11 23:02:48 -06:00
Add(image.Pt(this.handle.Bounds().Dx() / 2, 0))
}
}
type sliderLayout struct {
vertical bool
value float64
}
2024-07-21 09:48:28 -06:00
func (sliderLayout) MinimumSize (hints tomo.LayoutHints, boxes tomo.BoxQuerier) image.Point {
if boxes.Len() != 1 { return image.Pt(0, 0) }
return boxes.MinimumSize(0)
2023-08-11 23:02:48 -06:00
}
2024-07-21 09:48:28 -06:00
func (this sliderLayout) Arrange (hints tomo.LayoutHints, boxes tomo.BoxArranger) {
if boxes.Len() != 1 { return }
handle := image.Rectangle { Max: boxes.MinimumSize(0) }
2023-08-11 23:02:48 -06:00
gutter := hints.Bounds
2023-08-24 22:09:34 -06:00
2023-08-11 23:02:48 -06:00
if this.vertical {
height := gutter.Dy() - handle.Dy()
2023-08-24 22:09:34 -06:00
offset := int(float64(height) * (1 - this.value))
2023-08-20 15:41:14 -06:00
handle.Max.X = gutter.Dx()
2024-07-21 09:48:28 -06:00
boxes.SetBounds (
0,
handle.Add(image.Pt(0, offset)).Add(gutter.Min))
2023-08-11 23:02:48 -06:00
} else {
width := gutter.Dx() - handle.Dx()
offset := int(float64(width) * this.value)
2023-08-20 15:41:14 -06:00
handle.Max.Y = gutter.Dy()
2024-07-21 09:48:28 -06:00
boxes.SetBounds (
0,
handle.Add(image.Pt(offset, 0)).Add(gutter.Min))
2023-08-11 23:02:48 -06:00
}
2023-08-10 15:52:01 -06:00
}
2024-06-11 15:17:11 -06:00
2024-07-21 09:48:28 -06:00
func (this sliderLayout) RecommendedHeight (hints tomo.LayoutHints, boxes tomo.BoxQuerier, width int) int {
return this.MinimumSize(hints, boxes).X
2024-06-11 15:17:11 -06:00
}
2024-07-21 09:48:28 -06:00
func (this sliderLayout) RecommendedWidth (hints tomo.LayoutHints, boxes tomo.BoxQuerier, height int) int {
return this.MinimumSize(hints, boxes).Y
2024-06-11 15:17:11 -06:00
}