Update internal system
This commit is contained in:
parent
e2b3b84993
commit
2af42a3568
@ -1,26 +1,48 @@
|
|||||||
package system
|
package system
|
||||||
|
|
||||||
import "git.tebibyte.media/tomo/tomo"
|
import "git.tebibyte.media/tomo/tomo"
|
||||||
|
import "git.tebibyte.media/tomo/backend/internal/util"
|
||||||
|
|
||||||
type attrHierarchy [T tomo.Attr] struct {
|
type attrHierarchy [T tomo.Attr] struct {
|
||||||
style T
|
fallback T
|
||||||
user T
|
style util.Optional[T]
|
||||||
userExists bool
|
user util.Optional[T]
|
||||||
|
}
|
||||||
|
|
||||||
|
func (this *attrHierarchy[T]) SetFallback (fallback T) {
|
||||||
|
this.fallback = fallback
|
||||||
}
|
}
|
||||||
|
|
||||||
func (this *attrHierarchy[T]) SetStyle (style T) (different bool) {
|
func (this *attrHierarchy[T]) SetStyle (style T) (different bool) {
|
||||||
styleEquals := this.style.Equals(style)
|
styleEquals := false
|
||||||
this.style = style
|
if previous, ok := this.style.Value(); ok {
|
||||||
return !styleEquals && !this.userExists
|
styleEquals = previous.Equals(style)
|
||||||
|
}
|
||||||
|
this.style.Set(style)
|
||||||
|
return !styleEquals && !this.user.Exists()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (this *attrHierarchy[T]) UnsetStyle () (different bool) {
|
||||||
|
different = this.style.Exists()
|
||||||
|
this.style.Unset()
|
||||||
|
return different
|
||||||
}
|
}
|
||||||
|
|
||||||
func (this *attrHierarchy[T]) SetUser (user T) (different bool) {
|
func (this *attrHierarchy[T]) SetUser (user T) (different bool) {
|
||||||
userEquals := this.user.Equals(user)
|
userEquals := false
|
||||||
this.user = user
|
if previous, ok := this.user.Value(); ok {
|
||||||
this.userExists = true
|
userEquals = previous.Equals(user)
|
||||||
|
}
|
||||||
|
this.user.Set(user)
|
||||||
return !userEquals
|
return !userEquals
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (this *attrHierarchy[T]) UnsetUser () (different bool) {
|
||||||
|
different = this.user.Exists()
|
||||||
|
this.user.Unset()
|
||||||
|
return different
|
||||||
|
}
|
||||||
|
|
||||||
func (this *attrHierarchy[T]) Set (attr T, user bool) (different bool) {
|
func (this *attrHierarchy[T]) Set (attr T, user bool) (different bool) {
|
||||||
if user {
|
if user {
|
||||||
return this.SetUser(attr)
|
return this.SetUser(attr)
|
||||||
@ -29,10 +51,20 @@ func (this *attrHierarchy[T]) Set (attr T, user bool) (different bool) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (this *attrHierarchy[T]) Value () T {
|
func (this *attrHierarchy[T]) Unset (user bool) (different bool) {
|
||||||
if this.userExists {
|
if user {
|
||||||
return this.user
|
return this.UnsetUser()
|
||||||
} else {
|
} else {
|
||||||
return this.style
|
return this.UnsetStyle()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (this *attrHierarchy[T]) Value () T {
|
||||||
|
if user, ok := this.user.Value(); ok {
|
||||||
|
return user
|
||||||
|
} else if style, ok := this.style.Value(); ok{
|
||||||
|
return style
|
||||||
|
} else {
|
||||||
|
return this.fallback
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -14,11 +14,11 @@ type box struct {
|
|||||||
parent parent
|
parent parent
|
||||||
outer anyBox
|
outer anyBox
|
||||||
|
|
||||||
tags util.Set[string]
|
tags util.Set[string]
|
||||||
role tomo.Role
|
role tomo.Role
|
||||||
lastStyleNonce int
|
lastStyleNonce int
|
||||||
lastIconsNonce int
|
lastIconSetNonce int
|
||||||
styleApplicator *styleApplicator
|
styleApplicator *styleApplicator
|
||||||
|
|
||||||
minSize util.Memo[image.Point]
|
minSize util.Memo[image.Point]
|
||||||
bounds image.Rectangle
|
bounds image.Rectangle
|
||||||
@ -41,7 +41,6 @@ type box struct {
|
|||||||
focused bool
|
focused bool
|
||||||
pressed bool
|
pressed bool
|
||||||
|
|
||||||
|
|
||||||
canvas util.Memo[canvas.Canvas]
|
canvas util.Memo[canvas.Canvas]
|
||||||
drawer canvas.Drawer
|
drawer canvas.Drawer
|
||||||
|
|
||||||
@ -71,6 +70,7 @@ func (this *System) newBox (outer anyBox) *box {
|
|||||||
drawer: outer,
|
drawer: outer,
|
||||||
tags: make(util.Set[string]),
|
tags: make(util.Set[string]),
|
||||||
}
|
}
|
||||||
|
box.attrColor.SetFallback(tomo.AColor(color.Transparent))
|
||||||
box.canvas = util.NewMemo (func () canvas.Canvas {
|
box.canvas = util.NewMemo (func () canvas.Canvas {
|
||||||
if box.parent == nil { return nil }
|
if box.parent == nil { return nil }
|
||||||
parentCanvas := box.parent.getCanvas()
|
parentCanvas := box.parent.getCanvas()
|
||||||
@ -141,6 +141,10 @@ func (this *box) SetAttr (attr tomo.Attr) {
|
|||||||
this.outer.setAttr(attr, true)
|
this.outer.setAttr(attr, true)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (this *box) UnsetAttr (kind tomo.AttrKind) {
|
||||||
|
this.outer.unsetAttr(kind, true)
|
||||||
|
}
|
||||||
|
|
||||||
func (this *box) SetDNDData (dat data.Data) {
|
func (this *box) SetDNDData (dat data.Data) {
|
||||||
this.dndData = dat
|
this.dndData = dat
|
||||||
}
|
}
|
||||||
@ -201,18 +205,7 @@ func (this *box) setAttr (attr tomo.Attr, user bool) {
|
|||||||
case tomo.AttrBorder:
|
case tomo.AttrBorder:
|
||||||
previousBorderSum := this.borderSum()
|
previousBorderSum := this.borderSum()
|
||||||
different := this.attrBorder.Set(attr, user)
|
different := this.attrBorder.Set(attr, user)
|
||||||
|
this.handleBorderChange(previousBorderSum, different)
|
||||||
// only invalidate the layout if the border is sized differently
|
|
||||||
if this.borderSum() != previousBorderSum {
|
|
||||||
this.invalidateLayout()
|
|
||||||
this.invalidateMinimum()
|
|
||||||
}
|
|
||||||
|
|
||||||
// if the border takes up the same amount of space, only invalidate the
|
|
||||||
// drawing if it looks different
|
|
||||||
if different {
|
|
||||||
this.invalidateDraw()
|
|
||||||
}
|
|
||||||
|
|
||||||
case tomo.AttrMinimumSize:
|
case tomo.AttrMinimumSize:
|
||||||
if this.attrMinimumSize.Set(attr, user) {
|
if this.attrMinimumSize.Set(attr, user) {
|
||||||
@ -220,7 +213,42 @@ func (this *box) setAttr (attr tomo.Attr, user bool) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
case tomo.AttrPadding:
|
case tomo.AttrPadding:
|
||||||
if this.attrPadding.Set(attr, true) {
|
if this.attrPadding.Set(attr, user) {
|
||||||
|
this.invalidateLayout()
|
||||||
|
this.invalidateMinimum()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (this *box) unsetAttr (kind tomo.AttrKind, user bool) {
|
||||||
|
switch kind {
|
||||||
|
case tomo.AttrKindColor:
|
||||||
|
if this.attrColor.Unset(user) {
|
||||||
|
this.invalidateDraw()
|
||||||
|
}
|
||||||
|
|
||||||
|
case tomo.AttrKindTexture:
|
||||||
|
if this.attrTexture.Unset(user) {
|
||||||
|
this.invalidateDraw()
|
||||||
|
}
|
||||||
|
|
||||||
|
case tomo.AttrKindTextureMode:
|
||||||
|
if this.attrTextureMode.Unset(user) {
|
||||||
|
this.invalidateDraw()
|
||||||
|
}
|
||||||
|
|
||||||
|
case tomo.AttrKindBorder:
|
||||||
|
previousBorderSum := this.borderSum()
|
||||||
|
different := this.attrBorder.Unset(user)
|
||||||
|
this.handleBorderChange(previousBorderSum, different)
|
||||||
|
|
||||||
|
case tomo.AttrKindMinimumSize:
|
||||||
|
if this.attrMinimumSize.Unset(user) {
|
||||||
|
this.invalidateMinimum()
|
||||||
|
}
|
||||||
|
|
||||||
|
case tomo.AttrKindPadding:
|
||||||
|
if this.attrPadding.Unset(user) {
|
||||||
this.invalidateLayout()
|
this.invalidateLayout()
|
||||||
this.invalidateMinimum()
|
this.invalidateMinimum()
|
||||||
}
|
}
|
||||||
@ -531,7 +559,21 @@ func (this *box) recursiveRedo () {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (this *box) recursiveLoseCanvas () {
|
func (this *box) recursiveLoseCanvas () {
|
||||||
this.canvas.InvalidateTo(nil)
|
this.canvas.Invalidate()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (this *box) handleBorderChange (previousBorderSum tomo.Inset, different bool) {
|
||||||
|
// only invalidate the layout if the border is sized differently
|
||||||
|
if this.borderSum() != previousBorderSum {
|
||||||
|
this.invalidateLayout()
|
||||||
|
this.invalidateMinimum()
|
||||||
|
}
|
||||||
|
|
||||||
|
// if the border takes up the same amount of space, only invalidate the
|
||||||
|
// drawing if it looks different
|
||||||
|
if different {
|
||||||
|
this.invalidateDraw()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (this *box) invalidateStyle () {
|
func (this *box) invalidateStyle () {
|
||||||
@ -578,9 +620,9 @@ func (this *box) recursiveReApply () {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// icons
|
// icons
|
||||||
hierarchyIconsNonce := this.getIconsNonce()
|
hierarchyIconSetNonce := this.getIconSetNonce()
|
||||||
if this.lastIconsNonce != hierarchyIconsNonce {
|
if this.lastIconSetNonce != hierarchyIconSetNonce {
|
||||||
this.lastIconsNonce = hierarchyIconsNonce
|
this.lastIconSetNonce = hierarchyIconSetNonce
|
||||||
this.on.iconSetChange.Broadcast()
|
this.on.iconSetChange.Broadcast()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -629,7 +671,7 @@ func (this *box) getStyleNonce () int {
|
|||||||
return this.getHierarchy().getStyleNonce()
|
return this.getHierarchy().getStyleNonce()
|
||||||
}
|
}
|
||||||
|
|
||||||
func (this *box) getIconsNonce () int {
|
func (this *box) getIconSetNonce () int {
|
||||||
// should panic if not in the tree
|
// should panic if not in the tree
|
||||||
return this.getHierarchy().getIconsNonce()
|
return this.getHierarchy().getIconSetNonce()
|
||||||
}
|
}
|
||||||
|
@ -186,6 +186,53 @@ func (this *containerBox) setAttr (attr tomo.Attr, user bool) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (this *containerBox) unsetAttr (kind tomo.AttrKind, user bool) {
|
||||||
|
switch kind {
|
||||||
|
case tomo.AttrKindColor:
|
||||||
|
if this.attrColor.Unset(user) {
|
||||||
|
this.invalidateTransparentChildren()
|
||||||
|
this.invalidateDraw()
|
||||||
|
}
|
||||||
|
|
||||||
|
case tomo.AttrKindTexture:
|
||||||
|
if this.attrTexture.Unset(user) {
|
||||||
|
this.invalidateTransparentChildren()
|
||||||
|
this.invalidateDraw()
|
||||||
|
}
|
||||||
|
|
||||||
|
case tomo.AttrKindTextureMode:
|
||||||
|
if this.attrTextureMode.Unset(user) {
|
||||||
|
this.invalidateTransparentChildren()
|
||||||
|
this.invalidateDraw()
|
||||||
|
}
|
||||||
|
|
||||||
|
case tomo.AttrKindGap:
|
||||||
|
if this.attrGap.Unset(user) {
|
||||||
|
this.invalidateLayout()
|
||||||
|
this.invalidateMinimum()
|
||||||
|
}
|
||||||
|
|
||||||
|
case tomo.AttrKindAlign:
|
||||||
|
if this.attrAlign.Unset(user) {
|
||||||
|
this.invalidateLayout()
|
||||||
|
}
|
||||||
|
|
||||||
|
case tomo.AttrKindOverflow:
|
||||||
|
if this.attrOverflow.Unset(user) {
|
||||||
|
this.invalidateLayout()
|
||||||
|
this.invalidateMinimum()
|
||||||
|
}
|
||||||
|
|
||||||
|
case tomo.AttrKindLayout:
|
||||||
|
if this.attrLayout.Unset(user) {
|
||||||
|
this.invalidateLayout()
|
||||||
|
this.invalidateMinimum()
|
||||||
|
}
|
||||||
|
|
||||||
|
default: this.box.unsetAttr(kind, user)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func (this *containerBox) recommendedHeight (width int) int {
|
func (this *containerBox) recommendedHeight (width int) int {
|
||||||
layout := this.attrLayout.Value().Layout
|
layout := this.attrLayout.Value().Layout
|
||||||
if layout == nil || !this.attrOverflow.Value().Y {
|
if layout == nil || !this.attrOverflow.Value().Y {
|
||||||
|
@ -4,6 +4,7 @@ import "image"
|
|||||||
import "git.tebibyte.media/tomo/tomo"
|
import "git.tebibyte.media/tomo/tomo"
|
||||||
import "git.tebibyte.media/tomo/tomo/input"
|
import "git.tebibyte.media/tomo/tomo/input"
|
||||||
import "git.tebibyte.media/tomo/tomo/canvas"
|
import "git.tebibyte.media/tomo/tomo/canvas"
|
||||||
|
import "git.tebibyte.media/tomo/backend/style"
|
||||||
import "git.tebibyte.media/tomo/backend/internal/util"
|
import "git.tebibyte.media/tomo/backend/internal/util"
|
||||||
|
|
||||||
// Hierarchy is coupled to a tomo.Window implementation, and manages a tree of
|
// Hierarchy is coupled to a tomo.Window implementation, and manages a tree of
|
||||||
@ -168,16 +169,24 @@ func (this *Hierarchy) getWindow () tomo.Window {
|
|||||||
return this.link.GetWindow()
|
return this.link.GetWindow()
|
||||||
}
|
}
|
||||||
|
|
||||||
func (this *Hierarchy) getStyle () *tomo.Style {
|
func (this *Hierarchy) getStyle () *style.Style {
|
||||||
return this.system.style
|
return this.system.style
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (this *Hierarchy) getIconSet () style.IconSet {
|
||||||
|
return this.system.iconSet
|
||||||
|
}
|
||||||
|
|
||||||
|
func (this *Hierarchy) getFaceSet () style.FaceSet {
|
||||||
|
return this.system.faceSet
|
||||||
|
}
|
||||||
|
|
||||||
func (this *Hierarchy) getStyleNonce () int {
|
func (this *Hierarchy) getStyleNonce () int {
|
||||||
return this.system.styleNonce
|
return this.system.styleNonce
|
||||||
}
|
}
|
||||||
|
|
||||||
func (this *Hierarchy) getIconsNonce () int {
|
func (this *Hierarchy) getIconSetNonce () int {
|
||||||
return this.system.iconsNonce
|
return this.system.iconSetNonce
|
||||||
}
|
}
|
||||||
|
|
||||||
func (this *Hierarchy) getCanvas () canvas.Canvas {
|
func (this *Hierarchy) getCanvas () canvas.Canvas {
|
||||||
|
@ -75,6 +75,9 @@ type anyBox interface {
|
|||||||
// setAttr sets an attribute at the user or style level depending
|
// setAttr sets an attribute at the user or style level depending
|
||||||
// on the value of user.
|
// on the value of user.
|
||||||
setAttr (attr tomo.Attr, user bool)
|
setAttr (attr tomo.Attr, user bool)
|
||||||
|
// unsetAttr unsets an attribute at the user or style level depending
|
||||||
|
// on the value of user.
|
||||||
|
unsetAttr (kind tomo.AttrKind, user bool)
|
||||||
|
|
||||||
// propagate recursively calls a function on this anyBox, and all of its
|
// propagate recursively calls a function on this anyBox, and all of its
|
||||||
// children (if applicable) The normal propagate behavior calls the
|
// children (if applicable) The normal propagate behavior calls the
|
||||||
|
@ -1,11 +1,12 @@
|
|||||||
package system
|
package system
|
||||||
|
|
||||||
import "git.tebibyte.media/tomo/tomo"
|
import "git.tebibyte.media/tomo/tomo"
|
||||||
|
import "git.tebibyte.media/tomo/backend/style"
|
||||||
|
|
||||||
type styleApplicator struct {
|
type styleApplicator struct {
|
||||||
style *tomo.Style
|
style *style.Style
|
||||||
role tomo.Role
|
role tomo.Role
|
||||||
rules []tomo.Rule
|
rules []style.Rule
|
||||||
}
|
}
|
||||||
|
|
||||||
func (this *styleApplicator) apply (box anyBox) {
|
func (this *styleApplicator) apply (box anyBox) {
|
||||||
@ -25,7 +26,7 @@ func (this *styleApplicator) apply (box anyBox) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// compile list of attributes by searching through the cached ruleset
|
// compile list of attributes by searching through the cached ruleset
|
||||||
attrs := make(tomo.AttrSet)
|
attrs := make(style.AttrSet)
|
||||||
for _, rule := range this.rules {
|
for _, rule := range this.rules {
|
||||||
satisifed := true
|
satisifed := true
|
||||||
for _, tag := range rule.Tags {
|
for _, tag := range rule.Tags {
|
||||||
|
@ -2,8 +2,8 @@ package system
|
|||||||
|
|
||||||
import "io"
|
import "io"
|
||||||
import "image"
|
import "image"
|
||||||
import "git.tebibyte.media/tomo/tomo"
|
|
||||||
import "git.tebibyte.media/tomo/tomo/canvas"
|
import "git.tebibyte.media/tomo/tomo/canvas"
|
||||||
|
import "git.tebibyte.media/tomo/backend/style"
|
||||||
import "git.tebibyte.media/tomo/backend/internal/util"
|
import "git.tebibyte.media/tomo/backend/internal/util"
|
||||||
|
|
||||||
// System is coupled to a tomo.Backend implementation, and manages Hierarchies
|
// System is coupled to a tomo.Backend implementation, and manages Hierarchies
|
||||||
@ -11,9 +11,11 @@ import "git.tebibyte.media/tomo/backend/internal/util"
|
|||||||
type System struct {
|
type System struct {
|
||||||
link BackendLink
|
link BackendLink
|
||||||
|
|
||||||
style *tomo.Style
|
style *style.Style
|
||||||
styleNonce int
|
iconSet style.IconSet
|
||||||
iconsNonce int
|
faceSet style.FaceSet
|
||||||
|
styleNonce int
|
||||||
|
iconSetNonce int
|
||||||
|
|
||||||
hierarchies util.Set[*Hierarchy]
|
hierarchies util.Set[*Hierarchy]
|
||||||
}
|
}
|
||||||
@ -43,9 +45,9 @@ func New (link BackendLink) *System {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// SetStyle sets the tomo.Style that is applied to objects, and notifies them
|
// SetStyle sets the style that is applied to objects, and notifies them
|
||||||
// that the style has changed.
|
// that the style has changed.
|
||||||
func (this *System) SetStyle (style *tomo.Style) {
|
func (this *System) SetStyle (style *style.Style) {
|
||||||
this.style = style
|
this.style = style
|
||||||
this.styleNonce ++
|
this.styleNonce ++
|
||||||
for hierarchy := range this.hierarchies {
|
for hierarchy := range this.hierarchies {
|
||||||
@ -53,14 +55,21 @@ func (this *System) SetStyle (style *tomo.Style) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// SetIconSet notifies objects that the icons have changed.
|
// SetIconSet sets the icon set that provides icon textures, and notifies
|
||||||
func (this *System) SetIconSet (iconSet tomo.IconSet) {
|
// objects that the icons have changed.
|
||||||
this.iconsNonce ++
|
func (this *System) SetIconSet (iconSet style.IconSet) {
|
||||||
|
this.iconSet = iconSet
|
||||||
|
this.iconSetNonce ++
|
||||||
for hierarchy := range this.hierarchies {
|
for hierarchy := range this.hierarchies {
|
||||||
hierarchy.setIconSet()
|
hierarchy.setIconSet()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// SetFaceSet sets the face set that provides font faces.
|
||||||
|
func (this *System) SetFaceSet (faceSet style.FaceSet) {
|
||||||
|
this.faceSet = faceSet
|
||||||
|
}
|
||||||
|
|
||||||
func (this *System) removeHierarchy (hierarchy *Hierarchy) {
|
func (this *System) removeHierarchy (hierarchy *Hierarchy) {
|
||||||
delete(this.hierarchies, hierarchy)
|
delete(this.hierarchies, hierarchy)
|
||||||
}
|
}
|
||||||
|
@ -2,6 +2,7 @@ package system
|
|||||||
|
|
||||||
import "image"
|
import "image"
|
||||||
import "image/color"
|
import "image/color"
|
||||||
|
import "golang.org/x/image/font"
|
||||||
import "git.tebibyte.media/tomo/tomo"
|
import "git.tebibyte.media/tomo/tomo"
|
||||||
import "golang.org/x/image/math/fixed"
|
import "golang.org/x/image/math/fixed"
|
||||||
import "git.tebibyte.media/tomo/typeset"
|
import "git.tebibyte.media/tomo/typeset"
|
||||||
@ -9,6 +10,7 @@ import "git.tebibyte.media/tomo/tomo/text"
|
|||||||
import "git.tebibyte.media/tomo/tomo/input"
|
import "git.tebibyte.media/tomo/tomo/input"
|
||||||
import "git.tebibyte.media/tomo/tomo/event"
|
import "git.tebibyte.media/tomo/tomo/event"
|
||||||
import "git.tebibyte.media/tomo/tomo/canvas"
|
import "git.tebibyte.media/tomo/tomo/canvas"
|
||||||
|
import "git.tebibyte.media/tomo/backend/internal/util"
|
||||||
|
|
||||||
type textBox struct {
|
type textBox struct {
|
||||||
*box
|
*box
|
||||||
@ -31,6 +33,7 @@ type textBox struct {
|
|||||||
dot text.Dot
|
dot text.Dot
|
||||||
|
|
||||||
drawer typeset.Drawer
|
drawer typeset.Drawer
|
||||||
|
face util.Cycler[font.Face]
|
||||||
|
|
||||||
on struct {
|
on struct {
|
||||||
contentBoundsChange event.FuncBroadcaster
|
contentBoundsChange event.FuncBroadcaster
|
||||||
@ -41,6 +44,8 @@ type textBox struct {
|
|||||||
func (this *System) NewTextBox () tomo.TextBox {
|
func (this *System) NewTextBox () tomo.TextBox {
|
||||||
box := &textBox { }
|
box := &textBox { }
|
||||||
box.box = this.newBox(box)
|
box.box = this.newBox(box)
|
||||||
|
box.attrTextColor.SetFallback(tomo.ATextColor(color.Black))
|
||||||
|
box.attrDotColor.SetFallback(tomo.ADotColor(color.RGBA { G: 255, B: 255, A: 255}))
|
||||||
return box
|
return box
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -106,9 +111,8 @@ func (this *textBox) OnDotChange (callback func ()) event.Cookie {
|
|||||||
func (this *textBox) Draw (can canvas.Canvas) {
|
func (this *textBox) Draw (can canvas.Canvas) {
|
||||||
if can == nil { return }
|
if can == nil { return }
|
||||||
|
|
||||||
texture := this.attrTexture.Value().Texture
|
texture := this.attrTexture.Value().Texture
|
||||||
col := this.attrColor.Value().Color
|
col := this.attrColor.Value().Color
|
||||||
if col == nil { col = color.Transparent }
|
|
||||||
|
|
||||||
this.drawBorders(can)
|
this.drawBorders(can)
|
||||||
|
|
||||||
@ -124,9 +128,8 @@ func (this *textBox) Draw (can canvas.Canvas) {
|
|||||||
this.drawDot(can)
|
this.drawDot(can)
|
||||||
}
|
}
|
||||||
|
|
||||||
if this.attrFace.Value().Face != nil {
|
if this.face.Value() != nil {
|
||||||
textColor := this.attrTextColor.Value().Color
|
textColor := this.attrTextColor.Value().Color
|
||||||
if textColor == nil { textColor = color.Black }
|
|
||||||
this.drawer.Draw(can, textColor, this.textOffset())
|
this.drawer.Draw(can, textColor, this.textOffset())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -145,23 +148,22 @@ func (this *textBox) setAttr (attr tomo.Attr, user bool) {
|
|||||||
|
|
||||||
case tomo.AttrFace:
|
case tomo.AttrFace:
|
||||||
if this.attrFace.Set(attr, user) {
|
if this.attrFace.Set(attr, user) {
|
||||||
this.drawer.SetFace(attr.Face)
|
this.handleFaceChange()
|
||||||
this.invalidateMinimum()
|
|
||||||
this.invalidateLayout()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
case tomo.AttrWrap:
|
case tomo.AttrWrap:
|
||||||
if this.attrWrap.Set(attr, user) {
|
if this.attrWrap.Set(attr, user) {
|
||||||
this.drawer.SetWrap(bool(attr))
|
this.drawer.SetWrap(bool(this.attrWrap.Value()))
|
||||||
this.invalidateMinimum()
|
this.invalidateMinimum()
|
||||||
this.invalidateLayout()
|
this.invalidateLayout()
|
||||||
}
|
}
|
||||||
|
|
||||||
case tomo.AttrAlign:
|
case tomo.AttrAlign:
|
||||||
if this.attrAlign.Set(attr, user) {
|
if this.attrAlign.Set(attr, user) {
|
||||||
|
align := this.attrAlign.Value()
|
||||||
this.drawer.SetAlign (
|
this.drawer.SetAlign (
|
||||||
typeset.Align(attr.X),
|
typeset.Align(align.X),
|
||||||
typeset.Align(attr.Y))
|
typeset.Align(align.Y))
|
||||||
this.invalidateDraw()
|
this.invalidateDraw()
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -175,6 +177,49 @@ func (this *textBox) setAttr (attr tomo.Attr, user bool) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (this *textBox) unsetAttr (kind tomo.AttrKind, user bool) {
|
||||||
|
switch kind {
|
||||||
|
case tomo.AttrKindTextColor:
|
||||||
|
if this.attrTextColor.Unset(user) && !this.dot.Empty() {
|
||||||
|
this.invalidateDraw()
|
||||||
|
}
|
||||||
|
|
||||||
|
case tomo.AttrKindDotColor:
|
||||||
|
if this.attrDotColor.Unset(user) && !this.dot.Empty() {
|
||||||
|
this.invalidateDraw()
|
||||||
|
}
|
||||||
|
|
||||||
|
case tomo.AttrKindFace:
|
||||||
|
if this.attrFace.Unset(user) {
|
||||||
|
this.handleFaceChange()
|
||||||
|
}
|
||||||
|
|
||||||
|
case tomo.AttrKindWrap:
|
||||||
|
if this.attrWrap.Unset(user) {
|
||||||
|
this.drawer.SetWrap(bool(this.attrWrap.Value()))
|
||||||
|
this.invalidateMinimum()
|
||||||
|
this.invalidateLayout()
|
||||||
|
}
|
||||||
|
|
||||||
|
case tomo.AttrKindAlign:
|
||||||
|
if this.attrAlign.Unset(user) {
|
||||||
|
align := this.attrAlign.Value()
|
||||||
|
this.drawer.SetAlign (
|
||||||
|
typeset.Align(align.X),
|
||||||
|
typeset.Align(align.Y))
|
||||||
|
this.invalidateDraw()
|
||||||
|
}
|
||||||
|
|
||||||
|
case tomo.AttrKindOverflow:
|
||||||
|
if this.attrOverflow.Unset(user) {
|
||||||
|
this.invalidateMinimum()
|
||||||
|
this.invalidateLayout()
|
||||||
|
}
|
||||||
|
|
||||||
|
default: this.box.unsetAttr(kind, user)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func roundPt (point fixed.Point26_6) image.Point {
|
func roundPt (point fixed.Point26_6) image.Point {
|
||||||
return image.Pt(point.X.Round(), point.Y.Round())
|
return image.Pt(point.X.Round(), point.Y.Round())
|
||||||
}
|
}
|
||||||
@ -184,13 +229,11 @@ func fixPt (point image.Point) fixed.Point26_6 {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (this *textBox) drawDot (can canvas.Canvas) {
|
func (this *textBox) drawDot (can canvas.Canvas) {
|
||||||
if this.attrFace.Value().Face == nil { return }
|
face := this.face.Value()
|
||||||
|
if face == nil { return }
|
||||||
|
|
||||||
face := this.attrFace.Value().Face
|
|
||||||
textColor := this.attrTextColor.Value().Color
|
textColor := this.attrTextColor.Value().Color
|
||||||
dotColor := this.attrDotColor.Value().Color
|
dotColor := this.attrDotColor.Value().Color
|
||||||
if textColor == nil { textColor = color.Black }
|
|
||||||
if dotColor == nil { dotColor = color.RGBA { G: 255, B: 255, A: 255 } }
|
|
||||||
|
|
||||||
pen := can.Pen()
|
pen := can.Pen()
|
||||||
|
|
||||||
@ -447,3 +490,16 @@ func (this *textBox) scrollToDot () {
|
|||||||
|
|
||||||
this.ScrollTo(scroll)
|
this.ScrollTo(scroll)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (this *textBox) handleFaceChange () {
|
||||||
|
hierarchy := this.getHierarchy()
|
||||||
|
if hierarchy != nil { return }
|
||||||
|
faceSet := hierarchy.getFaceSet()
|
||||||
|
if faceSet != nil { return }
|
||||||
|
|
||||||
|
face := faceSet.Face(tomo.Face(this.attrFace.Value()))
|
||||||
|
this.face.Set(face, face)
|
||||||
|
this.drawer.SetFace(face)
|
||||||
|
this.invalidateMinimum()
|
||||||
|
this.invalidateLayout()
|
||||||
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user