2021-09-07 17:36:50 +03:00
package rui
import (
"fmt"
2021-10-04 17:58:17 +03:00
"math"
2021-09-07 17:36:50 +03:00
"strconv"
2021-10-04 17:58:17 +03:00
"strings"
2021-09-07 17:36:50 +03:00
)
2024-09-12 14:05:11 +03:00
// Constants which related to view's animation
2021-10-04 17:58:17 +03:00
const (
2024-12-06 19:15:23 +03:00
// Animation is the constant for "animation" property tag.
2024-09-18 13:50:06 +03:00
//
2024-12-05 20:15:39 +03:00
// Used by View.
2024-09-18 13:50:06 +03:00
// Sets and starts animations.
//
2024-12-06 19:15:23 +03:00
// Supported types: AnimationProperty, []AnimationProperty.
2024-09-18 13:50:06 +03:00
//
2024-12-06 19:15:23 +03:00
// Internal type is []AnimationProperty, other types converted to it during assignment.
// See AnimationProperty description for more details.
Animation PropertyName = "animation"
2021-10-04 17:58:17 +03:00
2024-09-18 13:50:06 +03:00
// AnimationPaused is the constant for "animation-paused" property tag.
//
2024-12-06 19:15:23 +03:00
// Used by AnimationProperty.
2024-09-18 13:50:06 +03:00
// Controls whether the animation is running or paused.
//
2024-12-05 20:15:39 +03:00
// Supported types: bool, int, string.
2024-09-18 13:50:06 +03:00
//
// Values:
2024-12-05 20:15:39 +03:00
// - true, 1, "true", "yes", "on", or "1" - Animation is paused.
// - false, 0, "false", "no", "off", or "0" - Animation is playing.
2024-11-13 12:56:39 +03:00
AnimationPaused PropertyName = "animation-paused"
2021-10-04 17:58:17 +03:00
2024-09-18 13:50:06 +03:00
// Transition is the constant for "transition" property tag.
//
2024-12-05 20:15:39 +03:00
// Used by View.
//
2024-12-06 19:15:23 +03:00
// Sets transition animation of view properties. Each provided property must contain AnimationProperty which describe how
2024-11-13 12:56:39 +03:00
// particular property will be animated on property value change. Transition animation can be applied to properties of the
2024-12-05 20:15:39 +03:00
// type SizeUnit, Color, AngleUnit, float64 and composite properties that contain elements of the listed types(for
// example, "shadow", "border", etc.). If we'll try to animate other properties with internal type like bool or
// string no error will occur, simply there will be no animation.
2024-09-18 13:50:06 +03:00
//
2024-12-05 20:15:39 +03:00
// Supported types: Params.
2024-09-18 13:50:06 +03:00
//
2024-12-05 20:15:39 +03:00
// See Params description for more details.
2024-11-13 12:56:39 +03:00
Transition PropertyName = "transition"
2021-10-04 17:58:17 +03:00
2024-09-18 13:50:06 +03:00
// PropertyTag is the constant for "property" property tag.
//
2024-12-06 19:15:23 +03:00
// Used by AnimationProperty.
2024-12-05 20:15:39 +03:00
//
// Describes a scenario for changing a View's property. Used only for animation script.
2024-09-18 13:50:06 +03:00
//
2024-12-05 20:15:39 +03:00
// Supported types: []AnimatedProperty, AnimatedProperty.
2024-09-18 13:50:06 +03:00
//
2024-12-05 20:15:39 +03:00
// Internal type is []AnimatedProperty, other types converted to it during assignment.
// See AnimatedProperty description for more details.
2024-11-13 12:56:39 +03:00
PropertyTag PropertyName = "property"
2021-10-04 17:58:17 +03:00
2024-09-18 13:50:06 +03:00
// Duration is the constant for "duration" property tag.
//
2024-12-06 19:15:23 +03:00
// Used by AnimationProperty.
2024-12-05 20:15:39 +03:00
//
2024-09-18 13:50:06 +03:00
// Sets the length of time in seconds that an animation takes to complete one cycle.
//
2024-12-05 20:15:39 +03:00
// Supported types: float, int, string.
2024-09-18 13:50:06 +03:00
//
2024-12-05 20:15:39 +03:00
// Internal type is float, other types converted to it during assignment.
2024-11-13 12:56:39 +03:00
Duration PropertyName = "duration"
2021-10-04 17:58:17 +03:00
2024-09-18 13:50:06 +03:00
// Delay is the constant for "delay" property tag.
//
2024-12-06 19:15:23 +03:00
// Used by AnimationProperty.
2024-12-05 20:15:39 +03:00
//
2024-11-13 12:56:39 +03:00
// Specifies the amount of time in seconds to wait from applying the animation to an element before beginning to perform
// the animation. The animation can start later, immediately from its beginning or immediately and partway through the
2024-09-18 13:50:06 +03:00
// animation.
//
2024-12-05 20:15:39 +03:00
// Supported types: float, int, string.
2024-09-18 13:50:06 +03:00
//
2024-12-05 20:15:39 +03:00
// Internal type is float, other types converted to it during assignment.
2024-11-13 12:56:39 +03:00
Delay PropertyName = "delay"
2021-10-04 17:58:17 +03:00
2024-09-18 13:50:06 +03:00
// TimingFunction is the constant for "timing-function" property tag.
//
2024-12-06 19:15:23 +03:00
// Used by AnimationProperty.
2024-12-05 20:15:39 +03:00
//
2024-09-18 13:50:06 +03:00
// Set how an animation progresses through the duration of each cycle.
//
2024-12-05 20:15:39 +03:00
// Supported types: string.
2024-09-18 13:50:06 +03:00
//
// 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].
2024-11-13 12:56:39 +03:00
TimingFunction PropertyName = "timing-function"
2021-10-04 17:58:17 +03:00
2024-09-18 13:50:06 +03:00
// IterationCount is the constant for "iteration-count" property tag.
//
2024-12-06 19:15:23 +03:00
// Used by AnimationProperty.
2024-12-05 20:15:39 +03:00
//
2024-09-18 13:50:06 +03:00
// Sets the number of times an animation sequence should be played before stopping. Used only for animation script.
//
2024-12-05 20:15:39 +03:00
// Supported types: int, string.
2024-09-18 13:50:06 +03:00
//
2024-12-05 20:15:39 +03:00
// Internal type is int, other types converted to it during assignment.
2024-11-13 12:56:39 +03:00
IterationCount PropertyName = "iteration-count"
2021-10-04 17:58:17 +03:00
2024-09-18 13:50:06 +03:00
// AnimationDirection is the constant for "animation-direction" property tag.
//
2024-12-06 19:15:23 +03:00
// Used by AnimationProperty.
2024-12-05 20:15:39 +03:00
//
2024-11-13 12:56:39 +03:00
// Whether an animation should play forward, backward, or alternate back and forth between playing the sequence forward
2024-09-18 13:50:06 +03:00
// and backward. Used only for animation script.
//
2024-12-05 20:15:39 +03:00
// Supported types: int, string.
2024-09-18 13:50:06 +03:00
//
// Values:
2024-12-05 20:15:39 +03:00
// - 0 (NormalAnimation) or "normal" - The animation plays forward every iteration, that is, when the animation ends, it is immediately reset to its starting position and played again.
// - 1 (ReverseAnimation) or "reverse" - The animation plays backwards, from the last position to the first, and then resets to the final position and plays again.
// - 2 (AlternateAnimation) or "alternate" - The animation changes direction in each cycle, that is, in the first cycle, it starts from the start position, reaches the end position, and in the second cycle, it continues from the end position and reaches the start position, and so on.
// - 3 (AlternateReverseAnimation) or "alternate-reverse" - The animation starts playing from the end position and reaches the start position, and in the next cycle, continuing from the start position, it goes to the end position.
2024-11-13 12:56:39 +03:00
AnimationDirection PropertyName = "animation-direction"
2021-10-04 17:58:17 +03:00
// NormalAnimation is value of the "animation-direction" property.
2024-12-05 20:15:39 +03:00
//
2021-10-04 17:58:17 +03:00
// The animation plays forwards each cycle. In other words, each time the animation cycles,
// the animation will reset to the beginning state and start over again. This is the default value.
NormalAnimation = 0
2022-08-10 15:36:38 +03:00
2021-10-04 17:58:17 +03:00
// ReverseAnimation is value of the "animation-direction" property.
2024-12-05 20:15:39 +03:00
//
2021-10-04 17:58:17 +03:00
// The animation plays backwards each cycle. In other words, each time the animation cycles,
// the animation will reset to the end state and start over again. Animation steps are performed
// backwards, and timing functions are also reversed.
2024-12-05 20:15:39 +03:00
//
2021-10-04 17:58:17 +03:00
// For example, an "ease-in" timing function becomes "ease-out".
ReverseAnimation = 1
2022-08-10 15:36:38 +03:00
2021-10-04 17:58:17 +03:00
// AlternateAnimation is value of the "animation-direction" property.
2024-12-05 20:15:39 +03:00
//
2021-10-04 17:58:17 +03:00
// The animation reverses direction each cycle, with the first iteration being played forwards.
// The count to determine if a cycle is even or odd starts at one.
AlternateAnimation = 2
2022-08-10 15:36:38 +03:00
2021-10-04 17:58:17 +03:00
// AlternateReverseAnimation is value of the "animation-direction" property.
2024-12-05 20:15:39 +03:00
//
2021-10-04 17:58:17 +03:00
// The animation reverses direction each cycle, with the first iteration being played backwards.
// The count to determine if a cycle is even or odd starts at one.
AlternateReverseAnimation = 3
// EaseTiming - a timing function which increases in velocity towards the middle of the transition, slowing back down at the end
EaseTiming = "ease"
2024-09-18 13:50:06 +03:00
2021-10-04 17:58:17 +03:00
// EaseInTiming - a timing function which starts off slowly, with the transition speed increasing until complete
EaseInTiming = "ease-in"
2024-09-18 13:50:06 +03:00
2021-10-04 17:58:17 +03:00
// EaseOutTiming - a timing function which starts transitioning quickly, slowing down the transition continues.
EaseOutTiming = "ease-out"
2024-09-18 13:50:06 +03:00
2021-10-04 17:58:17 +03:00
// EaseInOutTiming - a timing function which starts transitioning slowly, speeds up, and then slows down again.
EaseInOutTiming = "ease-in-out"
2024-09-18 13:50:06 +03:00
2021-10-04 17:58:17 +03:00
// LinearTiming - a timing function at an even speed
LinearTiming = "linear"
)
2022-11-23 15:10:29 +03:00
// StepsTiming return a timing function along stepCount stops along the transition, displaying each stop for equal lengths of time
2021-10-04 17:58:17 +03:00
func StepsTiming ( stepCount int ) string {
return "steps(" + strconv . Itoa ( stepCount ) + ")"
}
// CubicBezierTiming return a cubic-Bezier curve timing function. x1 and x2 must be in the range [0, 1].
func CubicBezierTiming ( x1 , y1 , x2 , y2 float64 ) string {
if x1 < 0 {
x1 = 0
} else if x1 > 1 {
x1 = 1
}
if x2 < 0 {
x2 = 0
} else if x2 > 1 {
x2 = 1
}
return fmt . Sprintf ( "cubic-bezier(%g, %g, %g, %g)" , x1 , y1 , x2 , y2 )
2021-09-07 17:36:50 +03:00
}
2021-10-04 17:58:17 +03:00
// AnimatedProperty describes the change script of one property
type AnimatedProperty struct {
// Tag is the name of the property
2024-11-13 12:56:39 +03:00
Tag PropertyName
2021-10-04 17:58:17 +03:00
// From is the initial value of the property
2022-07-26 18:36:00 +03:00
From any
2021-10-04 17:58:17 +03:00
// To is the final value of the property
2022-07-26 18:36:00 +03:00
To any
2021-10-04 17:58:17 +03:00
// KeyFrames is intermediate property values
2022-07-26 18:36:00 +03:00
KeyFrames map [ int ] any
2021-09-07 17:36:50 +03:00
}
2021-10-04 17:58:17 +03:00
type animationData struct {
2024-11-13 12:56:39 +03:00
dataProperty
2021-10-04 17:58:17 +03:00
keyFramesName string
2024-07-01 19:17:03 +03:00
usageCounter int
2024-07-05 16:41:07 +03:00
view View
2024-12-06 19:15:23 +03:00
listener func ( view View , animation AnimationProperty , event PropertyName )
2024-11-13 12:56:39 +03:00
oldListeners map [ PropertyName ] [ ] func ( View , PropertyName )
2024-12-06 19:15:23 +03:00
oldAnimation [ ] AnimationProperty
2021-10-04 17:58:17 +03:00
}
2024-12-06 19:15:23 +03:00
// AnimationProperty interface is used to set animation parameters. Used properties:
2024-12-05 20:15:39 +03:00
//
2021-10-04 17:58:17 +03:00
// "property", "id", "duration", "delay", "timing-function", "iteration-count", and "animation-direction"
2024-12-06 19:15:23 +03:00
type AnimationProperty interface {
2021-10-04 17:58:17 +03:00
Properties
2021-09-07 17:36:50 +03:00
fmt . Stringer
2024-07-05 16:41:07 +03:00
// Start starts the animation for the view specified by the first argument.
// The second argument specifies the animation event listener (can be nil)
2024-12-06 19:15:23 +03:00
Start ( view View , listener func ( view View , animation AnimationProperty , event PropertyName ) ) bool
2024-07-05 16:41:07 +03:00
// Stop stops the animation
Stop ( )
// Pause pauses the animation
Pause ( )
// Resume resumes an animation that was stopped using the Pause method
Resume ( )
2024-11-13 12:56:39 +03:00
writeTransitionString ( tag PropertyName , buffer * strings . Builder )
2021-10-04 17:58:17 +03:00
animationCSS ( session Session ) string
transitionCSS ( buffer * strings . Builder , session Session )
2022-11-23 15:10:29 +03:00
hasAnimatedProperty ( ) bool
2021-10-04 17:58:17 +03:00
animationName ( ) string
2024-07-01 19:17:03 +03:00
used ( )
unused ( session Session )
2021-09-07 17:36:50 +03:00
}
2024-12-06 19:15:23 +03:00
func parseAnimation ( obj DataObject ) AnimationProperty {
2021-10-04 17:58:17 +03:00
animation := new ( animationData )
animation . init ( )
for i := 0 ; i < obj . PropertyCount ( ) ; i ++ {
if node := obj . Property ( i ) ; node != nil {
2024-11-13 12:56:39 +03:00
tag := PropertyName ( node . Tag ( ) )
2021-10-04 17:58:17 +03:00
if node . Type ( ) == TextNode {
2024-11-13 12:56:39 +03:00
animation . Set ( tag , node . Text ( ) )
2021-10-04 17:58:17 +03:00
} else {
2024-11-13 12:56:39 +03:00
animation . Set ( tag , node )
2021-10-04 17:58:17 +03:00
}
}
}
return animation
2021-09-07 17:36:50 +03:00
}
2024-12-06 19:52:57 +03:00
// NewAnimationProperty creates a new animation object and return its interface
//
// The following properties can be used:
// - "id" (ID) - specifies the animation identifier. Used only for animation script.
// - "duration" (Duration) - specifies the length of time in seconds that an animation takes to complete one cycle;
// - "delay" (Delay) - specifies the amount of time in seconds to wait from applying the animation to an element before beginning to perform
// the animation. The animation can start later, immediately from its beginning or immediately and partway through the animation;
// - "timing-function" (TimingFunction) - specifies how an animation progresses through the duration of each cycle;
// - "iteration-count" (IterationCount) - specifies the number of times an animation sequence should be played before stopping. Used only for animation script;
// - "animation-direction" (AnimationDirection) - specifies whether an animation should play forward, backward,
// or alternate back and forth between playing the sequence forward and backward. Used only for animation script;
// - "property" (PropertyTag) - describes a scenario for changing a View's property. Used only for animation script.
func NewAnimationProperty ( params Params ) AnimationProperty {
2021-10-04 17:58:17 +03:00
animation := new ( animationData )
animation . init ( )
2021-09-07 17:36:50 +03:00
2021-10-04 17:58:17 +03:00
for tag , value := range params {
animation . Set ( tag , value )
}
return animation
2021-09-07 17:36:50 +03:00
}
2024-12-06 19:52:57 +03:00
// NewTransitionAnimation creates animation data for the transition.
// - timingFunc - specifies how an animation progresses through the duration of each cycle. If it is "" then "easy" function is used;
// - duration - specifies the length of time in seconds that an animation takes to complete one cycle. Must be > 0;
// - delay - specifies the amount of time in seconds to wait from applying the animation to an element before beginning to perform
// the animation. The animation can start later, immediately from its beginning or immediately and partway through the animation.
func NewTransitionAnimation ( timingFunc string , duration float64 , delay float64 ) AnimationProperty {
animation := new ( animationData )
animation . init ( )
if duration <= 0 {
ErrorLog ( "Animation duration must be greater than 0" )
return nil
}
if ! animation . Set ( Duration , duration ) {
return nil
}
if timingFunc != "" {
if ! animation . Set ( TimingFunction , timingFunc ) {
return nil
}
}
if delay != 0 {
animation . Set ( Delay , delay )
}
return animation
}
// NewTransitionAnimation creates the animation scenario.
// - id - specifies the animation identifier.
// - timingFunc - specifies how an animation progresses through the duration of each cycle. If it is "" then "easy" function is used;
// - duration - specifies the length of time in seconds that an animation takes to complete one cycle. Must be > 0;
// - delay - specifies the amount of time in seconds to wait from applying the animation to an element before beginning to perform
// the animation. The animation can start later, immediately from its beginning or immediately and partway through the animation.
// - direction - specifies whether an animation should play forward, backward,
// or alternate back and forth between playing the sequence forward and backward. Only the following values can be used:
// 0 (NormalAnimation), 1 (ReverseAnimation), 2 (AlternateAnimation), and 3 (AlternateReverseAnimation);
2024-12-06 19:56:51 +03:00
// - iterationCount - specifies the number of times an animation sequence should be played before stopping. A negative value specifies infinite repetition;
2024-12-06 19:52:57 +03:00
// - property - describes a scenario for changing a View's property.
func NewAnimation ( id string , timingFunc string , duration float64 , delay float64 , direction int , iterationCount int , property ... AnimatedProperty ) AnimationProperty {
animation := new ( animationData )
animation . init ( )
if duration <= 0 {
ErrorLog ( "Animation duration must be greater than 0" )
return nil
}
if ! animation . Set ( Duration , duration ) {
return nil
}
if id != "" {
animation . Set ( ID , id )
}
if timingFunc != "" {
animation . Set ( TimingFunction , timingFunc )
}
if delay != 0 {
animation . Set ( Delay , delay )
}
if direction > 0 {
animation . Set ( AnimationDirection , direction )
}
2024-12-06 19:56:51 +03:00
if iterationCount != 0 {
2024-12-06 19:52:57 +03:00
animation . Set ( IterationCount , iterationCount )
}
if len ( property ) > 0 {
animation . Set ( PropertyTag , property )
}
return animation
}
2024-11-13 12:56:39 +03:00
func ( animation * animationData ) init ( ) {
animation . dataProperty . init ( )
2024-12-06 19:15:23 +03:00
animation . normalize = normalizeAnimation
2024-11-13 12:56:39 +03:00
animation . set = animationSet
animation . supportedProperties = [ ] PropertyName { ID , PropertyTag , Duration , Delay , TimingFunction , IterationCount , AnimationDirection }
}
2024-07-05 16:41:07 +03:00
func ( animation * animationData ) animatedProperties ( ) [ ] AnimatedProperty {
value := animation . getRaw ( PropertyTag )
if value == nil {
2021-10-04 17:58:17 +03:00
ErrorLog ( "There are no animated properties." )
2024-07-05 16:41:07 +03:00
return nil
2021-09-07 17:36:50 +03:00
}
2024-07-05 16:41:07 +03:00
props , ok := value . ( [ ] AnimatedProperty )
if ! ok {
2021-10-04 17:58:17 +03:00
ErrorLog ( "Invalid animated properties." )
2024-07-05 16:41:07 +03:00
return nil
}
if len ( props ) == 0 {
ErrorLog ( "There are no animated properties." )
return nil
2021-09-07 17:36:50 +03:00
}
2024-07-05 16:41:07 +03:00
return props
}
func ( animation * animationData ) hasAnimatedProperty ( ) bool {
return animation . animatedProperties ( ) != nil
2021-09-07 17:36:50 +03:00
}
2021-10-04 17:58:17 +03:00
func ( animation * animationData ) animationName ( ) string {
return animation . keyFramesName
}
2021-09-07 17:36:50 +03:00
2024-07-01 19:17:03 +03:00
func ( animation * animationData ) used ( ) {
animation . usageCounter ++
}
func ( animation * animationData ) unused ( session Session ) {
animation . usageCounter --
if animation . usageCounter <= 0 && animation . keyFramesName != "" {
session . removeAnimation ( animation . keyFramesName )
}
}
2024-12-06 19:15:23 +03:00
func normalizeAnimation ( tag PropertyName ) PropertyName {
2024-11-13 12:56:39 +03:00
tag = defaultNormalize ( tag )
2022-05-22 12:54:02 +03:00
if tag == Direction {
return AnimationDirection
}
return tag
}
2024-11-13 12:56:39 +03:00
func animationSet ( properties Properties , tag PropertyName , value any ) [ ] PropertyName {
switch tag {
2021-10-04 17:58:17 +03:00
case ID :
if text , ok := value . ( string ) ; ok {
text = strings . Trim ( text , " \t\n\r" )
if text == "" {
2024-11-13 12:56:39 +03:00
properties . setRaw ( tag , nil )
2021-10-04 17:58:17 +03:00
} else {
2024-11-13 12:56:39 +03:00
properties . setRaw ( tag , text )
2021-10-04 17:58:17 +03:00
}
2024-11-13 12:56:39 +03:00
return [ ] PropertyName { tag }
2021-10-04 17:58:17 +03:00
}
notCompatibleType ( tag , value )
2024-11-13 12:56:39 +03:00
return nil
2021-10-04 17:58:17 +03:00
case PropertyTag :
switch value := value . ( type ) {
case AnimatedProperty :
if value . From == nil && value . KeyFrames != nil {
if val , ok := value . KeyFrames [ 0 ] ; ok {
value . From = val
delete ( value . KeyFrames , 0 )
}
}
if value . To == nil && value . KeyFrames != nil {
if val , ok := value . KeyFrames [ 100 ] ; ok {
value . To = val
delete ( value . KeyFrames , 100 )
}
}
if value . From == nil {
ErrorLog ( "AnimatedProperty.From is nil" )
} else if value . To == nil {
ErrorLog ( "AnimatedProperty.To is nil" )
} else {
2024-11-13 12:56:39 +03:00
properties . setRaw ( tag , [ ] AnimatedProperty { value } )
return [ ] PropertyName { tag }
2021-10-04 17:58:17 +03:00
}
case [ ] AnimatedProperty :
props := [ ] AnimatedProperty { }
for _ , val := range value {
if val . From == nil && val . KeyFrames != nil {
if v , ok := val . KeyFrames [ 0 ] ; ok {
val . From = v
delete ( val . KeyFrames , 0 )
}
}
if val . To == nil && val . KeyFrames != nil {
if v , ok := val . KeyFrames [ 100 ] ; ok {
val . To = v
delete ( val . KeyFrames , 100 )
}
}
if val . From == nil {
ErrorLog ( "AnimatedProperty.From is nil" )
} else if val . To == nil {
ErrorLog ( "AnimatedProperty.To is nil" )
} else {
props = append ( props , val )
}
}
if len ( props ) > 0 {
2024-11-13 12:56:39 +03:00
properties . setRaw ( tag , props )
return [ ] PropertyName { tag }
2021-10-04 17:58:17 +03:00
} else {
ErrorLog ( "[]AnimatedProperty is empty" )
}
case DataNode :
parseObject := func ( obj DataObject ) ( AnimatedProperty , bool ) {
result := AnimatedProperty { }
for i := 0 ; i < obj . PropertyCount ( ) ; i ++ {
if node := obj . Property ( i ) ; node . Type ( ) == TextNode {
propTag := strings . ToLower ( node . Tag ( ) )
switch propTag {
case "from" , "0" , "0%" :
result . From = node . Text ( )
case "to" , "100" , "100%" :
result . To = node . Text ( )
default :
tagLen := len ( propTag )
if tagLen > 0 && propTag [ tagLen - 1 ] == '%' {
propTag = propTag [ : tagLen - 1 ]
}
n , err := strconv . Atoi ( propTag )
if err != nil {
ErrorLog ( err . Error ( ) )
} else if n < 0 || n > 100 {
ErrorLogF ( ` key-frame "%d" is out of range ` , n )
} else {
if result . KeyFrames == nil {
2022-07-26 18:36:00 +03:00
result . KeyFrames = map [ int ] any { n : node . Text ( ) }
2021-10-04 17:58:17 +03:00
} else {
result . KeyFrames [ n ] = node . Text ( )
}
}
}
}
}
if result . From != nil && result . To != nil {
return result , true
}
return result , false
}
switch value . Type ( ) {
case ObjectNode :
if prop , ok := parseObject ( value . Object ( ) ) ; ok {
2024-11-13 12:56:39 +03:00
properties . setRaw ( tag , [ ] AnimatedProperty { prop } )
return [ ] PropertyName { tag }
2021-10-04 17:58:17 +03:00
}
case ArrayNode :
props := [ ] AnimatedProperty { }
for _ , val := range value . ArrayElements ( ) {
if val . IsObject ( ) {
if prop , ok := parseObject ( val . Object ( ) ) ; ok {
props = append ( props , prop )
}
} else {
notCompatibleType ( tag , val )
}
}
if len ( props ) > 0 {
2024-11-13 12:56:39 +03:00
properties . setRaw ( tag , props )
return [ ] PropertyName { tag }
2021-10-04 17:58:17 +03:00
}
default :
notCompatibleType ( tag , value )
}
default :
notCompatibleType ( tag , value )
}
case Duration :
2024-11-13 12:56:39 +03:00
return setFloatProperty ( properties , tag , value , 0 , math . MaxFloat64 )
2021-10-04 17:58:17 +03:00
case Delay :
2024-11-13 12:56:39 +03:00
return setFloatProperty ( properties , tag , value , - math . MaxFloat64 , math . MaxFloat64 )
2021-10-04 17:58:17 +03:00
case TimingFunction :
if text , ok := value . ( string ) ; ok {
2024-11-13 12:56:39 +03:00
properties . setRaw ( tag , text )
return [ ] PropertyName { tag }
2021-10-04 17:58:17 +03:00
}
case IterationCount :
2024-11-13 12:56:39 +03:00
return setIntProperty ( properties , tag , value )
2021-10-04 17:58:17 +03:00
2022-05-22 12:54:02 +03:00
case AnimationDirection :
2024-11-13 12:56:39 +03:00
return setEnumProperty ( properties , AnimationDirection , value , enumProperties [ AnimationDirection ] . values )
2021-10-04 17:58:17 +03:00
default :
ErrorLogF ( ` The "%s" property is not supported by Animation ` , tag )
2021-09-07 17:36:50 +03:00
}
2024-11-13 12:56:39 +03:00
return nil
2021-09-07 17:36:50 +03:00
}
2021-10-04 17:58:17 +03:00
func ( animation * animationData ) String ( ) string {
2022-05-22 12:54:02 +03:00
buffer := allocStringBuilder ( )
defer freeStringBuilder ( buffer )
buffer . WriteString ( "animation {" )
2021-09-07 17:36:50 +03:00
2024-07-06 13:04:12 +03:00
for _ , tag := range animation . AllTags ( ) {
if tag != PropertyTag {
if value , ok := animation . properties [ tag ] ; ok && value != nil {
buffer . WriteString ( "\n\t" )
2024-11-13 12:56:39 +03:00
buffer . WriteString ( string ( tag ) )
2024-07-06 13:04:12 +03:00
buffer . WriteString ( " = " )
writePropertyValue ( buffer , tag , value , "\t" )
buffer . WriteRune ( ',' )
}
}
}
writeProperty := func ( prop AnimatedProperty , indent string ) {
2024-11-13 12:56:39 +03:00
buffer . WriteString ( string ( prop . Tag ) )
2024-07-06 13:04:12 +03:00
buffer . WriteString ( "{\n" )
buffer . WriteString ( indent )
buffer . WriteString ( "from = " )
writePropertyValue ( buffer , "from" , prop . From , indent )
buffer . WriteString ( ",\n" )
buffer . WriteString ( indent )
buffer . WriteString ( "to = " )
writePropertyValue ( buffer , "to" , prop . To , indent )
for key , value := range prop . KeyFrames {
buffer . WriteString ( ",\n" )
buffer . WriteString ( indent )
tag := strconv . Itoa ( key ) + "%"
buffer . WriteString ( tag )
buffer . WriteString ( " = " )
2024-11-13 12:56:39 +03:00
writePropertyValue ( buffer , PropertyName ( tag ) , value , indent )
2024-07-06 13:04:12 +03:00
}
buffer . WriteString ( "\n" )
buffer . WriteString ( indent [ 1 : ] )
buffer . WriteString ( "}" )
}
if props := animation . animatedProperties ( ) ; len ( props ) > 0 {
buffer . WriteString ( "\n\t" )
2024-11-13 12:56:39 +03:00
buffer . WriteString ( string ( PropertyTag ) )
2024-07-06 13:04:12 +03:00
buffer . WriteString ( " = " )
if len ( props ) > 1 {
buffer . WriteString ( "[\n" )
for _ , prop := range props {
buffer . WriteString ( "\t\t" )
writeProperty ( prop , "\t\t\t" )
buffer . WriteString ( ",\n" )
}
buffer . WriteString ( "\t]" )
} else {
writeProperty ( props [ 0 ] , "\t\t" )
}
}
2022-05-22 12:54:02 +03:00
2024-07-06 13:04:12 +03:00
buffer . WriteString ( "\n}" )
2022-05-22 12:54:02 +03:00
return buffer . String ( )
2021-09-07 17:36:50 +03:00
}
2021-10-04 17:58:17 +03:00
func ( animation * animationData ) animationCSS ( session Session ) string {
if animation . keyFramesName == "" {
2024-07-05 16:41:07 +03:00
if props := animation . animatedProperties ( ) ; props != nil {
animation . keyFramesName = session . registerAnimation ( props )
} else {
2021-10-04 17:58:17 +03:00
return ""
}
2021-09-07 17:36:50 +03:00
}
2021-10-04 17:58:17 +03:00
buffer := allocStringBuilder ( )
defer freeStringBuilder ( buffer )
buffer . WriteString ( animation . keyFramesName )
2022-08-18 18:18:36 +03:00
if duration , ok := floatProperty ( animation , Duration , session , 1 ) ; ok && duration > 0 {
2021-10-04 17:58:17 +03:00
buffer . WriteString ( fmt . Sprintf ( " %gs " , duration ) )
} else {
buffer . WriteString ( " 1s " )
}
2024-11-24 09:52:43 +03:00
buffer . WriteString ( timingFunctionCSS ( animation , TimingFunction , session ) )
2021-10-04 17:58:17 +03:00
2022-08-18 18:18:36 +03:00
if delay , ok := floatProperty ( animation , Delay , session , 0 ) ; ok && delay > 0 {
2021-10-04 17:58:17 +03:00
buffer . WriteString ( fmt . Sprintf ( " %gs" , delay ) )
} else {
buffer . WriteString ( " 0s" )
}
if iterationCount , _ := intProperty ( animation , IterationCount , session , 0 ) ; iterationCount >= 0 {
if iterationCount == 0 {
iterationCount = 1
2021-09-07 17:36:50 +03:00
}
2021-10-04 17:58:17 +03:00
buffer . WriteString ( fmt . Sprintf ( " %d " , iterationCount ) )
} else {
buffer . WriteString ( " infinite " )
2021-09-07 17:36:50 +03:00
}
2021-10-04 17:58:17 +03:00
direction , _ := enumProperty ( animation , AnimationDirection , session , 0 )
values := enumProperties [ AnimationDirection ] . cssValues
if direction < 0 || direction >= len ( values ) {
direction = 0
}
buffer . WriteString ( values [ direction ] )
// TODO "animation-fill-mode"
buffer . WriteString ( " forwards" )
return buffer . String ( )
}
func ( animation * animationData ) transitionCSS ( buffer * strings . Builder , session Session ) {
2022-08-18 18:18:36 +03:00
if duration , ok := floatProperty ( animation , Duration , session , 1 ) ; ok && duration > 0 {
2021-10-04 17:58:17 +03:00
buffer . WriteString ( fmt . Sprintf ( " %gs " , duration ) )
} else {
buffer . WriteString ( " 1s " )
}
2024-11-24 09:52:43 +03:00
buffer . WriteString ( timingFunctionCSS ( animation , TimingFunction , session ) )
2021-10-04 17:58:17 +03:00
2022-08-18 18:18:36 +03:00
if delay , ok := floatProperty ( animation , Delay , session , 0 ) ; ok && delay > 0 {
2021-10-04 17:58:17 +03:00
buffer . WriteString ( fmt . Sprintf ( " %gs" , delay ) )
}
}
2024-11-13 12:56:39 +03:00
func ( animation * animationData ) writeTransitionString ( tag PropertyName , buffer * strings . Builder ) {
buffer . WriteString ( string ( tag ) )
2022-05-22 12:54:02 +03:00
buffer . WriteString ( "{" )
lead := " "
2024-11-13 12:56:39 +03:00
writeFloatProperty := func ( name PropertyName ) bool {
2022-05-22 12:54:02 +03:00
if value := animation . getRaw ( name ) ; value != nil {
buffer . WriteString ( lead )
2024-11-13 12:56:39 +03:00
buffer . WriteString ( string ( name ) )
2022-05-22 12:54:02 +03:00
buffer . WriteString ( " = " )
writePropertyValue ( buffer , name , value , "" )
lead = ", "
return true
}
return false
}
if ! writeFloatProperty ( Duration ) {
buffer . WriteString ( " duration = 1" )
lead = ", "
}
writeFloatProperty ( Delay )
if value := animation . getRaw ( TimingFunction ) ; value != nil {
if timingFunction , ok := value . ( string ) ; ok && timingFunction != "" {
buffer . WriteString ( lead )
2024-11-13 12:56:39 +03:00
buffer . WriteString ( string ( TimingFunction ) )
2022-05-22 12:54:02 +03:00
buffer . WriteString ( " = " )
if strings . ContainsAny ( timingFunction , " ,()" ) {
buffer . WriteRune ( '"' )
buffer . WriteString ( timingFunction )
buffer . WriteRune ( '"' )
2022-08-17 16:23:45 +03:00
} else {
buffer . WriteString ( timingFunction )
2022-05-22 12:54:02 +03:00
}
}
}
buffer . WriteString ( " }" )
}
2024-11-24 09:52:43 +03:00
func timingFunctionCSS ( properties Properties , tag PropertyName , session Session ) string {
if timingFunction , ok := stringProperty ( properties , tag , session ) ; ok {
2022-08-11 19:18:36 +03:00
if timingFunction , ok = session . resolveConstants ( timingFunction ) ; ok && isTimingFunctionValid ( timingFunction ) {
2021-10-04 17:58:17 +03:00
return timingFunction
2021-09-07 17:36:50 +03:00
}
2021-10-04 17:58:17 +03:00
}
return ( "ease" )
}
2022-08-11 19:18:36 +03:00
func isTimingFunctionValid ( timingFunction string ) bool {
2021-10-04 17:58:17 +03:00
switch timingFunction {
case "" , EaseTiming , EaseInTiming , EaseOutTiming , EaseInOutTiming , LinearTiming :
return true
}
2022-08-11 19:18:36 +03:00
size := len ( timingFunction )
if size > 0 && timingFunction [ size - 1 ] == ')' {
if index := strings . IndexRune ( timingFunction , '(' ) ; index > 0 {
args := timingFunction [ index + 1 : size - 1 ]
switch timingFunction [ : index ] {
case "steps" :
if _ , err := strconv . Atoi ( strings . Trim ( args , " \t\n" ) ) ; err == nil {
return true
}
2021-10-04 17:58:17 +03:00
2022-08-11 19:18:36 +03:00
case "cubic-bezier" :
if params := strings . Split ( args , "," ) ; len ( params ) == 4 {
for _ , param := range params {
if _ , err := strconv . ParseFloat ( strings . Trim ( param , " \t\n" ) , 64 ) ; err != nil {
return false
2021-10-04 17:58:17 +03:00
}
}
2022-08-11 19:18:36 +03:00
return true
2021-10-04 17:58:17 +03:00
}
2021-09-07 17:36:50 +03:00
}
}
2021-10-04 17:58:17 +03:00
}
return false
}
2022-08-11 19:18:36 +03:00
// IsTimingFunctionValid returns "true" if the "timingFunction" argument is the valid timing function.
func IsTimingFunctionValid ( timingFunction string , session Session ) bool {
if timingFunc , ok := session . resolveConstants ( strings . Trim ( timingFunction , " \t\n" ) ) ; ok {
return isTimingFunctionValid ( timingFunc )
}
return false
}
2021-10-04 17:58:17 +03:00
func ( session * sessionData ) registerAnimation ( props [ ] AnimatedProperty ) string {
session . animationCounter ++
name := fmt . Sprintf ( "kf%06d" , session . animationCounter )
var cssBuilder cssStyleBuilder
2024-07-01 19:17:03 +03:00
cssBuilder . init ( 0 )
2021-10-04 17:58:17 +03:00
cssBuilder . startAnimation ( name )
fromParams := Params { }
toParams := Params { }
frames := [ ] int { }
for _ , prop := range props {
fromParams [ prop . Tag ] = prop . From
toParams [ prop . Tag ] = prop . To
if len ( prop . KeyFrames ) > 0 {
for frame := range prop . KeyFrames {
needAppend := true
for i , n := range frames {
if n == frame {
needAppend = false
break
} else if frame < n {
needAppend = false
frames = append ( append ( frames [ : i ] , frame ) , frames [ i + 1 : ] ... )
break
}
}
if needAppend {
frames = append ( frames , frame )
}
2021-09-07 17:36:50 +03:00
}
}
2021-10-04 17:58:17 +03:00
}
cssBuilder . startAnimationFrame ( "from" )
NewViewStyle ( fromParams ) . cssViewStyle ( & cssBuilder , session )
cssBuilder . endAnimationFrame ( )
if len ( frames ) > 0 {
for _ , frame := range frames {
params := Params { }
for _ , prop := range props {
if prop . KeyFrames != nil {
if value , ok := prop . KeyFrames [ frame ] ; ok {
params [ prop . Tag ] = value
}
}
2021-09-07 17:36:50 +03:00
}
2021-10-04 17:58:17 +03:00
if len ( params ) > 0 {
cssBuilder . startAnimationFrame ( strconv . Itoa ( frame ) + "%" )
NewViewStyle ( params ) . cssViewStyle ( & cssBuilder , session )
cssBuilder . endAnimationFrame ( )
2021-09-07 17:36:50 +03:00
}
}
2021-10-04 17:58:17 +03:00
}
2021-09-07 17:36:50 +03:00
2021-10-04 17:58:17 +03:00
cssBuilder . startAnimationFrame ( "to" )
NewViewStyle ( toParams ) . cssViewStyle ( & cssBuilder , session )
cssBuilder . endAnimationFrame ( )
2021-09-07 17:36:50 +03:00
2021-10-04 17:58:17 +03:00
cssBuilder . endAnimation ( )
2024-07-01 19:17:03 +03:00
session . addAnimationCSS ( cssBuilder . finish ( ) )
2021-09-07 17:36:50 +03:00
2021-10-04 17:58:17 +03:00
return name
}
2021-09-07 17:36:50 +03:00
2024-12-06 19:15:23 +03:00
func ( view * viewData ) SetAnimated ( tag PropertyName , value any , animation AnimationProperty ) bool {
2021-10-04 17:58:17 +03:00
if animation == nil {
return view . Set ( tag , value )
}
2021-09-07 17:36:50 +03:00
2022-07-22 13:10:55 +03:00
session := view . Session ( )
htmlID := view . htmlID ( )
session . startUpdateScript ( htmlID )
2022-10-30 17:22:33 +03:00
session . updateProperty ( htmlID , "ontransitionend" , "transitionEndEvent(this, event)" )
session . updateProperty ( htmlID , "ontransitioncancel" , "transitionCancelEvent(this, event)" )
2021-10-04 17:58:17 +03:00
2024-11-13 12:56:39 +03:00
transitions := getTransitionProperty ( view )
2024-12-06 19:15:23 +03:00
var prevAnimation AnimationProperty = nil
2024-11-13 12:56:39 +03:00
if transitions != nil {
if prev , ok := transitions [ tag ] ; ok {
prevAnimation = prev
}
2021-10-04 17:58:17 +03:00
}
2024-11-13 12:56:39 +03:00
view . singleTransition [ tag ] = prevAnimation
setTransition ( view , tag , animation )
view . session . updateCSSProperty ( view . htmlID ( ) , "transition" , transitionCSS ( view , view . session ) )
2021-10-04 17:58:17 +03:00
2022-07-22 13:10:55 +03:00
session . finishUpdateScript ( htmlID )
2021-10-04 17:58:17 +03:00
result := view . Set ( tag , value )
if ! result {
delete ( view . singleTransition , tag )
2024-11-13 12:56:39 +03:00
view . session . updateCSSProperty ( view . htmlID ( ) , "transition" , transitionCSS ( view , view . session ) )
2021-10-04 17:58:17 +03:00
}
return result
}
2024-11-13 12:56:39 +03:00
func animationCSS ( properties Properties , session Session ) string {
2024-12-06 19:15:23 +03:00
if value := properties . getRaw ( Animation ) ; value != nil {
if animations , ok := value . ( [ ] AnimationProperty ) ; ok {
2021-10-04 17:58:17 +03:00
buffer := allocStringBuilder ( )
defer freeStringBuilder ( buffer )
for _ , animation := range animations {
if css := animation . animationCSS ( session ) ; css != "" {
if buffer . Len ( ) > 0 {
buffer . WriteString ( ", " )
}
buffer . WriteString ( css )
2021-09-07 17:36:50 +03:00
}
}
2021-10-04 17:58:17 +03:00
return buffer . String ( )
2021-09-07 17:36:50 +03:00
}
}
2021-10-04 17:58:17 +03:00
2021-09-07 17:36:50 +03:00
return ""
}
2024-11-13 12:56:39 +03:00
func transitionCSS ( properties Properties , session Session ) string {
if transitions := getTransitionProperty ( properties ) ; len ( transitions ) > 0 {
buffer := allocStringBuilder ( )
defer freeStringBuilder ( buffer )
convert := map [ PropertyName ] string {
CellHeight : "grid-template-rows" ,
CellWidth : "grid-template-columns" ,
Row : "grid-row" ,
Column : "grid-column" ,
Clip : "clip-path" ,
Shadow : "box-shadow" ,
ColumnSeparator : "column-rule" ,
FontName : "font" ,
TextSize : "font-size" ,
TextLineThickness : "text-decoration-thickness" ,
2021-09-07 17:36:50 +03:00
}
2022-08-17 16:58:07 +03:00
2024-11-13 12:56:39 +03:00
for tag , animation := range transitions {
if buffer . Len ( ) > 0 {
buffer . WriteString ( ", " )
}
if cssTag , ok := convert [ tag ] ; ok {
buffer . WriteString ( cssTag )
} else {
buffer . WriteString ( string ( tag ) )
}
animation . transitionCSS ( buffer , session )
2022-08-17 16:58:07 +03:00
}
2024-11-13 12:56:39 +03:00
return buffer . String ( )
2021-10-04 17:58:17 +03:00
}
2024-11-13 12:56:39 +03:00
return ""
2021-10-04 17:58:17 +03:00
}
2021-09-07 17:36:50 +03:00
2024-11-13 12:56:39 +03:00
/ *
2021-10-04 17:58:17 +03:00
func ( view * viewData ) updateTransitionCSS ( ) {
2024-11-13 12:56:39 +03:00
view . session . updateCSSProperty ( view . htmlID ( ) , "transition" , transitionCSS ( view , view . session ) )
2021-10-04 17:58:17 +03:00
}
2024-11-13 12:56:39 +03:00
* /
2021-09-07 17:36:50 +03:00
2024-12-06 19:15:23 +03:00
func ( style * viewStyle ) Transition ( tag PropertyName ) AnimationProperty {
2024-11-13 12:56:39 +03:00
if transitions := getTransitionProperty ( style ) ; transitions != nil {
if anim , ok := transitions [ tag ] ; ok {
2022-08-10 15:36:38 +03:00
return anim
}
}
return nil
}
2024-12-06 19:15:23 +03:00
func ( style * viewStyle ) Transitions ( ) map [ PropertyName ] AnimationProperty {
result := map [ PropertyName ] AnimationProperty { }
2024-11-13 12:56:39 +03:00
for tag , animation := range getTransitionProperty ( style ) {
2021-10-04 17:58:17 +03:00
result [ tag ] = animation
}
return result
}
2021-09-07 17:36:50 +03:00
2024-12-06 19:15:23 +03:00
func ( style * viewStyle ) SetTransition ( tag PropertyName , animation AnimationProperty ) {
2024-11-13 12:56:39 +03:00
setTransition ( style , style . normalize ( tag ) , animation )
}
2024-12-06 19:15:23 +03:00
func ( view * viewData ) SetTransition ( tag PropertyName , animation AnimationProperty ) {
2024-11-13 12:56:39 +03:00
setTransition ( view , view . normalize ( tag ) , animation )
if view . created {
view . session . updateCSSProperty ( view . htmlID ( ) , "transition" , transitionCSS ( view , view . session ) )
}
}
2024-12-06 19:15:23 +03:00
func setTransition ( properties Properties , tag PropertyName , animation AnimationProperty ) {
2024-11-13 12:56:39 +03:00
transitions := getTransitionProperty ( properties )
2022-08-10 15:36:38 +03:00
if animation == nil {
2024-11-13 12:56:39 +03:00
if transitions != nil {
delete ( transitions , tag )
if len ( transitions ) == 0 {
properties . setRaw ( Transition , nil )
}
}
} else if transitions != nil {
transitions [ tag ] = animation
2022-08-10 15:36:38 +03:00
} else {
2024-12-06 19:15:23 +03:00
properties . setRaw ( Transition , map [ PropertyName ] AnimationProperty { tag : animation } )
2022-08-10 15:36:38 +03:00
}
}
2024-12-06 19:15:23 +03:00
func getTransitionProperty ( properties Properties ) map [ PropertyName ] AnimationProperty {
2024-11-13 12:56:39 +03:00
if value := properties . getRaw ( Transition ) ; value != nil {
2024-12-06 19:15:23 +03:00
if transitions , ok := value . ( map [ PropertyName ] AnimationProperty ) ; ok {
2024-11-13 12:56:39 +03:00
return transitions
}
}
return nil
}
func setAnimationProperty ( properties Properties , tag PropertyName , value any ) bool {
2024-12-06 19:15:23 +03:00
set := func ( animations [ ] AnimationProperty ) {
2024-11-13 12:56:39 +03:00
properties . setRaw ( tag , animations )
for _ , animation := range animations {
animation . used ( )
}
}
switch value := value . ( type ) {
2024-12-06 19:15:23 +03:00
case AnimationProperty :
set ( [ ] AnimationProperty { value } )
2024-11-13 12:56:39 +03:00
return true
2024-12-06 19:15:23 +03:00
case [ ] AnimationProperty :
2024-11-13 12:56:39 +03:00
set ( value )
return true
case DataObject :
if animation := parseAnimation ( value ) ; animation . hasAnimatedProperty ( ) {
2024-12-06 19:15:23 +03:00
set ( [ ] AnimationProperty { animation } )
2024-11-13 12:56:39 +03:00
return true
}
case DataNode :
2024-12-06 19:15:23 +03:00
animations := [ ] AnimationProperty { }
2024-11-13 12:56:39 +03:00
result := true
for i := 0 ; i < value . ArraySize ( ) ; i ++ {
if obj := value . ArrayElement ( i ) . Object ( ) ; obj != nil {
if anim := parseAnimation ( obj ) ; anim . hasAnimatedProperty ( ) {
animations = append ( animations , anim )
} else {
result = false
}
} else {
notCompatibleType ( tag , value . ArrayElement ( i ) )
result = false
}
}
if result && len ( animations ) > 0 {
set ( animations )
}
return result
2022-08-10 15:36:38 +03:00
}
2024-11-13 12:56:39 +03:00
notCompatibleType ( tag , value )
return false
2022-08-10 15:36:38 +03:00
}
2021-10-04 17:58:17 +03:00
// SetAnimated sets the property with name "tag" of the "rootView" subview with "viewID" id by value. Result:
2021-11-04 21:13:34 +03:00
// true - success,
// false - error (incompatible type or invalid format of a string value, see AppLog).
2024-12-06 19:15:23 +03:00
func SetAnimated ( rootView View , viewID string , tag PropertyName , value any , animation AnimationProperty ) bool {
2021-10-04 17:58:17 +03:00
if view := ViewByID ( rootView , viewID ) ; view != nil {
return view . SetAnimated ( tag , value , animation )
}
return false
}
// IsAnimationPaused returns "true" if an animation of the subview is paused, "false" otherwise.
2022-08-31 22:17:46 +03:00
// If the second argument (subviewID) is not specified or it is "" then a value from the first argument (view) is returned.
func IsAnimationPaused ( view View , subviewID ... string ) bool {
2022-07-28 12:11:27 +03:00
return boolStyledProperty ( view , subviewID , AnimationPaused , false )
2021-10-04 17:58:17 +03:00
}
2021-09-07 17:36:50 +03:00
2022-08-10 15:36:38 +03:00
// GetTransitions returns the subview transitions. The result is always non-nil.
2022-08-31 22:17:46 +03:00
// If the second argument (subviewID) is not specified or it is "" then transitions of the first argument (view) is returned
2024-12-06 19:15:23 +03:00
func GetTransitions ( view View , subviewID ... string ) map [ PropertyName ] AnimationProperty {
2024-11-24 16:43:31 +03:00
if view = getSubview ( view , subviewID ) ; view != nil {
2022-08-10 15:36:38 +03:00
return view . Transitions ( )
2021-10-04 17:58:17 +03:00
}
2024-12-06 19:15:23 +03:00
return map [ PropertyName ] AnimationProperty { }
2021-10-04 17:58:17 +03:00
}
2022-08-10 15:36:38 +03:00
// GetTransition returns the subview property transition. If there is no transition for the given property then nil is returned.
2022-08-31 22:17:46 +03:00
// If the second argument (subviewID) is not specified or it is "" then transitions of the first argument (view) is returned
2024-12-06 19:15:23 +03:00
func GetTransition ( view View , subviewID string , tag PropertyName ) AnimationProperty {
2021-10-04 17:58:17 +03:00
if subviewID != "" {
view = ViewByID ( view , subviewID )
}
2022-08-10 15:36:38 +03:00
if view != nil {
return view . Transition ( tag )
2021-10-04 17:58:17 +03:00
}
2022-08-10 15:36:38 +03:00
return nil
}
// AddTransition adds the transition for the subview property.
2022-08-31 22:17:46 +03:00
// If the second argument (subviewID) is not specified or it is "" then the transition is added to the first argument (view)
2024-12-06 19:15:23 +03:00
func AddTransition ( view View , subviewID string , tag PropertyName , animation AnimationProperty ) bool {
2022-08-10 15:36:38 +03:00
if tag != "" {
if subviewID != "" {
view = ViewByID ( view , subviewID )
}
if view != nil {
view . SetTransition ( tag , animation )
return true
}
}
return false
2021-10-04 17:58:17 +03:00
}
// GetAnimation returns the subview animations. The result is always non-nil.
2022-08-31 22:17:46 +03:00
// If the second argument (subviewID) is not specified or it is "" then transitions of the first argument (view) is returned
2024-12-06 19:15:23 +03:00
func GetAnimation ( view View , subviewID ... string ) [ ] AnimationProperty {
2024-11-24 16:43:31 +03:00
if view = getSubview ( view , subviewID ) ; view != nil {
2024-12-06 19:15:23 +03:00
if value := view . getRaw ( Animation ) ; value != nil {
if animations , ok := value . ( [ ] AnimationProperty ) ; ok && animations != nil {
2021-10-04 17:58:17 +03:00
return animations
2021-09-07 17:36:50 +03:00
}
}
}
2024-12-06 19:15:23 +03:00
return [ ] AnimationProperty { }
2021-09-07 17:36:50 +03:00
}