1977 lines
		
	
	
		
			58 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			1977 lines
		
	
	
		
			58 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| /*****************************************************************
 | |
|  ** Author: Asvin Goel, goel@telematique.eu
 | |
|  **
 | |
|  ** A plugin for reveal.js adding a chalkboard.
 | |
|  **
 | |
|  ** Version: 2.1.0
 | |
|  **
 | |
|  ** License: MIT license (see LICENSE.md)
 | |
|  **
 | |
|  ** Credits:
 | |
|  ** Chalkboard effect by Mohamed Moustafa https://github.com/mmoustafa/Chalkboard
 | |
|  ** Multi color support initially added by Kurt Rinnert https://github.com/rinnert
 | |
|  ** Compatibility with reveal.js v4 by Hakim El Hattab https://github.com/hakimel
 | |
|  ******************************************************************/
 | |
| 
 | |
| window.RevealChalkboard = window.RevealChalkboard || {
 | |
| 	id: 'RevealChalkboard',
 | |
| 	init: function ( deck ) {
 | |
| 		initChalkboard( deck );
 | |
| 	},
 | |
| 	configure: function ( config ) {
 | |
| 		configure( config );
 | |
| 	},
 | |
| 	toggleNotesCanvas: function () {
 | |
| 		toggleNotesCanvas();
 | |
| 	},
 | |
| 	toggleChalkboard: function () {
 | |
| 		toggleChalkboard();
 | |
| 	},
 | |
| 	colorIndex: function () {
 | |
| 		colorIndex();
 | |
| 	},
 | |
| 	colorNext: function () {
 | |
| 		colorNext();
 | |
| 	},
 | |
| 	colorPrev: function () {
 | |
| 		colorPrev();
 | |
| 	},
 | |
| 	clear: function () {
 | |
| 		clear();
 | |
| 	},
 | |
| 	reset: function () {
 | |
| 		reset();
 | |
| 	},
 | |
| 	resetAll: function () {
 | |
| 		resetAll();
 | |
| 	},
 | |
| 	updateStorage: function () {
 | |
| 		updateStorage();
 | |
| 	},
 | |
| 	getData: function () {
 | |
| 		return getData();
 | |
| 	},
 | |
| 	download: function () {
 | |
| 		download();
 | |
| 	},
 | |
| };
 | |
| 
 | |
| function scriptPath() {
 | |
| 	// obtain plugin path from the script element
 | |
| 	var src;
 | |
| 	if ( document.currentScript ) {
 | |
| 		src = document.currentScript.src;
 | |
| 	} else {
 | |
| 		var sel = document.querySelector( 'script[src$="/chalkboard/plugin.js"]' )
 | |
| 		if ( sel ) {
 | |
| 			src = sel.src;
 | |
| 		}
 | |
| 	}
 | |
| 	var path = ( src === undefined ) ? "" : src.slice( 0, src.lastIndexOf( "/" ) + 1 );
 | |
| //console.log("Path: " + path);
 | |
| 	return path;
 | |
| }
 | |
| var path = scriptPath();
 | |
| 
 | |
