Elusien's Shotcut WebVfx Framework

Downloads - click on the relevant Javascript symbol
VersionDateDownloadDescription
1.82018-06-29
js logo
Fix a slight juddering when a cursor is used. Basically change form hiding the cursor by making it zero width to hiding it by making its color transparent.
1.72018-06-26
js logo
Updated to include a new feature: typewriter. This enables the user to output text that is staggered temporally, as if it were being produced by a typewriter.
1.62018-06-24
js logo
Fix to hide the <pre></pre> that contains the .srt file cues.
1.52018-06-22
js logo
Updated to include a new feature: subtitles. This enables the user to supply subtitles in "SubRip" format and specify how they appear in the video
1.42018-05-08
js logo
Updated to include a new feature: data-center. This facilitates centering HTML elements horizontally, vertically or in both dimensions, within their parent element, or within the screen (if the parent fills the screen).
1.32018-05-07
js logo
Updated to include a new feature: Fragmentation
1.22018-05-04
js logo
Updated to include a new slideshow type: 'splitting'
1.12018-04-29
js logo
Updated to include new features: Slideshow and Explosion
1.02018-03-03
js logo
Original version

Introduction

Shotcut is a free and open-source cross-platform video editing application for FreeBSD, Linux, OS X and Windows. It is based on the MLT Multimedia Framework. One of the filters that is available in Shotcut is the "Overlay HTML" filter. This enables the user to overlay a clip with an HTML page. This HTML page can be a static one, or it can be animated e.g. using the latest animation features of CSS, or via javascript.

To allow fine control of the HTML animation the "Overlay HTML" filter has an option to allow the user to specify that the WebVfx javascript framework of MLT be used. This anables the user to specify a routine that is called for each and every frame in the clip. The routine that is supplied by the user has a single parameter (time) which is the "normalised time" for that clip. The first frame in the clip has time 0.0 and the last frame in the clip has time 1.0 (actually it is 1.0 minus the normalised timelength of a frame, it never fully reaches 1.0).

Developing a WebVfx overlay HTML filter using Shotcut alone is not easy, as most errors result in a blank screen, with no indication of what caused the error, or even what the actual error is. This is where the Shotcut WebVfx Framework that I have developed comes in useful. It enables the development to be done using a modern browser, such as Chrome, or Edge, that has a comprehensive set of development tools (accessible using function key F12). See the following webpages for more details: Google Chrome Dev Tools Microsoft Edge Dev Tools

Basically, using the framework the user can use the browser to test an HTML page that will eventually be used as an (WebVfx-enabled) animated Overlay HTML filter which will use WebVfx to have perfect synchronicity with the frames in the clip. Then, once the filter is complete it can be used without modification in Shotcut. The framework uses the javascript "setTimeout" function to emulate WebVfx, but this is hidden from the user. When you use the HTML Overlay Filter in Shotcut itself, do not forget to check the box to say it uses WebVfx, and confirm that you know what you are doing.

Tip: If you save the project to a '.mlt' file and later on reload that project you may find that the filters appear to be missing from the clips. Don't worry, just click on the olive-coloured TRACK HEADER on the timeline [the far left-hand side], then when you click back on the clip you'll find the filter is there after all.

Frames_to_Time Utility

When creating some animations it helps to know what "normalised time" corresponds to a certain frame in the clip. A webpage has been created to enable users to input the frames in the format "HH:MM:SS;FF" (where "FF" is the number of the frame). This "framestamp" can be observed between the timeline and the preview screen. It is particularly useful for example if you are creating a "subtitle" Overlay HTML filter, so you can synchronise the subtitles precisely with the video. This utlity can be found at Frames_to_Time.

Centering HTML Elements

Frequently it is important that HTML elements are centered on the screen, either horizontally, vertically or in both dimensions. This can be done easily by using the 'data-center' attribute on the element that needs to be centered. (NOTE the American spelling 'er', not the British spelling 're'). This attribute can take the value of:

e.g.


<body style="{height: 100vh; width: 100vw; padding: 0; margin: 0; overflow: hidden;}">
	<div data-center="v" class='webvfx' data-animate='{0%: {opacity: 1;}, 100%: {opacity: 0%;}}'>
		I am fading
	</div>
</body>
<script src='webvfx'></script>		


While this will center the DIV vertically, the text will not be centered horizontally. This is because the DIV extends across the whole of the screen left to right, but by default text-align="left". To center the text itself you need to put text-align="center" as an attribute to the DIV element. If the width of the DIV were 50vw (half the width of the screen) then it would normally occupy the left-half of the screen. Using data-center="vh" the DIV would be centred precisely in the center of the screen. Actually, the element is always centered within its parent, but in this case the parent <body>occupies the whole of the screen, hence the DIV is centered on the screen.

Disclaimer & License (CC BY-SA 4.0)

Version 1.x Copyright © 2018 - Elusien Entertainment (www.elusien.co.uk)

Creative Commons Licence This work is licensed under a Creative Commons Attribution-ShareAlike 4.0 International License.

The following is a summary of the license. For the actual wording of the license see: https://creativecommons.org/licenses/by-sa/4.0/legalcode

Attribution — You must give appropriate credit, provide a link to the license, and indicate if changes were made. You may do so in any reasonable manner, but not in any way that suggests the licensor endorses you or your use.

ShareAlike — If you remix, transform, or build upon the material, you must distribute your contributions under the same license as the original.

No additional restrictions — You may not apply legal terms or technological measures that legally restrict others from doing anything the license permits.

Notices

Features

Animation

For animations that consist of manipulating CSS properties of HTML elements, such as colour, opacity, position on the page, height, width etc., all this can be done without the user having to resort to using javascript. The framework is designed to enable this to be done using attributes on the elements' HTML tags. e.g. to fadeout a <div> element you could write:


<div class='webvfx' data-animate='{0%: {opacity: 1;}, 100%: {opacity: 0%;}}'>
	I am fading
</div>		

Which says that at the beginning of the animation (keyframe 0%) the opacity is set to 1 (fully opaque), while at the end of the animation (keyframe 100%) the opacity is 0 (fully transparent). Additional keyframes can be added in between to fine-tune the fading, e.g. 50%: {opacity: 0.8;} would cause it to have faded to 0.8 opacity by the time the clip was halfway through and it would then fade more quickly in the second half of the clip. Another way to fine-tune is to use an "easing" function which varies the strength of the effect as a function of time. The default easing function is "linearTween", which is linear, but there are 24 others (such as easeInSine and easeInOutBounce) to choose from.

Stopwatch

Sometimes it is useful to show the frame-numbers as the video clip progresses, or show the time. This "Stopwatch" facility is also a feature of the framework, again specified solely in the HTML e.g.


<div class='webvfx' data-stopwatch=''>
	Frame <span>0</span> => <span></span><span>00</span>:<span>00</span><span></span>
</div>

This will show a "stopwatch" that has the frame-number followed by the time as MM:SS. It is possible to display the hour and the number of milliseconds as well. It is also possible, by the use of keyframes, to pause the stopwatch and resume or reset it. Obviously the stopwatch elements can all be styled using CSS as usual, so the size, colour, background, borders, border-radiuses etc. can all be tailored to the user's requirements.
e.g. the following HTML and CSS code produce the "stopwatch" below:


<div class='stopwatch webvfx' data-stopwatch=''>
	<span>0</span> &#9654; <span></span><span>00</span> : <span>00</span> . <span>000</span>
<div>

.stopwatch>span	{
	display         : inline-block;
	font-family     : Iceland, monospaced;
	font-weight     : bold;
	text-align      : center;
	color           : #fff;
	background-color: #333;
	width           : 2em;
	height          : 2em;
	line-height     : 2em;
	margin          : 0.5em;
	border          : 1px;
	border-radius   : 50%;
}
000 : 00 . 000

User-Supplied Animation

The framework also enables the user to develop his/her own features in javascript, again without having to know the intricacies of MLT's WebVfx and without having to alter the script depending on whether it is being run in a browser or in Shotcut.

Slideshow

There is a slideshow facility, where the user can specify a set of images and parameters, such as the type of transition between slides. The main problem is that it takes time to load an image and the slideshow should only be started once all the images have been loaded. The framework checks to see that all images have been loaded, if some have yet to be loaded it will ask what to do. It has been tested with 100 slides and although the slideshow with this number of slides is a bit jerky in the browser, it is fine in the video that is exported by Shotcut. A large number of slides may cause prolems due to lack of memory. Example:


	<div id='slides' data-control='60:30' class='webvfx' data-slideshow='{
		duration: 33%, type: splitting, magnify: 25%
	}'>
		<img src='../mhcommittee.elusien.co.uk/images/home.jpg'>
		<img src='../mhcommittee.elusien.co.uk/images/home_1.jpg'>
		<img src='../mhcommittee.elusien.co.uk/images/Melamine-wood-002.png'>
		<img src='../mhcommittee.elusien.co.uk/images/Melamine-wood-003.png'>
		<img src='../mhcommittee.elusien.co.uk/images/home_2.jpg'>
		<img src='../mhcommittee.elusien.co.uk/images/home.jpg'>
	</div>		

Which says that each slide will show for 10 seconds (60 divided by the number of slides (6)), each slide will be split in two horizontally and these two splits will take 33% of this 10 seconds to slide into place from the sides of the slideshow and will then grow by 25% for the other 67% of the time.

Explosion

This causes text to explode into the individual characters without the user having to resort to using javascript. The framework is designed to enable this to be done using attributes on the elements' HTML tags. e.g. to explode text in a <div> element you could write:


<div class='webvfx' data-explosion='{10%: explode, 50%: wait, 60%: implode, 90%: wait}'>
	<h1>BOOM</h2>
	<p>Watch me explode</p>
</div>		

Which says that just after the beginning of the animation (keyframe 10%) the text will explode, the explosion will cease halfway into the animation (50%), and a little while after that (60%) the characters will implode and remain stable, on the screen for a short while (90%) before the end of the animation.

Fragment

This causes an HTML to fragment into several pieces and then these peices appear to fly into and out of the screen. e.g. to fragment a <div> element you could write:


<div class='webvfx' data-fragment='{10%: fragment, 50%: wait, 60%: defragment, 90%: wait}'>
	<h1>Split and Flyaway/h2>
	<img src='myphoto.jpg'>
</div>		

Which says that just after the beginning of the animation (keyframe 10%) the div will fragment, the fragmentation will cease halfway into the animation (50%), and a little while after that (60%) the div will defragment and remain stable, on the screen for a short while (90%) before the end of the animation.

Subtitle

This causes an subtitles (in "subrip" file format - see en.wikipedia.org/wiki/SubRip) to appear according to the timings given. e.g. to display the timings in a <div> element you would copy the data in the '.srt' file into a <pre></pre> element such as:


<div class='webvfx' data-subtitles='SRT_subtitles'>
<pre id='SRT_subtitles'>
1
00:00:10,000 --> 00:00:15,400
This is the 1st subtitle 10sec to 15.4sec
Mary had a little lamb,
its <i>fleece</i> was white as snow
and everywhere that Mary went
that <b>lamb</b> was sure to go!</b>

