package rui import ( "strings" ) const ( // TopDownOrientation - subviews are arranged from top to bottom. Synonym of VerticalOrientation TopDownOrientation = 0 // StartToEndOrientation - subviews are arranged from left to right. Synonym of HorizontalOrientation StartToEndOrientation = 1 // BottomUpOrientation - subviews are arranged from bottom to top BottomUpOrientation = 2 // EndToStartOrientation - subviews are arranged from right to left EndToStartOrientation = 3 // ListWrapOff - subviews are scrolled and "true" if a new row/column starts ListWrapOff = 0 // ListWrapOn - the new row/column starts at bottom/right ListWrapOn = 1 // ListWrapReverse - the new row/column starts at top/left ListWrapReverse = 2 ) // ListLayout - list-container of View type ListLayout interface { ViewsContainer } type listLayoutData struct { viewsContainerData } // NewListLayout create new ListLayout object and return it func NewListLayout(session Session, params Params) ListLayout { view := new(listLayoutData) view.init(session) setInitParams(view, params) return view } func newListLayout(session Session) View { return NewListLayout(session, nil) } // Init initialize fields of ViewsAlignContainer by default values func (listLayout *listLayoutData) init(session Session) { listLayout.viewsContainerData.init(session) listLayout.tag = "ListLayout" listLayout.systemClass = "ruiListLayout" } func (listLayout *listLayoutData) String() string { return getViewString(listLayout, nil) } func (listLayout *listLayoutData) normalizeTag(tag string) string { tag = strings.ToLower(tag) switch tag { case "wrap": tag = ListWrap case "row-gap": return ListRowGap case ColumnGap: return ListColumnGap } return tag } func (listLayout *listLayoutData) Get(tag string) any { return listLayout.get(listLayout.normalizeTag(tag)) } func (listLayout *listLayoutData) get(tag string) any { if tag == Gap { if rowGap := GetListRowGap(listLayout); rowGap.Equal(GetListColumnGap(listLayout)) { return rowGap } return AutoSize() } return listLayout.viewsContainerData.get(tag) } func (listLayout *listLayoutData) Remove(tag string) { listLayout.remove(listLayout.normalizeTag(tag)) } func (listLayout *listLayoutData) remove(tag string) { if tag == Gap { listLayout.remove(ListRowGap) listLayout.remove(ListColumnGap) return } listLayout.viewsContainerData.remove(tag) if listLayout.created { switch tag { case Orientation, ListWrap, HorizontalAlign, VerticalAlign: updateCSSStyle(listLayout.htmlID(), listLayout.session) } } } func (listLayout *listLayoutData) Set(tag string, value any) bool { return listLayout.set(listLayout.normalizeTag(tag), value) } func (listLayout *listLayoutData) set(tag string, value any) bool { if value == nil { listLayout.remove(tag) return true } if tag == Gap { return listLayout.set(ListRowGap, value) && listLayout.set(ListColumnGap, value) } if listLayout.viewsContainerData.set(tag, value) { if listLayout.created { switch tag { case Orientation, ListWrap, HorizontalAlign, VerticalAlign: updateCSSStyle(listLayout.htmlID(), listLayout.session) } } return true } return false } func (listLayout *listLayoutData) htmlSubviews(self View, buffer *strings.Builder) { if listLayout.views != nil { for _, view := range listLayout.views { view.addToCSSStyle(map[string]string{`flex`: `0 0 auto`}) viewHTML(view, buffer) } } } // GetListVerticalAlign returns the vertical align of a ListLayout or ListView sibview: // TopAlign (0), BottomAlign (1), CenterAlign (2), or StretchAlign (3) // If the second argument (subviewID) is not specified or it is "" then a value from the first argument (view) is returned. func GetListVerticalAlign(view View, subviewID ...string) int { return enumStyledProperty(view, subviewID, VerticalAlign, TopAlign, false) } // GetListHorizontalAlign returns the vertical align of a ListLayout or ListView subview: // LeftAlign (0), RightAlign (1), CenterAlign (2), or StretchAlign (3) // If the second argument (subviewID) is not specified or it is "" then a value from the first argument (view) is returned. func GetListHorizontalAlign(view View, subviewID ...string) int { return enumStyledProperty(view, subviewID, HorizontalAlign, LeftAlign, false) } // GetListOrientation returns the orientation of a ListLayout or ListView subview: // TopDownOrientation (0), StartToEndOrientation (1), BottomUpOrientation (2), or EndToStartOrientation (3) // If the second argument (subviewID) is not specified or it is "" then a value from the first argument (view) is returned. func GetListOrientation(view View, subviewID ...string) int { if len(subviewID) > 0 && subviewID[0] != "" { view = ViewByID(view, subviewID[0]) } if view != nil { if orientation, ok := valueToOrientation(view.Get(Orientation), view.Session()); ok { return orientation } if value := valueFromStyle(view, Orientation); value != nil { if orientation, ok := valueToOrientation(value, view.Session()); ok { return orientation } } } return 0 } // GetListWrap returns the wrap type of a ListLayout or ListView subview: // ListWrapOff (0), ListWrapOn (1), or ListWrapReverse (2) // If the second argument (subviewID) is not specified or it is "" then a value from the first argument (view) is returned. func GetListWrap(view View, subviewID ...string) int { return enumStyledProperty(view, subviewID, ListWrap, ListWrapOff, false) } // GetListRowGap returns the gap between ListLayout or ListView rows. // If the second argument (subviewID) is not specified or it is "" then a value from the first argument (view) is returned. func GetListRowGap(view View, subviewID ...string) SizeUnit { return sizeStyledProperty(view, subviewID, ListRowGap, false) } // GetListColumnGap returns the gap between ListLayout or ListView columns. // If the second argument (subviewID) is not specified or it is "" then a value from the first argument (view) is returned. func GetListColumnGap(view View, subviewID ...string) SizeUnit { return sizeStyledProperty(view, subviewID, ListColumnGap, false) }