forked from joejulian/gio
f5985b5e7d
Add os.Exit to examples now that app.Main never returns. Signed-off-by: Elias Naur <mail@eliasnaur.com>
157 lines
3.1 KiB
Go
157 lines
3.1 KiB
Go
// SPDX-License-Identifier: Unlicense OR MIT
|
|
|
|
package main
|
|
|
|
import (
|
|
"fmt"
|
|
"image"
|
|
"image/color"
|
|
"log"
|
|
"math"
|
|
"os"
|
|
|
|
"gioui.org/app"
|
|
"gioui.org/f32"
|
|
"gioui.org/io/system"
|
|
"gioui.org/layout"
|
|
"gioui.org/op"
|
|
"gioui.org/op/paint"
|
|
"gioui.org/unit"
|
|
"gioui.org/widget"
|
|
"gioui.org/widget/material"
|
|
|
|
"gioui.org/font/gofont"
|
|
)
|
|
|
|
func main() {
|
|
go func() {
|
|
defer os.Exit(0)
|
|
w := app.NewWindow()
|
|
if err := loop(w); err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
}()
|
|
app.Main()
|
|
}
|
|
|
|
func loop(w *app.Window) error {
|
|
th := material.NewTheme(gofont.Collection())
|
|
var ops op.Ops
|
|
for {
|
|
e := <-w.Events()
|
|
switch e := e.(type) {
|
|
case system.DestroyEvent:
|
|
return e.Err
|
|
case system.FrameEvent:
|
|
gtx := layout.NewContext(&ops, e)
|
|
drawTabs(gtx, th)
|
|
e.Frame(gtx.Ops)
|
|
}
|
|
}
|
|
}
|
|
|
|
var tabs Tabs
|
|
var slider Slider
|
|
|
|
type Tabs struct {
|
|
list layout.List
|
|
tabs []Tab
|
|
selected int
|
|
}
|
|
|
|
type Tab struct {
|
|
btn widget.Clickable
|
|
Title string
|
|
}
|
|
|
|
func init() {
|
|
for i := 1; i <= 100; i++ {
|
|
tabs.tabs = append(tabs.tabs,
|
|
Tab{Title: fmt.Sprintf("Tab %d", i)},
|
|
)
|
|
}
|
|
}
|
|
|
|
type (
|
|
C = layout.Context
|
|
D = layout.Dimensions
|
|
)
|
|
|
|
func drawTabs(gtx layout.Context, th *material.Theme) layout.Dimensions {
|
|
return layout.Flex{Axis: layout.Vertical}.Layout(gtx,
|
|
layout.Rigid(func(gtx C) D {
|
|
return tabs.list.Layout(gtx, len(tabs.tabs), func(gtx C, tabIdx int) D {
|
|
t := &tabs.tabs[tabIdx]
|
|
if t.btn.Clicked() {
|
|
if tabs.selected < tabIdx {
|
|
slider.PushLeft()
|
|
} else if tabs.selected > tabIdx {
|
|
slider.PushRight()
|
|
}
|
|
tabs.selected = tabIdx
|
|
}
|
|
var tabWidth int
|
|
return layout.Stack{Alignment: layout.S}.Layout(gtx,
|
|
layout.Stacked(func(gtx C) D {
|
|
dims := material.Clickable(gtx, &t.btn, func(gtx C) D {
|
|
return layout.UniformInset(unit.Sp(12)).Layout(gtx,
|
|
material.H6(th, t.Title).Layout,
|
|
)
|
|
})
|
|
tabWidth = dims.Size.X
|
|
return dims
|
|
}),
|
|
layout.Stacked(func(gtx C) D {
|
|
if tabs.selected != tabIdx {
|
|
return layout.Dimensions{}
|
|
}
|
|
paint.ColorOp{Color: th.Color.Primary}.Add(gtx.Ops)
|
|
tabHeight := gtx.Px(unit.Dp(4))
|
|
paint.PaintOp{Rect: f32.Rectangle{
|
|
Max: f32.Point{
|
|
X: float32(tabWidth),
|
|
Y: float32(tabHeight),
|
|
},
|
|
}}.Add(gtx.Ops)
|
|
return layout.Dimensions{
|
|
Size: image.Point{X: tabWidth, Y: tabHeight},
|
|
}
|
|
}),
|
|
)
|
|
})
|
|
}),
|
|
layout.Flexed(1, func(gtx C) D {
|
|
return slider.Layout(gtx, func(gtx C) D {
|
|
fill(gtx, dynamicColor(tabs.selected))
|
|
return layout.Center.Layout(gtx,
|
|
material.H1(th, fmt.Sprintf("Tab content #%d", tabs.selected+1)).Layout,
|
|
)
|
|
})
|
|
}),
|
|
)
|
|
}
|
|
|
|
func bounds(gtx layout.Context) f32.Rectangle {
|
|
cs := gtx.Constraints
|
|
d := cs.Min
|
|
return f32.Rectangle{
|
|
Max: f32.Point{X: float32(d.X), Y: float32(d.Y)},
|
|
}
|
|
}
|
|
|
|
func fill(gtx layout.Context, col color.RGBA) {
|
|
dr := bounds(gtx)
|
|
paint.ColorOp{Color: col}.Add(gtx.Ops)
|
|
paint.PaintOp{Rect: dr}.Add(gtx.Ops)
|
|
}
|
|
|
|
func dynamicColor(i int) color.RGBA {
|
|
sn, cs := math.Sincos(float64(i) * math.Phi)
|
|
return color.RGBA{
|
|
R: 0xA0 + byte(0x30*sn),
|
|
G: 0xA0 + byte(0x30*cs),
|
|
B: 0xD0,
|
|
A: 0xFF,
|
|
}
|
|
}
|