2
00:00:16,600 --> 00:00:25,800
This is the 2nd subtitle 16.6sec to 25.8sec

3
00:00:20,600 --> 00:00:23,800
This is the 3rd subtitle 20.6sec to 23.8sec

</pre>		

Please note that it is imperative that clip length and frame-rate are specified correctly (see below) for the video being subtitled as both of these values are needed to create the correct timings for showing and deleting the subtitles at the correct frames within the video.

Typewriter

This causes text to appear, but staggered temporally, as if produced on a typewriter.


<div class='webvfx' data-typewriter='{start: 0.1, end: 0.9, stx: 10, etx: 10}'>
	Mary had a little lamb,<br>
	its <i>fleece</i> was white as snow<br>
	and everywhere that Mary went<br>
	that <b>lamb</b> was sure to go!
</div>		

The idividual characters in the text get wrapped inside <span><\span> tags which have a CSS classname of '_typewriter_'. The user may style these tags. If the CSS specifies a border for these elements e.g. ._typewriter_ {border: solid #f00 3px;} then a curser will appear to preceed the individual characters as they are "typed".

Details

Clip Length and Frame-Rate

The length of the clip and the frame-rate are known by Shotcut, but not by the browser, so we need to tell the framework what these are so that when the animation runs in the browser it runs for the same amount of time and uses the same number of frames. This makes the run in the browser as similar as possible as that in the video file exported from Shotcut. I say the exported file rather than the Shotcut viewing area since viewing animations in real-time depends on the complexity of the animation and the underlying clip and the processing power of the computer on which Shotcut is running. Animations and videos may look juddery and choppy in the real-time viewing area, where frames may be dropped, while they are more than likely to be smooth in the final exported version.

To provide the browser/framework with the information you need to include a 'data-control' attribute on one of the HTML elements giving the clip-length in seconds (default: 8) and the frame-rate in fps (default: 30). It does not matter which HTML element has this attribute, it does not have to be an animated one e.g.


<body data-control='15:24'>		<!-- 15 seconds clip at 24 fps -->

Documentation

Animation

Animation on an HTML element is requested by giving it the classname 'webvfx' and the attribute 'data-animate', which supplies the parameters for the animation. The 'webvfx' classname says it is to be operated on by the framework. The 'data-animate' attribute says it is an animation and has the options shown below. The framework now allows two animations to be run on an element. To reques a second animation use a 'data-animate2' attribute. See the 'text' example later on.


<element class='webvfx' data-animate='{
		start : normalised start time of the animation (default = 0.0) e.g.
				0.0  starts at the very beginning of the clip
				0.5  starts halfway through the clip
				0.75 starts three-quarters the way into the clip
				
		end	  : normalised  end  time of the animation (default = 1.0) e.g.
				1.0   ends  at the very end of the clip
				0.5   ends  halfway through the clip
				0.75  ends  three-quarters the way into the clip
				
		ease  : the easing function to use, which can be "linearTween" (default) or one of the following strings:
				"easeInQuad"		t^2         acceleration from zero velocity
                "easeOutQuad"		t^2         deceleration  to  zero velocity
                "easeInOutQuad"		t^2         acceleration until halfway, then deceleration
                "easeInCubic"		t^3         acceleration from zero velocity
                "easeOutCubic"		t^3         deceleration  to  zero velocity
                "easeInOutCubic"	t^3         acceleration until halfway, then deceleration
                "easeInQuartic"		t^4         acceleration from zero velocity
                "easeOutQuartic"	t^4         deceleration  to  zero velocity
                "easeInOutQuartic"	t^4         acceleration until halfway, then deceleration
                "easeInQuintic"		t^5         acceleration from zero velocity
                "easeOutQuintic"	t^5         deceleration  to  zero velocity
                "easeInOutQuintic"	t^5         acceleration until halfway, then deceleration
				"easeInSine"		sinusoidal  acceleration from zero velocity
				"easeOutSine"		sinusoidal  deceleration  to  zero velocity
                "easeInOutSine"		sinusoidal  acceleration until halfway, then deceleration
				"easeInExpo"		exponential acceleration from zero velocity
                "easeOutExpo"		exponential deceleration  to  zero velocity
                "easeInOutExpo"		exponential acceleration until halfway, then deceleration
                "easeInCirc"		circular    acceleration from zero velocity
                "easeOutCirc"		circular    deceleration  to  zero velocity
                "easeInOutCirc"		circular    acceleration until halfway, then deceleration
                "easeInBounce"		elastic     bounce, then acceleration from zero velocity
                "easeOutBounce"		elastic     deceleration  to  zero velocity, then bounce
                "easeInOutBounce"	elastic     bounce, then acceleration until halfway, then deceleration, then bounce
				Please be aware that these strings are CASE-SENSITIVE e.g. "easeInBounce", NOT "easeinbounce".
				
		  0%  : The keyframe at the start of the animation (REQUIRED - NO DEFAULT)
		100%  : The keyframe at the  end  of the animation (REQUIRED - NO DEFAULT)
		        Other keyframes (like 50%, 75%, 83.67% are optional)
				The keyframe specifies the CSS property/properties at the time represented by the percentage
				in a form very similar to CSS animation keyframes. The main differences being:
					- Unlike CSS keyframes, these ones (in the framework) must be separated by a comma (,).
					- Hyphenated CSS properties (e.g. background-color) are NOT ALLOWED, they have to be
					  "camelCased" (e.g. backgroundColor), the way the javascript "style" properties are.
					- a CSS property cannot be used more than once in a keyframe e.g. webTranform below:
					  0% : {webTransform: rotateX(180deg); webTransform: rotateY(180deg);}
					- The percentage is separated from the properties by a colon (:).
					- You cannot use "from" and "to", you have to use "0%" and "100%" instead.
					- For each property in the 0% keyframe there must be a corresponding property in
					  all the other keyframes.
					- Colors cannot be specified by name (e.g. "red"), they have to be specified either:
					- by their "#" value e.g. #F00 or ff0000
					  or by their "rgb" value e.g. eg: rgb(255,0,0) [you cannot mix these either]
				Please note that Shotcut WebVfx has some deficiences.
				It does not recognise some of the newer CSS properties, such as:
					transformStyle: preserve-3d;
				Also, some properties have to be presented with the "webkit" prefix, e.g.:
					{transform : rotateY(180deg);}  does not work, use  (webkitTransform : rotateY(180deg);}
		}'></element>

e.g.


