ConditionalFilter

From Avisynth wiki
Revision as of 17:01, 25 November 2021 by Pinterf (Talk | contribs)

Jump to: navigation, search

Contents

ConditionalFilter

ConditionalFilter(clip testclip, clip source1, clip source2, string expression1, string operator, string expression2 [, bool show] ) AVS+ConditionalFilter(clip testclip, clip source1, clip source2, string expression1 [, bool show] )


ConditionalFilter returns source1 when the condition formed by expression1+operator+expression2 is met for current frame, otherwise it returns source2. If any function in expression1 or expression2 is not explicitly applied to a clip, it will be applied on testclip. The audio is taken from source1.

AVS+The second form in Avisynth + is an abbreviated version which takes only four compulsory parameters, works like operator is "=" and expression2 is "true", similar to GConditionalFilter

An example: this will choose frames from vid_blur when the average luma value of a frame is less than 20. Otherwise frames from vid will be returned.

vid = AviSource("file")
vid_blur = vid.Blur(1.5)
ConditionalFilter(vid, vid_blur, vid, "AverageLuma()", "lessthan", "20")

Adding show="true" will display the actual values on the screen.

The strings expression1 and expression2 can be any numeric or boolean expressions, and may include internal or user functions, as well as some additional functions which are predefined (see Runtime Functions) and the special runtime variable current_frame (the frame number of the requested frame).

The string operator must be one of the following strings:

  • "greaterthan" or ">"
  • "lessthan" or "<"
  • "equals", "=" or "=="

ConditionalSelect

ConditionalSelect(clip testclip, string expression, clip source0 [, clip source1...] [, bool show] )

ConditionalSelect returns each one frame from several source clips based on an integer evaluator. If the expression evaluates to the integer j (starting at zero), the frame from the j-th source clip is returned. source0, ..., sourcen can be an array of clips.

This will return a frame from vid_blur2 when the average luma value of a frame (of vid) is less than 15, will return a frame from vid_blur when the average luma value of a frame (of vid) is higher than 15 but smaller than 25. Otherwise a frame from vid will be returned.

vid = AviSource("file")
vid_blur = vid.Blur(1.0)
vid_blur2 = vid.Blur(1.5)
ConditionalSelect(vid, 
\  "luma_av = AverageLuma()"+chr(13)+"luma_av<25 ? (luma_av<15 ? 2 : 1) : 0", 
\  vid, vid_blur, vid_blur2)

Adding show=true will display the actual values on the screen.

If a frame is requested from a non-existing source clip (say the expression evaluates to -1 or 3 in the example above, where 3 source clips are supplied), testclip will be returned.

Audio from testclip is passed through untouched.


ScriptClip

