Files
gio/widget/example_test.go
Pierre Curto 038ec2cc3f widget: fix handling of disabled context in Draggable
While here, add an example on how to use the Draggable
widget.

Signed-off-by: Pierre Curto <pierre.curto@gmail.com>
2021-12-10 06:25:43 +01:00

157 lines
3.8 KiB
Go

// SPDX-License-Identifier: Unlicense OR MIT
package widget_test
import (
"fmt"
"image"
"gioui.org/f32"
"gioui.org/io/pointer"
"gioui.org/io/router"
"gioui.org/io/transfer"
"gioui.org/layout"
"gioui.org/op"
"gioui.org/op/clip"
"gioui.org/widget"
)
func ExampleClickable_passthrough() {
// When laying out clickable widgets on top of each other,
// pointer events can be passed down for the underlying
// widgets to pick them up.
var button1, button2 widget.Clickable
var r router.Router
gtx := layout.Context{
Ops: new(op.Ops),
Constraints: layout.Exact(image.Pt(100, 100)),
Queue: &r,
}
// widget lays out two buttons on top of each other.
widget := func() {
content := func(gtx layout.Context) layout.Dimensions { return layout.Dimensions{Size: gtx.Constraints.Min} }
button1.Layout(gtx, content)
// button2 completely covers button1, but pass-through allows pointer
// events to pass through to button1.
defer pointer.PassOp{}.Push(gtx.Ops).Pop()
button2.Layout(gtx, content)
}
// The first layout and call to Frame declare the Clickable handlers
// to the input router, so the following pointer events are propagated.
widget()
r.Frame(gtx.Ops)
// Simulate one click on the buttons by sending a Press and Release event.
r.Queue(
pointer.Event{
Source: pointer.Mouse,
Buttons: pointer.ButtonPrimary,
Type: pointer.Press,
Position: f32.Pt(50, 50),
},
pointer.Event{
Source: pointer.Mouse,
Buttons: pointer.ButtonPrimary,
Type: pointer.Release,
Position: f32.Pt(50, 50),
},
)
// The second layout ensures that the click event is registered by the buttons.
widget()
if button1.Clicked() {
fmt.Println("button1 clicked!")
}
if button2.Clicked() {
fmt.Println("button2 clicked!")
}
// Output:
// button1 clicked!
// button2 clicked!
}
func ExampleDraggable_Layout() {
var r router.Router
gtx := layout.Context{
Ops: new(op.Ops),
Constraints: layout.Exact(image.Pt(100, 100)),
Queue: &r,
}
// mime is the type used to match drag and drop operations.
// It could be left empty in this example.
mime := "MyMime"
drag := &widget.Draggable{Type: mime}
var drop int
// widget lays out the drag and drop handlers and processes
// the transfer events.
widget := func() {
// Setup the draggable widget.
w := func(gtx layout.Context) layout.Dimensions {
sz := image.Pt(10, 10) // drag area
return layout.Dimensions{Size: sz}
}
drag.Layout(gtx, w, w)
// drag must respond with an Offer event when requested.
// Use the drag method for this.
if m, ok := drag.Requested(); ok {
drag.Offer(gtx.Ops, m, offer{Data: "hello world"})
}
// Setup the area for drops.
ds := clip.Rect{
Min: image.Pt(20, 20),
Max: image.Pt(40, 40),
}.Push(gtx.Ops)
transfer.TargetOp{
Tag: &drop,
Type: mime, // this must match the drag Type for the drop to succeed
}.Add(gtx.Ops)
ds.Pop()
// Check for the received data.
for _, ev := range gtx.Events(&drop) {
switch e := ev.(type) {
case transfer.DataEvent:
data := e.Open()
fmt.Println(data.(offer).Data)
}
}
}
// Register and lay out the widget.
widget()
r.Frame(gtx.Ops)
// Send drag and drop gesture events.
r.Queue(
pointer.Event{
Type: pointer.Press,
Position: f32.Pt(5, 5), // in the drag area
},
pointer.Event{
Type: pointer.Move,
Position: f32.Pt(5, 5), // in the drop area
},
pointer.Event{
Type: pointer.Release,
Position: f32.Pt(30, 30), // in the drop area
},
)
// Let the widget process the events.
widget()
r.Frame(gtx.Ops)
// Process the transfer.DataEvent.
widget()
// Output:
// hello world
}
type offer struct {
Data string
}
func (offer) Read([]byte) (int, error) { return 0, nil }
func (offer) Close() error { return nil }