<div class='webvfx' data-animate='{
	start : 0.0 ,
	 end  : 1.0 ,
	ease  : "easeInOutBounce" ,
      0%  : {top:   0px; left:   0px; backgroundColor: #f00;}    ,
     25%  : {top:   0px; left: 200px; backgroundColor: #00f;}    ,
     50%  : {top: 200px; left: 200px; backgroundColor: #f8f800;} ,
     75%  : {top: 200px; left:   0px; backgroundColor: #0f0;}    ,
    100%  : {top:   0px; left:   0px; backgroundColor: #f00;}
}'>
	Bouncing Colour-Changing Div
</div>	 

For more on CSS keyframes see: W3schools Keyframes.

Stopwatch

This enables you to have 1 or more stopwatches in the HTML Overlay filter. It also enables the use of "keyframes" for fine control of the stopwatches.

A stopwatch consists of 4 elements specified using HTML '<span></span>' tags.


Normally you would initialise each of the <span> elements by placing a zero (0) in it.
Placing any other number will initialise it to that value.
Leaving the <span> empty will hide it (see examples).

A Stopwatch is requested by giving it the classname 'webvfx' and the attribute 'data-stopwatch', which supplies the parameters for the animation. The 'webvfx' classname says it is to be operated on by the framework. The 'data-stopwatch' attribute has the following options:


<element class='webvfx' data-stopwatch='{
		start : normalised start time of the animation (default = 0.0) e.g.
				0.0  starts at the very beginning of the clip
				0.5  starts halfway through the clip
				0.75 starts three-quarters the way into the clip
				
		end	  : normalised  end  time of the animation (default = 1.0) e.g.
				1.0   ends  at the very end of the clip
				0.5   ends  halfway through the clip
				0.75  ends  three-quarters the way into the clip
				
				Keyframes may be provided to pause/resume/reset the stopwatch at certain points in the clip e.g.:
		 20%  : {pause;}			The stopwatch will pause one-fifth into the clip
		 40%  : {resume;}			The stopwatch will resume and the times will start up where they left off;
		 50%  : {pause;}
		 60%  : {resume: skip;}		The stopwatch will resume and the times will skip to the value they would
									have shown had not the stopwatch been paused.
		 70%  : {pause;}
		 80%  : {resume: <nnn>;}	The stopwatch will resume and the times will be reset to the time represented
		                            by the number <nnn> in milliseconds e.g.:
										resume: 0     will resume the stopwatch times from the beginning,
										resume: 50000 will resume the stopwatch times from 50 seconds.
										
		None of these parameters needs to be specified, the element could just state  data-stopwatch = ""
		}'></element>

User Supplied

The javascript needs to be in the form of a function that will be called for each and every frame in the clip and takes 4 arguments:

  1. the normalised time of the frame
  2. a logical value representing whether or we are running in a browser (true), or actually running in the Shotcut editor (false)
  3. the frame-number
  4. the frame-rate (in fps)
The user then puts this (and any other functions they have similarly developed and want to include) in a certain GLOBAL scope javascript array called "webvfx_add_to_frame" thus:


function flash(time, browser, frame_number, frame_rate) {
	bulb               = document.getElementById("bulb");
	bulb.style.opacity = ((frame_number % 10) < 5) ? 0.1 : 1.0;
}
webvfx_add_to_frame = [flash]; // Ensure "flash" is called once for every frame of the clip.

Slideshow

This creates a slideshow of a set of images. There are various parameters that enable the type of slideshow and transition between slides to be specified (see below). PLEASE NOTE you can have only 1 slideshow on the HTML web-page.

The code checks that all images have been loaded. If some have not been loaded yet it asks you to confirm whether or not you wish to carry on. If you "CANCEL", indicating that you don't want to carry on, it aborts, otherwise it then asks you to confirm whether or not you want to wait for the missing images. If you "CONFIRM" it will check again if all the images have been loaded. However if you ask it to "carry on"CANCEL" it will carry on with the slideshow. If there are still images not loaded it will print a list of these images on the javascript console in the browser, or in the "Application Log" in Shotcut, that you can view to see which have not been loaded. It could be that you made a mistake with the filename(s).

PLEASE NOTE if images have been rotated by manipulating the EXIF X-Y data the slideshow will not show that slide in the correct orientation. This is very complicated to do in javascript and is likely to be a permanent deficiency. If you have such an image you can use photo manipulation software to rotate it to the correct orientation if necessary, then save the file using an option to NOT save the EXIF data.

A Slideshow is requested by giving it the classname 'webvfx' and the attribute 'data-slideshow', which supplies the parameters for the animation. The 'webvfx' classname says it is to be operated on by the framework. The 'data-slideshow' attribute has the following options:


<element class='webvfx' data-slideshow='{
		type	   : the type of slideshow, which can be:
					 sliding    - the slides slide in from the left, right, top or bottom; (DEFAULT)
					 shuffling  - the slides slide in then out from alternate directions (left then right, or top then bottom);
					 appearing  - the slides just appear, they do not slide;
					 expanding  - the slides start out at zero size then expand to fill the slideshow window;
					 stretching - the slides stretch out (horizontally or vertically);
					 splitting  - the slides will be split in 2 (horizontally or vertically) and slide into place.
								  (This "splitting" option can be quite CPU-intensive)
					 
		from 	   : the side from which the slides appear when sliding/shuffling/stretching/splitting, which can be:
					 top;
					 bottom;
					 left;
					 right. (DEFAULT)
					 
		duration   : the (percentage) duration of the transition. DEFAULT = 25%
		
		magnify    : the percentage by which the slide will grow once the end of
					 the trnsition has been reached. DEFAULT = 25%
					 
		blur	   : the amount of Gaussian blur (25 = 25 pixels radius) to apply to the image while transitioning.
		             Be aware that large values for "blur" (say > 50) take a lot of processing power. DEFAULT = 0
				
		ease  : the easing function to use, which can be "linearTween" (DEFAULT) or one of the following strings:
				"easeInQuad"		t^2         acceleration from zero velocity
                "easeOutQuad"		t^2         deceleration  to  zero velocity
                "easeInOutQuad"		t^2         acceleration until halfway, then deceleration
                "easeInCubic"		t^3         acceleration from zero velocity
                "easeOutCubic"		t^3         deceleration  to  zero velocity
                "easeInOutCubic"	t^3         acceleration until halfway, then deceleration
                "easeInQuartic"		t^4         acceleration from zero velocity
                "easeOutQuartic"	t^4         deceleration  to  zero velocity
                "easeInOutQuartic"	t^4         acceleration until halfway, then deceleration
                "easeInQuintic"		t^5         acceleration from zero velocity
                "easeOutQuintic"	t^5         deceleration  to  zero velocity
                "easeInOutQuintic"	t^5         acceleration until halfway, then deceleration
				"easeInSine"		sinusoidal  acceleration from zero velocity
				"easeOutSine"		sinusoidal  deceleration  to  zero velocity
                "easeInOutSine"		sinusoidal  acceleration until halfway, then deceleration  (DEFAULT)
				"easeInExpo"		exponential acceleration from zero velocity
                "easeOutExpo"		exponential deceleration  to  zero velocity
                "easeInOutExpo"		exponential acceleration until halfway, then deceleration
                "easeInCirc"		circular    acceleration from zero velocity
                "easeOutCirc"		circular    deceleration  to  zero velocity
                "easeInOutCirc"		circular    acceleration until halfway, then deceleration
                "easeInBounce"		elastic     bounce, then acceleration from zero velocity
                "easeOutBounce"		elastic     deceleration  to  zero velocity, then bounce
                "easeInOutBounce"	elastic     bounce, then acceleration until halfway, then deceleration, then bounce
				Please be aware that these strings are CASE-SENSITIVE e.g. "easeInBounce", NOT "easeinbounce".
										
		None of these parameters needs to be specified, the element could just state data-slideshow = ""
		}'></element>

Explosion

This takes some text, breaks it down into the individual characters then simulates an explosion by animating these characters, making them fly over, into and out of the screen. There are various parameters that enable the timing and type of explosion (outwards or inwards) to be specified (see below).

An explosion is requested by giving it the classname 'webvfx' and the attribute 'data-explosion', which supplies the parameters for the animation. The 'webvfx' classname says it is to be operated on by the framework. The 'data-explosion' attribute has the following options:


<element class='webvfx' data-explosion='{
		start : normalised start time of the animation (default = 0.0) e.g.
				0.0  starts at the very beginning of the clip
				0.5  starts halfway through the clip
				0.75 starts three-quarters the way into the clip
				
		end	  : normalised  end  time of the animation (default = 1.0) e.g.
				1.0   ends  at the very end of the clip
				0.5   ends  halfway through the clip
				0.75  ends  three-quarters the way into the clip
				
		ease  : the easing function to use, which can be "linearTween" or one of the following strings:
				"easeInQuad"		t^2         acceleration from zero velocity
                "easeOutQuad"		t^2         deceleration  to  zero velocity
                "easeInOutQuad"		t^2         acceleration until halfway, then deceleration
                "easeInCubic"		t^3         acceleration from zero velocity
                "easeOutCubic"		t^3         deceleration  to  zero velocity
                "easeInOutCubic"	t^3         acceleration until halfway, then deceleration
                "easeInQuartic"		t^4         acceleration from zero velocity
                "easeOutQuartic"	t^4         deceleration  to  zero velocity
                "easeInOutQuartic"	t^4         acceleration until halfway, then deceleration
                "easeInQuintic"		t^5         acceleration from zero velocity
                "easeOutQuintic"	t^5         deceleration  to  zero velocity
                "easeInOutQuintic"	t^5         acceleration until halfway, then deceleration
				"easeInSine"		sinusoidal  acceleration from zero velocity
				"easeOutSine"		sinusoidal  deceleration  to  zero velocity
                "easeInOutSine"		sinusoidal  acceleration until halfway, then deceleration  (DEFAULT)
				"easeInExpo"		exponential acceleration from zero velocity
                "easeOutExpo"		exponential deceleration  to  zero velocity
                "easeInOutExpo"		exponential acceleration until halfway, then deceleration
                "easeInCirc"		circular    acceleration from zero velocity
                "easeOutCirc"		circular    deceleration  to  zero velocity
                "easeInOutCirc"		circular    acceleration until halfway, then deceleration
                "easeInBounce"		elastic     bounce, then acceleration from zero velocity
                "easeOutBounce"		elastic     deceleration  to  zero velocity, then bounce
                "easeInOutBounce"	elastic     bounce, then acceleration until halfway, then deceleration, then bounce
				Please be aware that these strings are CASE-SENSITIVE e.g. "easeInBounce", NOT "easeinbounce".
				
		begin : hidden			when the clip begins the text will be hidden. Use this if you intend to
						`		implode the text and don't want the intact text to show before it implodes.
								Also, if you want to explode text but don't want the text to be visible
								before the start of the clip.
								(DEFAULT = visible)
				
		finish : hidden			at the start of the clip the text will be hidden. Use this if you intend to
						`		implode the text and don't want the intact text to show after it implodes.
								e.g. if this text should vanish so a different text can be animated.
								(DEFAULT = visible)
		
		target: text			The individual characters will explode/implode (DEFAULT)
				element(nw,nh)	The whole HTML element will be divided into individual cells "nw" columns wide
								and "nh" columns high (i.e. nw x nw cells) and these cells will be the parts
								that explode/implode across the screen.
								
				Keyframes may be provided to specify when parts should implode, explode or remain static e.g.:
		  5%  : implode			The text/parts will implode onto the screen starting 5% of the way into the clip
		 20%  : wait			The implosion will last until one-fifth the way into the clip
		 75%  : explode			The text/parts will be static until three-quarters into the clip, when it will
								then explode. (DEFAULT)
										
		None of these parameters needs to be specified, the element could just state  data-explosion = ""
		}'></element>

Fragment

This takes an HTML element (such as a div), slices it into (nh x nv) peices then either causes these peices to fly apart (fragment) or fly together (defragment). There are various parameters that enable the timing and type of fragmentation (outwards or inwards) to be specified (see below).
PLEASE NOTE The elemnt being fragmented must have the CSS property "position: absolute;".

A fragmentation is requested by giving it the classname 'webvfx' and the attribute 'data-fragment', which supplies the parameters for the animation. The 'webvfx' classname says it is to be operated on by the framework. The 'data-fragment' attribute has the following options:


<element class='webvfx' data-fragment='{
		start : normalised start time of the animation (default = 0.0) e.g.
				0.0  starts at the very beginning of the clip
				0.5  starts halfway through the clip
				0.75 starts three-quarters the way into the clip
				
		end	  : normalised  end  time of the animation (default = 1.0) e.g.
				1.0   ends  at the very end of the clip
				0.5   ends  halfway through the clip
				0.75  ends  three-quarters the way into the clip
				
		ease  : the easing function to use, which can be "linearTween" or one of the following strings:
				"easeInQuad"		t^2         acceleration from zero velocity
                "easeOutQuad"		t^2         deceleration  to  zero velocity
                "easeInOutQuad"		t^2         acceleration until halfway, then deceleration
                "easeInCubic"		t^3         acceleration from zero velocity
                "easeOutCubic"		t^3         deceleration  to  zero velocity
                "easeInOutCubic"	t^3         acceleration until halfway, then deceleration
                "easeInQuartic"		t^4         acceleration from zero velocity
                "easeOutQuartic"	t^4         deceleration  to  zero velocity
                "easeInOutQuartic"	t^4         acceleration until halfway, then deceleration
                "easeInQuintic"		t^5         acceleration from zero velocity
                "easeOutQuintic"	t^5         deceleration  to  zero velocity
                "easeInOutQuintic"	t^5         acceleration until halfway, then deceleration
				"easeInSine"		sinusoidal  acceleration from zero velocity
				"easeOutSine"		sinusoidal  deceleration  to  zero velocity
                "easeInOutSine"		sinusoidal  acceleration until halfway, then deceleration  (DEFAULT)
				"easeInExpo"		exponential acceleration from zero velocity
                "easeOutExpo"		exponential deceleration  to  zero velocity
                "easeInOutExpo"		exponential acceleration until halfway, then deceleration
                "easeInCirc"		circular    acceleration from zero velocity
                "easeOutCirc"		circular    deceleration  to  zero velocity
                "easeInOutCirc"		circular    acceleration until halfway, then deceleration
                "easeInBounce"		elastic     bounce, then acceleration from zero velocity
                "easeOutBounce"		elastic     deceleration  to  zero velocity, then bounce
                "easeInOutBounce"	elastic     bounce, then acceleration until halfway, then deceleration, then bounce
				Please be aware that these strings are CASE-SENSITIVE e.g. "easeInBounce", NOT "easeinbounce".
				
		begin : hidden			when the clip begins the HTML element will be hidden. Use this if you intend to
						`		defragment the element and don't want the intact element to show before it defragments.
								Also, if you want to fragment an element but don't want it to be visible
								before the start of the clip.
								(DEFAULT = visible)
				
		finish : hidden			at the start of the clip the HTML element will be hidden. Use this if you intend to
						`		defragment the text and don't want the intact element to show after it defragments.
								e.g. if this element should vanish so a different element can be animated.
								(DEFAULT = visible)
		
		nh	   : <rows>			The whole HTML element will be divided horizontally into "nh" rows;
		nv	   : <cols>			The whole HTML element will be divided horizontally into "nh" rows;
								(i.e. there will be nh x nv cells to be (de)fragmented.
								
				Keyframes may be provided to specify when parts should implode, explode or remain static e.g.:
		  5%  : defragment		The cells will defragment onto the screen starting 5% of the way into the clip
		 20%  : wait			The defragmentation will last until one-fifth the way into the clip
		 75%  : fragment		The element will be static until three-quarters into the clip, when it will
								then fragment. (DEFAULT)
										
		None of these parameters needs to be specified, the element could just state  data-explosion = ""
		}'></element>

Subtitles

This takes an HTML element (such as a div) and SRT subtitles specified within a <pre></pre> element and displays those subtitles in the correct time sequence in that div. Normally you would apply the subtitles to the video they refer to, either on the video track or on a second composite track, but if you edit the video track (e.g. delete a slice fromit) it will not edit the subtitles. all the subtitle cues will stay but they will have new timings and will not be synchronised to the edited video.

One way of getting round this is to create a blank video of just the subtitles, crop it to the correct size and save it, then import this video as the second track and make the same changes to this track as you do to the original video track.

Another way is to create new subtitle timings, either by using the "subrip" utitlity on the edited video, or by using a subtitle editor. For more information see: en.wikipedia.org/wiki/SubRip


<element class='webvfx' data-subtitle='
		pre-element-id	: The id of the '<pre></pre>' element that holds the subtitles cues (see example).
'></element>

Typewriter

This takes the characters in an HTML element (such as a div) and outputs those characters staggered temporally as if they were being printed on a typewrite. The text can contain HTML tags, such as <b></b> for bold or <i></i> for italics, or even more complicated tags.

The characters are wrapped in <span class='_typewriter_'></span> tags. If the user uses CSS to style these tags and creates a border e.g. ._typewriter_ {border: solid #f00 3px;} then a curser will appear to preceed the individual characters as they are "typed". The parameters stx (start of text) and etx (end of text) specify how long the curser appears before typing starts and after it ends (see the example below).


<element class='webvfx' data-typewriter='{
		start : normalised start time of the "typing" (default = 0.0) e.g.
				0.0  starts at the very beginning of the clip
				0.5  starts halfway through the clip
				0.75 starts three-quarters the way into the clip
				
		end	  : normalised  end  time of the "typing" (default = 1.0) e.g.
				1.0   ends  at the very end of the clip
				0.5   ends  halfway through the clip
				0.75  ends  three-quarters the way into the clip
				
		stx   : the amount of time (in characters) for the cursor to appear
				before "typing" starts (default = 0) e.g.
				0     the cursor does not appear before typing starts
				10    the curser appears for the space of time taken to output
				      10 characters before typing starts. Assuming of course
					  that the user has specified CSS of a border on class "_typewriter_".
				
		etx   : the amount of time (in characters) for the cursor to appear
				after "typing" ends (default = 0) e.g.
				0     the cursor does not appear after typing ends
				10    the curser appears for the space of time taken to output
				      10 characters after typing ends. Assuming of course
					  that the user has specified CSS of a border on class "_typewriter_".				
'></element>

Overlay HTML Examples

All of these examples can be fairly easily tailored by modifying the CSS and if necessary the HTML. They are written to demonstrate the features of Elusien's WebVfx Framework and to show how quick and easy it is to use the framework to produce an overlay HTML filter that can be reasonably sophisticated.

Intro

This overlay produces the intro at the beginning of a movie/video. It demonstrates changes of opacity and color as well as rotation and expansion of text. To see the animation click on the button below:

The source code (HTML + CSS) that produced the animation is listed below. Use "copy & paste" to copy it to your own computer for you to modify to your requirements.


<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="utf-8">
<link href="https://fonts.googleapis.com/css?family=Audiowide|Iceland" rel="stylesheet">
<!-- WebVfx does not recognise Google web fonts in "<link>", so make sure you also have the font downloaded on your computer. -->
<title>Intro</title>
<style>
    html, body	{margin:0; padding:0; width: 100%; height: 100%; overflow: hidden;}
	table		{
/*				 width: 640px; height: 480px;	/*for a screen resolution of 640 x 480 */
				 width: 100% ; height: 100% ;
				 overflow        : 	hidden; border: solid 1px #ffffff;
				 background-color:	#222222;
				 background      :	repeating-linear-gradient(45deg, #303030 0%, #303030 10%, #202020 0%, #202020 50%) 0 / 15px 15px;
				}
	td			{text-align	: center; vertical-align: middle;}
	#elent		{
				 width		:	100%;
				 font-size	:	100px;
				 font-family:	'Iceland', cursive;
				 font-weight: 	bolder;
				 color		:	#228DFF;
				 text-shadow:	0 0 10px #fff, 0 0 20px #fff, 0 0 30px #fff, 0 0 40px #228DFF, 0 0 70px #228DFF, 0 0 80px #228DFF, 0 0 100px #228DFF, 0 0 150px #228DFF;
				 opacity	:	0;
	}
	#presents	{
				 width			  :	100%;
				 font-size		  :	35px;
				 color			  :	#aaa;
				 font-family	  :	Helvetica, Arial, sans-serif;
				 -webkit-transform: scale(0, 0) rotate(-180deg);
				    -moz-transform: scale(0, 0) rotate(-180deg);
				         transform: scale(0, 0) rotate(-180deg);
	}
	#title		{
				 width		:	100%;
				 font-size	:	50px;
				 font-family:	'Audiowide', cursive;
				 color		:   #00f;
				 opacity	:	0;
*/				 text-shadow:	1px 1px 0 #aaaaaa;
	}
	