ScriptClip(clip clip, string filter [, bool show, bool after_frame, AVS+ bool local]
AVS+ ScriptClip(clip clip, function func [, bool show, bool after_frame, bool local] )

ScriptClip returns the clip returned by the filter evaluated on every frame. The string filter can be any expression returning a clip, including internal or user clip functions, and may include line breaks, allowing a sequence of statements to be evaluated. Also, some functions which are predefined (see Runtime Functions) and the special runtime variable current_frame (the frame number of the requested frame) can be used in the filter expression. Adding show=true will display the actual values on the screen.

Some examples:

This will print the difference from the previous frame onto the current one:

clip = AviSource("c:\file.avi")
ScriptClip(clip, "Subtitle(String(YDifferenceFromPrevious))")

This will apply blur on each frame based on the difference from the previous. This will also show how errors are reported on some frames:

clip = AviSource("c:\file.avi")
ScriptClip(clip, "Blur(YDifferenceFromPrevious/20.0)")

This will apply TemporalSoften to very static scenes, and apply a variable blur on moving scenes. Blur is now capped properly. We also assign a variable - and this is why a line break is inserted:

function fmin(float f1, float f2) {
  return (f1<f2) ? f1 : f2
}
 
clip = AviSource("c:\file.avi")
T = clip.TemporalSoften(2, 7, 7, 3, 2)
ScriptClip(clip, "diff = YDifferenceToNext()"+chr(13)
\          +"diff > 2.5 ? Blur(fmin(diff/20,1.5)) : T")

Shows the frame-number in a clip:

ScriptClip("subtitle(string(current_frame))")

Shows 'frame = the frame-number' in a clip:

mpeg2source("F:\From_hell\from_hell.d2v").trim(100,124)
ScriptClip("""subtitle("frame = " + string(current_frame))""")

If after_frame=true, the script should be evaluated after the frame has been fetched from the filters above. Default is false.

Restrictions - the output of the script MUST be exactly like the clip delivered to ScriptClip (same colorspace, width and height). Your returned clip is allowed to have different length - but the length from clip is always used. Audio from clip is passed through untouched. For two very different sources (MPEG2DEC3 and AviSource) - you might run into colorspace mismatches. This is a known quirk.

AVS+ Avisynth+ specialities

There were always problems with ScriptClip, namely variable stability in multithreading. Compared to Avisynth Neo - which knew no mercy and made it correct but incompatible with old scripts - Avisynth+ extended it a bit when new ScriptClip features were ported back. Regarding visibility of variables: Avisynth+ changed back Neo's new default behaviour, which was similar to GRunT's local=true and resulted in incompatibility with some old scripts. So scripts written on the assumption of local = false did not work. Neo's (valid) point: they would not allow the following script to show the behavior to print "3".

     # prints 3 - Avisynth default but seems incorrect
     global foo=2
     function PrintFoo(clip c)
     { c.ScriptClip("Subtitle(string(foo))", local = false) }
     Version()
     PrintFoo()
     foo = 3
     last
     # prints 2 - Neo's default behaviour
     global foo=2
     function PrintFoo(clip c)
     { c.ScriptClip("Subtitle(string(foo))", local = true) }
     Version()
     PrintFoo()
     foo = 3
     last

So all runtime filters* accept a bool "local" parameter which acts same as in GRunT.

  • ConditionalSelect, ConditionalFilter, ScriptClip, ConditionalReader, FrameEvaluate, WriteFile, WriteFileIf, WriteFileStart, WriteFileEnd

If local=true (function-syntax default) the filter will evaluate its run-time script in a new variable scope (opens a new global variable frame), avoiding unintended sharing of variables between run-time scripts.

In our present Avisynth+ all runtime filters are compatible with the usual behaviour, but one can set the other mode by "local=true" parameter. Functions have a stricter variable scope. See the examples below.

Example on 'after_frame' and global variable visibility

 ColorbarsHD()
 Expr("frameno","","") # luma is 0..255 for the first 256 frames.
 # A global variable which ShowChannel is overwriting in each frame
 global SC_LMn_0 = 2 
 # ShowChannels (an external filter) sets some global variables 
 # (frame min/max/etc. statistics) e.g. "SC_LMn_0" in each frame
 ShowChannels(SetVar=True,show=false)  
 
 # Choose test method
 #method=0 # String-syntax
 method=1 # Function-Syntax
  
 if(method==0) {
   # "string"-syntax ScriptClip
   # 'local'=false is the default working mode for
   # 'after_frame'=true: script should be evaluated AFTER frame has been fetched, because we'd like to display the values set by ShowChannels
   ScriptClip( """subtitle(string(SC_LMn_0) + " current frame=" + String(current_frame)) """, after_frame = true)
    
   # Example of delayed display:
   # Statistics are from previously displayed frame which can be totally off when moving a slider in VirtualDub.
   # The reason of this asynchronous mode is because child (upper) script - ShowChannels in this case - runs _after_ the string 
   # evaluation. 'after_frame' is false by default, it is not necessary to specify, it's here for the sake of the example.
   # ScriptClip( """subtitle(string(SC_LMn_0) + " current frame=" + String(current_frame) ) """, after_frame = false)
 } else {
   # function-syntax ScriptClip
   # Here 'local'=false must be set because local=true is the default for this mode.
   # A 'function' inside a ScriptClip should have one clip argument or no argument.
   # 'c' is a parameter which must be passed to the function. Name is not important, it moves the actual clip into function's scope.
   # This is why we can SubTitle on it.
   # 'after_frame'=true: script should be evaluated AFTER frame has been fetched, because we'd like to display the values set by ShowChannels
   # A function can see only global variables, so it will see the value of 'SC_LMn_0' which ShowChannels set previously.
   ScriptClip( function [] (c) {c.subtitle(string(SC_LMn_0) ) } , local = false, after_frame = true)
 }

Function-like syntax: example on local=false

 # local=true -> new GlobalVar frame is created, 'current_frame' and 'last' is created in it, they are visible
 ScriptClip( function [] (c) {c.subtitle(string(current_frame) ) } , local = true) 

Function-like syntax: example on local=true

 # Error: "I don't know what 'current_frame' means.
 # local=false -> 'current_frame' and 'last' are set as a simple variables, but regular variables are not visible inside the function, only globals
 ScriptClip( function [] (c) { c.subtitle(string(current_frame) ) } , local = false) 

So when the script parameter is not a string but a function (see Function Objects - http://avisynth.nl/index.php/Function_objects) then classic Avisynth+ works in a MT correct way and the default value of "local" is true.


As a memo, ConditionalFilter still has the GRunT-like "shortcut" style parameter for bool results which implies "=" and "true" and parameter "condvarsuffix" for ConditionalReader)

See also: UseVar, special filter, opens a clean variable environment in which only the variables in the parameter list can be seen.

FrameEvaluate

FrameEvaluate(clip clip, string filter [, bool show, bool after_frame] )

Similar to ScriptClip, except the output of the filter is ignored. This can be used for assigning variables, etc. Frames are passed directly through from the supplied clip.

If after_frame=true, the script should be evaluated after the frame has been fetched from the filters above. Default is false.


ConditionalReader

ConditionalReader(clip clip, string filename, string variablename [, bool show] )

This filter allows you to import arbitrary information into a selectable variable.

See the dedicated ConditionalReader page.


Runtime Functions

These are the internal functions which are evaluated every frame.

These will return the average pixel value of a plane (requires ISSE):

AverageLuma(clip)
AverageChromaU(clip)
AverageChromaV(clip)

These return a float value between 0 and 255 of the absolute difference between two planes (requires ISSE): :RGBDifference(clip1, clip2)

LumaDifference(clip1, clip2)
ChromaUDifference(clip1, clip2)
ChromaVDifference(clip1, clip2)

When using these functions there is an "implicit last" clip (first parameter doesn't have to be specified), so the first parameter is replaced by the testclip.

These should be quite handy for detecting scene change transitions:

RGBDifferenceFromPrevious(clip)
YDifferenceFromPrevious(clip)
UDifferenceFromPrevious(clip)
VDifferenceFromPrevious(clip)
RGBDifferenceToNext(clip)
YDifferenceToNext(clip)
UDifferenceToNext(clip)
VDifferenceToNext(clip)

This will replace the last frame before a scene change with the first frame after the scene change:

ConditionalFilter(last, last, last.trim(1,0), 
\         "YDifferenceToNext()", ">", "10", true)

Other internal functions:

YPlaneMax(clip, float threshold)
UPlaneMax(clip, float threshold)
VPlaneMax(clip, float threshold)
YPlaneMin(clip, float threshold)
UPlaneMin(clip, float threshold)
VPlaneMin(clip, float threshold)
YPlaneMedian(clip)
UPlaneMedian(clip)
VPlaneMedian(clip)
YPlaneMinMaxDifference(clip, float threshold)
UPlaneMinMaxDifference(clip, float threshold)
VPlaneMinMaxDifference(clip, float threshold)

Threshold is a percentage, on how many percent of the pixels are allowed above or below minimum. The threshold is optional and defaults to 0.

If you understand the stuff above, you can proceed with "advanced conditional filtering", which tells you a little bit more about conditional filtering.


Advanced conditional filtering: part I

You will have to know a few things about the functionality of AviSynth to understand this section: Scripts are parsed from top to bottom, but when a frame is requested the last filter is actually being invoked first, requesting frames upwards in the filter chain. For example:

AviSource("myfile.avi")
ColorYUV(analyze=true)
Histogram()

When opening the script in VirtualDub the following happens:

  • When Vdub requests a frame, AviSynth requests the frame from Histogram.
  • Histogram requests a frame from ColorYUV,
  • ColorYUV in turn requests a frame from AviSource, which produces the frame, and delivers it to ColorYUV.
  • ColorYUV processes the image and sends it on to Histogram, which returns it to Virtualdub.

So the filter chain basically works backwards (the output is 'pulled' from below rather than 'pushed' from above), which gives each filter the possibility to request several frames from the source above. Conditional filters however, need to evaluate scripts before they request frames from the filter above, because they need to know which filter to call. Another important issue is that run-time scripts are evaluated in the same context as the main script. Hence only global defined variables in the conditional filter 'environment' can be used inside a function (and vice versa). Have a look at the following script:

v = AviSource("E:\Temp\Test3\atomic_kitten.avi").ConvertToYV12

function g(clip c)
{
  global w = c
  c2 = ScriptClip(c, "subtitle(t)")
  c3 = FrameEvaluate(c2, "t = String(text)")
  c4 = FrameEvaluate(c3, "text = YDifferenceFromPrevious(w)")
  return c4
}

g(v)

This filter chain works like this:

  • When Vdub requests a frame, AviSynth requests a frame from the second FrameEvaluate, the last filter in the chain generated by g().
  • The second FrameEvaluate evaluates YDifferenceFromPrevious(w), which leads to the following actions:
    • YDifferenceFromPrevious requests a frame from ConvertToYV12;
    • ConvertToYV12 requests a frame from AviSource, which produces the frame, and delivers it to ConvertToYV12;
    • ConvertToYV12 processes the image and returns it to YDifferenceFromPrevious;
    • YDifferenceFromPrevious requests a second frame from ConvertToYV12, which is obtained in a similar way to the first;
    • It then compares the two frames to calculate its result which it delivers to FrameEvaluate.
  • FrameEvaluate assigns this value to the variable text.
  • After this a frame is requested from the first FrameEvaluate.
  • The first FrameEvaluate, after evaluating String(text) and assigning this value to the variable t, requests a frame from ScriptClip.
  • ScriptClip sets last to the result of ConvertToYV12(), evaluates Subtitle(t) (creating a new, temporary, filter chain), and requests a frame from it.
    • Subtitle requests a frame from ConvertToYV12;
    • ConvertToYV12 requests a frame from AviSource, which produces the frame, and delivers it to ConvertToYV12;
    • ConvertToYV12 processes the image and returns it to Subtitle;
    • Subtitle adds the specified text to the frame and delivers the result to ScriptClip.
  • ScriptClip returns the subtitled frame to the first FrameEvaluate.
  • In turn this frame is returned to the second FrameEvaluate, and hence to Avisynth which returns it to VDub.

Notice how the addition of run-time filters and run-time functions makes the interactions between different parts of the filter chain more complex. This added complexity is managed internally by Avisynth, so you needn't worry about it. However, care is required when setting and using variables, as the order of events can be less obvious to the script writer (you!).

As can be seen, w is defined as a global variable. This way we can use it later in the script in the conditional environment. If we want to use the variables t and text in a different function (inside or outside the conditional environment), they must also be defined as global variables. Thus for example:

v = AviSource("E:\Temp\Test3\atomic_kitten.avi").ConvertToYV12
function g(clip c)
{
  global w = c
  c2 = ScriptClip(c, "subtitle(t)")
  c3 = FrameEvaluate(c2, "me()")
  c4 = FrameEvaluate(c3, "global text = YDifferenceFromPrevious(w)")
  return c4
}

function me()
{
  global t = String(text)
}

g(v)

This is just an illustration to demonstrate the various features. Much of the script above is redundant, and can be removed. The following two scripts give the same output

v = AviSource("c:\clip.avi")
# ScriptClip accepts multi-line scripts:
Scriptclip(v,"
	text = YDifferenceFromPrevious()
	t = string(text)
	subtitle(t)
")
v = AviSource("c:\clip.avi")
ScriptClip(v, "Subtitle(String(YDifferenceFromPrevious))")

In the following section an example of "adaptive motion/resizing filter" will be considered.


Advanced conditional filtering: part II

In the following example, some frame dependent info will be written to a text-file. The first variable "a" indicates whether the frame is combed (for a certain threshold). Note that IsCombed is a filter from the Decomb plugin. The second variable "b" indicates whether there is "much" movement in the frame.

global sep="."
global combedthreshold=25

function IsMoving()
{
  global b = (diff < 1.0) ? false : true
}

function CombingInfo(clip c)
{
 file = "F:\interlace.log"
 global clip = c

 c = WriteFile(c, file, "a", "sep", "b")
 c = FrameEvaluate(c, "global a = IsCombed(clip,combedthreshold)")
 c = FrameEvaluate(c, "IsMoving")
 c = FrameEvaluate(c,
 \      "global diff = 0.50*YDifferenceFromPrevious(clip) + "
 \    + "0.25*UDifferenceFromPrevious(clip) + "
 \    + "0.25*VDifferenceFromPrevious(clip)"
 \ ) 
 return c
}

v = mpeg2source("F:\From_hell\from_hell.d2v").trim(100,124)
CombingInfo(v)

We can tidy up the two functions, and remove global variables, by writing them as follows:

function IsMoving(float diff)
{
  return (diff >= 1.0)
}

function CombingInfo(clip c)
{
 file = "F:\interlace.log"

 c = FrameEvaluate(c,"
       diff = 0.50*YDifferenceFromPrevious()
       \  + 0.25*UDifferenceFromPrevious() 
       \  + 0.25*VDifferenceFromPrevious()
       b = IsMoving(diff)
       a = IsCombed(combedthreshold)
     ")
 c = WriteFile(c, file, "a", "sep", "b")

 return c
}


Advanced conditional filtering: part III

Some adaptive motion/resizing filters appeared on the forums. These filters discriminate between low, medium and high motion in a clip (on frame basis). By doing that, different filters can be used for different kind of motion in the clip. In general, one should use temporal smoothing in low motion scenes, spatial smoothing in high motion scenes and use spatio-temporal smoothing in medium motion scenes.

Below, a simplified version of QUANTIFIED MOTION FILTER v1.5 b1 (10/07/2003) by HomiE FR, is given:

# QUANTIFIED MOTION FILTER v1.3 
# LOADING AVISYNTH PLUGINS 
LoadPlugin("C:\PROGRA~1\GORDIA~1\mpeg2dec3.dll") 
LoadPlugin("C:\PROGRA~1\GORDIA~1\TemporalCleaner.dll") 
LoadPlugin("C:\PROGRA~1\GORDIA~1\FluxSmooth.dll") 
LoadPlugin("C:\PROGRA~1\GORDIA~1\UnFilter.dll") 

# LOADING QUANTIFIED MOTION FILTER SCRIPT
Import("E:\temp\QMF\qmf.avs")

# LOW MOTION FILTER FUNCTION
# -> SHARP RESIZING + TEMPORAL ONLY
function Low_Motion_Filter(clip c)
{
  c = TemporalCleaner(c, 5, 10)
  c = LanczosResize(c, 512, 272)
  return c
}

# MEDIUM MOTION FILTER FUNCTION
# -> NEUTRAL BICUBIC RESIZING + TEMPORAL & SPATIAL
function Medium_Motion_Filter(clip c)
{
  c = FluxSmooth(c, 7, 7)
  c = BicubicResize(c, 512, 272, 0.00, 0.50)
  return c
}

# HIGH MOTION FILTER FUNCTION
# -> SOFT RESIZING + SPATIAL ONLY
function High_Motion_Filter(clip c)
{
  c = FluxSmooth(c, -1, 14)
  c = UnFilter(c, -30, -30)
  c = BilinearResize(c, 512, 272)
  return c
}

# OPENING VIDEO SOURCE
AviSource("E:\temp\QMF\britney-I_love_rock_'n_roll.avi")
ConvertToYV12(interlaced=true)
Telecide(0)

# APPLYING ADAPTATIVE RESIZING FILTER (USING QMF)
QMF()

Put the following script in a separate file, called qmf.avs:

# QUANTIFIED MOTION FILTER (17/08/2003) by HomiE FR (homie.fr@wanadoo.fr)
# MOTION ESTIMATION FUNCTION
function ME()
{
  # SETTING MOTION LEVEL ACCORDING TO AVERAGE DIFFERENCE [1]
  global motion_level = (diff < threshold_lm) ? 0 : motion_level
  global motion_level = (diff >= threshold_lm && diff <= threshold_hm) ? 1 : motion_level
  global motion_level = (diff > threshold_hm) ? 2 : motion_level
}

# QUANTIFIED MOTION FILTER FUNCTION
function QMF(clip c, float "threshold_lm", float "threshold_hm", bool "debug")
{
  # SETTING MOTION LEVELS THRESHOLDS [2]
  threshold_lm = default(threshold_lm, 4.0)
  threshold_hm = default(threshold_hm, 12.0)
  global threshold_lm = threshold_lm
  global threshold_hm = threshold_hm

  # ENABLING/DISABLING DEBUG INFORMATION [3]
  debug = default(debug, false)

  # INITIALIZING MOTION LEVEL
  global motion_level = 0

  # SETTING PRESENT CLIP [4]
  global clip = c

  # GETTING OUTPUT RESOLUTION [5]
  width = Width(Low_Motion_Filter(c))
  height = Height(Low_Motion_Filter(c))
  global c_resized = PointResize(c, width, height)

  # APPLYING MOTION FILTER ACCORDING TO MOTION LEVEL [6]
  c = ConditionalFilter(c, Low_Motion_Filter(c), c_resized, "motion_level", "=", "0")  # [6a]
  c = ConditionalFilter(c, Medium_Motion_Filter(c), c, "motion_level", "=", "1")       # [6b]
  c = ConditionalFilter(c, High_Motion_Filter(c), c, "motion_level", "=", "2")         # [6c]

  # PRINTING DEBUG INFORMATION [7]
  c = (debug == true) ? ScriptClip(c, "Debug()") : c

  # GETTING MOTION LEVEL THROUGH MOTION ESTIMATION [8]
  c = FrameEvaluate(c, "ME()")

  # GETTING DIFFERENCES BETWEEN PAST/PRESENT FRAMES [9]
  c = FrameEvaluate(c, "global diff = 0.50*YDifferenceFromPrevious(clip)
      \ + 0.25*UDifferenceFromPrevious(clip) + 0.25*VDifferenceFromPrevious(clip)")
  return c
}

# DEBUG INFORMATION FUNCTION
function Debug(clip c)
{
  # PRINTING VERSION INFORMATION [10]
  c = Subtitle(c, "Quantified Motion Filter", x=20, y=30, font="lucida console", 
      \ size=18, text_color=$FFFFFF)
  c = Subtitle(c, "by HomiE FR (homie.fr@wanadoo.fr)", x=20, y=45, font="lucida console",
      \ size=14, text_color=$FFFFFF)
 
  # PRINTING MOTION ESTIMATION INFORMATION [11]
  c = Subtitle(c, "motion estimation", x=20, y=85, font="lucida console", size=18, 
      \ text_color=$FFFFFF)
  c = Subtitle(c, "diff = "+string(diff), x=20,y=110, font="lucida console", size=16, 
      \ text_color=$FFCCCC)
 
  # PRINTING QUANTIFIED MOTION FILTER INFORMATION [12]
  c = Subtitle(c, "quantified motion filter", x=20, y=135, font="lucida console", 
      \ size=18, text_color=$FFFFFF)
  c = (motion_level == 0) ? Subtitle(c, "scene type = low motion", x=20, y=160, 
      \ font="lucida console", size=16, text_color=$66FF66) : c
  c = (motion_level == 1) ? Subtitle(c, "scene type = medium motion", x=20, y=160, 
      \ font="lucida console", size=16, text_color=$66FF66) : c
  c = (motion_level == 2) ? Subtitle(c, "scene type = high motion", x=20, y=160, 
      \ font="lucida console", size=16, text_color=$66FF66) : c
  return c
}

This filter chain works like this:

  • When Vdub requests a frame, AviSynth requests a frame from QMF.
    • QMF request a frame from FrameEvaluate [9].
    • After doing this the script [9] is evaluated, and the global variable diff is assigned after requesting a frame from AviSource. FrameEvaluate [9] requests a frame from FrameEvaluate [8].
    • Once again the script [8] is evaluated:
      • when evaluating me(), the global variable motion_level is assigned for that frame [1]
    • If debug=true, a frame is requested from ScriptClip [7], and thus from Debug().
    • After that (and also when debug was set to false) a frame is requested from the last ConditionalFilter [6c], which requests a frame from [6b], which in turn requests a frame from [6a].
      • Note that in the end, a frame of High_Motion_filter, Medium_Motion_filter, or Low_Motion_filter is requested depending on the value of motion_level.
    • QMF request a frame from Telecide, Telecide from ConvertToYV12 and finally ConvertToYV12 from AviSource.
    • AviSource produces the frame and sends it to ConvertToYV12, etc.

A few details were omitted, but this is how the script basically works.


Changelog

v2.60 Added ConditionalSelect.
v2.55 ScriptClip: added after_frame.
v2.53 FrameEvaluate: added after_frame.
Personal tools