package rui import ( "fmt" "strings" ) type ClipShape string const ( InsetClip ClipShape = "inset" CircleClip ClipShape = "circle" EllipseClip ClipShape = "ellipse" PolygonClip ClipShape = "polygon" ) // ClipShapeProperty defines a View clipping area type ClipShapeProperty interface { Properties fmt.Stringer stringWriter // Shape returns the clip shape type Shape() ClipShape cssStyle(session Session) string valid(session Session) bool } type insetClipData struct { dataProperty } type ellipseClipData struct { dataProperty } type circleClipData struct { dataProperty } type polygonClipData struct { dataProperty } // NewClipShapeProperty creates ClipShapeProperty. // // The following properties can be used for shapes: // // InsetClip: // - "top" (Top) - offset (SizeUnit) from the top border of a View; // - "right" (Right) - offset (SizeUnit) from the right border of a View; // - "bottom" (Bottom) - offset (SizeUnit) from the bottom border of a View; // - "left" (Left) - offset (SizeUnit) from the left border of a View; // - "radius" (Radius) - corner radius (RadiusProperty). // // CircleClip: // - "x" (X) - x-axis position (SizeUnit) of the circle clip center; // - "y" (Y) - y-axis position (SizeUnit) of the circle clip center; // - "radius" (Radius) - radius (SizeUnit) of the circle clip center. // // EllipseClip: // - "x" (X) - x-axis position (SizeUnit) of the ellipse clip center; // - "y" (Y) - y-axis position (SizeUnit) of the ellipse clip center; // - "radius-x" (RadiusX) - x-axis radius (SizeUnit) of the ellipse clip center; // - "radius-y" (RadiusY) - y-axis radius (SizeUnit) of the ellipse clip center. // // PolygonClip: // - "points" (Points) - an array ([]SizeUnit) of corner points of the polygon in the following order: x1, y1, x2, y2, …. // // The function will return nil if no properties are specified, unsupported properties are specified, or at least one property has an invalid value. func NewClipShapeProperty(shape ClipShape, params Params) ClipShapeProperty { if len(params) == 0 { ErrorLog("No ClipShapeProperty params") return nil } var result ClipShapeProperty switch shape { case InsetClip: clip := new(insetClipData) clip.init() result = clip case CircleClip: clip := new(circleClipData) clip.init() result = clip case EllipseClip: clip := new(ellipseClipData) clip.init() result = clip case PolygonClip: clip := new(polygonClipData) clip.init() result = clip default: ErrorLog("Unknown ClipShape: " + string(shape)) return nil } for tag, value := range params { if !result.Set(tag, value) { return nil } } return result } // NewInsetClip creates a rectangle View clipping area. // - top - offset from the top border of a View; // - right - offset from the right border of a View; // - bottom - offset from the bottom border of a View; // - left - offset from the left border of a View; // - radius - corner radius, pass nil if you don't need to round corners func NewInsetClip(top, right, bottom, left SizeUnit, radius RadiusProperty) ClipShapeProperty { clip := new(insetClipData) clip.init() clip.setRaw(Top, top) clip.setRaw(Right, right) clip.setRaw(Bottom, bottom) clip.setRaw(Left, left) if radius != nil { clip.setRaw(Radius, radius) } return clip } // NewCircleClip creates a circle View clipping area. // - x - x-axis position of the circle clip center; // - y - y-axis position of the circle clip center; // - radius - radius of the circle clip center. func NewCircleClip(x, y, radius SizeUnit) ClipShapeProperty { clip := new(circleClipData) clip.init() clip.setRaw(X, x) clip.setRaw(Y, y) clip.setRaw(Radius, radius) return clip } // NewEllipseClip creates a ellipse View clipping area. // - x - x-axis position of the ellipse clip center; // - y - y-axis position of the ellipse clip center; // - rx - x-axis radius of the ellipse clip center; // - ry - y-axis radius of the ellipse clip center. func NewEllipseClip(x, y, rx, ry SizeUnit) ClipShapeProperty { clip := new(ellipseClipData) clip.init() clip.setRaw(X, x) clip.setRaw(Y, y) clip.setRaw(RadiusX, rx) clip.setRaw(RadiusY, ry) return clip } // NewPolygonClip creates a polygon View clipping area. // - points - an array of corner points of the polygon in the following order: x1, y1, x2, y2, … // // The elements of the function argument can be or text constants, // or the text representation of SizeUnit, or elements of SizeUnit type. func NewPolygonClip(points []any) ClipShapeProperty { clip := new(polygonClipData) clip.init() if polygonClipDataSet(clip, Points, points) != nil { return clip } return nil } // NewPolygonPointsClip creates a polygon View clipping area. // - points - an array of corner points of the polygon in the following order: x1, y1, x2, y2, … func NewPolygonPointsClip(points []SizeUnit) ClipShapeProperty { clip := new(polygonClipData) clip.init() if polygonClipDataSet(clip, Points, points) != nil { return clip } return nil } func (clip *insetClipData) init() { clip.dataProperty.init() clip.set = insetClipDataSet clip.supportedProperties = []PropertyName{ Top, Right, Bottom, Left, Radius, RadiusX, RadiusY, RadiusTopLeft, RadiusTopLeftX, RadiusTopLeftY, RadiusTopRight, RadiusTopRightX, RadiusTopRightY, RadiusBottomLeft, RadiusBottomLeftX, RadiusBottomLeftY, RadiusBottomRight, RadiusBottomRightX, RadiusBottomRightY, } } func (clip *insetClipData) Shape() ClipShape { return InsetClip } func insetClipDataSet(properties Properties, tag PropertyName, value any) []PropertyName { switch tag { case Top, Right, Bottom, Left: return setSizeProperty(properties, tag, value) case Radius: return setRadiusProperty(properties, value) case RadiusX, RadiusY, RadiusTopLeft, RadiusTopLeftX, RadiusTopLeftY, RadiusTopRight, RadiusTopRightX, RadiusTopRightY, RadiusBottomLeft, RadiusBottomLeftX, RadiusBottomLeftY, RadiusBottomRight, RadiusBottomRightX, RadiusBottomRightY: if setRadiusPropertyElement(properties, tag, value) { return []PropertyName{tag, Radius} } return nil } ErrorLogF(`"%s" property is not supported by the inset clip shape`, tag) return nil } func (clip *insetClipData) String() string { return runStringWriter(clip) } func (clip *insetClipData) writeString(buffer *strings.Builder, indent string) { buffer.WriteString("inset { ") comma := false for _, tag := range []PropertyName{Top, Right, Bottom, Left, Radius} { if value, ok := clip.properties[tag]; ok { if comma { buffer.WriteString(", ") } buffer.WriteString(string(tag)) buffer.WriteString(" = ") writePropertyValue(buffer, tag, value, indent) comma = true } } buffer.WriteString(" }") } func (clip *insetClipData) cssStyle(session Session) string { buffer := allocStringBuilder() defer freeStringBuilder(buffer) leadText := "inset(" for _, tag := range []PropertyName{Top, Right, Bottom, Left} { value, _ := sizeProperty(clip, tag, session) buffer.WriteString(leadText) buffer.WriteString(value.cssString("0px", session)) leadText = " " } if radius := getRadiusProperty(clip); radius != nil { buffer.WriteString(" round ") buffer.WriteString(radius.BoxRadius(session).cssString(session)) } buffer.WriteRune(')') return buffer.String() } func (clip *insetClipData) valid(session Session) bool { for _, tag := range []PropertyName{Top, Right, Bottom, Left, Radius, RadiusX, RadiusY} { if value, ok := sizeProperty(clip, tag, session); ok && value.Type != Auto && value.Value != 0 { return true } } return false } func (clip *circleClipData) init() { clip.dataProperty.init() clip.set = circleClipDataSet clip.supportedProperties = []PropertyName{X, Y, Radius} } func (clip *circleClipData) Shape() ClipShape { return CircleClip } func circleClipDataSet(properties Properties, tag PropertyName, value any) []PropertyName { switch tag { case X, Y, Radius: return setSizeProperty(properties, tag, value) } ErrorLogF(`"%s" property is not supported by the circle clip shape`, tag) return nil } func (clip *circleClipData) String() string { return runStringWriter(clip) } func (clip *circleClipData) writeString(buffer *strings.Builder, indent string) { buffer.WriteString("circle { ") comma := false for _, tag := range []PropertyName{Radius, X, Y} { if value, ok := clip.properties[tag]; ok { if comma { buffer.WriteString(", ") } buffer.WriteString(string(tag)) buffer.WriteString(" = ") writePropertyValue(buffer, tag, value, indent) comma = true } } buffer.WriteString(" }") } func (clip *circleClipData) cssStyle(session Session) string { buffer := allocStringBuilder() defer freeStringBuilder(buffer) buffer.WriteString("circle(") r, _ := sizeProperty(clip, Radius, session) buffer.WriteString(r.cssString("50%", session)) buffer.WriteString(" at ") x, _ := sizeProperty(clip, X, session) buffer.WriteString(x.cssString("50%", session)) buffer.WriteRune(' ') y, _ := sizeProperty(clip, Y, session) buffer.WriteString(y.cssString("50%", session)) buffer.WriteRune(')') return buffer.String() } func (clip *circleClipData) valid(session Session) bool { if value, ok := sizeProperty(clip, Radius, session); ok && value.Value == 0 { return false } return true } func (clip *ellipseClipData) init() { clip.dataProperty.init() clip.set = ellipseClipDataSet clip.supportedProperties = []PropertyName{X, Y, Radius, RadiusX, RadiusY} } func (clip *ellipseClipData) Shape() ClipShape { return EllipseClip } func ellipseClipDataSet(properties Properties, tag PropertyName, value any) []PropertyName { switch tag { case X, Y, RadiusX, RadiusY: return setSizeProperty(properties, tag, value) case Radius: if result := setSizeProperty(properties, RadiusX, value); result != nil { properties.setRaw(RadiusY, properties.getRaw(RadiusX)) return append(result, RadiusY) } return nil } ErrorLogF(`"%s" property is not supported by the ellipse clip shape`, tag) return nil } func (clip *ellipseClipData) String() string { return runStringWriter(clip) } func (clip *ellipseClipData) writeString(buffer *strings.Builder, indent string) { buffer.WriteString("ellipse { ") comma := false for _, tag := range []PropertyName{RadiusX, RadiusY, X, Y} { if value, ok := clip.properties[tag]; ok { if comma { buffer.WriteString(", ") } buffer.WriteString(string(tag)) buffer.WriteString(" = ") writePropertyValue(buffer, tag, value, indent) comma = true } } buffer.WriteString(" }") } func (clip *ellipseClipData) cssStyle(session Session) string { buffer := allocStringBuilder() defer freeStringBuilder(buffer) rx, _ := sizeProperty(clip, RadiusX, session) ry, _ := sizeProperty(clip, RadiusX, session) buffer.WriteString("ellipse(") buffer.WriteString(rx.cssString("50%", session)) buffer.WriteRune(' ') buffer.WriteString(ry.cssString("50%", session)) buffer.WriteString(" at ") x, _ := sizeProperty(clip, X, session) buffer.WriteString(x.cssString("50%", session)) buffer.WriteRune(' ') y, _ := sizeProperty(clip, Y, session) buffer.WriteString(y.cssString("50%", session)) buffer.WriteRune(')') return buffer.String() } func (clip *ellipseClipData) valid(session Session) bool { rx, _ := sizeProperty(clip, RadiusX, session) ry, _ := sizeProperty(clip, RadiusY, session) return rx.Value != 0 && ry.Value != 0 } func (clip *polygonClipData) init() { clip.dataProperty.init() clip.set = polygonClipDataSet clip.supportedProperties = []PropertyName{Points} } func (clip *polygonClipData) Shape() ClipShape { return PolygonClip } func polygonClipDataSet(properties Properties, tag PropertyName, value any) []PropertyName { if Points == tag { switch value := value.(type) { case []any: points := make([]any, len(value)) for i, val := range value { switch val := val.(type) { case string: if isConstantName(val) { points[i] = val } else if size, ok := StringToSizeUnit(val); ok { points[i] = size } else { notCompatibleType(tag, val) return nil } case SizeUnit: points[i] = val default: notCompatibleType(tag, val) points[i] = AutoSize() return nil } } properties.setRaw(Points, points) return []PropertyName{tag} case []SizeUnit: points := make([]any, len(value)) for i, point := range value { points[i] = point } properties.setRaw(Points, points) return []PropertyName{tag} case string: values := strings.Split(value, ",") points := make([]any, len(values)) for i, val := range values { val = strings.Trim(val, " \t\n\r") if isConstantName(val) { points[i] = val } else if size, ok := StringToSizeUnit(val); ok { points[i] = size } else { notCompatibleType(tag, val) return nil } } properties.setRaw(Points, points) return []PropertyName{tag} } } return nil } func (clip *polygonClipData) String() string { return runStringWriter(clip) } func (clip *polygonClipData) points() []any { if value := clip.getRaw(Points); value != nil { if points, ok := value.([]any); ok { return points } } return nil } func (clip *polygonClipData) writeString(buffer *strings.Builder, indent string) { buffer.WriteString("inset { ") if points := clip.points(); points != nil { buffer.WriteString(string(Points)) buffer.WriteString(` = "`) for i, value := range points { if i > 0 { buffer.WriteString(", ") } writePropertyValue(buffer, "", value, indent) } buffer.WriteString(`" `) } buffer.WriteRune('}') } func (clip *polygonClipData) cssStyle(session Session) string { points := clip.points() count := len(points) if count < 2 { return "" } buffer := allocStringBuilder() defer freeStringBuilder(buffer) writePoint := func(value any) { switch value := value.(type) { case string: if val, ok := session.resolveConstants(value); ok { if size, ok := StringToSizeUnit(val); ok { buffer.WriteString(size.cssString("0px", session)) return } } case SizeUnit: buffer.WriteString(value.cssString("0px", session)) return } buffer.WriteString("0px") } leadText := "polygon(" for i := 1; i < count; i += 2 { buffer.WriteString(leadText) writePoint(points[i-1]) buffer.WriteRune(' ') writePoint(points[i]) leadText = ", " } buffer.WriteRune(')') return buffer.String() } func (clip *polygonClipData) valid(session Session) bool { return len(clip.points()) > 0 } func parseClipShapeProperty(obj DataObject) ClipShapeProperty { switch obj.Tag() { case "inset": clip := new(insetClipData) clip.init() for _, tag := range []PropertyName{Top, Right, Bottom, Left, Radius, RadiusX, RadiusY} { if value, ok := obj.PropertyValue(string(tag)); ok { insetClipDataSet(clip, tag, value) } } return clip case "circle": clip := new(circleClipData) clip.init() for _, tag := range []PropertyName{X, Y, Radius} { if value, ok := obj.PropertyValue(string(tag)); ok { circleClipDataSet(clip, tag, value) } } return clip case "ellipse": clip := new(ellipseClipData) clip.init() for _, tag := range []PropertyName{X, Y, RadiusX, RadiusY} { if value, ok := obj.PropertyValue(string(tag)); ok { ellipseClipDataSet(clip, tag, value) } } return clip case "polygon": clip := new(polygonClipData) clip.init() if value, ok := obj.PropertyValue(string(Points)); ok { polygonClipDataSet(clip, Points, value) } return clip } return nil } func setClipShapePropertyProperty(properties Properties, tag PropertyName, value any) []PropertyName { switch value := value.(type) { case ClipShapeProperty: properties.setRaw(tag, value) return []PropertyName{tag} case string: if isConstantName(value) { properties.setRaw(tag, value) return []PropertyName{tag} } if obj := NewDataObject(value); obj == nil { if clip := parseClipShapeProperty(obj); clip != nil { properties.setRaw(tag, clip) return []PropertyName{tag} } } case DataObject: if clip := parseClipShapeProperty(value); clip != nil { properties.setRaw(tag, clip) return []PropertyName{tag} } case DataValue: if value.IsObject() { if clip := parseClipShapeProperty(value.Object()); clip != nil { properties.setRaw(tag, clip) return []PropertyName{tag} } } } notCompatibleType(tag, value) return nil } func getClipShapeProperty(prop Properties, tag PropertyName, session Session) ClipShapeProperty { if value := prop.getRaw(tag); value != nil { switch value := value.(type) { case ClipShapeProperty: return value case string: if text, ok := session.resolveConstants(value); ok { if obj := NewDataObject(text); obj == nil { return parseClipShapeProperty(obj) } } } } return nil } // GetClip returns a View clipping area. // If the second argument (subviewID) is not specified or it is "" then a top position of the first argument (view) is returned func GetClip(view View, subviewID ...string) ClipShapeProperty { if view = getSubview(view, subviewID); view != nil { return getClipShapeProperty(view, Clip, view.Session()) } return nil } // GetShapeOutside returns a shape around which adjacent inline content. // If the second argument (subviewID) is not specified or it is "" then a top position of the first argument (view) is returned func GetShapeOutside(view View, subviewID ...string) ClipShapeProperty { if view = getSubview(view, subviewID); view != nil { return getClipShapeProperty(view, ShapeOutside, view.Session()) } return nil }