</style>
</head>
<body data-control='10:30'>
	
<!--
	================================================
	        A SHOTCUT Overlay HTML Filter
			-----------------------------
	The "presentation" title screen for home movies.
	================================================
-->

	<table><tr><td>
		<p	class='webvfx' id='elent'
			data-animate='{ease: "easeInSine",
							  0%: {opacity:  0.15;},
						    100%: {opacity:  1   ;}
						  }'
		 >
			Elusien Entertainment
		</p>
		<p class='webvfx' id='presents'
		    data-animate='{start: 0.3, end: 0.5, ease: "easeInOutCirc",
							  0%: {webkitTransform: scale(0  , 0  ) rotate(-180deg);},
						     50%: {webkitTransform: scale(0.5, 0.5) rotate( -90deg);},
						    100%: {webkitTransform: scale(1  , 1  ) rotate(   0deg);}
						  }'
		 >
			Presents
		</p>
		<p class='webvfx' id='title'
			data-animate='{start: 0.6, end: 1.0, ease: "linearTween",
							  0%: {color:  #0000ff; opacity:  0.15;},
							 50%: {color:  #ff9966; opacity:  1   ;},
							100%: {color:  #99ff99; opacity:  1   ;}
						  }'
		 >
			An intro produced using<br>
			Elusien's WebVfx framework for<br>
			<b><u>MLT-SHOTCUT</u></b>
		</p>
	</td></tr></table>
</body>
<script src="WebVfx.js"></script>
</html>

Scrolling Credits

This overlay produces the scrolling credits at the end of a movie/video. To see the animation click on the button below:

The source code (HTML + CSS) that produced the animation is listed below. Use "copy & paste" to copy it to your own computer for you to modify to your requirements.


<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="utf-8">
<link href="https://fonts.googleapis.com/css?family=Open+Sans+Condensed:300" rel="stylesheet">
<!-- WebVfx does not recognise Google web fonts in "<link>", so make sure you also have the font downloaded on your computer. -->
<title>Credits 1</title>
<style>
    html, body    {margin: 0; width: 100%; height: 100%; overflow: hidden; background-color: #000;}
    #credits      {position: absolute; top: 100%; width: 100%; padding: 25px 0; margin:0; text-align: center;}
    	/****************************************/
    	/* You may change any styles below here */
    	/****************************************/
    #credits * {font: 25px Open Sans Condensed, Liberation Sans, Arial, sans-serif; color: #fff;
  	            text-shadow: -1px -1px 0 #444, 1px -1px 0 #444, -1px 1px 0 #444, 1px 1px 0 #444;
	} /* Alter the color especially according to your background, as well as the font */
    #credits h1	     {font-size: 3em;}                                             /* Alter the size especially according to your requirement */
    #credits .legal	 {font-size: 0.9em; width: 50%; margin: auto; color: #ddd;}    /* Alter the color especially according to your background */
	/********************************************************/
             /* Change CSS below only if you know what you are doing */
	/********************************************************/
    #credits table, #credits tr {width: 100%; font-size: 1em;}
    #credits td:first-child	{width: 45%; padding: 8px 0.5%; font-size: 1.0em; text-align: right;}
    #credits td:last-child	{width: 45%; padding: 8px 0.5%; font-size: 2.0em; text-align: left;}
</style>

</head>
<body>
	<div id="credits" data-control="15:30"> <!-- Do not change the ID of this div, anything in it will appear on the scrolling credits -->
		<h1>THE TERMINATOR</h1>
        <h1>Cast</h1>
        <table>			
            <tr><td>The Terminator</td><td>Arnold Schwarzenegger</td></tr>
            <tr><td>Kyle Reese</td><td>Michael Biehn</td></tr>
            <tr><td>Sarah Connor</td><td>Linda Hamilton</td></tr>
        </table>
        <h1>Credits</h1>
        <table>			
            <tr><td>Director</td><td>James Cameron</td></tr>
            <tr><td>Writer</td><td>James Cameron</td></tr>
            <tr><td>Writer</td><td>Gayle Anne Hurd</td></tr>
        </table>
        <h1>Locations</h1>
        <p>Elephant Rocks, New Zealand</p>
        <p>The Cotswolds, UK</p>
        <p>New York, USA</p>
        <p class="legal">
            No animals were harmed during the making of this film.<br>
            &copy;2018 - Elusien Entertainment.
            Except where otherwise noted, content is licensed under a
            Creative Commons Attribution 4.0 International Licence.
        </p>
		<hr>
		<p>
			<!-- Now display the shotcut logo -->
			Produced using the free, open source, cross-platform video editor:<br><br>
		    <img alt="shotcut logo" src=""><br>
            https://shotcut.org
		</p>
    </div>
</body>
<script src="webvfx.js"></script>
<script>
	
	webvfx_add_to_frame = [scroll];	// Tells Elusien's WebVfx framework to run the "scroll" routine for each frame.

	SC = {					// SC (Scrolling Cedits) has to be of GLOBAL scope
		credits       : {},
		style		  : {},
		initial_top   : 0,   
		total_height  : 0  
	};
	
	function scroll(time, browser, frame_number, frame_rate) {		
		if (time === 0.0) {
			SC.credits       = document.getElementById('credits');
			SC.style         = SC.credits.style;
			SC.initial_top   = SC.credits.offsetTop;       
			SC.total_height  = SC.credits.offsetHeight + SC.initial_top;
		} else {
			SC.style.top     = (SC.initial_top - SC.total_height * time) + "px"; // Scrolls up an increment
		}
	}
</script>
</html>

Logo

This overlay produces an animation involving the Shotcut logo. It has to use a small amount of user-supplied javascript to work around the fact that MLT-Shotcut does not recognise the "backface-visibility" CSS property. To see the animation click on the button below:

The source code (HTML + CSS) that produced the animation is listed below. Use "copy & paste" to copy it to your own computer for you to modify to your requirements.


<html lang="en">
<head>
<meta charset="utf-8">
<title>Logo</title>
<style>
    html, body {margin:0; padding:0; width: 100%; height: 100%; background-color: #000; overflow: hidden;}
    #flip-container, #flipper, #front, #back {width: 75%; height: 100%; margin: auto;}
    #flipper {
		position          : relative;
		transform-style   : preserve-3d;	/* ignored by MLT-Shotcut */		
	} 
    #front, #back {
       	color           : #ddd;
    	background-color: transparent;
    	font            : 77px Liberation Sans, Arial, sans-serif;
		text-shadow     : -1px -1px 0 #000, 1px -1px 0 #000, -1px 1px 0 #000, 1px 1px 0 #000;
    	text-align      : center;
    	vertical-align  : middle;
    	width           : 100%;
    	height          : 100%;
    	position        : absolute;
    	top             : 0px;
    	left            : 0px;
    }
	#l1 {color: violet;}
	#l2 {color: indigo;}
	#l3 {color: blue;}
	#l4 {color: green;}
	#l5 {color: yellow;}
	#l6 {color: orange;}
	#l7 {color: red;}
    #front	{
					        transform  : rotateX(  0deg) rotateY(  0deg);		/* ignored by MLT-Shotcut */
					-webkit-transform  : rotateX(  0deg) rotateY(  0deg);		/* fix using -webkit- prefix */
					backface-visibility: hidden;			/* ignored by MLT-Shotcut - fix by javascript */
			-webkit-backface-visibility: hidden;			/* also ignored by MLT-Shotcut - fix by javascript */
    }
    #back	{
							transform  : rotateX(180deg) rotateY(180deg);	/* ignored by MLT-Shotcut */
					-webkit-transform  : rotateX(180deg) rotateY(180deg);	/* fix using -webkit- prefix */
					backface-visibility: hidden;			/* ignored by MLT-Shotcut - fix by javascript */
			-webkit-backface-visibility: hidden;			/* also ignored by MLT-Shotcut - fix by javascript */
    }
