mirror of https://github.com/anoshenko/rui.git
Added Style functions to Theme
This commit is contained in:
parent
be2701e59d
commit
9c236ae102
265
theme.go
265
theme.go
|
@ -13,11 +13,11 @@ const (
|
||||||
LandscapeMedia = 2
|
LandscapeMedia = 2
|
||||||
)
|
)
|
||||||
|
|
||||||
type MediaStyle struct {
|
type mediaStyle struct {
|
||||||
Orientation int
|
orientation int
|
||||||
MaxWidth int
|
maxWidth int
|
||||||
MaxHeight int
|
maxHeight int
|
||||||
Styles map[string]ViewStyle
|
styles map[string]ViewStyle
|
||||||
}
|
}
|
||||||
|
|
||||||
type theme struct {
|
type theme struct {
|
||||||
|
@ -29,7 +29,7 @@ type theme struct {
|
||||||
images map[string]string
|
images map[string]string
|
||||||
darkImages map[string]string
|
darkImages map[string]string
|
||||||
styles map[string]ViewStyle
|
styles map[string]ViewStyle
|
||||||
mediaStyles []MediaStyle
|
mediaStyles []mediaStyle
|
||||||
}
|
}
|
||||||
|
|
||||||
type Theme interface {
|
type Theme interface {
|
||||||
|
@ -47,6 +47,11 @@ type Theme interface {
|
||||||
SetImage(tag, image, darkUIImage string)
|
SetImage(tag, image, darkUIImage string)
|
||||||
// ImageConstantTags returns the list of all available image constants
|
// ImageConstantTags returns the list of all available image constants
|
||||||
ImageConstantTags() []string
|
ImageConstantTags() []string
|
||||||
|
Style(tag string) ViewStyle
|
||||||
|
SetStyle(tag string, style ViewStyle)
|
||||||
|
MediaStyle(tag string, orientation, maxWidth, maxHeight int) ViewStyle
|
||||||
|
SetMediaStyle(tag string, orientation, maxWidth, maxHeight int, style ViewStyle)
|
||||||
|
StyleTags() []string
|
||||||
Append(anotherTheme Theme)
|
Append(anotherTheme Theme)
|
||||||
|
|
||||||
constant(tag string, touchUI bool) string
|
constant(tag string, touchUI bool) string
|
||||||
|
@ -57,11 +62,11 @@ type Theme interface {
|
||||||
data() *theme
|
data() *theme
|
||||||
}
|
}
|
||||||
|
|
||||||
func (rule MediaStyle) cssText() string {
|
func (rule mediaStyle) cssText() string {
|
||||||
builder := allocStringBuilder()
|
builder := allocStringBuilder()
|
||||||
defer freeStringBuilder(builder)
|
defer freeStringBuilder(builder)
|
||||||
|
|
||||||
switch rule.Orientation {
|
switch rule.orientation {
|
||||||
case PortraitMedia:
|
case PortraitMedia:
|
||||||
builder.WriteString(" and (orientation: portrait)")
|
builder.WriteString(" and (orientation: portrait)")
|
||||||
|
|
||||||
|
@ -69,45 +74,45 @@ func (rule MediaStyle) cssText() string {
|
||||||
builder.WriteString(" and (orientation: landscape)")
|
builder.WriteString(" and (orientation: landscape)")
|
||||||
}
|
}
|
||||||
|
|
||||||
if rule.MaxWidth > 0 {
|
if rule.maxWidth > 0 {
|
||||||
builder.WriteString(" and (max-width: ")
|
builder.WriteString(" and (max-width: ")
|
||||||
builder.WriteString(strconv.Itoa(rule.MaxWidth))
|
builder.WriteString(strconv.Itoa(rule.maxWidth))
|
||||||
builder.WriteString("px)")
|
builder.WriteString("px)")
|
||||||
}
|
}
|
||||||
|
|
||||||
if rule.MaxHeight > 0 {
|
if rule.maxHeight > 0 {
|
||||||
builder.WriteString(" and (max-height: ")
|
builder.WriteString(" and (max-height: ")
|
||||||
builder.WriteString(strconv.Itoa(rule.MaxHeight))
|
builder.WriteString(strconv.Itoa(rule.maxHeight))
|
||||||
builder.WriteString("px)")
|
builder.WriteString("px)")
|
||||||
}
|
}
|
||||||
|
|
||||||
return builder.String()
|
return builder.String()
|
||||||
}
|
}
|
||||||
|
|
||||||
func parseMediaRule(text string) (MediaStyle, bool) {
|
func parseMediaRule(text string) (mediaStyle, bool) {
|
||||||
rule := MediaStyle{
|
rule := mediaStyle{
|
||||||
Orientation: DefaultMedia,
|
orientation: DefaultMedia,
|
||||||
MaxWidth: 0,
|
maxWidth: 0,
|
||||||
MaxHeight: 0,
|
maxHeight: 0,
|
||||||
Styles: map[string]ViewStyle{},
|
styles: map[string]ViewStyle{},
|
||||||
}
|
}
|
||||||
|
|
||||||
elements := strings.Split(text, ":")
|
elements := strings.Split(text, ":")
|
||||||
for i := 1; i < len(elements); i++ {
|
for i := 1; i < len(elements); i++ {
|
||||||
switch element := elements[i]; element {
|
switch element := elements[i]; element {
|
||||||
case "portrait":
|
case "portrait":
|
||||||
if rule.Orientation != DefaultMedia {
|
if rule.orientation != DefaultMedia {
|
||||||
ErrorLog(`Duplicate orientation tag in the style section "` + text + `"`)
|
ErrorLog(`Duplicate orientation tag in the style section "` + text + `"`)
|
||||||
return rule, false
|
return rule, false
|
||||||
}
|
}
|
||||||
rule.Orientation = PortraitMedia
|
rule.orientation = PortraitMedia
|
||||||
|
|
||||||
case "landscape":
|
case "landscape":
|
||||||
if rule.Orientation != DefaultMedia {
|
if rule.orientation != DefaultMedia {
|
||||||
ErrorLog(`Duplicate orientation tag in the style section "` + text + `"`)
|
ErrorLog(`Duplicate orientation tag in the style section "` + text + `"`)
|
||||||
return rule, false
|
return rule, false
|
||||||
}
|
}
|
||||||
rule.Orientation = LandscapeMedia
|
rule.orientation = LandscapeMedia
|
||||||
|
|
||||||
default:
|
default:
|
||||||
elementSize := func(name string) (int, bool) {
|
elementSize := func(name string) (int, bool) {
|
||||||
|
@ -126,20 +131,20 @@ func parseMediaRule(text string) (MediaStyle, bool) {
|
||||||
if !ok {
|
if !ok {
|
||||||
return rule, false
|
return rule, false
|
||||||
}
|
}
|
||||||
if rule.MaxWidth != 0 {
|
if rule.maxWidth != 0 {
|
||||||
ErrorLog(`Duplicate "width" tag in the style section "` + text + `"`)
|
ErrorLog(`Duplicate "width" tag in the style section "` + text + `"`)
|
||||||
return rule, false
|
return rule, false
|
||||||
}
|
}
|
||||||
rule.MaxWidth = size
|
rule.maxWidth = size
|
||||||
} else if size, ok := elementSize("height"); !ok || size > 0 {
|
} else if size, ok := elementSize("height"); !ok || size > 0 {
|
||||||
if !ok {
|
if !ok {
|
||||||
return rule, false
|
return rule, false
|
||||||
}
|
}
|
||||||
if rule.MaxHeight != 0 {
|
if rule.maxHeight != 0 {
|
||||||
ErrorLog(`Duplicate "height" tag in the style section "` + text + `"`)
|
ErrorLog(`Duplicate "height" tag in the style section "` + text + `"`)
|
||||||
return rule, false
|
return rule, false
|
||||||
}
|
}
|
||||||
rule.MaxHeight = size
|
rule.maxHeight = size
|
||||||
} else {
|
} else {
|
||||||
ErrorLogF(`Unknown elemnet "%s" in the style section name "%s"`, element, text)
|
ErrorLogF(`Unknown elemnet "%s" in the style section name "%s"`, element, text)
|
||||||
return rule, false
|
return rule, false
|
||||||
|
@ -173,7 +178,7 @@ func (theme *theme) init() {
|
||||||
theme.images = map[string]string{}
|
theme.images = map[string]string{}
|
||||||
theme.darkImages = map[string]string{}
|
theme.darkImages = map[string]string{}
|
||||||
theme.styles = map[string]ViewStyle{}
|
theme.styles = map[string]ViewStyle{}
|
||||||
theme.mediaStyles = []MediaStyle{}
|
theme.mediaStyles = []mediaStyle{}
|
||||||
}
|
}
|
||||||
|
|
||||||
func (theme *theme) Name() string {
|
func (theme *theme) Name() string {
|
||||||
|
@ -240,6 +245,67 @@ func (theme *theme) SetImage(tag, image, darkUIImage string) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (theme *theme) Style(tag string) ViewStyle {
|
||||||
|
if style, ok := theme.styles[tag]; ok {
|
||||||
|
return style
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (theme *theme) SetStyle(tag string, style ViewStyle) {
|
||||||
|
if style != nil {
|
||||||
|
theme.styles[tag] = style
|
||||||
|
} else {
|
||||||
|
delete(theme.styles, tag)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (theme *theme) MediaStyle(tag string, orientation, maxWidth, maxHeight int) ViewStyle {
|
||||||
|
for _, styles := range theme.mediaStyles {
|
||||||
|
if styles.orientation == orientation && styles.maxWidth == maxWidth && styles.maxHeight == maxHeight {
|
||||||
|
if style, ok := styles.styles[tag]; ok {
|
||||||
|
return style
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (theme *theme) SetMediaStyle(tag string, orientation, maxWidth, maxHeight int, style ViewStyle) {
|
||||||
|
if maxWidth < 0 {
|
||||||
|
maxWidth = 0
|
||||||
|
}
|
||||||
|
if maxHeight < 0 {
|
||||||
|
maxHeight = 0
|
||||||
|
}
|
||||||
|
|
||||||
|
if orientation == DefaultMedia && maxWidth == 0 && maxHeight == 0 {
|
||||||
|
theme.SetStyle(tag, style)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
for i, styles := range theme.mediaStyles {
|
||||||
|
if styles.orientation == orientation && styles.maxWidth == maxWidth && styles.maxHeight == maxHeight {
|
||||||
|
if style != nil {
|
||||||
|
theme.mediaStyles[i].styles[tag] = style
|
||||||
|
} else {
|
||||||
|
delete(theme.mediaStyles[i].styles, tag)
|
||||||
|
}
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if style != nil {
|
||||||
|
theme.mediaStyles = append(theme.mediaStyles, mediaStyle{
|
||||||
|
orientation: orientation,
|
||||||
|
maxWidth: maxWidth,
|
||||||
|
maxHeight: maxHeight,
|
||||||
|
styles: map[string]ViewStyle{tag: style},
|
||||||
|
})
|
||||||
|
theme.sortMediaStyles()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func (theme *theme) ConstantTags() []string {
|
func (theme *theme) ConstantTags() []string {
|
||||||
keys := make([]string, 0, len(theme.constants))
|
keys := make([]string, 0, len(theme.constants))
|
||||||
for k := range theme.constants {
|
for k := range theme.constants {
|
||||||
|
@ -288,6 +354,49 @@ func (theme *theme) ImageConstantTags() []string {
|
||||||
return keys
|
return keys
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (theme *theme) StyleTags() []string {
|
||||||
|
keys := make([]string, 0, len(theme.styles)*2)
|
||||||
|
|
||||||
|
appendTag := func(k string) {
|
||||||
|
n := sort.SearchStrings(keys, k)
|
||||||
|
if n >= len(keys) {
|
||||||
|
keys = append(keys, k)
|
||||||
|
} else if keys[n] != k {
|
||||||
|
if n == 0 {
|
||||||
|
keys = append([]string{k}, keys...)
|
||||||
|
} else {
|
||||||
|
keys = append(keys[:n+1], keys[n:]...)
|
||||||
|
keys[n] = k
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for k := range theme.styles {
|
||||||
|
if index := strings.IndexRune(k, ':'); index < 0 {
|
||||||
|
keys = append(keys, k)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
sort.Strings(keys)
|
||||||
|
|
||||||
|
for k := range theme.styles {
|
||||||
|
if index := strings.IndexRune(k, ':'); index > 0 {
|
||||||
|
appendTag(k[:index])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, media := range theme.mediaStyles {
|
||||||
|
for k := range media.styles {
|
||||||
|
index := strings.IndexRune(k, ':')
|
||||||
|
if index > 0 {
|
||||||
|
appendTag(k[:index])
|
||||||
|
} else if index < 0 {
|
||||||
|
appendTag(k)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return keys
|
||||||
|
}
|
||||||
|
|
||||||
func (theme *theme) data() *theme {
|
func (theme *theme) data() *theme {
|
||||||
return theme
|
return theme
|
||||||
}
|
}
|
||||||
|
@ -329,11 +438,11 @@ func (theme *theme) Append(anotherTheme Theme) {
|
||||||
for _, anotherMedia := range another.mediaStyles {
|
for _, anotherMedia := range another.mediaStyles {
|
||||||
exists := false
|
exists := false
|
||||||
for _, media := range theme.mediaStyles {
|
for _, media := range theme.mediaStyles {
|
||||||
if anotherMedia.MaxHeight == media.MaxHeight &&
|
if anotherMedia.maxHeight == media.maxHeight &&
|
||||||
anotherMedia.MaxWidth == media.MaxWidth &&
|
anotherMedia.maxWidth == media.maxWidth &&
|
||||||
anotherMedia.Orientation == media.Orientation {
|
anotherMedia.orientation == media.orientation {
|
||||||
for tag, style := range anotherMedia.Styles {
|
for tag, style := range anotherMedia.styles {
|
||||||
media.Styles[tag] = style
|
media.styles[tag] = style
|
||||||
}
|
}
|
||||||
exists = true
|
exists = true
|
||||||
break
|
break
|
||||||
|
@ -355,11 +464,6 @@ func (theme *theme) cssText(session Session) string {
|
||||||
builder.init()
|
builder.init()
|
||||||
|
|
||||||
for tag, style := range theme.styles {
|
for tag, style := range theme.styles {
|
||||||
/*var style viewStyle
|
|
||||||
style.init()
|
|
||||||
for tag, value := range obj {
|
|
||||||
style.Set(tag, value)
|
|
||||||
}*/
|
|
||||||
builder.startStyle(tag)
|
builder.startStyle(tag)
|
||||||
style.cssViewStyle(&builder, session)
|
style.cssViewStyle(&builder, session)
|
||||||
builder.endStyle()
|
builder.endStyle()
|
||||||
|
@ -367,12 +471,7 @@ func (theme *theme) cssText(session Session) string {
|
||||||
|
|
||||||
for _, media := range theme.mediaStyles {
|
for _, media := range theme.mediaStyles {
|
||||||
builder.startMedia(media.cssText())
|
builder.startMedia(media.cssText())
|
||||||
for tag, style := range media.Styles {
|
for tag, style := range media.styles {
|
||||||
/*var style viewStyle
|
|
||||||
style.init()
|
|
||||||
for tag, value := range obj {
|
|
||||||
style.Set(tag, value)
|
|
||||||
}*/
|
|
||||||
builder.startStyle(tag)
|
builder.startStyle(tag)
|
||||||
style.cssViewStyle(&builder, session)
|
style.cssViewStyle(&builder, session)
|
||||||
builder.endStyle()
|
builder.endStyle()
|
||||||
|
@ -384,26 +483,15 @@ func (theme *theme) cssText(session Session) string {
|
||||||
}
|
}
|
||||||
|
|
||||||
func (theme *theme) addText(themeText string) bool {
|
func (theme *theme) addText(themeText string) bool {
|
||||||
data := ParseDataText(themeText)
|
|
||||||
if data == nil {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
theme.addData(data)
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
|
|
||||||
func (theme *theme) addData(data DataObject) {
|
|
||||||
if theme.constants == nil {
|
if theme.constants == nil {
|
||||||
theme.init()
|
theme.init()
|
||||||
}
|
}
|
||||||
|
|
||||||
if data.IsObject() && data.Tag() == "theme" {
|
data := ParseDataText(themeText)
|
||||||
theme.parseThemeData(data)
|
if data == nil || !data.IsObject() || data.Tag() != "theme" {
|
||||||
|
return false
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
func (theme *theme) parseThemeData(data DataObject) {
|
|
||||||
count := data.PropertyCount()
|
count := data.PropertyCount()
|
||||||
|
|
||||||
objToStyle := func(obj DataObject) ViewStyle {
|
objToStyle := func(obj DataObject) ViewStyle {
|
||||||
|
@ -519,7 +607,7 @@ func (theme *theme) parseThemeData(data DataObject) {
|
||||||
for k := 0; k < arraySize; k++ {
|
for k := 0; k < arraySize; k++ {
|
||||||
if element := d.ArrayElement(k); element != nil && element.IsObject() {
|
if element := d.ArrayElement(k); element != nil && element.IsObject() {
|
||||||
if obj := element.Object(); obj != nil {
|
if obj := element.Object(); obj != nil {
|
||||||
rule.Styles[obj.Tag()] = objToStyle(obj)
|
rule.styles[obj.Tag()] = objToStyle(obj)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -530,15 +618,20 @@ func (theme *theme) parseThemeData(data DataObject) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if len(theme.mediaStyles) > 0 {
|
theme.sortMediaStyles()
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
func (theme *theme) sortMediaStyles() {
|
||||||
|
if len(theme.mediaStyles) > 1 {
|
||||||
sort.SliceStable(theme.mediaStyles, func(i, j int) bool {
|
sort.SliceStable(theme.mediaStyles, func(i, j int) bool {
|
||||||
if theme.mediaStyles[i].Orientation != theme.mediaStyles[j].Orientation {
|
if theme.mediaStyles[i].orientation != theme.mediaStyles[j].orientation {
|
||||||
return theme.mediaStyles[i].Orientation < theme.mediaStyles[j].Orientation
|
return theme.mediaStyles[i].orientation < theme.mediaStyles[j].orientation
|
||||||
}
|
}
|
||||||
if theme.mediaStyles[i].MaxWidth != theme.mediaStyles[j].MaxWidth {
|
if theme.mediaStyles[i].maxWidth != theme.mediaStyles[j].maxWidth {
|
||||||
return theme.mediaStyles[i].MaxWidth < theme.mediaStyles[j].MaxWidth
|
return theme.mediaStyles[i].maxWidth < theme.mediaStyles[j].maxWidth
|
||||||
}
|
}
|
||||||
return theme.mediaStyles[i].MaxHeight < theme.mediaStyles[j].MaxHeight
|
return theme.mediaStyles[i].maxHeight < theme.mediaStyles[j].maxHeight
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -656,6 +749,50 @@ func (theme *theme) String() string {
|
||||||
writeConstants("constants", theme.constants)
|
writeConstants("constants", theme.constants)
|
||||||
writeConstants("constants:touch", theme.touchConstants)
|
writeConstants("constants:touch", theme.touchConstants)
|
||||||
|
|
||||||
|
writeStyles := func(orientation, maxWidth, maxHeihgt int, styles map[string]ViewStyle) bool {
|
||||||
|
count := len(styles)
|
||||||
|
if count == 0 {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
tags := make([]string, 0, count)
|
||||||
|
for name := range styles {
|
||||||
|
tags = append(tags, name)
|
||||||
|
}
|
||||||
|
sort.Strings(tags)
|
||||||
|
|
||||||
|
buffer.WriteString("\tstyles")
|
||||||
|
switch orientation {
|
||||||
|
case PortraitMedia:
|
||||||
|
buffer.WriteString(":portrait")
|
||||||
|
|
||||||
|
case LandscapeMedia:
|
||||||
|
buffer.WriteString(":landscape")
|
||||||
|
}
|
||||||
|
if maxWidth > 0 {
|
||||||
|
buffer.WriteString(fmt.Sprintf(":width%d", maxWidth))
|
||||||
|
}
|
||||||
|
if maxHeihgt > 0 {
|
||||||
|
buffer.WriteString(fmt.Sprintf(":heihgt%d", maxHeihgt))
|
||||||
|
}
|
||||||
|
buffer.WriteString(" = [\n")
|
||||||
|
|
||||||
|
for _, tag := range tags {
|
||||||
|
if style, ok := styles[tag]; ok {
|
||||||
|
buffer.WriteString("\t\t")
|
||||||
|
writeViewStyle(tag, style, buffer, "\t\t")
|
||||||
|
buffer.WriteString(",")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
buffer.WriteString("\t],\n")
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
writeStyles(0, 0, 0, theme.styles)
|
||||||
|
for _, media := range theme.mediaStyles {
|
||||||
|
writeStyles(media.orientation, media.maxWidth, media.maxHeight, media.styles)
|
||||||
|
}
|
||||||
|
|
||||||
buffer.WriteString("}\n")
|
buffer.WriteString("}\n")
|
||||||
return buffer.String()
|
return buffer.String()
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue