Changed BackgroundGradientPoint struct

This commit is contained in:
Alexei Anoshenko 2022-04-30 18:02:01 +03:00
parent 86271a3c6e
commit 26abd1f632
2 changed files with 113 additions and 197 deletions

View File

@ -243,84 +243,6 @@ func (gradient *backgroundConicGradient) setGradient(value interface{}) bool {
ErrorLogF(`Ivalid conic gradient: "%s"`, value) ErrorLogF(`Ivalid conic gradient: "%s"`, value)
return false return false
case []interface{}:
count := len(value)
if count < 2 {
ErrorLog("The gradient must contain at least 2 points")
return false
}
vector := make([]BackgroundGradientAngle, len(value))
for i, point := range value {
if point == nil {
ErrorLogF("Ivalid %d element of the conic gradient: nil", i)
return false
}
switch element := point.(type) {
case string:
if data, ok := gradient.stringToGradientPoint(element); ok {
vector[i] = data
} else {
ErrorLogF("Ivalid %d element of the conic gradient: %s", i, element)
return false
}
case Color:
vector[i].Color = element
case BackgroundGradientAngle:
if element.Color == nil {
ErrorLogF("Ivalid %d element of the conic gradient: Color is nil", i)
return false
}
switch color := element.Color.(type) {
case string:
if color == "" {
ErrorLogF("Ivalid %d element of the conic gradient: empty Color text", i)
return false
}
if color[0] != '@' {
if clr, ok := StringToColor(color); ok {
element.Color = clr
} else {
ErrorLogF("Ivalid %d element of the conic gradient: invalid Color text", i)
return false
}
}
case Color:
// do nothing
default:
ErrorLogF("Ivalid %d element of the conic gradient: unsupported Color type", i)
return false
}
if element.Angle != nil {
switch point := element.Angle.(type) {
case string:
if angle, ok := gradient.stringToAngle(point); ok {
element.Angle = angle
} else {
ErrorLogF("Ivalid %d element of the conic gradient: invalid Point text", i)
return false
}
case AngleUnit:
// do nothing
default:
ErrorLogF("Ivalid %d element of the conic gradient: unsupported Point type", i)
return false
}
}
vector[i] = element
}
}
gradient.properties[Gradient] = vector
return true
case []BackgroundGradientAngle: case []BackgroundGradientAngle:
count := len(value) count := len(value)
if count < 2 { if count < 2 {

View File

@ -48,10 +48,12 @@ const (
// BackgroundGradientPoint define point on gradient straight line // BackgroundGradientPoint define point on gradient straight line
type BackgroundGradientPoint struct { type BackgroundGradientPoint struct {
// Pos - the distance from the start of the gradient straight line // Color - the color of the point. Must not be nil.
Pos SizeUnit // Can take a value of Color type or string (color constant or textual description of the color)
// Color - the color of the point Color interface{}
Color Color // Pos - the distance from the start of the gradient straight line. Optional (may be nil).
// Can take a value of SizeUnit type or string (angle constant or textual description of the SizeUnit)
Pos interface{}
} }
type backgroundGradient struct { type backgroundGradient struct {
@ -86,6 +88,24 @@ func NewBackgroundRadialGradient(params Params) BackgroundElement {
return result return result
} }
func (gradient *backgroundGradient) parseGradientText(value string) []BackgroundGradientPoint {
elements := strings.Split(value, ",")
count := len(elements)
if count < 2 {
ErrorLog("The gradient must contain at least 2 points")
return nil
}
points := make([]BackgroundGradientPoint, count)
for i, element := range elements {
if !points[i].setValue(element) {
ErrorLogF(`Ivalid %d element of the conic gradient: "%s"`, i, element)
return nil
}
}
return points
}
func (gradient *backgroundGradient) Set(tag string, value interface{}) bool { func (gradient *backgroundGradient) Set(tag string, value interface{}) bool {
switch tag = strings.ToLower(tag); tag { switch tag = strings.ToLower(tag); tag {
@ -96,29 +116,13 @@ func (gradient *backgroundGradient) Set(tag string, value interface{}) bool {
switch value := value.(type) { switch value := value.(type) {
case string: case string:
if value != "" { if value != "" {
elements := strings.Split(value, `,`) if strings.Contains(value, " ") || strings.Contains(value, ",") {
if count := len(elements); count > 1 { if points := gradient.parseGradientText(value); len(points) >= 2 {
points := make([]interface{}, count)
for i, element := range elements {
if strings.Contains(element, "@") {
points[i] = element
} else {
var point BackgroundGradientPoint
if point.setValue(element) {
points[i] = point
} else {
ErrorLogF("Invalid gradient element #%d: %s", i, element)
return false
}
}
}
gradient.properties[Gradient] = points gradient.properties[Gradient] = points
return true return true
} }
} else if value[0] == '@' {
text := strings.Trim(value, " \n\r\t") gradient.properties[Gradient] = value
if text[0] == '@' {
gradient.properties[Gradient] = text
return true return true
} }
} }
@ -135,7 +139,6 @@ func (gradient *backgroundGradient) Set(tag string, value interface{}) bool {
points := make([]BackgroundGradientPoint, count) points := make([]BackgroundGradientPoint, count)
for i, color := range value { for i, color := range value {
points[i].Color = color points[i].Color = color
points[i].Pos = AutoSize()
} }
gradient.properties[Gradient] = points gradient.properties[Gradient] = points
return true return true
@ -152,74 +155,72 @@ func (gradient *backgroundGradient) Set(tag string, value interface{}) bool {
gradient.properties[Gradient] = points gradient.properties[Gradient] = points
return true return true
} }
case []interface{}:
if count := len(value); count > 1 {
points := make([]interface{}, count)
for i, element := range value {
switch element := element.(type) {
case string:
if strings.Contains(element, "@") {
points[i] = element
} else {
var point BackgroundGradientPoint
if !point.setValue(element) {
ErrorLogF("Invalid gradient element #%d: %s", i, element)
return false
}
points[i] = point
} }
case BackgroundGradientPoint:
points[i] = element
case GradientPoint:
points[i] = BackgroundGradientPoint{Color: element.Color, Pos: Percent(element.Offset * 100)}
case Color:
points[i] = BackgroundGradientPoint{Color: element, Pos: AutoSize()}
default:
ErrorLogF("Invalid gradient element #%d: %v", i, element)
return false
}
}
gradient.properties[Gradient] = points
return true
}
default:
ErrorLogF("Invalid gradient %v", value) ErrorLogF("Invalid gradient %v", value)
return false return false
} }
}
return gradient.backgroundElement.Set(tag, value) return gradient.backgroundElement.Set(tag, value)
} }
func (point *BackgroundGradientPoint) setValue(value string) bool { func (point *BackgroundGradientPoint) setValue(text string) bool {
var ok bool text = strings.Trim(text, " ")
switch elements := strings.Split(value, `:`); len(elements) { colorText := text
case 2: pointText := ""
if point.Color, ok = StringToColor(elements[0]); !ok {
return false if index := strings.Index(text, " "); index > 0 {
colorText = text[:index]
pointText = strings.Trim(text[index+1:], " ")
} }
if point.Pos, ok = StringToSizeUnit(elements[1]); !ok {
if colorText == "" {
return false return false
} }
case 1: if colorText[0] == '@' {
if point.Color, ok = StringToColor(elements[0]); !ok { point.Color = colorText
return false } else if color, ok := StringToColor(colorText); ok {
} point.Color = color
point.Pos = AutoSize() } else {
default:
return false return false
} }
if pointText == "" {
point.Pos = nil
} else if pointText[0] == '@' {
point.Pos = pointText
} else if pos, ok := StringToSizeUnit(pointText); ok {
point.Pos = pos
} else {
return false return false
}
return true
}
func (point *BackgroundGradientPoint) color(session Session) (Color, bool) {
if point.Color != nil {
switch color := point.Color.(type) {
case string:
if color != "" {
if color[0] == '@' {
if clr, ok := session.Color(color[1:]); ok {
return clr, true
}
} else {
if clr, ok := StringToColor(color); ok {
return clr, true
}
}
}
case Color:
return color, true
}
}
return 0, false
} }
func (gradient *backgroundGradient) writeGradient(session Session, buffer *strings.Builder) bool { func (gradient *backgroundGradient) writeGradient(session Session, buffer *strings.Builder) bool {
@ -229,46 +230,18 @@ func (gradient *backgroundGradient) writeGradient(session Session, buffer *strin
return false return false
} }
points := []BackgroundGradientPoint{} var points []BackgroundGradientPoint = nil
switch value := value.(type) { switch value := value.(type) {
case string: case string:
if text, ok := session.resolveConstants(value); ok && text != "" { if value != "" && value[0] == '@' {
elements := strings.Split(text, `,`) if text, ok := session.Constant(value[1:]); ok {
points := make([]BackgroundGradientPoint, len(elements)) points = gradient.parseGradientText(text)
for i, element := range elements {
if !points[i].setValue(element) {
ErrorLogF(`Invalid gradient point #%d: "%s"`, i, element)
return false
} }
} }
} else {
ErrorLog(`Invalid gradient: ` + value)
return false
}
case []BackgroundGradientPoint: case []BackgroundGradientPoint:
points = value points = value
case []interface{}:
points = make([]BackgroundGradientPoint, len(value))
for i, element := range value {
switch element := element.(type) {
case string:
if text, ok := session.resolveConstants(element); ok && text != "" {
if !points[i].setValue(text) {
ErrorLogF(`Invalid gradient point #%d: "%s"`, i, text)
return false
}
} else {
ErrorLogF(`Invalid gradient point #%d: "%s"`, i, text)
return false
}
case BackgroundGradientPoint:
points[i] = element
}
}
} }
if len(points) > 0 { if len(points) > 0 {
@ -277,10 +250,31 @@ func (gradient *backgroundGradient) writeGradient(session Session, buffer *strin
buffer.WriteString(`, `) buffer.WriteString(`, `)
} }
buffer.WriteString(point.Color.cssString()) if color, ok := point.color(session); ok {
if point.Pos.Type != Auto { buffer.WriteString(color.cssString())
} else {
return false
}
if point.Pos != nil {
switch value := point.Pos.(type) {
case string:
if value != "" {
if value[0] == '@' {
value, _ = session.Constant(value[1:])
}
if pos, ok := StringToSizeUnit(value); ok && pos.Type != Auto {
buffer.WriteRune(' ') buffer.WriteRune(' ')
buffer.WriteString(point.Pos.cssString("")) buffer.WriteString(pos.cssString(""))
}
}
case SizeUnit:
if value.Type != Auto {
buffer.WriteRune(' ')
buffer.WriteString(value.cssString(""))
}
}
} }
} }
return true return true