</style>
<script src='./webvfx.js'></script>
<script>
	function fix_backface_visibility(time, browser, framen_number, frame_rate) {
/*
	To work around the fact that MLT-Shotcut does not recognize the "backface-visibility" CSS property.
	We make the back hidden for the first half of the filter.
	Then for the second half we make the back visible and the front hidden.
	The variables 'front_style' and 'back_style' have to be GLOBAL in scope.
*/
		if (time === 0.0) {
			front_style            = document.getElementById('front').style;
			back_style             = document.getElementById('back' ).style;
			front_style.visibility = 'visible';
			back_style.visibility  = 'hidden';
		} else if (time > 0.5){
			front_style.visibility = 'hidden';
			back_style.visibility  = 'visible';	
		}
	}
	webvfx_add_to_frame = [fix_backface_visibility];
</script>

</head>
<body>
<div id="flip-container">
	<div id='flipper' class='webvfx' data-animate='{
		ease : "easeInOutBounce" ,
		  0% : {webkitTransform: rotateX(  0deg) rotateY(  0deg);} ,
		100% : {webkitTransform: rotateX(180deg) rotateY(180deg);}
	}'>
		<!-- Use tables as other constructs give problems with WebVfx -->
		
		<table id='front'>
			<tr><td>
			    <span id="l1">S</span><span id="l2">H</span><span id="l3">O</span><span id="l4">T</span><span id="l5">C</span><span id="l6">U</span><span id="l7">T</span>
			</td></tr>
		</table>
		<table id='back'>
			<!--the img below is the base64 encoded Shotcut logo -->
		    <tr><td><img alt="shotcut logo"
			    src="">
			</td></tr>
		</table>
	</div>
</div>
</body>
</html>

Credits 2

This overlay produces another set of credits, but instead of scrolling they zoom in from one side of the screen and zoom out of the other, with skewing to add to the realism. All this is done without any user-supplied javascript. To see the animation click on the button below:

The source code (HTML + CSS) that produced the animation is listed below. Use "copy & paste" to copy it to your own computer for you to modify to your requirements.


<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="utf-8">
<link href="https://fonts.googleapis.com/css?family=Open+Sans+Condensed:300" rel="stylesheet">
<!-- WebVfx does not recognise Google web fonts in "<link>", so make sure you also have the font downloaded on your computer. -->
<title>Credits 2</title>
<style>
    html, body	{margin: 0; width: 100%; height: 100%; overflow: hidden; background-color: #000;}
    section		{font: 25px Open Sans Condensed, Liberation Sans, Arial, sans-serif; color: #fff;
				 text-shadow: -1px -1px 0 #444, 1px -1px 0 #444, -1px 1px 0 #444, 1px 1px 0 #444;
				 position: absolute; top:0; left: 0; width: 100%; height:100%;
	} 
	table		{width: 100%; position: absolute; left: 100%; top:50%;
				         transform: perspective(1px) translateY(-50%); /* perspective stops blurring of text */
				 -webkit-transform: perspective(1px) translateY(-50%);
				}
    th	    	{font-size: 3em;} 
    td			{font-size: 0.9em; width: 50%; margin: auto; color: #ddd;}
	tr			{width: 100%; font-size: 1em;}
    td:last-child	{width: 45%; padding: 8px 0.5%; font-size: 2.0em; text-align: left;}
    td:first-child	{width: 45%; padding: 8px 0.5%; font-size: 1.0em; text-align: center;}
	#cast td:first-child, #credits td:first-child{text-align: right;}
</style>

</head>
<body data-control="30:30">
	<section id='title' class='webvfx'  data-animate='{start: 0.0, end: 0.125,
		ease : "easeOutBounce" ,
		  0% : {webkitTransform: translateX(-200%) skewX( 60deg);} ,
		 80% : {webkitTransform: translateX(-100%) skewX(  0deg);} ,
		100% : {webkitTransform: translateX(   0%) skewX( 60deg);}
	}'>
        <table>			
			<tr><th>THE TERMINATOR</th></tr>
		</table>
	</section>
	<section id='cast' class='webvfx'  data-animate='{start: 0.1, end: 0.35,
		ease : "easeOutBounce" ,
		  0% : {webkitTransform: translateX(   0%) skewX(-50deg);} ,
		 80% : {webkitTransform: translateX(-100%) skewX(  0deg);} ,
		100% : {webkitTransform: translateX(-200%) skewX(-50deg);}
	}'>
        <table>			
			<tr><th colspan='2'>Cast</th></tr>
            <tr><td>The Terminator</td><td>Arnold Schwarzenegger</td></tr>
            <tr><td>Kyle Reese</td><td>Michael Biehn</td></tr>
            <tr><td>Sarah Connor</td><td>Linda Hamilton</td></tr>
        </table>
	</section>
	<section id='credits' class='webvfx' data-animate='{start: 0.3, end: 0.6,
		ease : "easeOutBounce" ,
		  0% : {webkitTransform: translateX(   0%) skewX(-50deg);} ,
		 80% : {webkitTransform: translateX(-100%) skewX(  0deg);} ,
		100% : {webkitTransform: translateX(-200%) skewX(-50deg);}
	}'>
        <table>			
			<tr><th colspan='2'>Credits</th></tr>
            <tr><td>Director</td><td>James Cameron</td></tr>
            <tr><td>Writer</td><td>James Cameron</td></tr>
            <tr><td>Writer</td><td>Gayle Anne Hurd</td></tr>
        </table>
	</section>
	<section id='locations' class='webvfx' data-animate='{start: 0.55, end: 0.7,
		ease : "easeOutBounce" ,
		  0% : {webkitTransform: translateX(   0%) skewX(-50deg);} ,
		 80% : {webkitTransform: translateX(-100%) skewX(  0deg);} ,
		100% : {webkitTransform: translateX(-200%) skewX(-50deg);}
	}'>
		<table>
			<tr><th>Locations</th></tr>
        	<tr><td>Elephant Rocks, New Zealand</td></tr>
        	<tr><td>The Cotswolds, UK</td></tr>
			<tr><td>New York, USA</td></tr>
		</table>
	</section>
	<section id='legal' class='webvfx' data-animate='{start: 0.7, end: 0.8,
		ease : "easeOutBounce" ,
		  0% : {webkitTransform: translateX(   0%) skewX(-50deg);} ,
		 80% : {webkitTransform: translateX(-100%) skewX(  0deg);} ,
		100% : {webkitTransform: translateX(-200%) skewX(-50deg);}
	}'>
 		<table>
			<tr><th>Legal</th></tr>
			<tr><td>No animals were harmed during the making of this film.</td></tr>
            <tr><td>&copy;2018 - Elusien Entertainment.<br>
					Except where otherwise noted, content is licensed under a
					Creative Commons Attribution 4.0 International Licence.</td></tr>
		</table>
	</section>
	<section id='shotcut' class='webvfx' data-animate='{start: 0.8, end: 1.0,
		ease : "easeOutBounce" ,
		  0% : {webkitTransform: translateX(-200%) skewX( 50deg);} ,
		 80% : {webkitTransform: translateX(-100%) skewX(  0deg);} ,
		100% : {webkitTransform: translateX(   0%) skewX( 50deg);}
	}'>
 		<table><tr><td>			
			<!-- Now display the shotcut logo -->
			Produced using the free, open source, cross-platform video editor:<br><br>
		    <img alt="shotcut logo" src=""><br>
            https://shotcut.org
		</td></tr></table>
	</section>
</body>
<script src="webvfx.js"></script>
</html>

Stopwatch

This overlay is a bit of a mish-mash. It contains a wandering <div></div> element that morphs from a square to a circle and vice versa. It also contains several stopwatches of various precision, with different initialisation values, one of which pauses and resumes. To see the animation click on the button below:

The source code (HTML + CSS) that produced the animation is listed below. Use "copy & paste" to copy it to your own computer for you to modify to your requirements.


<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<link href="https://fonts.googleapis.com/css?family=Audiowide|Iceland" rel="stylesheet">
<!-- WebVfx does not recognise Google web fonts in "<link>", so make sure you also have the font downloaded on your computer. -->
<title>Stopwatch</title>
<style>
	body {margin: 0; width: 100%; height: 100%; overflow: hidden; background-color: #000;}
	body * {
		color:#fff;	font-family: 'Iceland', monospace; font-size: 30px;
	}
	#wanderer {
	    width: 150px; height: 150px;
	    background-color: #f00;
	    position: relative;
		z-index: -1;
		display: table-cell; 
		vertical-align: middle;
	}
	#wanderer>p	{text-align: center;}
/*		position: absolute; left: 50%; top:50%; font-size: 0.8em; text-align: center;
				         transform: perspective(1px) translate(-50%, -50%);
				 -webkit-transform: perspective(1px) translate(-50%, -50%);
	}*/
	[data-stopwatch]>span {
		display: inline-block;
		border: solid 1px #f00;
		padding: 2px; margin: 2px 0;
	}
	#middle {
		position: absolute;
		top:      50%; left:     50%;
				transform: perspective(1px) translate(-50%, -50%);
		-webkit-transform: perspective(1px) translate(-50%, -50%);
	}
