rui_orig/stackLayout.go

957 lines
30 KiB
Go
Raw Normal View History

2021-09-07 17:36:50 +03:00
package rui
import (
"fmt"
"strings"
)
// Constants which represent [StackLayout] animation type during pushing or popping views
2021-09-07 17:36:50 +03:00
const (
// PushTransform is the constant for "push-transform" property tag.
//
2024-12-05 20:15:39 +03:00
// Used by StackLayout.
// Specify start translation, scale and rotation over x, y and z axes as well as a distortion
// for an animated pushing of a child view.
//
2024-12-05 20:15:39 +03:00
// Supported types: TransformProperty, string.
//
2024-12-05 20:15:39 +03:00
// See TransformProperty description for more details.
//
// Conversion rules:
2024-12-05 20:15:39 +03:00
// - TransformProperty - stored as is, no conversion performed.
// - string - string representation of Transform interface. Example: "_{translate-x = 10px, scale-y = 1.1}".
PushTransform = "push-transform"
// PushDuration is the constant for "push-duration" property tag.
//
2024-12-05 20:15:39 +03:00
// Used by StackLayout.
// Sets the length of time in seconds that an push/pop animation takes to complete.
//
2024-12-05 20:15:39 +03:00
// Supported types: float, int, string.
//
2024-12-05 20:15:39 +03:00
// Internal type is float, other types converted to it during assignment.
PushDuration = "push-duration"
// PushTiming is the constant for "push-timing" property tag.
//
2024-12-05 20:15:39 +03:00
// Used by StackLayout.
// Set how an push/pop animation progresses through the duration of each cycle.
//
2024-12-05 20:15:39 +03:00
// Supported types: string.
//
// Values:
2024-12-05 20:15:39 +03:00
// - "ease" (EaseTiming) - Speed increases towards the middle and slows down at the end.
// - "ease-in" (EaseInTiming) - Speed is slow at first, but increases in the end.
// - "ease-out" (EaseOutTiming) - Speed is fast at first, but decreases in the end.
// - "ease-in-out" (EaseInOutTiming) - Speed is slow at first, but quickly increases and at the end it decreases again.
// - "linear" (LinearTiming) - Constant speed.
// - "step(n)" (StepTiming(n int) function) - Timing function along stepCount stops along the transition, displaying each stop for equal lengths of time.
// - "cubic-bezier(x1, y1, x2, y2)" (CubicBezierTiming(x1, y1, x2, y2 float64) function) - Cubic-Bezier curve timing function. x1 and x2 must be in the range [0, 1].
PushTiming = "push-timing"
// MoveToFrontAnimation is the constant for "move-to-front-animation" property tag.
//
2024-12-05 20:15:39 +03:00
// Used by StackLayout.
// Specifies whether animation is used when calling the MoveToFront/MoveToFrontByID method of StackLayout interface.
//
2024-12-05 20:15:39 +03:00
// Supported types: bool, int, string.
//
// Values:
2024-12-05 20:15:39 +03:00
// - true, 1, "true", "yes", "on", "1" - animation is used (default value).
// - false, 0, "false", "no", "off", "0" - animation is not used.
MoveToFrontAnimation = "move-to-front-animation"
// PushPerspective is the constant for "push-perspective" property tag.
//
2024-12-05 20:15:39 +03:00
// Used by StackLayout.
//
// Used to access the "perspective" property of StackLayout "push-transform" property:
// Distance between the z-plane and the user in order to give a 3D-positioned element some perspective. Each 3D element
// with z > 0 becomes larger, each 3D-element with z < 0 becomes smaller. The default value is 0 (no 3D effects).
//
2024-12-05 20:15:39 +03:00
// Supported types: SizeUnit, SizeFunc, string, float, int.
//
2024-12-05 20:15:39 +03:00
// Internal type is SizeUnit, other types converted to it during assignment.
// See SizeUnit description for more details.
PushPerspective PropertyName = "push-perspective"
// PushTranslateX is the constant for "push-translate-x" property tag.
//
2024-12-05 20:15:39 +03:00
// Used by StackLayout.
//
// Used to access the "translate-x" property of StackLayout "push-transform" property:
// x-axis translation value of a 2D/3D translation.
//
2024-12-05 20:15:39 +03:00
// Supported types: SizeUnit, SizeFunc, string, float, int.
//
2024-12-05 20:15:39 +03:00
// Internal type is SizeUnit, other types converted to it during assignment.
// See SizeUnit description for more details.
PushTranslateX PropertyName = "push-translate-x"
// PushTranslateY is the constant for "push-translate-y" property tag.
//
2024-12-05 20:15:39 +03:00
// Used by StackLayout.
//
// Used to access the "translate-y" property of StackLayout "push-transform" property:
// y-axis translation value of a 2D/3D translation.
//
2024-12-05 20:15:39 +03:00
// Supported types: SizeUnit, SizeFunc, string, float, int.
//
2024-12-05 20:15:39 +03:00
// Internal type is SizeUnit, other types converted to it during assignment.
// See SizeUnit description for more details.
PushTranslateY PropertyName = "push-translate-y"
// PushTranslateZ is the constant for "push-translate-z" property tag.
//
2024-12-05 20:15:39 +03:00
// Used by StackLayout.
//
// Used to access the "translate-z" property of StackLayout "push-transform" property:
// z-axis translation value of a 3D translation.
//
2024-12-05 20:15:39 +03:00
// Supported types: SizeUnit, SizeFunc, string, float, int.
//
2024-12-05 20:15:39 +03:00
// Internal type is SizeUnit, other types converted to it during assignment.
// See SizeUnit description for more details.
PushTranslateZ PropertyName = "push-translate-z"
// PushScaleX is the constant for "push-scale-x" property tag.
//
2024-12-05 20:15:39 +03:00
// Used by StackLayout.
//
// Used to access the "scale-x" property of StackLayout "push-transform" property:
// x-axis scaling value of a 2D/3D scale. The original scale is 1. Values between 0 and 1 are used to decrease original
// scale, more than 1 - to increase. The default value is 1.
//
2024-12-05 20:15:39 +03:00
// Supported types: float, int, string.
//
2024-12-05 20:15:39 +03:00
// Internal type is float, other types converted to it during assignment.
PushScaleX PropertyName = "push-scale-x"
// PushScaleY is the constant for "push-scale-y" property tag.
//
2024-12-05 20:15:39 +03:00
// Used by StackLayout.
//
// Used to access the "scale-y" property of StackLayout "push-transform" property:
// y-axis scaling value of a 2D/3D scale. The original scale is 1. Values between 0 and 1 are used to decrease original
// scale, more than 1 - to increase. The default value is 1.
//
2024-12-05 20:15:39 +03:00
// Supported types: float, int, string.
//
2024-12-05 20:15:39 +03:00
// Internal type is float, other types converted to it during assignment.
PushScaleY PropertyName = "push-scale-y"
// PushScaleZ is the constant for "push-scale-z" property tag.
//
2024-12-05 20:15:39 +03:00
// Used by StackLayout.
//
// Used to access the "scale-z" property of StackLayout "push-transform" property:
// z-axis scaling value of a 3D scale. The original scale is 1. Values between 0 and 1 are used to decrease original
// scale, more than 1 - to increase. The default value is 1.
//
2024-12-05 20:15:39 +03:00
// Supported types: float, int, string.
//
2024-12-05 20:15:39 +03:00
// Internal type is float, other types converted to it during assignment.
PushScaleZ PropertyName = "push-scale-z"
// PushRotate is the constant for "push-rotate" property tag.
//
2024-12-05 20:15:39 +03:00
// Used by StackLayout.
//
// Used to access the "rotate" property of StackLayout "push-transform" property:
// Angle of the view rotation. A positive angle denotes a clockwise rotation, a negative angle a counter-clockwise.
//
2024-12-05 20:15:39 +03:00
// Supported types: AngleUnit, string, float, int.
//
2024-12-05 20:15:39 +03:00
// Internal type is AngleUnit, other types will be converted to it during assignment.
// See AngleUnit description for more details.
//
// Conversion rules:
2024-12-05 20:15:39 +03:00
// - AngleUnit - stored as is, no conversion performed.
// - string - must contain string representation of AngleUnit. If numeric value will be provided without any suffix then AngleUnit with value and Radian value type will be created.
// - float - a new AngleUnit value will be created with Radian as a type.
// - int - a new AngleUnit value will be created with Radian as a type.
PushRotate PropertyName = "push-rotate"
// PushRotateX is the constant for "push-rotate-x" property tag.
//
2024-12-05 20:15:39 +03:00
// Used by StackLayout.
//
// Used to access the "rotate-x" property of StackLayout "push-transform" property:
// x-coordinate of the vector denoting the axis of rotation in range 0 to 1.
//
2024-12-05 20:15:39 +03:00
// Supported types: float, int, string.
//
2024-12-05 20:15:39 +03:00
// Internal type is float, other types converted to it during assignment.
PushRotateX PropertyName = "push-rotate-x"
// PushRotateY is the constant for "push-rotate-y" property tag.
//
2024-12-05 20:15:39 +03:00
// Used by StackLayout.
//
// Used to access the "rotate-y" property of StackLayout "push-transform" property:
// y-coordinate of the vector denoting the axis of rotation in range 0 to 1.
//
2024-12-05 20:15:39 +03:00
// Supported types: float, int, string.
//
2024-12-05 20:15:39 +03:00
// Internal type is float, other types converted to it during assignment.
PushRotateY PropertyName = "push-rotate-y"
// PushRotateZ is the constant for "push-rotate-z" property tag.
//
2024-12-05 20:15:39 +03:00
// Used by StackLayout.
//
// Used to access the "rotate-z" property of StackLayout "push-transform" property:
// z-coordinate of the vector denoting the axis of rotation in range 0 to 1.
//
2024-12-05 20:15:39 +03:00
// Supported types: float, int, string.
//
2024-12-05 20:15:39 +03:00
// Internal type is float, other types converted to it during assignment.
PushRotateZ PropertyName = "push-rotate-z"
// PushSkewX is the constant for "push-skew-x" property tag.
//
2024-12-05 20:15:39 +03:00
// Used by StackLayout.
//
// Used to access the "skew-x" property of StackLayout "push-transform" property:
// Angle to use to distort the element along the abscissa. The default value is 0.
//
2024-12-05 20:15:39 +03:00
// Supported types: AngleUnit, string, float, int.
//
2024-12-05 20:15:39 +03:00
// Internal type is AngleUnit, other types will be converted to it during assignment.
// See AngleUnit description for more details.
//
// Conversion rules:
2024-12-05 20:15:39 +03:00
// - AngleUnit - stored as is, no conversion performed.
// - string - must contain string representation of AngleUnit. If numeric value will be provided without any suffix then AngleUnit with value and Radian value type will be created.
// - float - a new AngleUnit value will be created with Radian as a type.
// - int - a new AngleUnit value will be created with Radian as a type.
PushSkewX PropertyName = "push-skew-x"
// PushSkewY is the constant for "push-skew-y" property tag.
//
2024-12-05 20:15:39 +03:00
// Used by StackLayout.
//
// Used to access the "skew-y" property of StackLayout "push-transform" property:
// Angle to use to distort the element along the ordinate. The default value is 0.
//
2024-12-05 20:15:39 +03:00
// Supported types: AngleUnit, string, float, int.
//
2024-12-05 20:15:39 +03:00
// Internal type is AngleUnit, other types will be converted to it during assignment.
// See AngleUnit description for more details.
//
// Conversion rules:
2024-12-05 20:15:39 +03:00
// - AngleUnit - stored as is, no conversion performed.
// - string - must contain string representation of AngleUnit. If numeric value will be provided without any suffix then AngleUnit with value and Radian value type will be created.
// - float - a new AngleUnit value will be created with Radian as a type.
// - int - a new AngleUnit value will be created with Radian as a type.
PushSkewY PropertyName = "push-skew-y"
2021-09-07 17:36:50 +03:00
)
// StackLayout represents a StackLayout view
2021-09-07 17:36:50 +03:00
type StackLayout interface {
ViewsContainer
2024-04-23 19:34:36 +03:00
// Peek returns the current (visible) View. If StackLayout is empty then it returns nil.
2021-09-07 17:36:50 +03:00
Peek() View
2024-04-23 19:34:36 +03:00
// RemovePeek removes the current View and returns it. If StackLayout is empty then it doesn't do anything and returns nil.
RemovePeek() View
2024-11-24 22:14:35 +03:00
// MoveToFront makes the given View current.
2024-12-05 20:15:39 +03:00
//
2024-11-24 22:14:35 +03:00
// The second argument is a function called after the move to front animation ends.
2024-12-05 20:15:39 +03:00
//
2024-11-24 22:14:35 +03:00
// Returns true if successful, false otherwise.
MoveToFront(view View, onShown ...func(View)) bool
2024-04-23 19:34:36 +03:00
2024-11-24 22:14:35 +03:00
// MoveToFrontByID makes the View current by viewID.
2024-12-05 20:15:39 +03:00
//
2024-11-24 22:14:35 +03:00
// The second argument is a function called after the move to front animation ends.
2024-12-05 20:15:39 +03:00
//
2024-11-24 22:14:35 +03:00
// Returns true if successful, false otherwise.
MoveToFrontByID(viewID string, onShown ...func(View)) bool
2024-04-23 19:34:36 +03:00
// Push adds a new View to the container and makes it current.
2024-12-05 20:15:39 +03:00
//
2024-04-23 19:34:36 +03:00
// It is similar to Append, but the addition is done using an animation effect.
2024-12-05 20:15:39 +03:00
//
2024-04-23 19:34:36 +03:00
// The animation type is specified by the second argument and can take the following values:
2024-12-05 20:15:39 +03:00
// - DefaultAnimation (0) - Default animation. For the Push function it is EndToStartAnimation, for Pop - StartToEndAnimation;
// - StartToEndAnimation (1) - Animation from beginning to end. The beginning and the end are determined by the direction of the text output;
// - EndToStartAnimation (2) - End-to-Beginning animation;
// - TopDownAnimation (3) - Top-down animation;
// - BottomUpAnimation (4) - Bottom up animation.
2024-11-24 22:14:35 +03:00
// The second argument `onPushFinished` is the function to be called when the animation ends.
Push(view View, onPushFinished ...func())
2024-04-23 19:34:36 +03:00
// Pop removes the current View from the container using animation.
2024-12-05 20:15:39 +03:00
//
2024-11-24 22:14:35 +03:00
// The argument `onPopFinished` is the function to be called when the animation ends.
2024-12-05 20:15:39 +03:00
//
2024-04-23 19:34:36 +03:00
// The function will return false if the StackLayout is empty and true if the current item has been removed.
2024-11-24 22:14:35 +03:00
Pop(onPopFinished ...func(View)) bool
}
type pushFinished struct {
peekID string
2024-11-24 22:14:35 +03:00
listener []func()
}
type popFinished struct {
view View
2024-11-24 22:14:35 +03:00
listener []func(View)
2021-09-07 17:36:50 +03:00
}
type stackLayoutData struct {
viewsContainerData
onPushFinished map[string]pushFinished
onPopFinished map[string]popFinished
2024-11-24 22:14:35 +03:00
onMoveFinished map[string]popFinished
2021-09-07 17:36:50 +03:00
}
// NewStackLayout create new StackLayout object and return it
func NewStackLayout(session Session, params Params) StackLayout {
view := new(stackLayoutData)
view.init(session)
2021-09-07 17:36:50 +03:00
setInitParams(view, params)
return view
}
func newStackLayout(session Session) View {
2024-11-13 12:56:39 +03:00
//return NewStackLayout(session, nil)
return new(stackLayoutData)
2021-09-07 17:36:50 +03:00
}
// Init initialize fields of ViewsContainer by default values
func (layout *stackLayoutData) init(session Session) {
layout.viewsContainerData.init(session)
2021-09-07 17:36:50 +03:00
layout.tag = "StackLayout"
layout.systemClass = "ruiStackLayout"
layout.onPushFinished = map[string]pushFinished{}
layout.onPopFinished = map[string]popFinished{}
2024-11-24 22:14:35 +03:00
layout.onMoveFinished = map[string]popFinished{}
2024-11-13 12:56:39 +03:00
layout.set = layout.setFunc
layout.remove = layout.removeFunc
layout.changed = layout.propertyChanged
layout.setRaw(TransitionEndEvent, []func(View, PropertyName){layout.transitionFinished})
if session.TextDirection() == RightToLeftDirection {
layout.setRaw(PushTransform, NewTransformProperty(Params{TranslateX: Percent(-100)}))
} else {
layout.setRaw(PushTransform, NewTransformProperty(Params{TranslateX: Percent(100)}))
}
2022-05-22 12:54:02 +03:00
}
func (layout *stackLayoutData) transitionFinished(view View, tag PropertyName) {
if tags := strings.Split(string(tag), "-"); len(tags) >= 2 {
session := layout.Session()
viewID := tags[1]
switch tags[0] {
case "push":
if finished, ok := layout.onPushFinished[viewID]; ok {
if finished.peekID != "" {
pageID := finished.peekID + "page"
session.startUpdateScript(pageID)
session.updateCSSProperty(pageID, "visibility", "hidden")
session.updateCSSProperty(pageID, "transition", "")
session.updateCSSProperty(pageID, "transform", "")
session.removeProperty(pageID, "ontransitionend")
session.removeProperty(pageID, "ontransitioncancel")
session.finishUpdateScript(pageID)
}
pageID := viewID + "page"
session.startUpdateScript(pageID)
session.updateCSSProperty(pageID, "z-index", "auto")
session.updateCSSProperty(pageID, "transition", "")
session.removeProperty(pageID, "ontransitionend")
session.removeProperty(pageID, "ontransitioncancel")
session.finishUpdateScript(pageID)
2024-11-24 22:14:35 +03:00
for _, listener := range finished.listener {
if listener != nil {
listener()
}
}
delete(layout.onPushFinished, viewID)
layout.contentChanged()
}
case "pop":
if finished, ok := layout.onPopFinished[viewID]; ok {
2024-11-24 18:23:24 +03:00
session.callFunc("removeView", viewID+"page")
2024-11-24 22:14:35 +03:00
for _, listener := range finished.listener {
if listener != nil {
listener(finished.view)
}
}
delete(layout.onPopFinished, viewID)
if count := len(layout.views); count > 0 {
peekID := layout.views[count-1].htmlID() + "page"
session.startUpdateScript(peekID)
session.removeProperty(peekID, "ontransitionend")
session.removeProperty(peekID, "ontransitioncancel")
session.finishUpdateScript(peekID)
}
2021-09-07 17:36:50 +03:00
}
case "move":
if count := len(layout.views); count > 1 {
pageID := layout.views[count-2].htmlID() + "page"
session.startUpdateScript(pageID)
session.updateCSSProperty(pageID, "visibility", "hidden")
session.updateCSSProperty(pageID, "transition", "")
session.updateCSSProperty(pageID, "transform", "")
session.finishUpdateScript(pageID)
}
2021-09-07 17:36:50 +03:00
pageID := viewID + "page"
session.startUpdateScript(pageID)
session.updateCSSProperty(pageID, "z-index", "auto")
session.updateCSSProperty(pageID, "transition", "")
session.removeProperty(pageID, "ontransitionend")
session.removeProperty(pageID, "ontransitioncancel")
session.finishUpdateScript(pageID)
layout.contentChanged()
2024-11-24 22:14:35 +03:00
if finished, ok := layout.onMoveFinished[viewID]; ok {
for _, listener := range finished.listener {
if listener != nil {
listener(finished.view)
}
}
delete(layout.onMoveFinished, viewID)
}
2021-09-07 17:36:50 +03:00
}
}
}
func (layout *stackLayoutData) setFunc(tag PropertyName, value any) []PropertyName {
switch tag {
case TransitionEndEvent:
// TODO
2024-11-22 16:36:08 +03:00
listeners, ok := valueToOneArgEventListeners[View, PropertyName](value)
2022-07-27 20:31:57 +03:00
if ok && listeners != nil {
listeners = append(listeners, layout.transitionFinished)
layout.setRaw(TransitionEndEvent, listeners)
2024-11-13 12:56:39 +03:00
return []PropertyName{tag}
}
2024-11-13 12:56:39 +03:00
return nil
case PushTiming:
if text, ok := value.(string); ok {
layout.setRaw(tag, text)
return []PropertyName{tag}
2024-11-13 12:56:39 +03:00
}
}
return layout.viewsContainerData.setFunc(tag, value)
}
func (layout *stackLayoutData) propertyChanged(tag PropertyName) {
switch tag {
case PushTransform, PushTiming, PushDuration, MoveToFrontAnimation,
PushPerspective, PushRotateX, PushRotateY, PushRotateZ, PushRotate, PushSkewX, PushSkewY,
PushScaleX, PushScaleY, PushScaleZ, PushTranslateX, PushTranslateY, PushTranslateZ:
// do nothing
default:
layout.viewsContainerData.propertyChanged(tag)
}
}
func (layout *stackLayoutData) removeFunc(tag PropertyName) []PropertyName {
switch tag {
case TransitionEndEvent:
layout.setRaw(TransitionEndEvent, []func(View, PropertyName){layout.transitionFinished})
2024-11-13 12:56:39 +03:00
return []PropertyName{tag}
}
return layout.viewsContainerData.removeFunc(tag)
}
2021-09-07 17:36:50 +03:00
func (layout *stackLayoutData) Peek() View {
if count := len(layout.views); count > 0 {
return layout.views[count-1]
2021-09-07 17:36:50 +03:00
}
return nil
}
2024-11-24 22:14:35 +03:00
func (layout *stackLayoutData) MoveToFront(view View, onShown ...func(View)) bool {
if view == nil {
ErrorLog(`MoveToFront(nil) forbidden`)
return false
}
2021-09-07 17:36:50 +03:00
htmlID := view.htmlID()
switch count := len(layout.views); count {
case 0:
// do nothing
2021-09-07 17:36:50 +03:00
case 1:
if layout.views[0].htmlID() == htmlID {
2021-09-07 17:36:50 +03:00
return true
}
default:
for i, view := range layout.views {
if view.htmlID() == htmlID {
2024-11-24 22:14:35 +03:00
layout.moveToFrontByIndex(i, onShown)
return true
}
}
2021-09-07 17:36:50 +03:00
}
ErrorLog(`MoveToFront() fail. Subview not found.`)
2021-09-07 17:36:50 +03:00
return false
}
2024-11-24 22:14:35 +03:00
func (layout *stackLayoutData) MoveToFrontByID(viewID string, onShown ...func(View)) bool {
switch count := len(layout.views); count {
case 0:
// do nothing
2021-09-07 17:36:50 +03:00
case 1:
if layout.views[0].ID() == viewID {
2021-09-07 17:36:50 +03:00
return true
}
default:
for i, view := range layout.views {
if view.ID() == viewID {
2024-11-24 22:14:35 +03:00
layout.moveToFrontByIndex(i, onShown)
return true
}
}
2021-09-07 17:36:50 +03:00
}
ErrorLogF(`MoveToFront("%s") fail. Subview with "%s" not found.`, viewID, viewID)
2021-09-07 17:36:50 +03:00
return false
}
2024-11-24 22:14:35 +03:00
func (layout *stackLayoutData) moveToFrontByIndex(index int, onShow []func(View)) {
count := len(layout.views)
if index == count-1 {
return
2021-09-07 17:36:50 +03:00
}
view := layout.views[index]
peekID := layout.views[count-1].htmlID()
if index == 0 {
layout.views = append(layout.views[1:], view)
2021-09-07 17:36:50 +03:00
} else {
layout.views = append(append(layout.views[:index], layout.views[index+1:]...), view)
2021-09-07 17:36:50 +03:00
}
2024-11-25 23:13:29 +03:00
if !layout.created {
return
}
session := layout.Session()
pageID := view.htmlID() + "page"
peekPageID := peekID + "page"
animated := IsMoveToFrontAnimation(layout)
var transform TransformProperty = nil
if animated {
transform = GetPushTransform(layout)
}
if transform == nil {
session.updateCSSProperty(peekPageID, "visibility", "hidden")
session.updateCSSProperty(pageID, "visibility", "visible")
layout.contentChanged()
2024-11-24 22:14:35 +03:00
for _, listener := range onShow {
if listener != nil {
listener(view)
}
}
return
}
2024-11-24 22:14:35 +03:00
layout.onMoveFinished[view.htmlID()] = popFinished{
view: view,
listener: onShow,
}
buffer := allocStringBuilder()
defer freeStringBuilder(buffer)
buffer.WriteString(`stackTransitionEndEvent('`)
buffer.WriteString(layout.htmlID())
buffer.WriteString(`', 'move-`)
buffer.WriteString(view.htmlID())
buffer.WriteString(`', event)`)
listener := buffer.String()
transformCSS := transformMirror(transform, session).transformCSS(session)
transitionCSS := layout.pushTransitionCSS()
session.updateCSSProperty(peekPageID, "transition", transitionCSS)
session.startUpdateScript(pageID)
session.updateProperty(pageID, "ontransitionend", listener)
session.updateProperty(pageID, "ontransitioncancel", listener)
session.updateCSSProperty(pageID, "transform", transformCSS)
session.updateCSSProperty(pageID, "z-index", "100")
session.updateCSSProperty(pageID, "visibility", "visible")
session.finishUpdateScript(pageID)
session.updateCSSProperty(pageID, "transition", transitionCSS)
session.updateCSSProperty(pageID, "transform", "")
session.updateCSSProperty(peekPageID, "transform", transformCSS)
}
func (layout *stackLayoutData) contentChanged() {
if listener, ok := layout.changeListener[Content]; ok {
listener(layout, Content)
2021-09-07 17:36:50 +03:00
}
}
2024-04-23 19:34:36 +03:00
func (layout *stackLayoutData) RemovePeek() View {
return layout.RemoveView(len(layout.views) - 1)
}
func (layout *stackLayoutData) pushTransitionCSS() string {
return fmt.Sprintf("transform %.2fs %s", GetPushDuration(layout), GetPushTiming(layout))
}
func transformMirror(transform TransformProperty, session Session) TransformProperty {
result := NewTransformProperty(nil)
for _, tag := range []PropertyName{Perspective, RotateX, RotateY, RotateZ, ScaleX, ScaleY, ScaleZ, TranslateZ} {
if value := transform.getRaw(tag); value != nil {
result.Set(tag, value)
}
}
for _, tag := range []PropertyName{Rotate, SkewX, SkewY} {
if angle, ok := angleProperty(transform, tag, session); ok {
angle.Value = -angle.Value
result.Set(tag, angle)
}
}
for _, tag := range []PropertyName{TranslateX, TranslateY} {
if size, ok := sizeProperty(transform, tag, session); ok {
size.Value = -size.Value
result.Set(tag, size)
}
}
return result
}
2024-11-24 18:23:24 +03:00
// Append appends a view to the end of the list of a view children
func (layout *stackLayoutData) Append(view View) {
if view == nil {
ErrorLog("StackLayout.Append(nil) is forbidden")
return
}
stackID := layout.htmlID()
view.setParentID(stackID)
count := len(layout.views)
if count == 0 {
layout.views = []View{view}
} else {
layout.views = append(layout.views, view)
}
2024-11-25 23:13:29 +03:00
if layout.created {
buffer := allocStringBuilder()
defer freeStringBuilder(buffer)
2024-11-24 18:23:24 +03:00
2024-11-25 23:13:29 +03:00
buffer.WriteString(`<div id="`)
buffer.WriteString(view.htmlID())
buffer.WriteString(`page" class="ruiStackPageLayout">`)
viewHTML(view, buffer, "")
buffer.WriteString(`</div>`)
2024-11-24 18:23:24 +03:00
2024-11-25 23:13:29 +03:00
session := layout.Session()
if count > 0 {
session.updateCSSProperty(layout.views[count-1].htmlID()+"page", "visibility", "hidden")
}
session.appendToInnerHTML(stackID, buffer.String())
2024-11-24 18:23:24 +03:00
2024-11-25 23:13:29 +03:00
if listener, ok := layout.changeListener[Content]; ok {
listener(layout, Content)
}
2024-11-24 18:23:24 +03:00
}
}
// Insert inserts a view to the "index" position in the list of a view children
func (layout *stackLayoutData) Insert(view View, index int) {
if view == nil {
ErrorLog("StackLayout.Insert(nil, ...) is forbidden")
return
}
if layout.views == nil || index < 0 || index >= len(layout.views) {
layout.Append(view)
return
}
stackID := layout.htmlID()
view.setParentID(stackID)
if index > 0 {
layout.views = append(layout.views[:index], append([]View{view}, layout.views[index:]...)...)
} else {
layout.views = append([]View{view}, layout.views...)
}
buffer := allocStringBuilder()
defer freeStringBuilder(buffer)
buffer.WriteString(`<div id="`)
buffer.WriteString(view.htmlID())
buffer.WriteString(`page" class="ruiStackPageLayout" style="visibility: hidden;">`)
viewHTML(view, buffer, "")
buffer.WriteString(`</div>`)
session := layout.Session()
session.appendToInnerHTML(stackID, buffer.String())
if listener, ok := layout.changeListener[Content]; ok {
listener(layout, Content)
}
}
// Remove removes view from list and return it
func (layout *stackLayoutData) RemoveView(index int) View {
if layout.views == nil {
layout.views = []View{}
return nil
}
count := len(layout.views)
if index < 0 || index >= count {
return nil
}
session := layout.Session()
view := layout.views[index]
view.setParentID("")
if count == 1 {
layout.views = []View{}
} else if index == 0 {
layout.views = layout.views[1:]
} else if index == count-1 {
layout.views = layout.views[:index]
session.updateCSSProperty(layout.views[count-2].htmlID()+"page", "visibility", "visible")
} else {
layout.views = append(layout.views[:index], layout.views[index+1:]...)
}
layout.Session().callFunc("removeView", view.htmlID()+"page")
if listener, ok := layout.changeListener[Content]; ok {
listener(layout, Content)
}
return view
}
2024-11-24 22:14:35 +03:00
func (layout *stackLayoutData) Push(view View, onPushFinished ...func()) {
2021-09-07 17:36:50 +03:00
if view == nil {
ErrorLog("StackLayout.Push(nil, ....) is forbidden")
return
}
transform := GetPushTransform(layout)
if transform == nil {
layout.Append(view)
2024-11-24 22:14:35 +03:00
for _, listener := range onPushFinished {
if listener != nil {
listener()
}
}
return
}
prevPeek := ""
finished := pushFinished{
listener: onPushFinished,
}
if count := len(layout.views); count > 0 {
finished.peekID = layout.views[count-1].htmlID()
prevPeek = finished.peekID + "page"
}
htmlID := view.htmlID()
layout.onPushFinished[htmlID] = finished
view.setParentID(layout.htmlID())
layout.views = append(layout.views, view)
2021-09-07 17:36:50 +03:00
session := layout.Session()
buffer := allocStringBuilder()
defer freeStringBuilder(buffer)
buffer.WriteString(`<div id="`)
buffer.WriteString(htmlID)
buffer.WriteString(`page" class="ruiStackPageLayout" ontransitionend="stackTransitionEndEvent('`)
buffer.WriteString(layout.htmlID())
buffer.WriteString(`', 'push-`)
2021-09-07 17:36:50 +03:00
buffer.WriteString(htmlID)
buffer.WriteString(`', event)" style="z-index: 100; transform: `)
buffer.WriteString(transform.transformCSS(layout.session))
buffer.WriteRune(';')
2021-09-07 17:36:50 +03:00
transitionCSS := layout.pushTransitionCSS()
buffer.WriteString(" transition: ")
buffer.WriteString(transitionCSS)
buffer.WriteString(`;">`)
2021-09-07 17:36:50 +03:00
viewHTML(view, buffer, "")
2021-09-07 17:36:50 +03:00
buffer.WriteString(`</div>`)
session.appendToInnerHTML(layout.htmlID(), buffer.String())
2024-11-13 12:56:39 +03:00
if prevPeek != "" {
mirror := transformMirror(transform, session)
layout.session.updateCSSProperty(prevPeek, "transition", transitionCSS)
layout.session.updateCSSProperty(prevPeek, "transform", mirror.transformCSS(session))
2024-11-13 12:56:39 +03:00
}
layout.session.updateCSSProperty(htmlID+"page", "transform", "")
2021-09-07 17:36:50 +03:00
}
2024-11-24 22:14:35 +03:00
func (layout *stackLayoutData) Pop(onPopFinished ...func(View)) bool {
count := len(layout.views)
if count == 0 {
2021-09-07 17:36:50 +03:00
ErrorLog("StackLayout is empty")
return false
}
transform := GetPushTransform(layout)
if transform == nil {
if view := layout.RemovePeek(); view != nil {
2024-11-24 22:14:35 +03:00
for _, listener := range onPopFinished {
if listener != nil {
listener(view)
}
}
return true
}
return false
}
2021-09-07 17:36:50 +03:00
peek := count - 1
view := layout.views[peek]
view.setParentID("")
2021-09-07 17:36:50 +03:00
layout.views = layout.views[:peek]
layout.contentChanged()
layout.onPopFinished[view.htmlID()] = popFinished{
view: view,
listener: onPopFinished,
}
htmlID := view.htmlID()
2021-09-07 17:36:50 +03:00
session := layout.Session()
buffer := allocStringBuilder()
defer freeStringBuilder(buffer)
buffer.WriteString(`stackTransitionEndEvent('`)
buffer.WriteString(layout.htmlID())
buffer.WriteString(`', 'pop-`)
2021-09-07 17:36:50 +03:00
buffer.WriteString(htmlID)
buffer.WriteString(`', event)`)
listener := buffer.String()
pageID := htmlID + "page"
transitionCSS := layout.pushTransitionCSS()
session.startUpdateScript(pageID)
session.updateProperty(pageID, "ontransitionend", listener)
session.updateProperty(pageID, "ontransitioncancel", listener)
session.updateCSSProperty(pageID, "z-index", "100")
session.updateCSSProperty(pageID, "transition", transitionCSS)
session.finishUpdateScript(pageID)
peek--
if peek >= 0 {
peekID := layout.views[peek].htmlID() + "page"
session.updateCSSProperty(peekID, "transition", "")
session.startUpdateScript(peekID)
session.updateCSSProperty(peekID, "transform", transformMirror(transform, session).transformCSS(session))
session.updateCSSProperty(peekID, "visibility", "visible")
session.finishUpdateScript(peekID)
session.updateCSSProperty(peekID, "transition", transitionCSS)
session.updateCSSProperty(peekID, "transform", "")
2021-09-07 17:36:50 +03:00
}
session.updateCSSProperty(pageID, "transform", transform.transformCSS(session))
2021-09-07 17:36:50 +03:00
return true
}
func (layout *stackLayoutData) htmlSubviews(self View, buffer *strings.Builder) {
if count := len(layout.views); count > 0 {
peek := count - 1
2021-09-07 17:36:50 +03:00
for i, view := range layout.views {
buffer.WriteString(`<div id="`)
buffer.WriteString(view.htmlID())
2021-09-07 17:36:50 +03:00
buffer.WriteString(`page`)
buffer.WriteString(`" class="ruiStackPageLayout"`)
if i != peek {
buffer.WriteString(` style="visibility: hidden;"`)
}
buffer.WriteString(`>`)
2024-11-21 09:25:46 +03:00
viewHTML(view, buffer, "")
2021-09-07 17:36:50 +03:00
buffer.WriteString(`</div>`)
}
}
}
// IsMoveToFrontAnimation returns "true" if an animation is used when calling the MoveToFront/MoveToFrontByID method of StackLayout interface.
// If the second argument (subviewID) is not specified or it is "" then a value from the first argument (view) is returned.
func IsMoveToFrontAnimation(view View, subviewID ...string) bool {
2024-11-24 16:43:31 +03:00
if view = getSubview(view, subviewID); view != nil {
if value, ok := boolProperty(view, MoveToFrontAnimation, view.Session()); ok {
return value
}
if value := valueFromStyle(view, MoveToFrontAnimation); value != nil {
if b, ok := valueToBool(value, view.Session()); ok {
return b
}
}
}
return true
}
// GetPushDuration returns the length of time in seconds that an push/pop StackLayout animation takes to complete.
// If the second argument (subviewID) is not specified or it is "" then a width of the first argument (view) is returned
func GetPushDuration(view View, subviewID ...string) float64 {
return floatStyledProperty(view, subviewID, PushDuration, 1)
}
// GetPushTiming returns the function which sets how an push/pop animation progresses.
// If the second argument (subviewID) is not specified or it is "" then a width of the first argument (view) is returned
func GetPushTiming(view View, subviewID ...string) string {
result := stringStyledProperty(view, subviewID, PushTiming, false)
if isTimingFunctionValid(result) {
return result
}
return "easy"
}
2024-12-02 15:47:06 +03:00
// GetPushTransform returns the start transform (translation, scale and rotation over x, y and z axes as well as a distortion)
// for an animated pushing of a child view.
// The default value is nil (no transform).
// If the second argument (subviewID) is not specified or it is "" then a value from the first argument (view) is returned.
func GetPushTransform(view View, subviewID ...string) TransformProperty {
return transformStyledProperty(view, subviewID, PushTransform)
}