| const initChalkboard = function ( Reveal ) {
 | |
| //console.warn(path);
 | |
| 	/* Feature detection for passive event handling*/
 | |
| 	var passiveSupported = false;
 | |
| 
 | |
| 	try {
 | |
| 		window.addEventListener( 'test', null, Object.defineProperty( {}, 'passive', {
 | |
| 			get: function () {
 | |
| 				passiveSupported = true;
 | |
| 			}
 | |
| 		} ) );
 | |
| 	} catch ( err ) {}
 | |
| 
 | |
| 
 | |
| /*****************************************************************
 | |
|  ** Configuration
 | |
|  ******************************************************************/
 | |
| 	var background, pen, draw, color;
 | |
| 	var grid = false;
 | |
| 	var boardmarkerWidth = 3;
 | |
| 	var chalkWidth = 7;
 | |
| 	var chalkEffect = 1.0;
 | |
| 	var rememberColor = [ true, false ];
 | |
| 	var eraser = {
 | |
| 		src: path + 'img/sponge.png',
 | |
| 		radius: 20
 | |
| 	};
 | |
| 	var boardmarkers = [ {
 | |
| 			color: 'rgba(100,100,100,1)',
 | |
| 			cursor: 'url(' + path + 'img/boardmarker-black.png), auto'
 | |
| 		},
 | |
| 		{
 | |
| 			color: 'rgba(30,144,255, 1)',
 | |
| 			cursor: 'url(' + path + 'img/boardmarker-blue.png), auto'
 | |
| 		},
 | |
| 		{
 | |
| 			color: 'rgba(220,20,60,1)',
 | |
| 			cursor: 'url(' + path + 'img/boardmarker-red.png), auto'
 | |
| 		},
 | |
| 		{
 | |
| 			color: 'rgba(50,205,50,1)',
 | |
| 			cursor: 'url(' + path + 'img/boardmarker-green.png), auto'
 | |
| 		},
 | |
| 		{
 | |
| 			color: 'rgba(255,140,0,1)',
 | |
| 			cursor: 'url(' + path + 'img/boardmarker-orange.png), auto'
 | |
| 		},
 | |
| 		{
 | |
| 			color: 'rgba(150,0,20150,1)',
 | |
| 			cursor: 'url(' + path + 'img/boardmarker-purple.png), auto'
 | |
| 		},
 | |
| 		{
 | |
| 			color: 'rgba(255,220,0,1)',
 | |
| 			cursor: 'url(' + path + 'img/boardmarker-yellow.png), auto'
 | |
| 		}
 | |
| 	];
 | |
| 	var chalks = [ {
 | |
| 			color: 'rgba(255,255,255,0.5)',
 | |
| 			cursor: 'url(' + path + 'img/chalk-white.png), auto'
 | |
| 		},
 | |
| 		{
 | |
| 			color: 'rgba(96, 154, 244, 0.5)',
 | |
| 			cursor: 'url(' + path + 'img/chalk-blue.png), auto'
 | |
| 		},
 | |
| 		{
 | |
| 			color: 'rgba(237, 20, 28, 0.5)',
 | |
| 			cursor: 'url(' + path + 'img/chalk-red.png), auto'
 | |
| 		},
 | |
| 		{
 | |
| 			color: 'rgba(20, 237, 28, 0.5)',
 | |
| 			cursor: 'url(' + path + 'img/chalk-green.png), auto'
 | |
| 		},
 | |
| 		{
 | |
| 			color: 'rgba(220, 133, 41, 0.5)',
 | |
| 			cursor: 'url(' + path + 'img/chalk-orange.png), auto'
 | |
| 		},
 | |
| 		{
 | |
| 			color: 'rgba(220,0,220,0.5)',
 | |
| 			cursor: 'url(' + path + 'img/chalk-purple.png), auto'
 | |
| 		},
 | |
| 		{
 | |
| 			color: 'rgba(255,220,0,0.5)',
 | |
| 			cursor: 'url(' + path + 'img/chalk-yellow.png), auto'
 | |
| 		}
 | |
| 	];
 | |
| 	var keyBindings = {
 | |
| 		toggleNotesCanvas: {
 | |
| 			keyCode: 67,
 | |
| 			key: 'C',
 | |
| 			description: 'Toggle notes canvas'
 | |
| 		},
 | |
| 		toggleChalkboard: {
 | |
| 			keyCode: 66,
 | |
| 			key: 'B',
 | |
| 			description: 'Toggle chalkboard'
 | |
| 		},
 | |
| 		clear: {
 | |
| 			keyCode: 46,
 | |
| 			key: 'DEL',
 | |
| 			description: 'Clear drawings on slide'
 | |
| 		},
 | |
| /*
 | |
| 		reset: {
 | |
| 			keyCode: 173,
 | |
| 			key: '-',
 | |
| 			description: 'Reset drawings on slide'
 | |
| 		},
 | |
| */
 | |
| 		resetAll: {
 | |
| 			keyCode: 8,
 | |
| 			key: 'BACKSPACE',
 | |
| 			description: 'Reset all drawings'
 | |
| 		},
 | |
| 		colorNext: {
 | |
| 			keyCode: 88,
 | |
| 			key: 'X',
 | |
| 			description: 'Next color'
 | |
| 		},
 | |
| 		colorPrev: {
 | |
| 			keyCode: 89,
 | |
| 			key: 'Y',
 | |
| 			description: 'Previous color'
 | |
| 		},
 | |
| 		download: {
 | |
| 			keyCode: 68,
 | |
| 			key: 'D',
 | |
| 			description: 'Download drawings'
 | |
| 		}
 | |
| 	};
 | |
| 
 | |
| 
 | |
| 	var theme = 'chalkboard';
 | |
| 	var color = [ 0, 0 ];
 | |
| 	var toggleChalkboardButton = false;
 | |
| 	var toggleNotesButton = false;
 | |
| 	var colorButtons = true;
 | |
| 	var boardHandle = true;
 | |
| 	var transition = 800;
 | |
| 
 | |
| 	var readOnly = false;
 | |
| 	var messageType = 'broadcast';
 | |
| 
 | |
| 	var config = configure( Reveal.getConfig().chalkboard || {} );
 | |
| 	if ( config.keyBindings ) {
 | |
| 		for ( var key in config.keyBindings ) {
 | |
| 			keyBindings[ key ] = config.keyBindings[ key ];
 | |
| 		};
 | |
| 	}
 | |
| 
 | |
| 	function configure( config ) {
 | |
| 
 | |
| 		if ( config.boardmarkerWidth || config.penWidth ) boardmarkerWidth = config.boardmarkerWidth || config.penWidth;
 | |
| 		if ( config.chalkWidth ) chalkWidth = config.chalkWidth;
 | |
| 		if ( config.chalkEffect ) chalkEffect = config.chalkEffect;
 | |
| 		if ( config.rememberColor ) rememberColor = config.rememberColor;
 | |
| 		if ( config.eraser ) eraser = config.eraser;
 | |
| 		if ( config.boardmarkers ) boardmarkers = config.boardmarkers;
 | |
| 		if ( config.chalks ) chalks = config.chalks;
 | |
| 
 | |
| 		if ( config.theme ) theme = config.theme;
 | |
| 		switch ( theme ) {
 | |
| 		case 'whiteboard':
 | |
| 			background = [ 'rgba(127,127,127,.1)', path + 'img/whiteboard.png' ];
 | |
| 			draw = [ drawWithBoardmarker, drawWithBoardmarker ];
 | |
| 			pens = [ boardmarkers, boardmarkers ];
 | |
| 			grid = {
 | |
| 				color: 'rgb(127,127,255,0.1)',
 | |
| 				distance: 40,
 | |
| 				width: 2
 | |
| 			};
 | |
| 			break;
 | |
| 		case 'chalkboard':
 | |
| 		default:
 | |
| 			background = [ 'rgba(127,127,127,.1)', path + 'img/blackboard.png' ];
 | |
| 			draw = [ drawWithBoardmarker, drawWithChalk ];
 | |
| 			pens = [ boardmarkers, chalks ];
 | |
| 			grid = {
 | |
| 				color: 'rgb(50,50,10,0.5)',
 | |
| 				distance: 80,
 | |
| 				width: 2
 | |
| 			};
 | |
| 		}
 | |
| 
 | |
| 		if ( config.background ) background = config.background;
 | |
| 		if ( config.grid != undefined ) grid = config.grid;
 | |
| 
 | |
| 		if ( config.toggleChalkboardButton != undefined ) toggleChalkboardButton = config.toggleChalkboardButton;
 | |
| 		if ( config.toggleNotesButton != undefined ) toggleNotesButton = config.toggleNotesButton;
 | |
| 		if ( config.colorButtons != undefined ) colorButtons = config.colorButtons;
 | |
| 		if ( config.boardHandle != undefined ) boardHandle = config.boardHandle;
 | |
| 		if ( config.transition ) transition = config.transition;
 | |
| 
 | |
| 		if ( config.readOnly != undefined ) readOnly = config.readOnly;
 | |
| 		if ( config.messageType ) messageType = config.messageType;
 | |
| 
 | |
| 		if ( drawingCanvas && ( config.theme || config.background || config.grid ) ) {
 | |
| 			var canvas = document.getElementById( drawingCanvas[ 1 ].id );
 | |
| 			canvas.style.background = 'url("' + background[ 1 ] + '") repeat';
 | |
| 			clearCanvas( 1 );
 | |
| 			drawGrid();
 | |
| 		}
 | |
| 
 | |
| 		return config;
 | |
| 	}
 | |
| /*****************************************************************
 | |
|  ** Setup
 | |
|  ******************************************************************/
 | |
| 
 | |
| 	function whenReady( callback ) {
 | |
| 		// wait for markdown to be parsed and code to be highlighted
 | |
| 		if ( !document.querySelector( 'section[data-markdown]:not([data-markdown-parsed])' ) 
 | |
| 		     && !document.querySelector( 'code[data-line-numbers*="|"]') 	
 | |
| 		) {
 | |
| 			callback();
 | |
| 		} else {
 | |
| 			console.log( "Wait for markdown to be parsed and code to be highlighted" );
 | |
| 			setTimeout( whenReady, 500, callback )
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	function whenLoaded( callback ) {
 | |
| 		// wait for drawings to be loaded and markdown to be parsed
 | |
| 		if ( loaded !== null ) {
 | |
| 			callback();
 | |
| 		} else {
 | |
| 			console.log( "Wait for drawings to be loaded" );
 | |
| 			setTimeout( whenLoaded, 500, callback )
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if ( toggleChalkboardButton ) {
 | |
| console.warn( "toggleChalkboardButton is deprecated, use customcontrols plugin instead!" );
 | |
| //console.log("toggleChalkboardButton")
 | |
| 		var button = document.createElement( 'div' );
 | |
| 		button.className = "chalkboard-button";
 | |
| 		button.id = "toggle-chalkboard";
 | |
| 		button.style.visibility = "visible";
 | |
| 		button.style.position = "absolute";
 | |
| 		button.style.zIndex = 30;
 | |
| 		button.style.fontSize = "24px";
 | |
| 
 | |
| 		button.style.left = toggleChalkboardButton.left || "30px";
 | |
| 		button.style.bottom = toggleChalkboardButton.bottom || "30px";
 | |
| 		button.style.top = toggleChalkboardButton.top || "auto";
 | |
| 		button.style.right = toggleChalkboardButton.right || "auto";
 | |
| 
 | |
| 		button.innerHTML = '<a href="#" title="Toggle chalkboard (' + keyBindings.toggleChalkboard.key + ')" onclick="RevealChalkboard.toggleChalkboard(); return false;"><i class="fa fa-pen-square"></i></a>'
 | |
| 		document.querySelector( ".reveal" ).appendChild( button );
 | |
| 	}
 | |
| 	if ( toggleNotesButton ) {
 | |
| console.warn( "toggleNotesButton is deprecated, use customcontrols plugin instead!" );
 | |
| //console.log("toggleNotesButton")
 | |
| 		var button = document.createElement( 'div' );
 | |
| 		button.className = "chalkboard-button";
 | |
| 		button.id = "toggle-notes";
 | |
| 		button.style.position = "absolute";
 | |
| 		button.style.zIndex = 30;
 | |
| 		button.style.fontSize = "24px";
 | |
| 
 | |
| 		button.style.left = toggleNotesButton.left || "70px";
 | |
| 		button.style.bottom = toggleNotesButton.bottom || "30px";
 | |
| 		button.style.top = toggleNotesButton.top || "auto";
 | |
| 		button.style.right = toggleNotesButton.right || "auto";
 | |
| 
 | |
| 		button.innerHTML = '<a href="#" title="Toggle slide annotation (' + keyBindings.toggleNotesCanvas.key + ')" onclick="RevealChalkboard.toggleNotesCanvas(); return false;"><i class="fa fa-pen"></i></a>'
 | |
| 		document.querySelector( ".reveal" ).appendChild( button );
 | |
| 	}
 | |
| 
 | |
| 	var drawingCanvas = [ {
 | |
| 		id: 'notescanvas'
 | |
| 	}, {
 | |
| 		id: 'chalkboard'
 | |
| 	} ];
 | |
| 	setupDrawingCanvas( 0 );
 | |
| 	setupDrawingCanvas( 1 );
 | |
| 
 | |
| 	var mode = 0; // 0: notes canvas, 1: chalkboard
 | |
| 	var board = 0; // board index (only for chalkboard)
 | |
| 
 | |
| 	var mouseX = 0;
 | |
| 	var mouseY = 0;
 | |
| 	var lastX = null;
 | |
| 	var lastY = null;
 | |
| 
 | |
| 	var drawing = false;
 | |
| 	var erasing = false;
 | |
| 
 | |
| 	var slideStart = Date.now();
 | |
| 	var slideIndices = {
 | |
| 		h: 0,
 | |
| 		v: 0
 | |
| 	};
 | |
| 
 | |
| 	var timeouts = [
 | |
| 		[],
 | |
| 		[]
 | |
| 	];
 | |
| 	var touchTimeout = null;
 | |
| 	var slidechangeTimeout = null;
 | |
| 	var updateStorageTimeout = null;
 | |
| 	var playback = false;
 | |
| 
 | |
| 	function createPalette( colors, length ) {
 | |
| 		if ( length === true || length > colors.length ) {
 | |
| 			length = colors.length;
 | |
| 		}
 | |
| 		var palette = document.createElement( 'div' );
 | |
| 		palette.classList.add( 'palette' );
 | |
| 		var list = document.createElement( 'ul' );
 | |
| 		// color pickers
 | |
| 		for ( var i = 0; i < length; i++ ) {
 | |
| 			var colorButton = document.createElement( 'li' );
 | |
| 			colorButton.setAttribute( 'data-color', i );
 | |
| 			colorButton.innerHTML = '<i class="fa fa-square"></i>';
 | |
| 			colorButton.style.color = colors[ i ].color;
 | |
| 			colorButton.addEventListener( 'click', function ( e ) {
 | |
| 				var element = e.target;
 | |
| 				while ( !element.hasAttribute( 'data-color' ) ) {
 | |
| 					element = element.parentElement;
 | |
| 				}
 | |
| 				colorIndex( parseInt( element.getAttribute( 'data-color' ) ) );
 | |
| 			} );
 | |
| 			colorButton.addEventListener( 'touchstart', function ( e ) {
 | |
| 				var element = e.target;
 | |
| 				while ( !element.hasAttribute( 'data-color' ) ) {
 | |
| 					element = element.parentElement;
 | |
| 				}
 | |
| 				colorIndex( parseInt( element.getAttribute( 'data-color' ) ) );
 | |
| 			} );
 | |
| 			list.appendChild( colorButton );
 | |
| 		}
 | |
| 		palette.appendChild( list );
 | |
| 		return palette;
 | |
| 	};
 | |
| 
 | |
| 	function switchBoard( boardIdx ) {
 | |
| 		selectBoard( boardIdx, true );
 | |
| 		// broadcast
 | |
| 		var message = new CustomEvent( messageType );
 | |
| 			message.content = {
 | |
| 			sender: 'chalkboard-plugin',
 | |
| 			type: 'selectboard',
 | |
| 			timestamp: Date.now() - slideStart,
 | |
| 			mode,
 | |
| 			board
 | |
| 		};
 | |
| 		document.dispatchEvent( message );	
 | |
| 	}
 | |
| 
 | |
| 	function setupDrawingCanvas( id ) {
 | |
| 		var container = document.createElement( 'div' );
 | |
| 		container.id = drawingCanvas[ id ].id;
 | |
| 		container.classList.add( 'overlay' );
 | |
| 		container.setAttribute( 'data-prevent-swipe', 'true' );
 | |
| 		container.oncontextmenu = function () {
 | |
| 			return false;
 | |
| 		}
 | |
| 		container.style.cursor = pens[ id ][ color[ id ] ].cursor;
 | |
| 
 | |
| 		drawingCanvas[ id ].width = window.innerWidth;
 | |
| 		drawingCanvas[ id ].height = window.innerHeight;
 | |
| 		drawingCanvas[ id ].scale = 1;
 | |
| 		drawingCanvas[ id ].xOffset = 0;
 | |
| 		drawingCanvas[ id ].yOffset = 0;
 | |
| 
 | |
| 		if ( id == "0" ) {
 | |
| 			container.style.background = 'rgba(0,0,0,0)';
 | |
| 			container.style.zIndex = 24;
 | |
| 			container.style.opacity = 1;
 | |
| 			container.style.visibility = 'visible';
 | |
| 			container.style.pointerEvents = 'none';
 | |
| 
 | |
| 			var slides = document.querySelector( '.slides' );
 | |
| 			var aspectRatio = Reveal.getConfig().width / Reveal.getConfig().height;
 | |
| 			if ( drawingCanvas[ id ].width > drawingCanvas[ id ].height * aspectRatio ) {
 | |
| 				drawingCanvas[ id ].xOffset = ( drawingCanvas[ id ].width - drawingCanvas[ id ].height * aspectRatio ) / 2;
 | |
| 			} else if ( drawingCanvas[ id ].height > drawingCanvas[ id ].width / aspectRatio ) {
 | |
| 				drawingCanvas[ id ].yOffset = ( drawingCanvas[ id ].height - drawingCanvas[ id ].width / aspectRatio ) / 2;
 | |
| 			}
 | |
| 
 | |
| 			if ( colorButtons ) {
 | |
| 				var palette = createPalette( boardmarkers, colorButtons );
 | |
| 				palette.style.visibility = 'hidden'; // only show palette in drawing mode
 | |
| 				container.appendChild( palette );
 | |
| 			}
 | |
| 		} else {
 | |
| 			container.style.background = 'url("' + background[ id ] + '") repeat';
 | |
| 			container.style.zIndex = 26;
 | |
| 			container.style.opacity = 0;
 | |
| 			container.style.visibility = 'hidden';
 | |
| 
 | |
| 			if ( colorButtons ) {
 | |
| 				var palette = createPalette( chalks, colorButtons );
 | |
| 				container.appendChild( palette );
 | |
| 			}
 | |
| 			if ( boardHandle ) {
 | |
| 				var handle = document.createElement( 'div' );
 | |
| 				handle.classList.add( 'boardhandle' );
 | |
| 				handle.innerHTML = '<ul><li><a id="previousboard" href="#" title="Previous board"><i class="fas fa-chevron-up"></i></a></li><li><a id="nextboard" href="#" title="Next board"><i class="fas fa-chevron-down"></i></a></li></ul>';
 | |
| 				handle.querySelector( '#previousboard' ).addEventListener( 'click', function ( e ) {
 | |
| 					e.preventDefault();
 | |
| 					switchBoard( board - 1 );
 | |
| 				} );
 | |
| 				handle.querySelector( '#nextboard' ).addEventListener( 'click', function ( e ) {
 | |
| 					e.preventDefault();
 | |
| 					switchBoard( board + 1 );
 | |
| 				} );
 | |
| 				handle.querySelector( '#previousboard' ).addEventListener( 'touchstart', function ( e ) {
 | |
| 					e.preventDefault();
 | |
| 					switchBoard( board - 1 );
 | |
| 				} );
 | |
| 				handle.querySelector( '#nextboard' ).addEventListener( 'touchstart', function ( e ) {
 | |
| 					e.preventDefault();
 | |
| 					switchBoard( board + 1 );
 | |
| 				} );
 | |
| 
 | |
| 				container.appendChild( handle );
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 
 | |
| 		var sponge = document.createElement( 'img' );
 | |
| 		sponge.src = eraser.src;
 | |
| 		sponge.id = 'sponge';
 | |
| 		sponge.style.visibility = 'hidden';
 | |
| 		sponge.style.position = 'absolute';
 | |
| 		container.appendChild( sponge );
 | |
| 		drawingCanvas[ id ].sponge = sponge;
 | |
| 
 | |
| 		var canvas = document.createElement( 'canvas' );
 | |
| 		canvas.width = drawingCanvas[ id ].width;
 | |
| 		canvas.height = drawingCanvas[ id ].height;
 | |
| 		canvas.setAttribute( 'data-chalkboard', id );
 | |
| 		canvas.style.cursor = pens[ id ][ color[ id ] ].cursor;
 | |
| 		container.appendChild( canvas );
 | |
| 		drawingCanvas[ id ].canvas = canvas;
 | |
| 
 | |
| 		drawingCanvas[ id ].context = canvas.getContext( '2d' );
 | |
| 
 | |
| 		setupCanvasEvents( container );
 | |
| 
 | |
| 		document.querySelector( '.reveal' ).appendChild( container );
 | |
| 		drawingCanvas[ id ].container = container;
 | |
| 	}
 | |
| 
 | |
| 
 | |
| /*****************************************************************
 | |
|  ** Storage
 | |
|  ******************************************************************/
 | |
| 
 | |
| 	var storage = [ {
 | |
| 			width: Reveal.getConfig().width,
 | |
| 			height: Reveal.getConfig().height,
 | |
| 			data: []
 | |
| 		},
 | |
| 		{
 | |
| 			width: Reveal.getConfig().width,
 | |
| 			height: Reveal.getConfig().height,
 | |
| 			data: []
 | |
| 		}
 | |
| 	];
 | |
| 
 | |
| 	var loaded = null;
 | |
| 
 | |
| 	if ( config.storage ) {
 | |
| 		// Get chalkboard drawings from session storage
 | |
| 		loaded = initStorage( sessionStorage.getItem( config.storage ) );
 | |
| 	}
 | |
| 
 | |
| 	if ( !loaded && config.src != null ) {
 | |
| 		// Get chalkboard drawings from the given file
 | |
| 		loadData( config.src );
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Initialize storage.
 | |
| 	 */
 | |
| 	function initStorage( json ) {
 | |
| 		var success = false;
 | |
| 		try {
 | |
| 			var data = JSON.parse( json );
 | |
| 			for ( var id = 0; id < data.length; id++ ) {
 | |
| 				if ( drawingCanvas[ id ].width != data[ id ].width || drawingCanvas[ id ].height != data[ id ].height ) {
 | |
| 					drawingCanvas[ id ].scale = Math.min( drawingCanvas[ id ].width / data[ id ].width, drawingCanvas[ id ].height / data[ id ].height );
 | |
| 					drawingCanvas[ id ].xOffset = ( drawingCanvas[ id ].width - data[ id ].width * drawingCanvas[ id ].scale ) / 2;
 | |
| 					drawingCanvas[ id ].yOffset = ( drawingCanvas[ id ].height - data[ id ].height * drawingCanvas[ id ].scale ) / 2;
 | |
| 				}
 | |
| 				if ( config.readOnly ) {
 | |
| 					drawingCanvas[ id ].container.style.cursor = 'default';
 | |
| 					drawingCanvas[ id ].canvas.style.cursor = 'default';
 | |
| 				}
 | |
| 			}
 | |
| 			success = true;
 | |
| 			storage = data;
 | |
| 		} catch ( err ) {
 | |
| 			console.warn( "Cannot initialise storage!" );
 | |
| 		}
 | |
| 		return success;
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * Load data.
 | |
| 	 */
 | |
| 	function loadData( filename ) {
 | |
| 		var xhr = new XMLHttpRequest();
 | |
| 		xhr.onload = function () {
 | |
| 			if ( xhr.readyState === 4 && xhr.status != 404 ) {
 | |
| 				loaded = initStorage( xhr.responseText );
 | |
| 				updateStorage();
 | |
| 				console.log( "Drawings loaded from file" );
 | |
| 			} else {
 | |
| 				config.readOnly = undefined;
 | |
| 				readOnly = undefined;
 | |
| 				console.warn( 'Failed to get file ' + filename + '. ReadyState: ' + xhr.readyState + ', Status: ' + xhr.status );
 | |
| 				loaded = false;
 | |
| 			}
 | |
| 		};
 | |
| 
 | |
| 		xhr.open( 'GET', filename, true );
 | |
| 		try {
 | |
| 			xhr.send();
 | |
| 		} catch ( error ) {
 | |
| 			config.readOnly = undefined;
 | |
| 			readOnly = undefined;
 | |
| 			console.warn( 'Failed to get file ' + filename + '. Make sure that the presentation and the file are served by a HTTP server and the file can be found there. ' + error );
 | |
| 			loaded = false;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	function storageChanged( now ) {
 | |
| 		if ( !now ) {
 | |
| 			// create or update timer
 | |
| 			if ( updateStorageTimeout ) {
 | |
| 				clearTimeout( updateStorageTimeout );
 | |
| 			}
 | |
| 			updateStorageTimeout = setTimeout( storageChanged, 1000, true);
 | |
| 		}
 | |
| 		else {
 | |
| // console.log("Update storage", updateStorageTimeout,  Date.now());
 | |
| 			updateStorage();
 | |
| 			updateStorageTimeout = null;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	function updateStorage() {
 | |
| 		var json = JSON.stringify( storage )
 | |
| 		if ( config.storage ) {
 | |
| 			sessionStorage.setItem( config.storage, json )
 | |
| 		}
 | |
| 		return json;
 | |
| 	}
 | |
| 
 | |
| 	function recordEvent( event ) {
 | |
| //console.log(event);
 | |
| 		event.time = Date.now() - slideStart;
 | |
| 		if ( mode == 1 ) event.board = board;
 | |
| 		var slideData = getSlideData();
 | |
| 		var i = slideData.events.length;
 | |
| 		while ( i > 0 && event.time < slideData.events[ i - 1 ].time ) {
 | |
| 			i--;
 | |
| 		}
 | |
| 		slideData.events.splice( i, 0, event );
 | |
| 		slideData.duration = Math.max( slideData.duration, Date.now() - slideStart ) + 1;
 | |
| 
 | |
| 		storageChanged();
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Get data as json string.
 | |
| 	 */
 | |
| 	function getData() {
 | |
| 		// cleanup slide data without events
 | |
| 		for ( var id = 0; id < 2; id++ ) {
 | |
| 			for ( var i = storage[ id ].data.length - 1; i >= 0; i-- ) {
 | |
| 				if ( storage[ id ].data[ i ].events.length == 0 ) {
 | |
| 					storage[ id ].data.splice( i, 1 );
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		return updateStorage();
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Download data.
 | |
| 	 */
 | |
| 	function downloadData() {
 | |
| 		var a = document.createElement( 'a' );
 | |
| 		document.body.appendChild( a );
 | |
| 		try {
 | |
| 			a.download = 'chalkboard.json';
 | |
| 			var blob = new Blob( [ getData() ], {
 | |
| 				type: 'application/json'
 | |
| 			} );
 | |
| 			a.href = window.URL.createObjectURL( blob );
 | |
| 		} catch ( error ) {
 | |
| 			a.innerHTML += ' (' + error + ')';
 | |
| 		}
 | |
| 		a.click();
 | |
| 		document.body.removeChild( a );
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Returns data object for the slide with the given indices.
 | |
| 	 */
 | |
| 	function getSlideData( indices, id ) {
 | |
| 		if ( id == undefined ) id = mode;
 | |
| 		if ( !indices ) indices = slideIndices;
 | |
| 		var data;
 | |
| 		for ( var i = 0; i < storage[ id ].data.length; i++ ) {
 | |
| 			if ( storage[ id ].data[ i ].slide.h === indices.h && storage[ id ].data[ i ].slide.v === indices.v && storage[ id ].data[ i ].slide.f === indices.f ) {
 | |
| 				data = storage[ id ].data[ i ];
 | |
| 				return data;
 | |
| 			}
 | |
| 		}
 | |
| 		var page = Number( Reveal.getCurrentSlide().getAttribute('data-pdf-page-number') ); 
 | |
| //console.log( indices, Reveal.getCurrentSlide() );
 | |
| 		storage[ id ].data.push( {
 | |
| 			slide: indices,
 | |
| 			page,
 | |
| 			events: [],
 | |
| 			duration: 0
 | |
| 		} );
 | |
| 		data = storage[ id ].data[ storage[ id ].data.length - 1 ];
 | |
| 		return data;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Returns maximum duration of slide playback for both modes
 | |
| 	 */
 | |
| 	function getSlideDuration( indices ) {
 | |
| 		if ( !indices ) indices = slideIndices;
 | |
| 		var duration = 0;
 | |
| 		for ( var id = 0; id < 2; id++ ) {
 | |
| 			for ( var i = 0; i < storage[ id ].data.length; i++ ) {
 | |
| 				if ( storage[ id ].data[ i ].slide.h === indices.h && storage[ id ].data[ i ].slide.v === indices.v && storage[ id ].data[ i ].slide.f === indices.f ) {
 | |
| 					duration = Math.max( duration, storage[ id ].data[ i ].duration );
 | |
| 					break;
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| //console.log( duration );
 | |
| 		return duration;
 | |
| 	}
 | |
| 
 | |
| /*****************************************************************
 | |
|  ** Print
 | |
|  ******************************************************************/
 | |
| 	var printMode = ( /print-pdf/gi ).test( window.location.search );
 | |
| //console.log("createPrintout" + printMode)
 | |
| 
 | |
| 	function addPageNumbers() {
 | |
| 		// determine page number for printouts with fragments serialised
 | |
| 		var slides = Reveal.getSlides();
 | |
| 		var page = 0;
 | |
| 		for ( var i=0; i < slides.length; i++) {
 | |
| 			slides[i].setAttribute('data-pdf-page-number',page.toString());
 | |
| 			// add number of fragments without fragment indices
 | |
| 			var count = slides[i].querySelectorAll('.fragment:not([data-fragment-index])').length;
 | |
| 			var fragments = slides[i].querySelectorAll('.fragment[data-fragment-index]');
 | |
| 			for ( var j=0; j < fragments.length; j++) {
 | |
| 				// increasenumber of fragments by highest fragment index (which start at 0)
 | |
| 				if ( Number(fragments[j].getAttribute('data-fragment-index')) + 1 > count ) {
 | |
| 					count = Number(fragments[j].getAttribute('data-fragment-index')) + 1;
 | |
| 				}
 | |
| 			}
 | |
| //console.log(count,fragments.length,( slides[i].querySelector('h1,h2,h3,h4')||{}).innerHTML, page); 
 | |
| 			page += count + 1;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	function createPrintout() {
 | |
| 		//console.warn(Reveal.getTotalSlides(),Reveal.getSlidesElement());
 | |
| 		if ( storage[ 1 ].data.length == 0 ) return;
 | |
| 		console.log( 'Create printout(s) for ' + storage[ 1 ].data.length + " slides" );
 | |
| 		drawingCanvas[ 0 ].container.style.opacity = 0; // do not print notes canvas
 | |
| 		drawingCanvas[ 0 ].container.style.visibility = 'hidden';
 | |
| 
 | |
| 		var patImg = new Image();
 | |
| 		patImg.onload = function () {
 | |
| 			var slides = Reveal.getSlides();
 | |
| //console.log(slides);
 | |
| 			for ( var i = storage[ 1 ].data.length - 1; i >= 0; i-- ) {
 | |
| 				console.log( 'Create printout for slide ' + storage[ 1 ].data[ i ].slide.h + '.' + storage[ 1 ].data[ i ].slide.v );
 | |
| 				var slideData = getSlideData( storage[ 1 ].data[ i ].slide, 1 );
 | |
| 				var drawings = createDrawings( slideData, patImg );
 | |
| //console.log("Page:", storage[ 1 ].data[ i ].page );
 | |
| //console.log("Slide:", slides[storage[ 1 ].data[ i ].page] );
 | |
| 				addDrawings( slides[storage[ 1 ].data[ i ].page], drawings );
 | |
| 
 | |
| 			}
 | |
| //			Reveal.sync();
 | |
| 		};
 | |
| 		patImg.src = background[ 1 ];
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	function cloneCanvas( oldCanvas ) {
 | |
| 		//create a new canvas
 | |
| 		var newCanvas = document.createElement( 'canvas' );
 | |
| 		var context = newCanvas.getContext( '2d' );
 | |
| 		//set dimensions
 | |
| 		newCanvas.width = oldCanvas.width;
 | |
| 		newCanvas.height = oldCanvas.height;
 | |
| 		//apply the old canvas to the new one
 | |
| 		context.drawImage( oldCanvas, 0, 0 );
 | |
| 		//return the new canvas
 | |
| 		return newCanvas;
 | |
| 	}
 | |
| 
 | |
| 	function getCanvas( template, container, board ) {
 | |
| 		var idx = container.findIndex( element => element.board === board );
 | |
| 		if ( idx === -1 ) {
 | |
| 			var canvas = cloneCanvas( template );
 | |
| 			if ( !container.length ) {
 | |
| 				idx = 0;
 | |
| 				container.push( {
 | |
| 					board,
 | |
| 					canvas
 | |
| 				} );
 | |
| 			} else if ( board < container[ 0 ].board ) {
 | |
| 				idx = 0;
 | |
| 				container.unshift( {
 | |
| 					board,
 | |
| 					canvas
 | |
| 				} );
 | |
| 			} else if ( board > container[ container.length - 1 ].board ) {
 | |
| 				idx = container.length;
 | |
| 				container.push( {
 | |
| 					board,
 | |
| 					canvas
 | |
| 				} );
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		return container[ idx ].canvas;
 | |
| 	}
 | |
| 
 | |
| 	function createDrawings( slideData, patImg ) {
 | |
| 		var width = Reveal.getConfig().width;
 | |
| 		var height = Reveal.getConfig().height;
 | |
| 		var scale = 1;
 | |
| 		var xOffset = 0;
 | |
| 		var yOffset = 0;
 | |
| 		if ( width != storage[ 1 ].width || height != storage[ 1 ].height ) {
 | |
| 			scale = Math.min( width / storage[ 1 ].width, height / storage[ 1 ].height );
 | |
| 			xOffset = ( width - storage[ 1 ].width * scale ) / 2;
 | |
| 			yOffset = ( height - storage[ 1 ].height * scale ) / 2;
 | |
| 		}
 | |
| 		mode = 1;
 | |
| 		board = 0;
 | |
| //		console.log( 'Create printout(s) for slide ', slideData );
 | |
| 
 | |
| 		var drawings = [];
 | |
| 		var template = document.createElement( 'canvas' );
 | |
| 		template.width = width;
 | |
| 		template.height = height;
 | |
| 
 | |
| 		var imgCtx = template.getContext( '2d' );
 | |
| 		imgCtx.fillStyle = imgCtx.createPattern( patImg, 'repeat' );
 | |
| 		imgCtx.rect( 0, 0, width, height );
 | |
| 		imgCtx.fill();
 | |
| 
 | |
| 		for ( var j = 0; j < slideData.events.length; j++ ) {
 | |
| 			switch ( slideData.events[ j ].type ) {
 | |
| 			case 'draw':
 | |
| 				draw[ 1 ]( getCanvas( template, drawings, board ).getContext( '2d' ),
 | |
| 					xOffset + slideData.events[ j ].x1 * scale,
 | |
| 					yOffset + slideData.events[ j ].y1 * scale,
 | |
| 					xOffset + slideData.events[ j ].x2 * scale,
 | |
| 					yOffset + slideData.events[ j ].y2 * scale,
 | |
| 					yOffset + slideData.events[ j ].color
 | |
| 				);
 | |
| 				break;
 | |
| 			case 'erase':
 | |
| 				eraseWithSponge( getCanvas( template, drawings, board ).getContext( '2d' ),
 | |
| 					xOffset + slideData.events[ j ].x * scale,
 | |
| 					yOffset + slideData.events[ j ].y * scale
 | |
| 				);
 | |
| 				break;
 | |
| 			case 'selectboard':
 | |
| 				selectBoard( slideData.events[ j ].board );
 | |
| 				break;
 | |
| 			case 'clear':
 | |
| 				getCanvas( template, drawings, board ).getContext( '2d' ).clearRect( 0, 0, width, height );
 | |
| 				getCanvas( template, drawings, board ).getContext( '2d' ).fill();
 | |
| 				break;
 | |
| 			default:
 | |
| 				break;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		drawings = drawings.sort( ( a, b ) => a.board > b.board && 1 || -1 );
 | |
| 
 | |
| 		mode = 0;
 | |
| 
 | |
| 		return drawings;
 | |
| 	}
 | |
| 
 | |
| 	function addDrawings( slide, drawings ) {
 | |
| 		var parent = slide.parentElement.parentElement;
 | |
| 		var nextSlide = slide.parentElement.nextElementSibling;
 | |
| 
 | |
| 		for ( var i = 0; i < drawings.length; i++ ) {
 | |
| 			var newPDFPage = document.createElement( 'div' );
 | |
| 			newPDFPage.classList.add( 'pdf-page' );
 | |
| 			newPDFPage.style.height = Reveal.getConfig().height;
 | |
| 			newPDFPage.append( drawings[ i ].canvas );
 | |
| //console.log("Add drawing", newPDFPage);
 | |
| 			if ( nextSlide != null ) {
 | |
| 				parent.insertBefore( newPDFPage, nextSlide );
 | |
| 			} else {
 | |
| 				parent.append( newPDFPage );
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/*****************************************************************
 | |
| 	 ** Drawings
 | |
| 	 ******************************************************************/
 | |
| 
 | |
| 	function drawWithBoardmarker( context, fromX, fromY, toX, toY, colorIdx ) {
 | |
| 		if ( colorIdx == undefined ) colorIdx = color[ mode ];
 | |
| 		context.lineWidth = boardmarkerWidth;
 | |
| 		context.lineCap = 'round';
 | |
| 		context.strokeStyle = boardmarkers[ colorIdx ].color;
 | |
| 		context.beginPath();
 | |
| 		context.moveTo( fromX, fromY );
 | |
| 		context.lineTo( toX, toY );
 | |
| 		context.stroke();
 | |
| 	}
 | |
| 
 | |
| 	function drawWithChalk( context, fromX, fromY, toX, toY, colorIdx ) {
 | |
| 		if ( colorIdx == undefined ) colorIdx = color[ mode ];
 | |
| 		var brushDiameter = chalkWidth;
 | |
| 		context.lineWidth = brushDiameter;
 | |
| 		context.lineCap = 'round';
 | |
| 		context.fillStyle = chalks[ colorIdx ].color; // 'rgba(255,255,255,0.5)';
 | |
| 		context.strokeStyle = chalks[ colorIdx ].color;
 | |
| 		/*var opacity = Math.min(0.8, Math.max(0,color[1].replace(/^.*,(.+)\)/,'$1') - 0.1)) + Math.random()*0.2;*/
 | |
| 		var opacity = 1.0;
 | |
| 		context.strokeStyle = context.strokeStyle.replace( /[\d\.]+\)$/g, opacity + ')' );
 | |
| 		context.beginPath();
 | |
| 		context.moveTo( fromX, fromY );
 | |
| 		context.lineTo( toX, toY );
 | |
| 		context.stroke();
 | |
| 		// Chalk Effect
 | |
| 		var length = Math.round( Math.sqrt( Math.pow( toX - fromX, 2 ) + Math.pow( toY - fromY, 2 ) ) / ( 5 / brushDiameter ) );
 | |
| 		var xUnit = ( toX - fromX ) / length;
 | |
| 		var yUnit = ( toY - fromY ) / length;
 | |
| 		for ( var i = 0; i < length; i++ ) {
 | |
| 			if ( chalkEffect > ( Math.random() * 0.9 ) ) {
 | |
| 				var xCurrent = fromX + ( i * xUnit );
 | |
| 				var yCurrent = fromY + ( i * yUnit );
 | |
| 				var xRandom = xCurrent + ( Math.random() - 0.5 ) * brushDiameter * 1.2;
 | |
| 				var yRandom = yCurrent + ( Math.random() - 0.5 ) * brushDiameter * 1.2;
 | |
| 				context.clearRect( xRandom, yRandom, Math.random() * 2 + 2, Math.random() + 1 );
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
|  
 | |
| 	function eraseWithSponge( context, x, y ) {
 | |
| 		context.save();
 | |
| 		context.beginPath();
 | |
| 		context.arc( x, y, eraser.radius, 0, 2 * Math.PI, false );
 | |
| 		context.clip();
 | |
| 		context.clearRect( x - eraser.radius - 1, y - eraser.radius - 1, eraser.radius * 2 + 2, eraser.radius * 2 + 2 );
 | |
| 		context.restore();
 | |
| 		if ( mode == 1 && grid ) {
 | |
| 			redrawGrid( x, y, eraser.radius );
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * Show an overlay for the chalkboard.
 | |
| 	 */
 | |
| 	function showChalkboard() {
 | |
| //console.log("showChalkboard");
 | |
| 		clearTimeout( touchTimeout );
 | |
| 		touchTimeout = null;
 | |
| 		drawingCanvas[ 0 ].sponge.style.visibility = 'hidden'; // make sure that the sponge from touch events is hidden
 | |
| 		drawingCanvas[ 1 ].sponge.style.visibility = 'hidden'; // make sure that the sponge from touch events is hidden
 | |
| 		drawingCanvas[ 1 ].container.style.opacity = 1;
 | |
| 		drawingCanvas[ 1 ].container.style.visibility = 'visible';
 | |
| 		mode = 1;
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * Closes open chalkboard.
 | |
| 	 */
 | |
| 	function closeChalkboard() {
 | |
| 		clearTimeout( touchTimeout );
 | |
| 		touchTimeout = null;
 | |
| 		drawingCanvas[ 0 ].sponge.style.visibility = 'hidden'; // make sure that the sponge from touch events is hidden
 | |
| 		drawingCanvas[ 1 ].sponge.style.visibility = 'hidden'; // make sure that the sponge from touch events is hidden
 | |
| 		drawingCanvas[ 1 ].container.style.opacity = 0;
 | |
| 		drawingCanvas[ 1 ].container.style.visibility = 'hidden';
 | |
| 		lastX = null;
 | |
| 		lastY = null;
 | |
| 		mode = 0;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Clear current canvas.
 | |
| 	 */
 | |
| 	function clearCanvas( id ) {
 | |
| 		if ( id == 0 ) clearTimeout( slidechangeTimeout );
 | |
| 		drawingCanvas[ id ].context.clearRect( 0, 0, drawingCanvas[ id ].width, drawingCanvas[ id ].height );
 | |
| 		if ( id == 1 && grid ) drawGrid();
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Draw grid on background
 | |
| 	 */
 | |
| 	function drawGrid() {
 | |
| 		var context = drawingCanvas[ 1 ].context;
 | |
| 
 | |
| 		drawingCanvas[ 1 ].scale = Math.min( drawingCanvas[ 1 ].width / storage[ 1 ].width, drawingCanvas[ 1 ].height / storage[ 1 ].height );
 | |
| 		drawingCanvas[ 1 ].xOffset = ( drawingCanvas[ 1 ].width - storage[ 1 ].width * drawingCanvas[ 1 ].scale ) / 2;
 | |
| 		drawingCanvas[ 1 ].yOffset = ( drawingCanvas[ 1 ].height - storage[ 1 ].height * drawingCanvas[ 1 ].scale ) / 2;
 | |
| 
 | |
| 		var scale = drawingCanvas[ 1 ].scale;
 | |
| 		var xOffset = drawingCanvas[ 1 ].xOffset;
 | |
| 		var yOffset = drawingCanvas[ 1 ].yOffset;
 | |
| 
 | |
| 		var distance = grid.distance * scale;
 | |
| 
 | |
| 		var fromX = drawingCanvas[ 1 ].width / 2 - distance / 2 - Math.floor( ( drawingCanvas[ 1 ].width - distance ) / 2 / distance ) * distance;
 | |
| 		for ( var x = fromX; x < drawingCanvas[ 1 ].width; x += distance ) {
 | |
| 			context.beginPath();
 | |
| 			context.lineWidth = grid.width * scale;
 | |
| 			context.lineCap = 'round';
 | |
| 			context.fillStyle = grid.color;
 | |
| 			context.strokeStyle = grid.color;
 | |
| 			context.moveTo( x, 0 );
 | |
| 			context.lineTo( x, drawingCanvas[ 1 ].height );
 | |
| 			context.stroke();
 | |
| 		}
 | |
| 		var fromY = drawingCanvas[ 1 ].height / 2 - distance / 2 - Math.floor( ( drawingCanvas[ 1 ].height - distance ) / 2 / distance ) * distance;
 | |
| 
 | |
| 		for ( var y = fromY; y < drawingCanvas[ 1 ].height; y += distance ) {
 | |
| 			context.beginPath();
 | |
| 			context.lineWidth = grid.width * scale;
 | |
| 			context.lineCap = 'round';
 | |
| 			context.fillStyle = grid.color;
 | |
| 			context.strokeStyle = grid.color;
 | |
| 			context.moveTo( 0, y );
 | |
| 			context.lineTo( drawingCanvas[ 1 ].width, y );
 | |
| 			context.stroke();
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	function redrawGrid( centerX, centerY, diameter ) {
 | |
| 		var context = drawingCanvas[ 1 ].context;
 | |
| 
 | |
| 		drawingCanvas[ 1 ].scale = Math.min( drawingCanvas[ 1 ].width / storage[ 1 ].width, drawingCanvas[ 1 ].height / storage[ 1 ].height );
 | |
| 		drawingCanvas[ 1 ].xOffset = ( drawingCanvas[ 1 ].width - storage[ 1 ].width * drawingCanvas[ 1 ].scale ) / 2;
 | |
| 		drawingCanvas[ 1 ].yOffset = ( drawingCanvas[ 1 ].height - storage[ 1 ].height * drawingCanvas[ 1 ].scale ) / 2;
 | |
| 
 | |
| 		var scale = drawingCanvas[ 1 ].scale;
 | |
| 		var xOffset = drawingCanvas[ 1 ].xOffset;
 | |
| 		var yOffset = drawingCanvas[ 1 ].yOffset;
 | |
| 
 | |
| 		var distance = grid.distance * scale;
 | |
| 
 | |
| 		var fromX = drawingCanvas[ 1 ].width / 2 - distance / 2 - Math.floor( ( drawingCanvas[ 1 ].width - distance ) / 2 / distance ) * distance;
 | |
| 
 | |
| 		for ( var x = fromX + distance * Math.ceil( ( centerX - diameter - fromX ) / distance ); x <= fromX + distance * Math.floor( ( centerX + diameter - fromX ) / distance ); x += distance ) {
 | |
| 			context.beginPath();
 | |
| 			context.lineWidth = grid.width * scale;
 | |
| 			context.lineCap = 'round';
 | |
| 			context.fillStyle = grid.color;
 | |
| 			context.strokeStyle = grid.color;
 | |
| 			context.moveTo( x, centerY - Math.sqrt( diameter * diameter - ( centerX - x ) * ( centerX - x ) ) );
 | |
| 			context.lineTo( x, centerY + Math.sqrt( diameter * diameter - ( centerX - x ) * ( centerX - x ) ) );
 | |
| 			context.stroke();
 | |
| 		}
 | |
| 		var fromY = drawingCanvas[ 1 ].height / 2 - distance / 2 - Math.floor( ( drawingCanvas[ 1 ].height - distance ) / 2 / distance ) * distance;
 | |
| 		for ( var y = fromY + distance * Math.ceil( ( centerY - diameter - fromY ) / distance ); y <= fromY + distance * Math.floor( ( centerY + diameter - fromY ) / distance ); y += distance ) {
 | |
| 			context.beginPath();
 | |
| 			context.lineWidth = grid.width * scale;
 | |
| 			context.lineCap = 'round';
 | |
| 			context.fillStyle = grid.color;
 | |
| 			context.strokeStyle = grid.color;
 | |
| 			context.moveTo( centerX - Math.sqrt( diameter * diameter - ( centerY - y ) * ( centerY - y ) ), y );
 | |
| 			context.lineTo( centerX + Math.sqrt( diameter * diameter - ( centerY - y ) * ( centerY - y ) ), y );
 | |
| 			context.stroke();
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Set the  color
 | |
| 	 */
 | |
| 	function setColor( index, record ) {
 | |
| 		// protect against out of bounds (this could happen when
 | |
| 		// replaying events recorded with different color settings).
 | |
| 		if ( index >= pens[ mode ].length ) index = 0;
 | |
| 		color[ mode ] = index;
 | |
| 		drawingCanvas[ mode ].canvas.style.cursor = pens[ mode ][ color[ mode ] ].cursor;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Set the  board
 | |
| 	 */
 | |
| 	function selectBoard( boardIdx, record ) {
 | |
| //console.log("Set board",boardIdx);
 | |
| 		if ( board == boardIdx ) return;
 | |
| 
 | |
| 		board = boardIdx;
 | |
| 		redrawChalkboard( boardIdx );
 | |
| 		if ( record ) {
 | |
| 			recordEvent( { type: 'selectboard' } );
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	function redrawChalkboard( boardIdx ) {
 | |
| 		clearCanvas( 1 );
 | |
| 		var slideData = getSlideData( slideIndices, 1 );
 | |
| 		var index = 0;
 | |
| 		var play = ( boardIdx == 0 );
 | |
| 		while ( index < slideData.events.length && slideData.events[ index ].time < Date.now() - slideStart ) {
 | |
| 			if ( boardIdx == slideData.events[ index ].board ) {
 | |
| 				playEvent( 1, slideData.events[ index ], Date.now() - slideStart );
 | |
| 			}
 | |
| 
 | |
| 			index++;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 
 | |
| 	/**
 | |
| 	 * Forward cycle color
 | |
| 	 */
 | |
| 	function cycleColorNext() {
 | |
| 		color[ mode ] = ( color[ mode ] + 1 ) % pens[ mode ].length;
 | |
| 		return color[ mode ];
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Backward cycle color
 | |
| 	 */
 | |
| 	function cycleColorPrev() {
 | |
| 		color[ mode ] = ( color[ mode ] + ( pens[ mode ].length - 1 ) ) % pens[ mode ].length;
 | |
| 		return color[ mode ];
 | |
| 	}
 | |
| 
 | |
| /*****************************************************************
 | |
|  ** Broadcast
 | |
|  ******************************************************************/
 | |
| 
 | |
| 	var eventQueue = [];
 | |
| 
 | |
| 	document.addEventListener( 'received', function ( message ) {
 | |
| 		if ( message.content && message.content.sender == 'chalkboard-plugin' ) {
 | |
| 			// add message to queue
 | |
| 			eventQueue.push( message );
 | |
| 			console.log( JSON.stringify( message ) );
 | |
| 		}
 | |
| 		if ( eventQueue.length == 1 ) processQueue();
 | |
| 	} );
 | |
| 
 | |
| 	function processQueue() {
 | |
| 		// take first message from queue
 | |
| 		var message = eventQueue.shift();
 | |
| 
 | |
| 		// synchronize time with seminar host
 | |
| 		slideStart = Date.now() - message.content.timestamp;
 | |
| 		// set status
 | |
| 		if ( mode < message.content.mode ) {
 | |
| 			// open chalkboard
 | |
| 			showChalkboard();
 | |
| 		} else if ( mode > message.content.mode ) {
 | |
| 			// close chalkboard
 | |
| 			closeChalkboard();
 | |
| 		}
 | |
| 		if ( board != message.content.board ) {
 | |
| 			board = message.content.board;
 | |
| 			redrawChalkboard( board );
 | |
| 		};
 | |
| 
 | |
| 		switch ( message.content.type ) {
 | |
| 		case 'showChalkboard':
 | |
| 			showChalkboard();
 | |
| 			break;
 | |
| 		case 'closeChalkboard':
 | |
| 			closeChalkboard();
 | |
| 			break;
 | |
| 		case 'erase':
 | |
| 			erasePoint( message.content.x, message.content.y );
 | |
| 			break;
 | |
| 		case 'draw':
 | |
| 			drawSegment( message.content.fromX, message.content.fromY, message.content.toX, message.content.toY, message.content.color );
 | |
| 			break;
 | |
| 		case 'clear':
 | |
| 			clearSlide();
 | |
| 			break;
 | |
| 		case 'selectboard':
 | |
| 			selectBoard( message.content.board, true );
 | |
| 			break;
 | |
| 		case 'resetSlide':
 | |
| 			resetSlideDrawings();
 | |
| 			break;
 | |
| 		case 'init':
 | |
| 			storage = message.content.storage;
 | |
| 			for ( var id = 0; id < 2; id++ ) {
 | |
| 				drawingCanvas[ id ].scale = Math.min( drawingCanvas[ id ].width / storage[ id ].width, drawingCanvas[ id ].height / storage[ id ].height );
 | |
| 				drawingCanvas[ id ].xOffset = ( drawingCanvas[ id ].width - storage[ id ].width * drawingCanvas[ id ].scale ) / 2;
 | |
| 				drawingCanvas[ id ].yOffset = ( drawingCanvas[ id ].height - storage[ id ].height * drawingCanvas[ id ].scale ) / 2;
 | |
| 			}
 | |
| 			clearCanvas( 0 );
 | |
| 			clearCanvas( 1 );
 | |
| 			if ( !playback ) {
 | |
| 				slidechangeTimeout = setTimeout( startPlayback, transition, getSlideDuration(), 0 );
 | |
| 			}
 | |
| 			if ( mode == 1 && message.content.mode == 0 ) {
 | |
| 				setTimeout( closeChalkboard, transition + 50 );
 | |
| 			}
 | |
| 			if ( mode == 0 && message.content.mode == 1 ) {
 | |
| 				setTimeout( showChalkboard, transition + 50 );
 | |
| 			}
 | |
| 			mode = message.content.mode;
 | |
| 			board = message.content.board;
 | |
| 			break;
 | |
| 		default:
 | |
| 			break;
 | |
| 		}
 | |
| 
 | |
| 		// continue with next message if queued
 | |
| 		if ( eventQueue.length > 0 ) {
 | |
| 			processQueue();
 | |
| 		} else {
 | |
| 			storageChanged();
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	document.addEventListener( 'welcome', function ( user ) {
 | |
| 		// broadcast storage
 | |
| 		var message = new CustomEvent( messageType );
 | |
| 		message.content = {
 | |
| 			sender: 'chalkboard-plugin',
 | |
| 			recipient: user.id,
 | |
| 			type: 'init',
 | |
| 			timestamp: Date.now() - slideStart,
 | |
| 			storage: storage,
 | |
| 			mode,
 | |
| 			board
 | |
| 		};
 | |
| 		document.dispatchEvent( message );
 | |
| 	} );
 | |
| 
 | |
| 	/*****************************************************************
 | |
| 	 ** Playback
 | |
| 	 ******************************************************************/
 | |
| 
 | |
| 	document.addEventListener( 'seekplayback', function ( event ) {
 | |
| //console.log('event seekplayback ' + event.timestamp);
 | |
| 		stopPlayback();
 | |
| 		if ( !playback || event.timestamp == 0 ) {
 | |
| 			// in other cases startplayback fires after seeked
 | |
| 			startPlayback( event.timestamp );
 | |
| 		}
 | |
| 		//console.log('seeked');
 | |
| 	} );
 | |
| 
 | |
| 
 | |
| 	document.addEventListener( 'startplayback', function ( event ) {
 | |
| //console.log('event startplayback ' + event.timestamp);
 | |
| 		stopPlayback();
 | |
| 		playback = true;
 | |
| 		startPlayback( event.timestamp );
 | |
| 	} );
 | |
| 
 | |
| 	document.addEventListener( 'stopplayback', function ( event ) {
 | |
| //console.log('event stopplayback ' + (Date.now() - slideStart) );
 | |
| 		playback = false;
 | |
| 		stopPlayback();
 | |
| 	} );
 | |
| 
 | |
| 	document.addEventListener( 'startrecording', function ( event ) {
 | |
| //console.log('event startrecording ' + event.timestamp);
 | |
| 		startRecording();
 | |
| 	} );
 | |
| 
 | |
| 
 | |
| 	function startRecording() {
 | |
| 		resetSlide( true );
 | |
| 		slideStart = Date.now();
 | |
| 	}
 | |
| 
 | |
| 	function startPlayback( timestamp, finalMode ) {
 | |
| //console.log("playback " + timestamp );
 | |
| 		slideStart = Date.now() - timestamp;
 | |
| 		closeChalkboard();
 | |
| 		mode = 0;
 | |
| 		board = 0;
 | |
| 		for ( var id = 0; id < 2; id++ ) {
 | |
| 			clearCanvas( id );
 | |
| 			var slideData = getSlideData( slideIndices, id );
 | |
| //console.log( timestamp +" / " + JSON.stringify(slideData));
 | |
| 			var index = 0;
 | |
| 			while ( index < slideData.events.length && slideData.events[ index ].time < ( Date.now() - slideStart ) ) {
 | |
| 				playEvent( id, slideData.events[ index ], timestamp );
 | |
| 				index++;
 | |
| 			}
 | |
| 
 | |
| 			while ( playback && index < slideData.events.length ) {
 | |
| 				timeouts[ id ].push( setTimeout( playEvent, slideData.events[ index ].time - ( Date.now() - slideStart ), id, slideData.events[ index ], timestamp ) );
 | |
| 				index++;
 | |
| 			}
 | |
| 		}
 | |
| //console.log("Mode: " + finalMode + "/" + mode );
 | |
| 		if ( finalMode != undefined ) {
 | |
| 			mode = finalMode;
 | |
| 		}
 | |
| 		if ( mode == 1 ) showChalkboard();
 | |
| //console.log("playback (ok)");
 | |
| 
 | |
| 	};
 | |
| 
 | |
| 	function stopPlayback() {
 | |
| //console.log("stopPlayback");
 | |
| //console.log("Timeouts: " + timeouts[0].length + "/"+ timeouts[1].length);
 | |
| 		for ( var id = 0; id < 2; id++ ) {
 | |
| 			for ( var i = 0; i < timeouts[ id ].length; i++ ) {
 | |
| 				clearTimeout( timeouts[ id ][ i ] );
 | |
| 			}
 | |
| 			timeouts[ id ] = [];
 | |
| 		}
 | |
| 	};
 | |
| 
 | |
| 	function playEvent( id, event, timestamp ) {
 | |
| //console.log( timestamp +" / " + JSON.stringify(event));
 | |
| //console.log( id + ": " + timestamp +" / " +  event.time +" / " + event.type +" / " + mode );
 | |
| 		switch ( event.type ) {
 | |
| 		case 'open':
 | |
| 			if ( timestamp <= event.time ) {
 | |
| 				showChalkboard();
 | |
| 			} else {
 | |
| 				mode = 1;
 | |
| 			}
 | |
| 
 | |
| 			break;
 | |
| 		case 'close':
 | |
| 			if ( timestamp < event.time ) {
 | |
| 				closeChalkboard();
 | |
| 			} else {
 | |
| 				mode = 0;
 | |
| 			}
 | |
| 			break;
 | |
| 		case 'clear':
 | |
| 			clearCanvas( id );
 | |
| 			break;
 | |
| 		case 'selectboard':
 | |
| 			selectBoard( event.board );
 | |
| 			break;
 | |
| 		case 'draw':
 | |
| 			drawLine( id, event, timestamp );
 | |
| 			break;
 | |
| 		case 'erase':
 | |
| 			eraseCircle( id, event, timestamp );
 | |
| 			break;
 | |
| 		}
 | |
| 	};
 | |
| 
 | |
| 	function drawLine( id, event, timestamp ) {
 | |
| 		var ctx = drawingCanvas[ id ].context;
 | |
| 		var scale = drawingCanvas[ id ].scale;
 | |
| 		var xOffset = drawingCanvas[ id ].xOffset;
 | |
| 		var yOffset = drawingCanvas[ id ].yOffset;
 | |
| 		draw[ id ]( ctx, xOffset + event.x1 * scale, yOffset + event.y1 * scale, xOffset + event.x2 * scale, yOffset + event.y2 * scale, event.color );
 | |
| 	};
 | |
| 
 | |
| 	function eraseCircle( id, event, timestamp ) {
 | |
| 		var ctx = drawingCanvas[ id ].context;
 | |
| 		var scale = drawingCanvas[ id ].scale;
 | |
| 		var xOffset = drawingCanvas[ id ].xOffset;
 | |
| 		var yOffset = drawingCanvas[ id ].yOffset;
 | |
| 
 | |
| 		eraseWithSponge( ctx, xOffset + event.x * scale, yOffset + event.y * scale );
 | |
| 	};
 | |
| 
 | |
| 	function startErasing( x, y ) {
 | |
| 		drawing = false;
 | |
| 		erasing = true;
 | |
| 		drawingCanvas[ mode ].sponge.style.visibility = 'visible';
 | |
| 		erasePoint( x, y );
 | |
| 	}
 | |
| 
 | |
| 	function erasePoint( x, y ) {
 | |
| 		var ctx = drawingCanvas[ mode ].context;
 | |
| 		var scale = drawingCanvas[ mode ].scale;
 | |
| 		var xOffset = drawingCanvas[ mode ].xOffset;
 | |
| 		var yOffset = drawingCanvas[ mode ].yOffset;
 | |
| 
 | |
| 		// move sponge image
 | |
| 		drawingCanvas[ mode ].sponge.style.left = ( x * scale + xOffset - eraser.radius ) + 'px';
 | |
| 		drawingCanvas[ mode ].sponge.style.top = ( y * scale + yOffset - 2 * eraser.radius ) + 'px';
 | |
| 
 | |
| 		recordEvent( {
 | |
| 			type: 'erase',
 | |
| 			x,
 | |
| 			y
 | |
| 		} );
 | |
| 
 | |
| 		if (
 | |
| 			x * scale + xOffset > 0 &&
 | |
| 			y * scale + yOffset > 0 &&
 | |
| 			x * scale + xOffset < drawingCanvas[ mode ].width &&
 | |
| 			y * scale + yOffset < drawingCanvas[ mode ].height
 | |
| 		) {
 | |
| 			eraseWithSponge( ctx, x * scale + xOffset, y * scale + yOffset );
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	function stopErasing() {
 | |
| 		erasing = false;
 | |
| 		// hide sponge
 | |
| 		drawingCanvas[ mode ].sponge.style.visibility = 'hidden';
 | |
| 	}
 | |
| 
 | |
| 	function startDrawing( x, y ) {
 | |
| 		drawing = true;
 | |
| 
 | |
| 		var ctx = drawingCanvas[ mode ].context;
 | |
| 		var scale = drawingCanvas[ mode ].scale;
 | |
| 		var xOffset = drawingCanvas[ mode ].xOffset;
 | |
| 		var yOffset = drawingCanvas[ mode ].yOffset;
 | |
| 		lastX = x * scale + xOffset;
 | |
| 		lastY = y * scale + yOffset;
 | |
| 	}
 | |
| 
 | |
| 	function drawSegment( fromX, fromY, toX, toY, colorIdx ) {
 | |
| 		var ctx = drawingCanvas[ mode ].context;
 | |
| 		var scale = drawingCanvas[ mode ].scale;
 | |
| 		var xOffset = drawingCanvas[ mode ].xOffset;
 | |
| 		var yOffset = drawingCanvas[ mode ].yOffset;
 | |
| 
 | |
| 		recordEvent( {
 | |
| 			type: 'draw',
 | |
| 			color: colorIdx,
 | |
| 			x1: fromX,
 | |
| 			y1: fromY,
 | |
| 			x2: toX,
 | |
| 			y2: toY
 | |
| 		} );
 | |
| 
 | |
| 		if (
 | |
| 			fromX * scale + xOffset > 0 &&
 | |
| 			fromY * scale + yOffset > 0 &&
 | |
| 			fromX * scale + xOffset < drawingCanvas[ mode ].width &&
 | |
| 			fromY * scale + yOffset < drawingCanvas[ mode ].height &&
 | |
| 			toX * scale + xOffset > 0 &&
 | |
| 			toY * scale + yOffset > 0 &&
 | |
| 			toX * scale + xOffset < drawingCanvas[ mode ].width &&
 | |
| 			toY * scale + yOffset < drawingCanvas[ mode ].height
 | |
| 		) {
 | |
| 			draw[ mode ]( ctx, fromX * scale + xOffset, fromY * scale + yOffset, toX * scale + xOffset, toY * scale + yOffset, colorIdx );
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	function stopDrawing() {
 | |
| 		drawing = false;
 | |
| 	}
 | |
| 
 | |
| 
 | |
| /*****************************************************************
 | |
|  ** User interface
 | |
|  ******************************************************************/
 | |
| 
 | |
| 	function setupCanvasEvents( canvas ) {
 | |
| // TODO: check all touchevents
 | |
| 		canvas.addEventListener( 'touchstart', function ( evt ) {
 | |
| 			evt.preventDefault();
 | |
| //console.log("Touch start");
 | |
| 			if ( !readOnly && evt.target.getAttribute( 'data-chalkboard' ) == mode ) {
 | |
| 				var scale = drawingCanvas[ mode ].scale;
 | |
| 				var xOffset = drawingCanvas[ mode ].xOffset;
 | |
| 				var yOffset = drawingCanvas[ mode ].yOffset;
 | |
| 
 | |
| 				var touch = evt.touches[ 0 ];
 | |
| 				mouseX = touch.pageX;
 | |
| 				mouseY = touch.pageY;
 | |
| 				startDrawing( ( mouseX - xOffset ) / scale, ( mouseY - yOffset ) / scale );
 | |
| 				touchTimeout = setTimeout( startErasing, 500,  ( mouseX - xOffset ) / scale, ( mouseY - yOffset ) / scale );
 | |
| 			}
 | |
| 		}, passiveSupported ? {
 | |
| 			passive: false
 | |
| 		} : false );
 | |
| 
 | |
| 		canvas.addEventListener( 'touchmove', function ( evt ) {
 | |
| 			evt.preventDefault();
 | |
| //console.log("Touch move");
 | |
| 			clearTimeout( touchTimeout );
 | |
| 			touchTimeout = null;
 | |
| 			if ( drawing || erasing ) {
 | |
| 				var scale = drawingCanvas[ mode ].scale;
 | |
| 				var xOffset = drawingCanvas[ mode ].xOffset;
 | |
| 				var yOffset = drawingCanvas[ mode ].yOffset;
 | |
| 
 | |
| 				var touch = evt.touches[ 0 ];
 | |
| 				mouseX = touch.pageX;
 | |
| 				mouseY = touch.pageY;
 | |
| 				if ( mouseY < drawingCanvas[ mode ].height && mouseX < drawingCanvas[ mode ].width ) {
 | |
| 					// move sponge
 | |
| 					if ( event.type == 'erase' ) {
 | |
| 						drawingCanvas[ mode ].sponge.style.left = ( mouseX - eraser.radius ) + 'px';
 | |
| 						drawingCanvas[ mode ].sponge.style.top = ( mouseY - eraser.radius ) + 'px';
 | |
| 					}
 | |
| 				}
 | |
| 
 | |
| 				if ( drawing ) {
 | |
| 					drawSegment( ( lastX - xOffset ) / scale, ( lastY - yOffset ) / scale, ( mouseX - xOffset ) / scale, ( mouseY - yOffset ) / scale, color[ mode ] );
 | |
| 					// broadcast
 | |
| 					var message = new CustomEvent( messageType );
 | |
| 					message.content = {
 | |
| 						sender: 'chalkboard-plugin',
 | |
| 						type: 'draw',
 | |
| 						timestamp: Date.now() - slideStart,
 | |
| 						mode,
 | |
| 						board,
 | |
| 						fromX: ( lastX - xOffset ) / scale,
 | |
| 						fromY: ( lastY - yOffset ) / scale,
 | |
| 						toX: ( mouseX - xOffset ) / scale,
 | |
| 						toY: ( mouseY - yOffset ) / scale,
 | |
| 						color: color[ mode ]
 | |
| 					};
 | |
| 					document.dispatchEvent( message );
 | |
| 
 | |
| 					lastX = mouseX;
 | |
| 					lastY = mouseY;
 | |
| 				} else {
 | |
| 					erasePoint( ( mouseX - xOffset ) / scale, ( mouseY - yOffset ) / scale );
 | |
| 					// broadcast
 | |
| 					var message = new CustomEvent( messageType );
 | |
| 					message.content = {
 | |
| 						sender: 'chalkboard-plugin',
 | |
| 						type: 'erase',
 | |
| 						timestamp: Date.now() - slideStart,
 | |
| 						mode,
 | |
| 						board,
 | |
| 						x: ( mouseX - xOffset ) / scale,
 | |
| 						y: ( mouseY - yOffset ) / scale
 | |
| 					};
 | |
| 					document.dispatchEvent( message );
 | |
| 				}
 | |
| 
 | |
| 			}
 | |
| 		}, false );
 | |
| 
 | |
| 
 | |
| 		canvas.addEventListener( 'touchend', function ( evt ) {
 | |
| 			evt.preventDefault();
 | |
| 			clearTimeout( touchTimeout );
 | |
| 			touchTimeout = null;
 | |
| 			// hide sponge image
 | |
| 			drawingCanvas[ mode ].sponge.style.visibility = 'hidden';
 | |
| 			stopDrawing();
 | |
| 		}, false );
 | |
| 
 | |
| 		canvas.addEventListener( 'mousedown', function ( evt ) {
 | |
| 			evt.preventDefault();
 | |
| 			if ( !readOnly && evt.target.getAttribute( 'data-chalkboard' ) == mode ) {
 | |
| //console.log( "mousedown: " + evt.button );
 | |
| 				var scale = drawingCanvas[ mode ].scale;
 | |
| 				var xOffset = drawingCanvas[ mode ].xOffset;
 | |
| 				var yOffset = drawingCanvas[ mode ].yOffset;
 | |
| 
 | |
| 				mouseX = evt.pageX;
 | |
| 				mouseY = evt.pageY;
 | |
| 
 | |
| 				if ( evt.button == 2 || evt.button == 1 ) {
 | |
| 					startErasing( ( mouseX - xOffset ) / scale, ( mouseY - yOffset ) / scale );
 | |
| 					// broadcast
 | |
| 					var message = new CustomEvent( messageType );
 | |
| 					message.content = {
 | |
| 						sender: 'chalkboard-plugin',
 | |
| 						type: 'erase',
 | |
| 						timestamp: Date.now() - slideStart,
 | |
| 						mode,
 | |
| 						board,
 | |
| 						x: ( mouseX - xOffset ) / scale,
 | |
| 						y: ( mouseY - yOffset ) / scale
 | |
| 					};
 | |
| 					document.dispatchEvent( message );
 | |
| 				} else {
 | |
| 					startDrawing( ( mouseX - xOffset ) / scale, ( mouseY - yOffset ) / scale );
 | |
| 				}
 | |
| 			}
 | |
| 		} );
 | |
| 
 | |
| 		canvas.addEventListener( 'mousemove', function ( evt ) {
 | |
| 			evt.preventDefault();
 | |
| //console.log("Mouse move");
 | |
| 			if ( drawing || erasing ) {
 | |
| 				var scale = drawingCanvas[ mode ].scale;
 | |
| 				var xOffset = drawingCanvas[ mode ].xOffset;
 | |
| 				var yOffset = drawingCanvas[ mode ].yOffset;
 | |
| 
 | |
| 				mouseX = evt.pageX;
 | |
| 				mouseY = evt.pageY;
 | |
| 
 | |
| 				if ( drawing ) {
 | |
| 					drawSegment( ( lastX - xOffset ) / scale, ( lastY - yOffset ) / scale, ( mouseX - xOffset ) / scale, ( mouseY - yOffset ) / scale, color[ mode ] );
 | |
| 					// broadcast
 | |
| 					var message = new CustomEvent( messageType );
 | |
| 					message.content = {
 | |
| 						sender: 'chalkboard-plugin',
 | |
| 						type: 'draw',
 | |
| 						timestamp: Date.now() - slideStart,
 | |
| 						mode,
 | |
| 						board,
 | |
| 						fromX: ( lastX - xOffset ) / scale,
 | |
| 						fromY: ( lastY - yOffset ) / scale,
 | |
| 						toX: ( mouseX - xOffset ) / scale,
 | |
| 						toY: ( mouseY - yOffset ) / scale,
 | |
| 						color: color[ mode ]
 | |
| 					};
 | |
| 					document.dispatchEvent( message );
 | |
| 
 | |
| 					lastX = mouseX;
 | |
| 					lastY = mouseY;
 | |
| 				} else {
 | |
| 					erasePoint( ( mouseX - xOffset ) / scale, ( mouseY - yOffset ) / scale );
 | |
| 					// broadcast
 | |
| 					var message = new CustomEvent( messageType );
 | |
| 					message.content = {
 | |
| 						sender: 'chalkboard-plugin',
 | |
| 						type: 'erase',
 | |
| 						timestamp: Date.now() - slideStart,
 | |
| 						mode,
 | |
| 						board,
 | |
| 						x: ( mouseX - xOffset ) / scale,
 | |
| 						y: ( mouseY - yOffset ) / scale
 | |
| 					};
 | |
| 					document.dispatchEvent( message );
 | |
| 				}
 | |
| 
 | |
| 			}
 | |
| 		} );
 | |
| 
 | |
| 
 | |
| 		canvas.addEventListener( 'mouseup', function ( evt ) {
 | |
| 			evt.preventDefault();
 | |
| 			drawingCanvas[ mode ].canvas.style.cursor = pens[ mode ][ color[ mode ] ].cursor;
 | |
| 			if ( drawing || erasing ) {
 | |
| 				stopDrawing();
 | |
| 				stopErasing();
 | |
| 			}
 | |
| 		} );
 | |
| 	}
 | |
| 
 | |
| 	function resize() {
 | |
| //console.log("resize");
 | |
| 		// Resize the canvas and draw everything again
 | |
| 		var timestamp = Date.now() - slideStart;
 | |
| 		if ( !playback ) {
 | |
| 			timestamp = getSlideDuration();
 | |
| 		}
 | |
| 
 | |
| //console.log( drawingCanvas[0].scale + "/" + drawingCanvas[0].xOffset + "/" +drawingCanvas[0].yOffset );
 | |
| 		for ( var id = 0; id < 2; id++ ) {
 | |
| 			drawingCanvas[ id ].width = window.innerWidth;
 | |
| 			drawingCanvas[ id ].height = window.innerHeight;
 | |
| 			drawingCanvas[ id ].canvas.width = drawingCanvas[ id ].width;
 | |
| 			drawingCanvas[ id ].canvas.height = drawingCanvas[ id ].height;
 | |
| 			drawingCanvas[ id ].context.canvas.width = drawingCanvas[ id ].width;
 | |
| 			drawingCanvas[ id ].context.canvas.height = drawingCanvas[ id ].height;
 | |
| 
 | |
| 			drawingCanvas[ id ].scale = Math.min( drawingCanvas[ id ].width / storage[ id ].width, drawingCanvas[ id ].height / storage[ id ].height );
 | |
| 			drawingCanvas[ id ].xOffset = ( drawingCanvas[ id ].width - storage[ id ].width * drawingCanvas[ id ].scale ) / 2;
 | |
| 			drawingCanvas[ id ].yOffset = ( drawingCanvas[ id ].height - storage[ id ].height * drawingCanvas[ id ].scale ) / 2;
 | |
| //console.log( drawingCanvas[id].scale + "/" + drawingCanvas[id].xOffset + "/" +drawingCanvas[id].yOffset );
 | |
| 		}
 | |
| //console.log( window.innerWidth + "/" + window.innerHeight);
 | |
| 		startPlayback( timestamp, mode, true );
 | |
| 	}
 | |
| 
 | |
| 	Reveal.addEventListener( 'pdf-ready', function ( evt ) {
 | |
| //		console.log( "Create printouts when ready" );
 | |
| 		whenLoaded( createPrintout );
 | |
| 	});
 | |
| 
 | |
| 	Reveal.addEventListener( 'ready', function ( evt ) {
 | |
| //console.log('ready');
 | |
| 		if ( !printMode ) {
 | |
| 			window.addEventListener( 'resize', resize );
 | |
| 
 | |
| 			slideStart = Date.now() - getSlideDuration();
 | |
| 			slideIndices = Reveal.getIndices();
 | |
| 			if ( !playback ) {
 | |
| 				startPlayback( getSlideDuration(), 0 );
 | |
| 			}
 | |
| 			if ( Reveal.isAutoSliding() ) {
 | |
| 				var event = new CustomEvent( 'startplayback' );
 | |
| 				event.timestamp = 0;
 | |
| 				document.dispatchEvent( event );
 | |
| 			}
 | |
| 			updateStorage();
 | |
| 			whenReady( addPageNumbers );
 | |
| 		}
 | |
| 	} );
 | |
| 	Reveal.addEventListener( 'slidechanged', function ( evt ) {
 | |
| //		clearTimeout( slidechangeTimeout );
 | |
| //console.log('slidechanged');
 | |
| 		if ( !printMode ) {
 | |
| 			slideStart = Date.now() - getSlideDuration();
 | |
| 			slideIndices = Reveal.getIndices();
 | |
| 			closeChalkboard();
 | |
| 			board = 0;
 | |
| 			clearCanvas( 0 );
 | |
| 			clearCanvas( 1 );
 | |
| 			if ( !playback ) {
 | |
| 				slidechangeTimeout = setTimeout( startPlayback, transition, getSlideDuration(), 0 );
 | |
| 			}
 | |
| 			if ( Reveal.isAutoSliding() ) {
 | |
| 				var event = new CustomEvent( 'startplayback' );
 | |
| 				event.timestamp = 0;
 | |
| 				document.dispatchEvent( event );
 | |
| 			}
 | |
| 		}
 | |
| 	} );
 | |
| 	Reveal.addEventListener( 'fragmentshown', function ( evt ) {
 | |
| //		clearTimeout( slidechangeTimeout );
 | |
| //console.log('fragmentshown');
 | |
| 		if ( !printMode ) {
 | |
| 			slideStart = Date.now() - getSlideDuration();
 | |
| 			slideIndices = Reveal.getIndices();
 | |
| 			closeChalkboard();
 | |
| 			board = 0;
 | |
| 			clearCanvas( 0 );
 | |
| 			clearCanvas( 1 );
 | |
| 			if ( Reveal.isAutoSliding() ) {
 | |
| 				var event = new CustomEvent( 'startplayback' );
 | |
| 				event.timestamp = 0;
 | |
| 				document.dispatchEvent( event );
 | |
| 			} else if ( !playback ) {
 | |
| 				startPlayback( getSlideDuration(), 0 );
 | |
| //				closeChalkboard();
 | |
| 			}
 | |
| 		}
 | |
| 	} );
 | |
| 	Reveal.addEventListener( 'fragmenthidden', function ( evt ) {
 | |
| //		clearTimeout( slidechangeTimeout );
 | |
| //console.log('fragmenthidden');
 | |
| 		if ( !printMode ) {
 | |
| 			slideStart = Date.now() - getSlideDuration();
 | |
| 			slideIndices = Reveal.getIndices();
 | |
| 			closeChalkboard();
 | |
| 			board = 0;
 | |
| 			clearCanvas( 0 );
 | |
| 			clearCanvas( 1 );
 | |
| 			if ( Reveal.isAutoSliding() ) {
 | |
| 				document.dispatchEvent( new CustomEvent( 'stopplayback' ) );
 | |
| 			} else if ( !playback ) {
 | |
| 				startPlayback( getSlideDuration() );
 | |
| 				closeChalkboard();
 | |
| 			}
 | |
| 		}
 | |
| 	} );
 | |
| 
 | |
| 	Reveal.addEventListener( 'autoslideresumed', function ( evt ) {
 | |
| //console.log('autoslideresumed');
 | |
| 		var event = new CustomEvent( 'startplayback' );
 | |
| 		event.timestamp = 0;
 | |
| 		document.dispatchEvent( event );
 | |
| 	} );
 | |
| 	Reveal.addEventListener( 'autoslidepaused', function ( evt ) {
 | |
| //console.log('autoslidepaused');
 | |
| 		document.dispatchEvent( new CustomEvent( 'stopplayback' ) );
 | |
| 
 | |
| 		// advance to end of slide
 | |
| //		closeChalkboard();
 | |
| 		startPlayback( getSlideDuration(), 0 );
 | |
| 	} );
 | |
| 
 | |
| 	function toggleNotesCanvas() {
 | |
| 		if ( !readOnly ) {
 | |
| 			if ( mode == 1 ) {
 | |
| 				toggleChalkboard();
 | |
| 				notescanvas.style.background = background[ 0 ]; //'rgba(255,0,0,0.5)';
 | |
| 				notescanvas.style.pointerEvents = 'auto';
 | |
| 			}
 | |
| 			else {
 | |
| 				if ( notescanvas.style.pointerEvents != 'none' ) {
 | |
| 					// hide notes canvas
 | |
| 					if ( colorButtons ) {
 | |
| 						notescanvas.querySelector( '.palette' ).style.visibility = 'hidden';
 | |
| 					}
 | |
| 					notescanvas.style.background = 'rgba(0,0,0,0)';
 | |
| 					notescanvas.style.pointerEvents = 'none';
 | |
| 				}
 | |
| 				else {
 | |
| 					// show notes canvas
 | |
| 					if ( colorButtons ) {
 | |
| 						notescanvas.querySelector( '.palette' ).style.visibility = 'visible';
 | |
| 					}
 | |
| 					notescanvas.style.background = background[ 0 ]; //'rgba(255,0,0,0.5)';
 | |
| 					notescanvas.style.pointerEvents = 'auto';
 | |
| 
 | |
| 					var idx = 0;
 | |
| 					if ( color[ mode ] ) {
 | |
| 						idx = color[ mode ];
 | |
| 					}
 | |
| 
 | |
| 					setColor( idx, true );
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	};
 | |
| 
 | |
| 	function toggleChalkboard() {
 | |
| //console.log("toggleChalkboard " + mode);
 | |
| 		if ( mode == 1 ) {
 | |
| 			if ( !readOnly ) {
 | |
| 				recordEvent( { type: 'close' } );
 | |
| 			}
 | |
| 			closeChalkboard();
 | |
| 
 | |
| 			// broadcast
 | |
| 			var message = new CustomEvent( messageType );
 | |
| 			message.content = {
 | |
| 				sender: 'chalkboard-plugin',
 | |
| 				type: 'closeChalkboard',
 | |
| 				timestamp: Date.now() - slideStart,
 | |
| 				mode: 0,
 | |
| 				board
 | |
| 			};
 | |
| 			document.dispatchEvent( message );
 | |
| 
 | |
| 
 | |
| 		} else {
 | |
| 			showChalkboard();
 | |
| 			if ( !readOnly ) {
 | |
| 				recordEvent( { type: 'open' } );
 | |
| 				// broadcast
 | |
| 				var message = new CustomEvent( messageType );
 | |
| 				message.content = {
 | |
| 					sender: 'chalkboard-plugin',
 | |
| 					type: 'showChalkboard',
 | |
| 					timestamp: Date.now() - slideStart,
 | |
| 					mode: 1,
 | |
| 					board
 | |
| 				};
 | |
| 				document.dispatchEvent( message );
 | |
| 
 | |
| 				var idx = 0;
 | |
| 
 | |
| 				if ( rememberColor[ mode ] ) {
 | |
| 					idx = color[ mode ];
 | |
| 				}
 | |
| 
 | |
| 				setColor( idx, true );
 | |
| 			}
 | |
| 		}
 | |
| 	};
 | |
| 
 | |
| 	function clearSlide() {
 | |
| 		recordEvent( { type: 'clear' } );
 | |
| 		clearCanvas( mode );
 | |
| 	}
 | |
| 
 | |
| 	function clear() {
 | |
| 		if ( !readOnly ) {
 | |
| 			clearSlide();
 | |
| 			// broadcast
 | |
| 			var message = new CustomEvent( messageType );
 | |
| 			message.content = {
 | |
| 				sender: 'chalkboard-plugin',
 | |
| 				type: 'clear',
 | |
| 				timestamp: Date.now() - slideStart,
 | |
| 				mode,
 | |
| 				board
 | |
| 			};
 | |
| 			document.dispatchEvent( message );
 | |
| 		}
 | |
| 	};
 | |
| 
 | |
| 	function colorIndex( idx ) {
 | |
| 		if ( !readOnly ) {
 | |
| 			setColor( idx, true );
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	function colorNext() {
 | |
| 		if ( !readOnly ) {
 | |
| 			let idx = cycleColorNext();
 | |
| 			setColor( idx, true );
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	function colorPrev() {
 | |
| 		if ( !readOnly ) {
 | |
| 			let idx = cycleColorPrev();
 | |
| 			setColor( idx, true );
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	function resetSlideDrawings() {
 | |
| 		slideStart = Date.now();
 | |
| 		closeChalkboard();
 | |
| 
 | |
| 		clearCanvas( 0 );
 | |
| 		clearCanvas( 1 );
 | |
| 
 | |
| 		mode = 1;
 | |
| 		var slideData = getSlideData();
 | |
| 		slideData.duration = 0;
 | |
| 		slideData.events = [];
 | |
| 		mode = 0;
 | |
| 		var slideData = getSlideData();
 | |
| 		slideData.duration = 0;
 | |
| 		slideData.events = [];
 | |
| 
 | |
| 		updateStorage();
 | |
| 	}
 | |
| 
 | |
| 	function resetSlide( force ) {
 | |
| 		var ok = force || confirm( "Please confirm to delete chalkboard drawings on this slide!" );
 | |
| 		if ( ok ) {
 | |
| //console.log("resetSlide ");
 | |
| 			stopPlayback();
 | |
| 			resetSlideDrawings();
 | |
| 			// broadcast
 | |
| 			var message = new CustomEvent( messageType );
 | |
| 			message.content = {
 | |
| 				sender: 'chalkboard-plugin',
 | |
| 				type: 'resetSlide',
 | |
| 				timestamp: Date.now() - slideStart,
 | |
| 				mode,
 | |
| 				board
 | |
| 			};
 | |
| 			document.dispatchEvent( message );
 | |
| 		}
 | |
| 	};
 | |
| 
 | |
| 	function resetStorage( force ) {
 | |
| 		var ok = force || confirm( "Please confirm to delete all chalkboard drawings!" );
 | |
| 		if ( ok ) {
 | |
| 			stopPlayback();
 | |
| 			slideStart = Date.now();
 | |
| 			clearCanvas( 0 );
 | |
| 			clearCanvas( 1 );
 | |
| 			if ( mode == 1 ) {
 | |
| 				closeChalkboard();
 | |
| 			}
 | |
| 
 | |
| 			storage = [ {
 | |
| 					width: Reveal.getConfig().width,
 | |
| 					height: Reveal.getConfig().height,
 | |
| 					data: []
 | |
| 				},
 | |
| 				{
 | |
| 					width: Reveal.getConfig().width,
 | |
| 					height: Reveal.getConfig().height,
 | |
| 					data: []
 | |
| 				}
 | |
| 			];
 | |
| 
 | |
| 			if ( config.storage ) {
 | |
| 				sessionStorage.setItem( config.storage, null )
 | |
| 			}
 | |
| 			// broadcast
 | |
| 			var message = new CustomEvent( messageType );
 | |
| 			message.content = {
 | |
| 				sender: 'chalkboard-plugin',
 | |
| 				type: 'init',
 | |
| 				timestamp: Date.now() - slideStart,
 | |
| 				storage,
 | |
| 				mode,
 | |
| 				board
 | |
| 			};
 | |
| 			document.dispatchEvent( message );
 | |
| 		}
 | |
| 	};
 | |
| 
 | |
| 	this.toggleNotesCanvas = toggleNotesCanvas;
 | |
| 	this.toggleChalkboard = toggleChalkboard;
 | |
| 	this.colorIndex = colorIndex;
 | |
| 	this.colorNext = colorNext;
 | |
| 	this.colorPrev = colorPrev;
 | |
| 	this.clear = clear;
 | |
| 	this.reset = resetSlide;
 | |
| 	this.resetAll = resetStorage;
 | |
| 	this.download = downloadData;
 | |
| 	this.updateStorage = updateStorage;
 | |
| 	this.getData = getData;
 | |
| 	this.configure = configure;
 | |
| 
 | |
| 
 | |
| 	for ( var key in keyBindings ) {
 | |
| 		if ( keyBindings[ key ] ) {
 | |
| 			Reveal.addKeyBinding( keyBindings[ key ], RevealChalkboard[ key ] );
 | |
| 		}
 | |
| 	};
 | |
| 
 | |
| 	return this;
 | |
| };
 |