</style>
</head>

<body data-control='20:30'>
	<div id='wanderer' class='webvfx'
		 data-animate='{start: 0.1, end: 1.0, ease: "easeInOutBounce",
					       0%: {backgroundColor: #f00; left:  0px; top:  0px; borderRadius:  0%;},
					      25%: {backgroundColor: #00f; left:100px; top:  0px; borderRadius: 50%;},
						  50%: {backgroundColor: #0f0; left:200px; top:200px; borderRadius:  0%;},
						  75%: {backgroundColor: #f0f; left:  0px; top:200px; borderRadius: 50%;},
						 100%: {backgroundColor: #ff0; left:100px; top:  0px; borderRadius:  0%;}
					   }'>
		<p>WANDERER</p>
	</div>
	
	<div class='webvfx'
		 data-stopwatch='{start: 0.2, end: 0.8,
							10%: {pause;},
							20%: {resume;},
							40%: {pause;},
							60%: {resume: skip;},
							80%: {pause;},
							90%: {resume: 100000;}
						 }'>
		Frame <span>0</span> => </span><span>00</span>h <span>00</span>m <span>00</span>s <span>000</span>ms
	</div>
	
	<div class='webvfx' data-stopwatch=''>
	   <span>0</span> => <span></span><span>00</span>:<span>00</span>.<span>000</span>
	</div>
	
	<div id='middle' class="webvfx" data-stopwatch=''>						
	   <span>9000</span> => <span></span><span>08</span>:<span>10</span><span></span>
	</div>
	
	<p>	Assuming a framerate of 30fps. </p>
</body>

<script src="WebVfx.js"></script>
</html>

Gauge

This overlay is a simulation of a radiation gauge with some LEDs thrown in for good measure. It uses the excellent canvas gauge framework of Mykhailo Stadnyk. (see canvas-gauges.com)

We actually do different things depending on whether we are running in a browser or in Shotcut. Updating the canvas takes quite a bit of CPU time and the Canvas Gauge library does this for various points between the current position of the needle and the new one. If we are in the browser it can take longer than the setTimeout value, so we only update the gauge value every 50 frames or so (more frequently than that can cause problems). The browser will still show smooth motion of the needle between values since the gauge library interpolates values in between. If we are in Shotcut it is a different matter. We will only see the end value of the needle, not the interpolations in-between. But CPU time isn't a problem as we have lots to spare when rendering, so we can update the gauge value for every frame if we want to get smooth motion of the needle (hence the "if" test at the beginning of the code). I may look into modifying the Canvas Gauge library itself to introduce into the Shotcut filter the same smooth needle movement that is seen in the browser.

To see the animation click on the button below:

The source code (HTML + CSS) that produced the animation is listed below. Use "copy & paste" to copy it to your own computer for you to modify to your requirements.


<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<link href="https://fonts.googleapis.com/css?family=Audiowide|Iceland" rel="stylesheet">
<!-- WebVfx does not recognise Google web fonts in "<link>", so make sure you also have the font downloaded on your computer. -->
<title>Gauge</title>
<style>
/* LEDs */
.led_box	{height: 30px; width: 75px; margin: 10px 0; float: left;}
.led		{margin: 0 auto; width: 24px; height: 24px; background-color: #F00; border-radius: 50%;}
#led_red.led_red_on			{background-color: #F00;    box-shadow: rgba(0, 0, 0, 0.2) 0 -1px 7px 1px, inset #441313 0 -1px 9px, rgba(255, 0, 0, 0.5) 0 2px 12px;}
#led_yellow.led_yellow_on	{background-color: #FF0;    box-shadow: rgba(0, 0, 0, 0.2) 0 -1px 7px 1px, inset #808002 0 -1px 9px, #FF0                 0 2px 12px;}
#led_blue.led_blue_on		{background-color: #24E0FF; box-shadow: rgba(0, 0, 0, 0.2) 0 -1px 7px 1px, inset #006    0 -1px 9px, #3F8CFF              0 2px 14px;}	
#led_green.led_green_on		{background-color: #ABFF00; box-shadow: rgba(0, 0, 0, 0.2) 0 -1px 7px 1px, inset #304701 0 -1px 9px, #89FF00              0 2px 12px;}

#led_red 					{background-color: #800;    box-shadow: rgba(0, 0, 0, 0.2) 0 -1px 7px 1px, inset #441313 0 -1px 9px, rgba(255, 0, 0, 0.5) 0 2px 0;}
#led_yellow 				{background-color: #880;    box-shadow: rgba(0, 0, 0, 0.2) 0 -1px 7px 1px, inset #808002 0 -1px 9px, #FF0                 0 2px 0;}
#led_blue					{background-color: #007088; box-shadow: rgba(0, 0, 0, 0.2) 0 -1px 7px 1px, inset #006    0 -1px 9px, #3F8CFF              0 2px 0;}	
#led_green					{background-color: #378800; box-shadow: rgba(0, 0, 0, 0.2) 0 -1px 7px 1px, inset #304701 0 -1px 9px, #89FF00              0 2px 0;}
</style>
<body data-control="20:30">
	<h2>https://canvas-gauges.com/</h1>
	<canvas id="gauge"></canvas>
	<div>
		<div class="led_box"><div id="led_green"  class="led led_green_on"	></div></div>
		<div class="led_box"><div id="led_blue"   class="led"  				></div></div>
		<div class="led_box"><div id="led_yellow" class="led"				></div></div>
		<div class="led_box"><div id="led_red"    class="led"				></div></div>
	</div>
</body>

<script>
/*
	The "repeat" String method is used in the Gauge javascript. However, it was
	only defined in the ECMA-262 6th Edition (June 2015) specification. The
	version of ECMAScript used by MLT-Shotcut does not have this method, so we
	create one.
*/
	if (typeof String.prototype.repeat == 'undefined') {
		String.prototype.repeat = function(n){
			var x = this;
			return Array(n+1).join(x);
		};
	}
//	alert("+".repeat(4));
</script>

<!-- <script src="http://cdn.rawgit.com/Mikhus/canvas-gauges/gh-pages/download/2.1.4/all/gauge.min.js"></script> NOT WITH MLT-SHOTCUT -->
<script src="gauge.min.js"></script>
<script src="WebVfx.js"></script>
<script>

//	Create the gauge on the canvas

		old_colour = "green";
		leds = {
			green  : document.getElementById("led_green" ).classList ,
			blue   : document.getElementById("led_blue"  ).classList ,
			yellow : document.getElementById("led_yellow").classList ,
			red    : document.getElementById("led_red"   ).classList 
		};
		gauge = new RadialGauge({
			renderTo         : 'gauge',
			width            : 300,
			height           : 300,
			units            : "Radiation (Bq)",
			value            : 0,
			minValue         : 0,
			startAngle       : 90,
			ticksAngle       : 180,
			valueBox         : true,
			valueInt         : 2,
			valueDec         : 1,
			maxValue         : 100,
			majorTicks       : ["0", "20", "40", "60", "80", "100"],
			minorTicks       : 2,
			strokeTicks      : true,
			highlights       : [
				{"from":  0, "to":  15, "color": "rgba(100, 255, 100, .20)"},
				{"from": 15, "to":  60, "color": "rgba(100, 100, 255, .20)"},
				{"from": 60, "to":  80, "color": "rgba(220, 200,   0, .75)"},
				{"from": 80, "to": 100, "color": "rgba(200,  50,  50, .75)"}
			],
			colorPlate       : "#ddd",
			borderShadowWidth: 3,
			borders          : false,
			needleType       : "arrow",
			needleWidth      : 5,
			needleCircleSize : 2,
			needleCircleOuter: true,
			needleCircleInner: false,
			animationDuration: 1500,
			animationRule    : "linear"
		}).draw();

//	Create the animation function called by Elusien's WebVfx Framework

function animate_gauge(time, browser, frame_number, frame_rate){
// Update every 50th frame in the browser and every frame in Shotcut.
	if (!browser || ((frame_number % 50) === 0)){
		var value  = (Math.abs(Math.sin(time*Math.PI*2.3)) * 99).toFixed(1);
		gauge.value = value;	// update the canvas (needle and value)

//	Now decide which LEDs to update.

		var colour  = "red";
		if (value < 80) { colour = "yellow";}
		if (value < 60) { colour = "blue"  ;}
		if (value < 15) { colour = "green" ;}
		if (colour == old_colour) {return;}
		leds[old_colour].remove("led_" + old_colour + "_on");
		old_colour = colour;
		leds[old_colour].add   ("led_" + old_colour + "_on");
  
// Update the declarative chart...
//		document.getElementById("gauge").setAttribute("data-value", value);		
	}
}

	webvfx_add_to_frame = [animate_gauge];	// Tell the framework to call the animation function.

</script>
</html>

Text

This overlay is based on an example CSS animation provided by "Jonray". The CSS version can be seen here (https://codepen.io/jonray1/pen/QQoNWg). There is a slight modification in the text-shadow to workaround what appears to be an MLT/Shotcut bug. Basically, text-shadows work in all situation but one: when the shadow is offset and the blur is non zero. I have left the original CSS keyframes in the HTML for comparison, but have obviously not called CSS animation. To see the animation click on the button below:

The source code (HTML + CSS) that produced the animation is listed below. Use "copy & paste" to copy it to your own computer for you to modify to your requirements.


<html lang="en">
<head>
<meta charset="utf-8">
<title>Bounville</title>
<style>
body {
background-color: powderblue;
}
div {overflow: visible; font-family: arial; opacity: 0; position: absolute;
/*				 text-shadow: 0 0 12px #222; */
				 text-shadow:	0 0 10px #fff, 0 0 20px #fff, 0 0 30px #fff, 0 0 40px #228DFF, 0 0 70px #228DFF, 0 0 80px #228DFF, 0 0 100px #228DFF, 0 0 150px #228DFF;
	}
		
		#letter01 {
		font-size:200px;
		color:blue;
/*		animation: fadeindrop01 1.5s ease forwards 1s,
		moveright01 2s ease forwards 2.6s ;*/
		}
		
		
		@keyframes fadeindrop01 {
0%   { opacity:0; left:370px; top:-100px;}
100% {opacity:1; left:370px; top:210px; }
}

		@keyframes moveright01 {
0%   {left:370px; top:210px;}
100% {left:410px; top:210px; }
		}

#letter02 {
		font-size:150px;
		color:green;
/*		animation: fadeinup02 1s ease forwards 1.1s,
		moveright02 1s ease forwards 2.2s ; */
		}
		
		@keyframes fadeinup02 {
0%   {opacity:0; left:580px; top:500px;}
100% {opacity:1; left:580px; top:250px;}
}

		@keyframes moveright02 {
0%   {left:580px; top:250px;}
100% {left:540px; top:250px;}
}

#letter03 {
		font-size:150px;
		color:orange;
/*		animation: fadeindown03 1s ease forwards 1.3s,
		moveright03 1s ease forwards 2.4s ; */
		}
		
		@keyframes fadeindown03 {
0%   {opacity:0; left:700px; top:50px;}
100% {opacity:1; left:700px; top:250px;}
}

		@keyframes moveright03 {
0%   {left:700px; top:250px;}
100% {left:620px; top:250px;}
}

#letter04 {
		font-size:150px;
		color:grey;
/*		animation: fadeindown04 1.3s ease forwards 2.6s,
		moveright04 1.6s ease forwards 4s ; */
		}
		
		@keyframes fadeindown04 {
0%   {opacity:0; left:680px; top:230px;}
100% {opacity:1; left:700px; top:250px;}
}

		@keyframes moveright04 {
0%   {left:700px; top:250px;}
100% {left:700px; top:250px;}
}

#letter05 {
		font-size:150px;
		color:red;
		z-index: 2;
/*		animation: fadeindown05 2s ease forwards 2.8s,
		moveright05 1.5s ease forwards 4.9s ; */
		}
		
		@keyframes fadeindown05 {
0%   {opacity:0; left:750px; top:300px;}
100% {opacity:1; left:750px; top:250px;}
}

		@keyframes moveright05 {
0%   {left:750px; top:250px;}
100% {left:750px; top:250px;}
}

