2024-06-11 22:19:12 -06:00
|
|
|
// Package style provides a data-driven style implementation.
|
|
|
|
package style
|
2024-05-03 13:25:34 -06:00
|
|
|
|
|
|
|
import "image"
|
|
|
|
import "image/color"
|
|
|
|
import "git.tebibyte.media/tomo/tomo"
|
|
|
|
import "git.tebibyte.media/tomo/tomo/event"
|
|
|
|
import "git.tebibyte.media/tomo/tomo/input"
|
|
|
|
import "git.tebibyte.media/tomo/tomo/canvas"
|
|
|
|
|
2024-05-03 17:58:10 -06:00
|
|
|
// this is CSS's bastard child
|
|
|
|
|
2024-06-11 22:19:12 -06:00
|
|
|
// Style allows the use of data to define a visual style.
|
|
|
|
type Style struct {
|
2024-05-03 13:25:34 -06:00
|
|
|
// Textures maps texture names to image textures.
|
|
|
|
Textures map[string] image.Image
|
|
|
|
textures map[string] canvas.TextureCloser // private texture cache
|
|
|
|
missing canvas.TextureCloser // cache for "missing" texture
|
2024-05-03 17:58:10 -06:00
|
|
|
|
2024-05-03 13:25:34 -06:00
|
|
|
// Rules determines which styles get applied to which Objects.
|
2024-05-03 17:58:10 -06:00
|
|
|
Rules []Rule
|
2024-05-03 13:25:34 -06:00
|
|
|
|
2024-05-27 14:02:39 -06:00
|
|
|
// Colors maps tomo.Color values to color.RGBA values.
|
|
|
|
Colors map[tomo.Color] color.Color
|
2024-05-03 13:25:34 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
// Rule describes under what circumstances should certain style attributes be
|
|
|
|
// active.
|
|
|
|
type Rule struct {
|
2024-05-27 14:02:39 -06:00
|
|
|
Role tomo.Role
|
2024-05-03 17:58:10 -06:00
|
|
|
Default AttrSet
|
|
|
|
Hovered AttrSet
|
|
|
|
Pressed AttrSet
|
|
|
|
Focused AttrSet
|
|
|
|
}
|
|
|
|
|
|
|
|
// AttrSet is a set of attributes wherein only one/zero of each attribute type
|
|
|
|
// can exist. I deserve to be imprisoned for the way I made this work (look in
|
|
|
|
// attribute.go). Its zero value can be used safely, and you can copy it if you
|
|
|
|
// want, but it will point to the same set of attributes.
|
|
|
|
type AttrSet struct {
|
|
|
|
set map[int] Attr
|
|
|
|
}
|
|
|
|
|
|
|
|
// AS builds an AttrSet out of a vararg list of Attr values.
|
|
|
|
func AS (attrs ...Attr) AttrSet {
|
|
|
|
set := AttrSet { }
|
|
|
|
set.Add(attrs...)
|
|
|
|
return set
|
|
|
|
}
|
|
|
|
|
|
|
|
// Add adds attributes to the set.
|
|
|
|
func (this *AttrSet) Add (attrs ...Attr) {
|
|
|
|
this.ensure()
|
|
|
|
for _, attr := range attrs {
|
|
|
|
this.set[attr.attr()] = attr
|
|
|
|
}
|
2024-05-03 13:25:34 -06:00
|
|
|
}
|
|
|
|
|
2024-05-03 17:58:10 -06:00
|
|
|
// MergeUnder takes attributes from another set and adds them if they don't
|
|
|
|
// already exist in this one.
|
|
|
|
func (this *AttrSet) MergeUnder (other AttrSet) {
|
|
|
|
this.ensure()
|
|
|
|
if other.set == nil { return }
|
|
|
|
for _, attr := range other.set {
|
|
|
|
if _, exists := this.set[attr.attr()]; !exists {
|
|
|
|
this.Add(attr)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// MergeOver takes attributes from another set and adds them, overriding this
|
|
|
|
// one.
|
|
|
|
func (this *AttrSet) MergeOver (other AttrSet) {
|
|
|
|
this.ensure()
|
|
|
|
if other.set == nil { return }
|
|
|
|
for _, attr := range other.set {
|
|
|
|
this.Add(attr)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (this *AttrSet) ensure () {
|
|
|
|
if this.set == nil { this.set = make(map[int] Attr) }
|
|
|
|
}
|
|
|
|
|
2024-06-11 22:19:12 -06:00
|
|
|
func (this *Style) execute (object tomo.Object, set AttrSet) {
|
2024-05-03 13:25:34 -06:00
|
|
|
box := object.GetBox()
|
|
|
|
|
2024-05-03 17:58:10 -06:00
|
|
|
for _, attr := range set.set {
|
2024-05-03 13:25:34 -06:00
|
|
|
switch attr := attr.(type) {
|
|
|
|
case AttrColor:
|
|
|
|
box.SetColor(attr.Color)
|
|
|
|
case AttrTexture:
|
2024-05-27 14:02:39 -06:00
|
|
|
box.SetTextureTile(this.texture(string(attr)))
|
2024-05-03 13:25:34 -06:00
|
|
|
case AttrBorder:
|
|
|
|
box.SetBorder([]tomo.Border(attr)...)
|
|
|
|
case AttrMinimumSize:
|
|
|
|
box.SetMinimumSize(image.Point(attr))
|
|
|
|
case AttrPadding:
|
|
|
|
box.SetPadding(tomo.Inset(attr))
|
|
|
|
case AttrGap:
|
|
|
|
if box, ok := box.(tomo.ContainerBox); ok {
|
|
|
|
box.SetGap(image.Point(attr))
|
|
|
|
}
|
|
|
|
case AttrTextColor:
|
|
|
|
if box, ok := box.(tomo.TextBox); ok {
|
|
|
|
box.SetTextColor(attr.Color)
|
|
|
|
}
|
2024-05-03 17:58:10 -06:00
|
|
|
case AttrDotColor:
|
|
|
|
if box, ok := box.(tomo.TextBox); ok {
|
|
|
|
box.SetDotColor(attr.Color)
|
|
|
|
}
|
2024-05-03 13:25:34 -06:00
|
|
|
case AttrFace:
|
|
|
|
if box, ok := box.(tomo.TextBox); ok {
|
|
|
|
box.SetFace(attr)
|
|
|
|
}
|
2024-05-03 17:58:10 -06:00
|
|
|
case AttrAlign:
|
|
|
|
if box, ok := box.(tomo.ContentBox); ok {
|
|
|
|
box.SetAlign(attr.X, attr.Y)
|
|
|
|
}
|
|
|
|
default:
|
2024-05-27 14:02:39 -06:00
|
|
|
panic("bug: nasin/internal/tomo.Theme: unexpected attribute")
|
2024-05-03 13:25:34 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-06-11 22:19:12 -06:00
|
|
|
func (this *Style) texture (name string) canvas.Texture {
|
2024-05-03 13:25:34 -06:00
|
|
|
this.ensureTextureCache()
|
|
|
|
if texture, ok := this.textures[name]; ok {
|
|
|
|
return texture
|
|
|
|
}
|
|
|
|
if this.Textures == nil {
|
|
|
|
if source, ok := this.Textures[name]; ok {
|
|
|
|
texture := tomo.NewTexture(source)
|
|
|
|
this.textures[name] = texture
|
|
|
|
return texture
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return this.missingTexture()
|
|
|
|
}
|
|
|
|
|
2024-06-11 22:19:12 -06:00
|
|
|
func (this *Style) missingTexture () canvas.Texture {
|
2024-05-03 13:25:34 -06:00
|
|
|
if this.missing == nil {
|
|
|
|
this.missing = tomo.NewTexture(missingTexture(16))
|
|
|
|
}
|
|
|
|
return this.missing
|
|
|
|
}
|
|
|
|
|
2024-06-11 22:19:12 -06:00
|
|
|
func (this *Style) ensureTextureCache () {
|
2024-05-03 13:25:34 -06:00
|
|
|
if this.textures == nil { this.textures = make(map[string] canvas.TextureCloser) }
|
|
|
|
}
|
|
|
|
|
2024-05-03 17:58:10 -06:00
|
|
|
// setsFor builds flattened attr sets for a specific role based on the rules list
|
2024-06-11 22:19:12 -06:00
|
|
|
func (this *Style) setsFor (role tomo.Role) (defaul, hovered, pressed, focused AttrSet) {
|
2024-05-03 17:58:10 -06:00
|
|
|
for _, current := range this.Rules {
|
|
|
|
// check for a match
|
|
|
|
packageMatch := current.Role.Package == role.Package || current.Role.Package == ""
|
|
|
|
objectMatch := current.Role.Object == role.Object || current.Role.Object == ""
|
|
|
|
variantMatch := current.Role.Variant == role.Variant || current.Role.Variant == ""
|
|
|
|
if packageMatch && objectMatch && variantMatch {
|
|
|
|
// if found, merge and override
|
|
|
|
defaul.MergeOver(current.Default)
|
|
|
|
hovered.MergeOver(current.Hovered)
|
|
|
|
pressed.MergeOver(current.Pressed)
|
|
|
|
focused.MergeOver(current.Focused)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// hovered and pressed are mutually exclusive states, so we compress
|
|
|
|
// them with the default state.
|
|
|
|
hovered.MergeUnder(defaul)
|
|
|
|
pressed.MergeUnder(defaul)
|
|
|
|
return defaul, hovered, pressed, focused
|
|
|
|
}
|
|
|
|
|
2024-06-11 22:19:12 -06:00
|
|
|
func (this *Style) Apply (object tomo.Object) event.Cookie {
|
2024-05-03 13:25:34 -06:00
|
|
|
pressed := false
|
|
|
|
hovered := false
|
2024-06-03 19:59:28 -06:00
|
|
|
box := object.GetBox()
|
|
|
|
role := box.Role()
|
2024-05-03 13:25:34 -06:00
|
|
|
|
2024-05-03 17:58:10 -06:00
|
|
|
defaultSet, hoveredSet, pressedSet, focusedSet := this.setsFor(role)
|
2024-05-03 13:25:34 -06:00
|
|
|
|
|
|
|
updateStyle := func () {
|
|
|
|
if pressed {
|
2024-05-03 17:58:10 -06:00
|
|
|
this.execute(object, pressedSet)
|
|
|
|
} else if hovered {
|
|
|
|
this.execute(object, hoveredSet)
|
|
|
|
} else {
|
|
|
|
this.execute(object, defaultSet)
|
2024-05-03 13:25:34 -06:00
|
|
|
}
|
2024-05-03 17:58:10 -06:00
|
|
|
if box.Focused() && !pressed {
|
|
|
|
this.execute(object, focusedSet)
|
2024-05-03 13:25:34 -06:00
|
|
|
}
|
|
|
|
}
|
2024-05-03 14:23:21 -06:00
|
|
|
updateStyle()
|
2024-05-03 13:25:34 -06:00
|
|
|
|
|
|
|
return event.MultiCookie (
|
|
|
|
box.OnFocusEnter(updateStyle),
|
|
|
|
box.OnFocusLeave(updateStyle),
|
|
|
|
box.OnMouseDown(func (button input.Button) {
|
2024-05-03 17:58:10 -06:00
|
|
|
if button != input.ButtonLeft { return }
|
2024-05-03 13:25:34 -06:00
|
|
|
pressed = true
|
|
|
|
updateStyle()
|
|
|
|
}),
|
|
|
|
box.OnMouseUp(func (button input.Button) {
|
2024-05-03 17:58:10 -06:00
|
|
|
if button != input.ButtonLeft { return }
|
2024-05-03 13:25:34 -06:00
|
|
|
pressed = false
|
|
|
|
updateStyle()
|
|
|
|
}),
|
|
|
|
box.OnMouseEnter(func () {
|
|
|
|
hovered = true
|
|
|
|
updateStyle()
|
|
|
|
}),
|
|
|
|
box.OnMouseLeave(func () {
|
|
|
|
hovered = false
|
|
|
|
updateStyle()
|
|
|
|
}))
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2024-06-11 22:19:12 -06:00
|
|
|
func (this *Style) RGBA (c tomo.Color) (r, g, b, a uint32) {
|
2024-05-03 13:25:34 -06:00
|
|
|
if this.Colors == nil { return 0xFFFF, 0, 0xFFFF, 0xFFFF }
|
|
|
|
color, ok := this.Colors[c]
|
|
|
|
if !ok { return 0xFFFF, 0, 0xFFFF, 0xFFFF }
|
|
|
|
return color.RGBA()
|
|
|
|
}
|
|
|
|
|
2024-06-11 22:19:12 -06:00
|
|
|
// Close closes all cached textures this style has open. Do not call this while
|
|
|
|
// the style is in use.
|
|
|
|
func (this *Style) Close () error {
|
2024-05-03 13:25:34 -06:00
|
|
|
this.missing.Close()
|
|
|
|
this.missing = nil
|
|
|
|
for _, texture := range this.textures {
|
|
|
|
texture.Close()
|
|
|
|
}
|
|
|
|
this.textures = nil
|
|
|
|
return nil
|
|
|
|
}
|