2021-09-07 17:36:50 +03:00
|
|
|
|
package rui
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"fmt"
|
|
|
|
|
"strings"
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
const (
|
2022-07-08 13:16:42 +03:00
|
|
|
|
// LoadedEvent is the constant for the "loaded-event" property tag.
|
|
|
|
|
// The "loaded-event" event occurs event occurs when the image has been loaded.
|
|
|
|
|
LoadedEvent = "loaded-event"
|
|
|
|
|
// ErrorEvent is the constant for the "error-event" property tag.
|
|
|
|
|
// The "error-event" event occurs event occurs when the image loading failed.
|
|
|
|
|
ErrorEvent = "error-event"
|
|
|
|
|
|
2021-09-07 17:36:50 +03:00
|
|
|
|
// NoneFit - value of the "object-fit" property of an ImageView. The replaced content is not resized
|
|
|
|
|
NoneFit = 0
|
|
|
|
|
// ContainFit - value of the "object-fit" property of an ImageView. The replaced content
|
|
|
|
|
// is scaled to maintain its aspect ratio while fitting within the element’s content box.
|
|
|
|
|
// The entire object is made to fill the box, while preserving its aspect ratio, so the object
|
|
|
|
|
// will be "letterboxed" if its aspect ratio does not match the aspect ratio of the box.
|
|
|
|
|
ContainFit = 1
|
|
|
|
|
// CoverFit - value of the "object-fit" property of an ImageView. The replaced content
|
|
|
|
|
// is sized to maintain its aspect ratio while filling the element’s entire content box.
|
|
|
|
|
// If the object's aspect ratio does not match the aspect ratio of its box, then the object will be clipped to fit.
|
|
|
|
|
CoverFit = 2
|
|
|
|
|
// FillFit - value of the "object-fit" property of an ImageView. The replaced content is sized
|
|
|
|
|
// to fill the element’s content box. The entire object will completely fill the box.
|
|
|
|
|
// If the object's aspect ratio does not match the aspect ratio of its box, then the object will be stretched to fit.
|
|
|
|
|
FillFit = 3
|
|
|
|
|
// ScaleDownFit - value of the "object-fit" property of an ImageView. The content is sized as
|
|
|
|
|
// if NoneFit or ContainFit were specified, whichever would result in a smaller concrete object size.
|
|
|
|
|
ScaleDownFit = 4
|
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
// ImageView - image View
|
|
|
|
|
type ImageView interface {
|
|
|
|
|
View
|
2022-07-08 13:16:42 +03:00
|
|
|
|
// NaturalSize returns the intrinsic, density-corrected size (width, height) of the image in pixels.
|
|
|
|
|
// If the image hasn't been loaded yet or an load error has occurred, then (0, 0) is returned.
|
|
|
|
|
NaturalSize() (float64, float64)
|
|
|
|
|
// CurrentSource() return the full URL of the image currently visible in the ImageView.
|
|
|
|
|
// If the image hasn't been loaded yet or an load error has occurred, then "" is returned.
|
|
|
|
|
CurrentSource() string
|
2021-09-07 17:36:50 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
type imageViewData struct {
|
|
|
|
|
viewData
|
2022-07-08 13:16:42 +03:00
|
|
|
|
naturalWidth float64
|
|
|
|
|
naturalHeight float64
|
|
|
|
|
currentSrc string
|
2021-09-07 17:36:50 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// NewImageView create new ImageView object and return it
|
|
|
|
|
func NewImageView(session Session, params Params) ImageView {
|
|
|
|
|
view := new(imageViewData)
|
2022-09-01 11:04:50 +03:00
|
|
|
|
view.init(session)
|
2021-09-07 17:36:50 +03:00
|
|
|
|
setInitParams(view, params)
|
|
|
|
|
return view
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func newImageView(session Session) View {
|
|
|
|
|
return NewImageView(session, nil)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Init initialize fields of imageView by default values
|
2022-09-01 11:04:50 +03:00
|
|
|
|
func (imageView *imageViewData) init(session Session) {
|
|
|
|
|
imageView.viewData.init(session)
|
2021-09-07 17:36:50 +03:00
|
|
|
|
imageView.tag = "ImageView"
|
|
|
|
|
//imageView.systemClass = "ruiImageView"
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
2022-05-22 12:54:02 +03:00
|
|
|
|
func (imageView *imageViewData) String() string {
|
|
|
|
|
return getViewString(imageView)
|
|
|
|
|
}
|
|
|
|
|
|
2021-09-07 17:36:50 +03:00
|
|
|
|
func (imageView *imageViewData) normalizeTag(tag string) string {
|
|
|
|
|
tag = strings.ToLower(tag)
|
|
|
|
|
switch tag {
|
|
|
|
|
case "source":
|
|
|
|
|
tag = Source
|
|
|
|
|
|
|
|
|
|
case VerticalAlign:
|
|
|
|
|
tag = ImageVerticalAlign
|
|
|
|
|
|
|
|
|
|
case HorizontalAlign:
|
|
|
|
|
tag = ImageHorizontalAlign
|
|
|
|
|
|
2021-11-20 11:15:28 +03:00
|
|
|
|
case altTag:
|
2021-09-07 17:36:50 +03:00
|
|
|
|
tag = AltText
|
|
|
|
|
}
|
|
|
|
|
return tag
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (imageView *imageViewData) Remove(tag string) {
|
|
|
|
|
imageView.remove(imageView.normalizeTag(tag))
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (imageView *imageViewData) remove(tag string) {
|
|
|
|
|
imageView.viewData.remove(tag)
|
2021-11-20 11:15:28 +03:00
|
|
|
|
if imageView.created {
|
|
|
|
|
switch tag {
|
|
|
|
|
case Source:
|
|
|
|
|
updateProperty(imageView.htmlID(), "src", "", imageView.session)
|
|
|
|
|
removeProperty(imageView.htmlID(), "srcset", imageView.session)
|
2021-09-07 17:36:50 +03:00
|
|
|
|
|
2021-11-20 11:15:28 +03:00
|
|
|
|
case AltText:
|
|
|
|
|
updateInnerHTML(imageView.htmlID(), imageView.session)
|
2021-09-07 17:36:50 +03:00
|
|
|
|
|
2021-11-20 11:15:28 +03:00
|
|
|
|
case ImageVerticalAlign, ImageHorizontalAlign:
|
|
|
|
|
updateCSSStyle(imageView.htmlID(), imageView.session)
|
|
|
|
|
}
|
2021-09-07 17:36:50 +03:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-07-26 18:36:00 +03:00
|
|
|
|
func (imageView *imageViewData) Set(tag string, value any) bool {
|
2021-09-07 17:36:50 +03:00
|
|
|
|
return imageView.set(imageView.normalizeTag(tag), value)
|
|
|
|
|
}
|
|
|
|
|
|
2022-07-26 18:36:00 +03:00
|
|
|
|
func (imageView *imageViewData) set(tag string, value any) bool {
|
2021-09-07 17:36:50 +03:00
|
|
|
|
if value == nil {
|
|
|
|
|
imageView.remove(tag)
|
|
|
|
|
return true
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
switch tag {
|
|
|
|
|
case Source:
|
|
|
|
|
if text, ok := value.(string); ok {
|
|
|
|
|
imageView.properties[Source] = text
|
2021-11-20 11:15:28 +03:00
|
|
|
|
if imageView.created {
|
2022-04-23 18:13:35 +03:00
|
|
|
|
src := text
|
|
|
|
|
if src != "" && src[0] == '@' {
|
|
|
|
|
src, _ = imageProperty(imageView, Source, imageView.session)
|
|
|
|
|
}
|
|
|
|
|
updateProperty(imageView.htmlID(), "src", src, imageView.session)
|
|
|
|
|
if srcset := imageView.srcSet(src); srcset != "" {
|
2021-11-20 11:15:28 +03:00
|
|
|
|
updateProperty(imageView.htmlID(), "srcset", srcset, imageView.session)
|
|
|
|
|
} else {
|
|
|
|
|
removeProperty(imageView.htmlID(), "srcset", imageView.session)
|
|
|
|
|
}
|
2021-09-07 17:36:50 +03:00
|
|
|
|
}
|
2021-11-20 11:15:28 +03:00
|
|
|
|
imageView.propertyChangedEvent(Source)
|
2021-09-07 17:36:50 +03:00
|
|
|
|
return true
|
|
|
|
|
}
|
2021-11-20 11:15:28 +03:00
|
|
|
|
notCompatibleType(Source, value)
|
2021-09-07 17:36:50 +03:00
|
|
|
|
|
|
|
|
|
case AltText:
|
|
|
|
|
if text, ok := value.(string); ok {
|
|
|
|
|
imageView.properties[AltText] = text
|
2021-11-20 11:15:28 +03:00
|
|
|
|
if imageView.created {
|
|
|
|
|
updateInnerHTML(imageView.htmlID(), imageView.session)
|
|
|
|
|
}
|
|
|
|
|
imageView.propertyChangedEvent(Source)
|
2021-09-07 17:36:50 +03:00
|
|
|
|
return true
|
|
|
|
|
}
|
|
|
|
|
notCompatibleType(tag, value)
|
|
|
|
|
|
2022-07-08 13:16:42 +03:00
|
|
|
|
case LoadedEvent, ErrorEvent:
|
2022-07-27 20:31:57 +03:00
|
|
|
|
if listeners, ok := valueToNoParamListeners[ImageView](value); ok {
|
|
|
|
|
if listeners == nil {
|
|
|
|
|
delete(imageView.properties, tag)
|
|
|
|
|
} else {
|
|
|
|
|
imageView.properties[tag] = listeners
|
|
|
|
|
}
|
2022-07-08 13:16:42 +03:00
|
|
|
|
return true
|
|
|
|
|
}
|
|
|
|
|
|
2021-09-07 17:36:50 +03:00
|
|
|
|
default:
|
|
|
|
|
if imageView.viewData.set(tag, value) {
|
2021-11-20 11:15:28 +03:00
|
|
|
|
if imageView.created {
|
|
|
|
|
switch tag {
|
|
|
|
|
case ImageVerticalAlign, ImageHorizontalAlign:
|
|
|
|
|
updateCSSStyle(imageView.htmlID(), imageView.session)
|
|
|
|
|
}
|
2021-09-07 17:36:50 +03:00
|
|
|
|
}
|
|
|
|
|
return true
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return false
|
|
|
|
|
}
|
|
|
|
|
|
2022-07-26 18:36:00 +03:00
|
|
|
|
func (imageView *imageViewData) Get(tag string) any {
|
2021-09-07 17:36:50 +03:00
|
|
|
|
return imageView.viewData.get(imageView.normalizeTag(tag))
|
|
|
|
|
}
|
|
|
|
|
|
2022-07-08 13:16:42 +03:00
|
|
|
|
func (imageView *imageViewData) imageListeners(tag string) []func(ImageView) {
|
|
|
|
|
if value := imageView.getRaw(tag); value != nil {
|
|
|
|
|
if listeners, ok := value.([]func(ImageView)); ok {
|
|
|
|
|
return listeners
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return []func(ImageView){}
|
|
|
|
|
}
|
|
|
|
|
|
2021-09-07 17:36:50 +03:00
|
|
|
|
func (imageView *imageViewData) srcSet(path string) string {
|
|
|
|
|
if srcset, ok := resources.imageSrcSets[path]; ok {
|
|
|
|
|
buffer := allocStringBuilder()
|
|
|
|
|
defer freeStringBuilder(buffer)
|
|
|
|
|
for i, src := range srcset {
|
|
|
|
|
if i > 0 {
|
|
|
|
|
buffer.WriteString(", ")
|
|
|
|
|
}
|
|
|
|
|
buffer.WriteString(src.path)
|
|
|
|
|
buffer.WriteString(fmt.Sprintf(" %gx", src.scale))
|
|
|
|
|
}
|
|
|
|
|
return buffer.String()
|
|
|
|
|
}
|
|
|
|
|
return ""
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (imageView *imageViewData) htmlTag() string {
|
|
|
|
|
return "img"
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
func (imageView *imageViewData) closeHTMLTag() bool {
|
|
|
|
|
return false
|
|
|
|
|
}
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
func (imageView *imageViewData) htmlProperties(self View, buffer *strings.Builder) {
|
|
|
|
|
imageView.viewData.htmlProperties(self, buffer)
|
2022-04-23 18:13:35 +03:00
|
|
|
|
|
|
|
|
|
if imageResource, ok := imageProperty(imageView, Source, imageView.Session()); ok && imageResource != "" {
|
|
|
|
|
if imageResource[0] == '@' {
|
|
|
|
|
if image, ok := imageView.Session().ImageConstant(imageResource[1:]); ok {
|
|
|
|
|
imageResource = image
|
|
|
|
|
} else {
|
|
|
|
|
imageResource = ""
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if imageResource != "" {
|
|
|
|
|
buffer.WriteString(` src="`)
|
|
|
|
|
buffer.WriteString(imageResource)
|
2021-09-07 17:36:50 +03:00
|
|
|
|
buffer.WriteString(`"`)
|
2022-04-23 18:13:35 +03:00
|
|
|
|
if srcset := imageView.srcSet(imageResource); srcset != "" {
|
|
|
|
|
buffer.WriteString(` srcset="`)
|
|
|
|
|
buffer.WriteString(srcset)
|
|
|
|
|
buffer.WriteString(`"`)
|
|
|
|
|
}
|
2021-09-07 17:36:50 +03:00
|
|
|
|
}
|
|
|
|
|
}
|
2022-04-17 11:21:01 +03:00
|
|
|
|
|
2022-08-31 22:17:46 +03:00
|
|
|
|
if text := GetImageViewAltText(imageView); text != "" {
|
2022-04-17 11:21:01 +03:00
|
|
|
|
buffer.WriteString(` alt="`)
|
|
|
|
|
buffer.WriteString(textToJS(text))
|
|
|
|
|
buffer.WriteString(`"`)
|
|
|
|
|
}
|
2022-07-08 13:16:42 +03:00
|
|
|
|
|
|
|
|
|
buffer.WriteString(` onload="imageLoaded(this, event)"`)
|
|
|
|
|
|
|
|
|
|
if len(imageView.imageListeners(ErrorEvent)) > 0 {
|
|
|
|
|
buffer.WriteString(` onerror="imageError(this, event)"`)
|
|
|
|
|
}
|
2021-09-07 17:36:50 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (imageView *imageViewData) cssStyle(self View, builder cssBuilder) {
|
|
|
|
|
imageView.viewData.cssStyle(self, builder)
|
|
|
|
|
|
|
|
|
|
if value, ok := enumProperty(imageView, Fit, imageView.session, 0); ok {
|
|
|
|
|
builder.add("object-fit", enumProperties[Fit].cssValues[value])
|
|
|
|
|
} else {
|
|
|
|
|
builder.add("object-fit", "none")
|
|
|
|
|
}
|
|
|
|
|
|
2022-08-31 22:17:46 +03:00
|
|
|
|
vAlign := GetImageViewVerticalAlign(imageView)
|
|
|
|
|
hAlign := GetImageViewHorizontalAlign(imageView)
|
2021-09-07 17:36:50 +03:00
|
|
|
|
if vAlign != CenterAlign || hAlign != CenterAlign {
|
|
|
|
|
var position string
|
|
|
|
|
switch hAlign {
|
|
|
|
|
case LeftAlign:
|
|
|
|
|
position = "left"
|
|
|
|
|
case RightAlign:
|
|
|
|
|
position = "right"
|
|
|
|
|
default:
|
|
|
|
|
position = "center"
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
switch vAlign {
|
|
|
|
|
case TopAlign:
|
|
|
|
|
position += " top"
|
|
|
|
|
case BottomAlign:
|
|
|
|
|
position += " bottom"
|
|
|
|
|
default:
|
|
|
|
|
position += " center"
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
builder.add("object-position", position)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-07-08 13:16:42 +03:00
|
|
|
|
func (imageView *imageViewData) handleCommand(self View, command string, data DataObject) bool {
|
|
|
|
|
switch command {
|
|
|
|
|
case "imageViewError":
|
|
|
|
|
for _, listener := range imageView.imageListeners(ErrorEvent) {
|
|
|
|
|
listener(imageView)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case "imageViewLoaded":
|
|
|
|
|
imageView.naturalWidth = dataFloatProperty(data, "natural-width")
|
|
|
|
|
imageView.naturalHeight = dataFloatProperty(data, "natural-height")
|
|
|
|
|
imageView.currentSrc, _ = data.PropertyValue("current-src")
|
|
|
|
|
|
|
|
|
|
for _, listener := range imageView.imageListeners(LoadedEvent) {
|
|
|
|
|
listener(imageView)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
|
return imageView.viewData.handleCommand(self, command, data)
|
|
|
|
|
}
|
|
|
|
|
return true
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (imageView *imageViewData) NaturalSize() (float64, float64) {
|
|
|
|
|
return imageView.naturalWidth, imageView.naturalHeight
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (imageView *imageViewData) CurrentSource() string {
|
|
|
|
|
return imageView.currentSrc
|
|
|
|
|
}
|
|
|
|
|
|
2021-09-07 17:36:50 +03:00
|
|
|
|
// GetImageViewSource returns the image URL of an ImageView subview.
|
2022-08-31 22:17:46 +03:00
|
|
|
|
// If the second argument (subviewID) is not specified or it is "" then a left position of the first argument (view) is returned
|
|
|
|
|
func GetImageViewSource(view View, subviewID ...string) string {
|
|
|
|
|
if len(subviewID) > 0 && subviewID[0] != "" {
|
|
|
|
|
view = ViewByID(view, subviewID[0])
|
2021-09-07 17:36:50 +03:00
|
|
|
|
}
|
2022-04-23 18:13:35 +03:00
|
|
|
|
|
|
|
|
|
if view != nil {
|
|
|
|
|
if image, ok := imageProperty(view, Source, view.Session()); ok {
|
|
|
|
|
return image
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-09-07 17:36:50 +03:00
|
|
|
|
return ""
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// GetImageViewAltText returns an alternative text description of an ImageView subview.
|
2022-08-31 22:17:46 +03:00
|
|
|
|
// If the second argument (subviewID) is not specified or it is "" then a left position of the first argument (view) is returned
|
|
|
|
|
func GetImageViewAltText(view View, subviewID ...string) string {
|
|
|
|
|
if len(subviewID) > 0 && subviewID[0] != "" {
|
|
|
|
|
view = ViewByID(view, subviewID[0])
|
2022-05-17 10:46:00 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if view != nil {
|
|
|
|
|
if value := view.getRaw(AltText); value != nil {
|
|
|
|
|
if text, ok := value.(string); ok {
|
|
|
|
|
text, _ = view.Session().GetString(text)
|
|
|
|
|
return text
|
|
|
|
|
}
|
|
|
|
|
}
|
2021-09-07 17:36:50 +03:00
|
|
|
|
}
|
|
|
|
|
return ""
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// GetImageViewFit returns how the content of a replaced ImageView subview:
|
|
|
|
|
// NoneFit (0), ContainFit (1), CoverFit (2), FillFit (3), or ScaleDownFit (4).
|
2022-08-31 22:17:46 +03:00
|
|
|
|
// If the second argument (subviewID) is not specified or it is "" then a left position of the first argument (view) is returned
|
|
|
|
|
func GetImageViewFit(view View, subviewID ...string) int {
|
2022-07-28 12:41:50 +03:00
|
|
|
|
return enumStyledProperty(view, subviewID, Fit, NoneFit, false)
|
2021-09-07 17:36:50 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// GetImageViewVerticalAlign return the vertical align of an ImageView subview: TopAlign (0), BottomAlign (1), CenterAlign (2)
|
2022-08-31 22:17:46 +03:00
|
|
|
|
// If the second argument (subviewID) is not specified or it is "" then a left position of the first argument (view) is returned
|
|
|
|
|
func GetImageViewVerticalAlign(view View, subviewID ...string) int {
|
2022-07-28 12:41:50 +03:00
|
|
|
|
return enumStyledProperty(view, subviewID, ImageVerticalAlign, LeftAlign, false)
|
2021-09-07 17:36:50 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// GetImageViewHorizontalAlign return the vertical align of an ImageView subview: LeftAlign (0), RightAlign (1), CenterAlign (2)
|
2022-08-31 22:17:46 +03:00
|
|
|
|
// If the second argument (subviewID) is not specified or it is "" then a left position of the first argument (view) is returned
|
|
|
|
|
func GetImageViewHorizontalAlign(view View, subviewID ...string) int {
|
2022-07-28 12:41:50 +03:00
|
|
|
|
return enumStyledProperty(view, subviewID, ImageHorizontalAlign, LeftAlign, false)
|
2021-09-07 17:36:50 +03:00
|
|
|
|
}
|