#letter06 {
		font-size:400px;
		color:#b3b3ff;
		z-index:1;
/*		animation: fadeindown06 1s ease forwards 3.5s; */
		
		}
		
		@keyframes fadeindown06 {
0%   {opacity:0; left:770px; top:60px;}
100% {opacity:1; left:770px; top:80px;}
}

#letter07 {
		font-size:150px;
		color:blue;
		z-index:2;
/*		animation: fadeindown07 2s ease forwards 4s,
		moveright07 1s ease forwards 8s ; */
		}
		
		@keyframes fadeindown07 {
0%   {opacity:0; left:900px; top:250px;}
100% {opacity:1; left:900px; top:250px;}
}

		@keyframes moveright07 {
0%   {left:900px; top:250px;}
100% {left:900px; top:250px;}
}

#letter08 {
		font-size:150px;
		color:green;
		z-index:2;
/*		animation: fadeindown08 2s ease forwards 4.2s,
		moveright08 1s ease forwards 8s ; */
		}
		
		@keyframes fadeindown08 {
0%   {opacity:0; left:930px; top:250px;}
100% {opacity:1; left:930px; top:250px;}
}

		@keyframes moveright08 {
0%   {left:930px; top:250px;}
100% {left:930px; top:250px;}
}


#letter09 {
		font-size:150px;
		color:grey;
		z-index:2;
/*		animation: fadeindown09 2s ease forwards 4.4s;
		moveright09 1s ease forwards 8s ;*/
		}
		
		@keyframes fadeindown09 {
0%   {opacity:0; left:960px; top:250px;}
100% {opacity:1; left:960px; top:250px;}
}

		@keyframes moveright09 {
0%   {left:960px; top:250px;}
100% {left:960px; top:250px;}
}

#letter10 {
		font-size:150px;
		color:red;
		z-index:2;
/*		animation: fadeindown10 2s ease forwards 4.6s; */
}
		
		@keyframes fadeindown10 {
0%   {opacity:0; left:990px; top:250px;}
100% {opacity:1; left:990px; top:250px;}
}

#line01 {
        width:390px;
		height:4px;
		background-color: grey;
		top:410px;
		left:430px;
/*		animation: justfade01 2s ease forwards 5s; */
}
		
		@keyframes justfade01 {
0%   {opacity:0; }
100% {opacity:1;}
}

#line02 {
		width:130px;
		height:4px;
		background-color: grey;
		top:410px;
		left:920px;
/*		animation: justfade02 2s ease forwards 5.1s; */
}

		@keyframes fadeindrop01 {
0%   { opacity:0; left:370px; top:-100px;}
100% {opacity:1; left:370px; top:210px; }
}

		@keyframes moveright01 {
0%   {left:370px; top:210px;}
100% {left:410px; top:210px; }


</style>
</head>
<body>
<div id="letter01" class='webvfx' data-animate='{start: 0.1, end: 0.25, ease: "easeInOutBounce",
	  0% : {opacity:0; left:370px; top:-100px;},
	100% : {opacity:1; left:370px; top:210px;}
}'
		data-animate2='{start: 0.26, end: 0.46, ease: "easeInOutBounce",
	  0% : {left:370px; top:210px;},
	100% : {left:410px; top:210px;}
}'>B</div>
<div id="letter02" class='webvfx' data-animate='{start: 0.11, end: 0.21, ease: "easeInOutBounce",
	  0% : {opacity:0; left:580px; top:500px;},
	100% : {opacity:1; left:580px; top:250px;}
}'
		data-animate2='{start: 0.22, end: 0.32, ease: "easeInOutBounce",
	  0% : {left:580px; top:250px;},
	100% : {left:540px; top:250px;}
}'>o</div>
<div id="letter03" class='webvfx' data-animate='{start: 0.13, end: 0.23, ease: "easeInOutBounce",
	  0% : {opacity:0; left:700px; top:50px;},
	100% : {opacity:1; left:700px; top:250px;}
}'
		data-animate2='{start: 0.24, end: 0.34, ease: "easeInOutBounce",
	  0% : {left:700px; top:250px;},
	100% : {left:620px; top:250px;}
}'>u</div>
<div id="letter04" class='webvfx' data-animate='{start: 0.26, end: 0.39, ease: "easeInOutBounce",
	  0% : {opacity:0; left:680px; top:230px;},
	100% : {opacity:1; left:700px; top:250px;}
}'
		data-animate2='{start: 0.4, end: 0.56, ease: "easeInOutBounce",
	  0% : {left:700px; top:250px;},
	100% : {left:700px; top:250px;}
}'>r</div>
<div id="letter05" class='webvfx' data-animate='{start: 0.28, end: 0.48, ease: "easeInOutBounce",
	  0% : {opacity:0; left:750px; top:300px;},
	100% : {opacity:1; left:750px; top:250px;}
}'
		data-animate2='{start: 0.49, end: 0.64, ease: "easeInOutBounce",
	  0% : {left:750px; top:250px;},
	100% : {left:750px; top:250px;}
}'>n</div>
<div id="letter06" class='webvfx' data-animate='{start: 0.25, end: 0.45, ease: "easeInOutBounce",
	  0% : {opacity:0; left:770px; top:60px;},
	100% : {opacity:1; left:770px; top:80px;}
}'>v</div>
<div id="letter07" class='webvfx' data-animate='{start: 0.4, end: 0.6, ease: "easeInOutBounce",
	  0% : {opacity:0; left:900px; top:250px;},
	100% : {opacity:1; left:900px; top:250px;}
}'
		data-animate2='{start: 0.1, end: 0.9, ease: "easeInOutBounce",
	  0% : {left:900px; top:250px;},
	100% : {left:900px; top:250px;}
}'>i</div>
<div id="letter08" class='webvfx' data-animate='{start: 0.46, end: 0.66, ease: "easeInOutBounce",
	  0% : {opacity:0; left:930px; top:250px;},
	100% : {opacity:1; left:930px; top:250px;}
}'
		data-animate2='{start: 0.1, end: 0.9, ease: "easeInOutBounce",
	  0% : {left:930px; top:250px;},
	100% : {left:930px; top:250px;}
}'>l</div>
<div id="letter09" class='webvfx' data-animate ='{start: 0.44, end: 0.64, ease: "easeInOutBounce",
      0% : {opacity:0; left:960px; top:250px;},
    100% : {opacity:1; left:960px; top:250px;}
}'
		data-animate2='{start: 0.8, end: 0.9, ease: "easeInOutBounce",
	  0% : {left:960px; top:250px;},
	100% : {left:960px; top:250px;}
}'>l</div>
<div id="letter10" class='webvfx' data-animate ='{start: 0.56, end: 0.76, ease: "easeInOutBounce",
      0% : {opacity:0; left:990px; top:250px;},
    100% : {opacity:1; left:990px; top:250px;}
}'>e</div>
<div id="line01" class='webvfx' data-animate='{start: 0.5, end: 0.7, ease: "easeInOutBounce",
	  0% : {opacity:0;},
	100% : {opacity:1;}
}'></div>
<div id="line02" class='webvfx' data-animate='{start: 0.51, end: 0.71, ease: "easeInOutBounce",
	  0% : {opacity:0;},
	100% : {opacity:1;}
}'></div>

</body>
<script src="WebVfx.js"></script>
</html>

Slideshow

This overlay is used to create a reasonably simple slideshow. It works best with a few slides, but has been tested on a 4GB RAM system using 100 slides. It provides for various transitions between slides and all of it is parameterised from the HTML and CSS. No knowledge of Javascript is required.

Please read the documentation as there are some deficiencies in MLT (used by Shotcut) that can only be worked-around by using CSS units based on the viewport (i.e. 'vw' and 'vh'), rather than the normal length units ('px' or '%'). To see the animation click on the button below:

The source code (HTML + CSS) that produced the animation is listed below. Use "copy & paste" to copy it to your own computer for you to modify to your requirements.


<!DOCTYPE html>
<html lang="en">
<head>
	<meta charset="utf-8">
	<title>Slideshow</title>
	<style>
/*
	The body background pattern (a sort of black metallic mesh) is courtesy of:
		https://www.toptal.com/designers/subtlepatterns/thumbnail-view/
*/
	    html, body	{margin:0; padding:0; width: 100vw; height: 100vh; overflow: hidden;}	/* Set Body = Viewport */
		      body  {background: center repeat url('./bo_play_pattern.png');}				/* (optionally) set a body background */
/*
	IMPORTANT:	Centre the <section> by using viewport units only ('vw' and 'vh')
				and make sure that both 'width' plus 2 x right-margin and
				'height' plus 2 x bottom-margin = 100
*/
		section		{position: relative; top: 0; left: 0; overflow: hidden;
					 width: 90vw; height: 90vh; margin: 5vh 5vw; padding: 0;}
					 
/*	IMPORTANT:	Make the #slides <div> the same size as the <section> */

		#slides		{top: 0; left: 0; height: 90vh; width: 90vw;
					 background:  url('./pull_focus_bg.jpg') center #666;}
