2021-09-07 17:36:50 +03:00
|
|
|
package rui
|
|
|
|
|
2024-04-22 19:14:58 +03:00
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
)
|
2021-09-07 17:36:50 +03:00
|
|
|
|
|
|
|
const (
|
2024-11-27 11:32:13 +03:00
|
|
|
// BorderBox is the value of the following properties:
|
|
|
|
//
|
|
|
|
// * BackgroundClip - The background extends to the outside edge of the border (but underneath the border in z-ordering).
|
|
|
|
//
|
|
|
|
// * BackgroundOrigin - The background is positioned relative to the border box.
|
|
|
|
//
|
|
|
|
// * MaskClip - The painted content is clipped to the border box.
|
|
|
|
//
|
|
|
|
// * MaskOrigin - The mask is positioned relative to the border box.
|
|
|
|
BorderBox = 0
|
|
|
|
|
|
|
|
// PaddingBox is value of the BackgroundClip and MaskClip property:
|
|
|
|
//
|
|
|
|
// * BackgroundClip - The background extends to the outside edge of the padding. No background is drawn beneath the border.
|
|
|
|
//
|
|
|
|
// * BackgroundOrigin - The background is positioned relative to the padding box.
|
|
|
|
//
|
|
|
|
// * MaskClip - The painted content is clipped to the padding box.
|
|
|
|
//
|
|
|
|
// * MaskOrigin - The mask is positioned relative to the padding box.
|
|
|
|
PaddingBox = 1
|
|
|
|
|
|
|
|
// ContentBox is value of the BackgroundClip and MaskClip property:
|
|
|
|
//
|
|
|
|
// * BackgroundClip - The background is painted within (clipped to) the content box.
|
|
|
|
//
|
|
|
|
// * BackgroundOrigin - The background is positioned relative to the content box.
|
|
|
|
//
|
|
|
|
// * MaskClip - The painted content is clipped to the content box.
|
|
|
|
//
|
|
|
|
// * MaskOrigin - The mask is positioned relative to the content box.
|
|
|
|
ContentBox = 2
|
2021-09-07 17:36:50 +03:00
|
|
|
)
|
|
|
|
|
2024-09-12 14:05:11 +03:00
|
|
|
// BackgroundElement describes the background element
|
2021-09-07 17:36:50 +03:00
|
|
|
type BackgroundElement interface {
|
|
|
|
Properties
|
2024-04-22 19:14:58 +03:00
|
|
|
fmt.Stringer
|
|
|
|
stringWriter
|
2021-10-04 17:58:17 +03:00
|
|
|
cssStyle(session Session) string
|
2024-09-12 14:05:11 +03:00
|
|
|
|
|
|
|
// Tag returns type of the background element.
|
|
|
|
// Possible values are: "image", "conic-gradient", "linear-gradient" and "radial-gradient"
|
2021-09-07 17:36:50 +03:00
|
|
|
Tag() string
|
2024-09-12 14:05:11 +03:00
|
|
|
|
|
|
|
// Clone creates a new copy of BackgroundElement
|
2022-04-30 12:13:16 +03:00
|
|
|
Clone() BackgroundElement
|
2021-09-07 17:36:50 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
type backgroundElement struct {
|
2024-11-13 12:56:39 +03:00
|
|
|
dataProperty
|
2021-09-07 17:36:50 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewBackgroundImage creates the new background image
|
|
|
|
func createBackground(obj DataObject) BackgroundElement {
|
|
|
|
var result BackgroundElement = nil
|
|
|
|
|
|
|
|
switch obj.Tag() {
|
|
|
|
case "image":
|
2024-11-13 12:56:39 +03:00
|
|
|
result = NewBackgroundImage(nil)
|
2021-09-07 17:36:50 +03:00
|
|
|
|
|
|
|
case "linear-gradient":
|
2024-11-13 12:56:39 +03:00
|
|
|
result = NewBackgroundLinearGradient(nil)
|
2021-09-07 17:36:50 +03:00
|
|
|
|
|
|
|
case "radial-gradient":
|
2024-11-13 12:56:39 +03:00
|
|
|
result = NewBackgroundRadialGradient(nil)
|
2021-09-07 17:36:50 +03:00
|
|
|
|
2022-04-30 12:13:16 +03:00
|
|
|
case "conic-gradient":
|
2024-11-13 12:56:39 +03:00
|
|
|
result = NewBackgroundConicGradient(nil)
|
2022-04-30 12:13:16 +03:00
|
|
|
|
2021-09-07 17:36:50 +03:00
|
|
|
default:
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
count := obj.PropertyCount()
|
|
|
|
for i := 0; i < count; i++ {
|
|
|
|
if node := obj.Property(i); node.Type() == TextNode {
|
|
|
|
if value := node.Text(); value != "" {
|
2024-11-13 12:56:39 +03:00
|
|
|
result.Set(PropertyName(node.Tag()), value)
|
2021-09-07 17:36:50 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
2024-11-27 11:32:13 +03:00
|
|
|
func parseBackgroundValue(value any) []BackgroundElement {
|
2024-11-13 12:56:39 +03:00
|
|
|
|
|
|
|
switch value := value.(type) {
|
|
|
|
case BackgroundElement:
|
2024-11-27 11:32:13 +03:00
|
|
|
return []BackgroundElement{value}
|
2024-11-13 12:56:39 +03:00
|
|
|
|
|
|
|
case []BackgroundElement:
|
2024-11-27 11:32:13 +03:00
|
|
|
return value
|
2024-11-13 12:56:39 +03:00
|
|
|
|
|
|
|
case []DataValue:
|
2024-11-27 11:32:13 +03:00
|
|
|
background := []BackgroundElement{}
|
2024-11-13 12:56:39 +03:00
|
|
|
for _, el := range value {
|
|
|
|
if el.IsObject() {
|
|
|
|
if element := createBackground(el.Object()); element != nil {
|
|
|
|
background = append(background, element)
|
|
|
|
} else {
|
2024-11-27 11:32:13 +03:00
|
|
|
return nil
|
2024-11-13 12:56:39 +03:00
|
|
|
}
|
|
|
|
} else if obj := ParseDataText(el.Value()); obj != nil {
|
|
|
|
if element := createBackground(obj); element != nil {
|
|
|
|
background = append(background, element)
|
|
|
|
} else {
|
2024-11-27 11:32:13 +03:00
|
|
|
return nil
|
2024-11-13 12:56:39 +03:00
|
|
|
}
|
|
|
|
} else {
|
2024-11-27 11:32:13 +03:00
|
|
|
return nil
|
2024-11-13 12:56:39 +03:00
|
|
|
}
|
|
|
|
}
|
2024-11-27 11:32:13 +03:00
|
|
|
return background
|
2024-11-13 12:56:39 +03:00
|
|
|
|
|
|
|
case DataObject:
|
|
|
|
if element := createBackground(value); element != nil {
|
2024-11-27 11:32:13 +03:00
|
|
|
return []BackgroundElement{element}
|
2024-11-13 12:56:39 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
case []DataObject:
|
2024-11-27 11:32:13 +03:00
|
|
|
background := []BackgroundElement{}
|
2024-11-13 12:56:39 +03:00
|
|
|
for _, obj := range value {
|
|
|
|
if element := createBackground(obj); element != nil {
|
|
|
|
background = append(background, element)
|
|
|
|
} else {
|
2024-11-27 11:32:13 +03:00
|
|
|
return nil
|
2024-11-13 12:56:39 +03:00
|
|
|
}
|
|
|
|
}
|
2024-11-27 11:32:13 +03:00
|
|
|
return background
|
2024-11-13 12:56:39 +03:00
|
|
|
|
|
|
|
case string:
|
|
|
|
if obj := ParseDataText(value); obj != nil {
|
|
|
|
if element := createBackground(obj); element != nil {
|
2024-11-27 11:32:13 +03:00
|
|
|
return []BackgroundElement{element}
|
2024-11-13 12:56:39 +03:00
|
|
|
}
|
|
|
|
}
|
2024-12-04 18:45:08 +03:00
|
|
|
|
|
|
|
case []string:
|
|
|
|
elements := make([]BackgroundElement, 0, len(value))
|
|
|
|
for _, element := range value {
|
|
|
|
if obj := ParseDataText(element); obj != nil {
|
|
|
|
if element := createBackground(obj); element != nil {
|
|
|
|
elements = append(elements, element)
|
|
|
|
} else {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return elements
|
|
|
|
|
|
|
|
case []any:
|
|
|
|
elements := make([]BackgroundElement, 0, len(value))
|
|
|
|
for _, element := range value {
|
|
|
|
switch element := element.(type) {
|
|
|
|
case BackgroundElement:
|
|
|
|
elements = append(elements, element)
|
|
|
|
|
|
|
|
case string:
|
|
|
|
if obj := ParseDataText(element); obj != nil {
|
|
|
|
if element := createBackground(obj); element != nil {
|
|
|
|
elements = append(elements, element)
|
|
|
|
} else {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
default:
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return elements
|
|
|
|
|
2024-11-13 12:56:39 +03:00
|
|
|
}
|
|
|
|
|
2024-11-27 11:32:13 +03:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func setBackgroundProperty(properties Properties, tag PropertyName, value any) []PropertyName {
|
|
|
|
|
|
|
|
background := parseBackgroundValue(value)
|
|
|
|
if background == nil {
|
|
|
|
notCompatibleType(tag, value)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2024-11-13 12:56:39 +03:00
|
|
|
if len(background) > 0 {
|
2024-11-27 11:32:13 +03:00
|
|
|
properties.setRaw(tag, background)
|
|
|
|
} else if properties.getRaw(tag) != nil {
|
|
|
|
properties.setRaw(tag, nil)
|
2024-11-13 12:56:39 +03:00
|
|
|
} else {
|
|
|
|
return []PropertyName{}
|
|
|
|
}
|
|
|
|
|
2024-11-27 11:32:13 +03:00
|
|
|
return []PropertyName{tag}
|
|
|
|
}
|
|
|
|
|
|
|
|
func backgroundCSS(properties Properties, session Session) string {
|
|
|
|
|
|
|
|
if value := properties.getRaw(Background); value != nil {
|
|
|
|
if backgrounds, ok := value.([]BackgroundElement); ok && len(backgrounds) > 0 {
|
|
|
|
buffer := allocStringBuilder()
|
|
|
|
defer freeStringBuilder(buffer)
|
|
|
|
|
|
|
|
for _, background := range backgrounds {
|
|
|
|
if value := background.cssStyle(session); value != "" {
|
|
|
|
if buffer.Len() > 0 {
|
|
|
|
buffer.WriteString(", ")
|
|
|
|
}
|
|
|
|
buffer.WriteString(value)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if buffer.Len() > 0 {
|
|
|
|
backgroundColor, _ := colorProperty(properties, BackgroundColor, session)
|
|
|
|
if backgroundColor != 0 {
|
|
|
|
buffer.WriteRune(' ')
|
|
|
|
buffer.WriteString(backgroundColor.cssString())
|
|
|
|
}
|
|
|
|
return buffer.String()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
func maskCSS(properties Properties, session Session) string {
|
|
|
|
|
|
|
|
if value := properties.getRaw(Mask); value != nil {
|
|
|
|
if backgrounds, ok := value.([]BackgroundElement); ok && len(backgrounds) > 0 {
|
|
|
|
buffer := allocStringBuilder()
|
|
|
|
defer freeStringBuilder(buffer)
|
|
|
|
|
|
|
|
for _, background := range backgrounds {
|
|
|
|
if value := background.cssStyle(session); value != "" {
|
|
|
|
if buffer.Len() > 0 {
|
|
|
|
buffer.WriteString(", ")
|
|
|
|
}
|
|
|
|
buffer.WriteString(value)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return buffer.String()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
func backgroundStyledPropery(view View, subviewID []string, tag PropertyName) []BackgroundElement {
|
|
|
|
var background []BackgroundElement = nil
|
|
|
|
|
|
|
|
if view = getSubview(view, subviewID); view != nil {
|
|
|
|
if value := view.getRaw(tag); value != nil {
|
|
|
|
if backgrounds, ok := value.([]BackgroundElement); ok {
|
|
|
|
background = backgrounds
|
|
|
|
}
|
|
|
|
} else if value := valueFromStyle(view, tag); value != nil {
|
|
|
|
background = parseBackgroundValue(value)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if count := len(background); count > 0 {
|
|
|
|
result := make([]BackgroundElement, count)
|
|
|
|
copy(result, background)
|
|
|
|
return result
|
|
|
|
}
|
|
|
|
|
|
|
|
return []BackgroundElement{}
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetBackground returns the view background.
|
|
|
|
// If the second argument (subviewID) is not specified or it is "" then a value from the first argument (view) is returned.
|
|
|
|
func GetBackground(view View, subviewID ...string) []BackgroundElement {
|
|
|
|
return backgroundStyledPropery(view, subviewID, Background)
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetMask returns the view mask.
|
|
|
|
// If the second argument (subviewID) is not specified or it is "" then a value from the first argument (view) is returned.
|
|
|
|
func GetMask(view View, subviewID ...string) []BackgroundElement {
|
|
|
|
return backgroundStyledPropery(view, subviewID, Mask)
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetBackgroundClip returns a "background-clip" of the subview. Returns one of next values:
|
|
|
|
//
|
|
|
|
// BorderBox (0), PaddingBox (1), ContentBox (2)
|
|
|
|
//
|
|
|
|
// If the second argument (subviewID) is not specified or it is "" then a value from the first argument (view) is returned.
|
|
|
|
func GetBackgroundClip(view View, subviewID ...string) int {
|
|
|
|
return enumStyledProperty(view, subviewID, BackgroundClip, 0, false)
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetBackgroundOrigin returns a "background-origin" of the subview. Returns one of next values:
|
|
|
|
//
|
|
|
|
// BorderBox (0), PaddingBox (1), ContentBox (2)
|
|
|
|
//
|
|
|
|
// If the second argument (subviewID) is not specified or it is "" then a value from the first argument (view) is returned.
|
|
|
|
func GetBackgroundOrigin(view View, subviewID ...string) int {
|
|
|
|
return enumStyledProperty(view, subviewID, BackgroundOrigin, 0, false)
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetMaskClip returns a "mask-clip" of the subview. Returns one of next values:
|
|
|
|
//
|
|
|
|
// BorderBox (0), PaddingBox (1), ContentBox (2)
|
|
|
|
//
|
|
|
|
// If the second argument (subviewID) is not specified or it is "" then a value from the first argument (view) is returned.
|
|
|
|
func GetMaskClip(view View, subviewID ...string) int {
|
|
|
|
return enumStyledProperty(view, subviewID, MaskClip, 0, false)
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetMaskOrigin returns a "mask-origin" of the subview. Returns one of next values:
|
|
|
|
//
|
|
|
|
// BorderBox (0), PaddingBox (1), ContentBox (2)
|
|
|
|
//
|
|
|
|
// If the second argument (subviewID) is not specified or it is "" then a value from the first argument (view) is returned.
|
|
|
|
func GetMaskOrigin(view View, subviewID ...string) int {
|
|
|
|
return enumStyledProperty(view, subviewID, MaskOrigin, 0, false)
|
2024-11-13 12:56:39 +03:00
|
|
|
}
|