/*
	IMPORTANT:	You can add a border (e.g. white to make it look like a printed photo)
				and a margin, if you like, but:
				MAKE SURE THE UNITS ARE ONLY 'px', NOT 'vw/vh' or '%'.
				(This is due to an MLT deficiency)
*/				
		img         {display: none; border: solid 10px #fff; margin: 2px;}
	</style>
</head>
<body>
	
<!--
	============================================================================
	        A SHOTCUT Overlay HTML Filter using Elusien's WebVfx framework
			--------------------------------------------------------------
					                  Slideshow
	============================================================================
-->
	<section>
		<div id='slides' data-control='60:30' class='webvfx' data-slideshow='{
			duration: 25% , type: splitting, from: "left", magnify: 25%, blur: 50, ease: easeInOutSine}'>
			<img src='http://mhcommittee.elusien.co.uk/images/home.jpg'                                  >
			<img src='http://mhcommittee.elusien.co.uk/images/home_1.jpg'                                >
			<img src='http://mhcommittee.elusien.co.uk/images/Melamine-wood-002.png'                     >
			<img src='http://mhcommittee.elusien.co.uk/images/Melamine-wood-003.png'                     >
			<img src='http://mhcommittee.elusien.co.uk/images/home_2.jpg'                                >
			<img src='http://mhcommittee.elusien.co.uk/images/home.jpg'                                  >
		    <img src='http://trips.elusien.co.uk/2017c-New_Zealand/images/Day10a+Gannet_Colony.JPG'      >                                              
		    <img src='http://trips.elusien.co.uk/2017c-New_Zealand/images/Day10b+Gannets.JPG'            >
		    <img src='http://trips.elusien.co.uk/2017c-New_Zealand/images/Day10d+Bethells_Beach_Cave.JPG'>
		    <img src='http://trips.elusien.co.uk/2017c-New_Zealand/images/Day10e+Karakare_Falls.JPG'     >
		</div>
	</section>
</body>
<script src="WebVfx.js"></script> <!-- Use Elusien's WebVfx framework (www.elusien.co.uk/shotcut) -->
</html>

Explosion

This overlay takes some text, in the form of movie credits, breaks it down into the individual characters then simulates an implosion followed by an explosion by animating these characters, making them fly over, into and out of the screen. To see the animation click on the button below:

The source code (HTML + CSS) that produced the animation is listed below. Use "copy & paste" to copy it to your own computer for you to modify to your requirements.


<!DOCTYPE html>
<html lang="en">
<head>
	<meta charset="utf-8">
	<link href="https://fonts.googleapis.com/css?family=Open+Sans+Condensed:300" rel="stylesheet">
	<!-- WebVfx does not recognise Google web fonts in "<link>", so make sure you also have the font downloaded on your computer. -->
	<title>Explosion</title>
	<style>
		html, body	{margin: 0; width: 100%; height: 100%; overflow: hidden; background-color: #000;}
		section		{font: 25px Open Sans Condensed, Liberation Sans, Arial, sans-serif; color: #fff;
					 text-shadow: -1px -1px 0 #444, 1px -1px 0 #444, -1px 1px 0 #444, 1px 1px 0 #444;
					 position: absolute; top:0; left: 0; width: 100%; height:100%;
					} 
		table		{width: 100%; position: absolute; left: 0; top:50%;
							 transform: perspective(1px) translateY(-50%); /* perspective reduces blurring of text */
					 -webkit-transform: perspective(1px) translateY(-50%);
					}
		th	    	{font-size:   3em;} 
		td			{font-size: 0.9em; width: 50%; margin: auto; color: #ddd;}
		tr			{font-size:   1em; width: 100%;}
		td:last-child	{width: 45%; padding: 8px 0.5%; font-size: 2.0em; text-align: left;}
		td:first-child	{width: 45%; padding: 8px 0.5%; font-size: 1.0em; text-align: center;}
		#cast td:first-child, #credits td:first-child					 {text-align: right;}
	</style>
</head>
<body data-control='45:30''>
	<section id='title' class='webvfx' data-explosion='{start: 0.1 , end: 0.2, finish: hidden, ease: easeInOutSine, 0%: explode}'>
        <table>			
			<tr><th>THE TERMINATOR</th></tr>
		</table>
	</section>
	<section id='cast' class='webvfx'  data-explosion='{start: 0.15, end: 0.3, begin:  hidden, ease: easeInOutSine, 0%: implode, 40%: wait, 60%: explode}'>
        <table>			
			<tr><th colspan='2'>Cast</th></tr>
            <tr><td>The Terminator</td><td>Arnold Schwarzenegger</td></tr>
            <tr><td>Kyle Reese</td><td>Michael Biehn</td></tr>
            <tr><td>Sarah Connor</td><td>Linda Hamilton</td></tr>
        </table>
	</section>
	<section id='credits' class='webvfx' data-explosion='{start: 0.3 , end: 0.6, begin:  hidden, ease: easeInOutSine, 0%: implode, 40%: wait, 60%: explode}'>
        <table>			
			<tr><th colspan='2'>Credits</th></tr>
            <tr><td>Director</td><td>James Cameron</td></tr>
            <tr><td>Writer</td><td>James Cameron</td></tr>
            <tr><td>Writer</td><td>Gayle Anne Hurd</td></tr>
        </table>
	</section>
	<section id='locations' class='webvfx' data-explosion='{start: 0.6 , end: 0.8, begin:  hidden, ease: easeInOutSine, 0%: implode, 40%: wait, 60%: explode}'>
		<table>
			<tr><th>Locations</th></tr>
        	<tr><td>Elephant Rocks, New Zealand</td></tr>
        	<tr><td>The Cotswolds, UK</td></tr>
			<tr><td>New York, USA</td></tr>
		</table>
	</section>
	<section id='legal' class='webvfx' data-explosion='{start: 0.8 , end: 1.0, begin:  hidden, ease: easeInOutSine, 0%: implode, 50%: wait}'>
 		<table>
			<tr><th>Legal</th></tr>
			<tr><td>No animals were harmed during the making of this film.</td></tr>
            <tr><td>&copy;2018 - Elusien Entertainment.<br>
					Except where otherwise noted, content is licensed under a 
					Creative Commons Attribution 4.0 International Licence.</td></tr>
		</table>
	</section>
</body>
<script src="webvfx.js"></script>
</html>

(De)Fragmentation

This overlay uses the "explosion" feature to animate a header, while at the same time defragmenting and then fragmenting the Shotcut logo. To see the animation click on the button below:

The source code (HTML + CSS) that produced the animation is listed below. Use "copy & paste" to copy it to your own computer for you to modify to your requirements.


<html lang="en">
<head>
<meta charset="utf-8">
<title>Fragmentation</title>
<style>
    html, body {margin:0; padding:0; width: 100vw; height: 100vh; background-color: #000; overflow: hidden; color: #fff;}
	h1	{color: #eee; text-align: center; padding: 10vh 0 0 0;}
</style>

</head>
<body data-control='15:30'>
	<h1  class='webvfx' data-explosion='{start: 0.05, end: 1.0, begin: hidden, ease: easeInOutSine,
		0%: implode, 40%: wait, 60%: explode}'>
		Fragmenting The Shotgun Logo (https://vimeo.com/268484362)
	</h1>
	<div class='webvfx' data-center='vh'
		 data-fragment='{start: 0.05, end: 1.0, nh: 5, nv: 20, begin: hidden, ease: easeInOutSine,
		0%: defragment, 40%: wait, 60%: fragment}'>
		<img alt="shotcut logo"
			    src="">
	</div>
</body>
<script src='./webvfx.js'></script>
</html>

Subtitles

This overlay uses the "subtitles" feature to display subtitles that have been copied from a Subrip file (.srt). To see the animation click on the button below:

The source code (HTML + CSS) that produced the animation is listed below. Use "copy & paste" to copy it to your own computer for you to modify to your requirements.


<!DOCTYPE html>
<html lang="en">
<head>
	<meta charset="utf-8">
	<title>Subtitles</title>
	<style>
	    html, body	{margin:0; padding:0; width: 100vw; height: 100vh; background-color: transparent; overflow: hidden; border: solid 1px #000;}
		#st1		{position: absolute;  width: 90vw; left: 0; top: 0   ; margin: 1vh 0 1vh 5vw; padding: 0.25em;
					 background-color: rgba(0, 0, 0, 0.5);
					 font: 18px Open Sans Condensed, Liberation Sans, Arial, sans-serif; color: #fff;
					 text-shadow: -1px -1px 0 #444, 1px -1px 0 #444, -1px 1px 0 #444, 1px 1px 0 #444;
					 text-align: center; border-radius: 0.5em;
					}
		#st2    	{position: absolute;  width: 100vw; left: 0; bottom: 0; margin: 1vh 0 1vh 5vw; padding: 0.25em;
					 background-color: transparent;
					 font: 18px Open Sans Condensed, Liberation Sans, Arial, sans-serif; color: #fff;
					 text-shadow: -1px -1px 0 #444, 1px -1px 0 #444, -1px 1px 0 #444, 1px 1px 0 #444;
					 text-align: center; border-radius: 0.5em; letter-spacing: 2px;
					}
	</style>
</head>
<body data-control='40:30'>
	The subtitles appear at the top (#srt1) of the screen and also at the bottom (#srt2).
	<div id='st1' class='webvfx' data-subtitles='subtitles'></div>
	<div id='st2' class='webvfx' data-subtitles='subtitles'></div>
	<pre id='subtitles'>
1
00:00:04,000 --> 00:00:15,400
This is the 1st subtitle 04sec to 15.4sec
Mary had a little lamb,
its <i>fleece</i> was white as snow
and everywhere that Mary went
that <b>lamb</b> was sure to go!

2
00:00:16,600 --> 00:00:25,800
This is the 2nd subtitle 16.6sec to 25.8sec

3
00:00:20,600 --> 00:00:23,800
This is the 3rd subtitle 20.6sec to 23.8sec

4
00:00:24,600 --> 00:00:30,800
This is the 4th subtitle 24.6sec to 30.8sec

</pre>
</body>
<script src='./webvfx.js'></script>
</html>

Typewriter

This overlay uses the "typewriter" feature to display text as if it were output on a typewriter. A cursor is displayed after each letter and at the beginning of the sequence, for the time it would take to output 10 letters and also at the end of the sequence for the time it would take to output 8 letters To see the animation click on the button below:

The source code (HTML + CSS) that produced the animation is listed below. Use "copy & paste" to copy it to your own computer for you to modify to your requirements.


<!DOCTYPE html>
<html lang="en">
<head>
	<meta charset="utf-8">
	<title>Subtitles</title>
	<style>
	    html, body	{margin:0; padding:0; width: 100vw; height: 100vh; background-color: transparent; overflow: hidden; border: solid 1px #000;}
		#st1		{position: absolute;  width: 90vw; left: 0; top:30vh   ; margin: 1vh 0 1vh 5vw; padding: 0.25em;
					 background-color: #444;
					 font: 32px Open Sans Condensed, Liberation Sans, Arial, sans-serif; color: #fff;
					 text-shadow: -1px -1px 0 #444, 1px -1px 0 #444, -1px 1px 0 #444, 1px 1px 0 #444;
					 text-align: center; border-radius: 0.5em;
					}
		._typewriter_ {border-right: solid #fff 2px; margin: 2px;}
	</style>
</head>
<body data-control='20:30'>
	<div id='st1' class='webvfx' data-typewriter='{start: 0.1, end: 0.9, stx: 10, etx: 20}'>
		Mary had a little lamb, <br>
		its <i>fleece</i> was white as snow <br>
		and everywhere that Mary went <br>
		that <b>lamb</b> was sure to go!
	</div>
</body>
<script src='./webvfx.js'></script>
</html>