/***************************************************************** ** 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 = '' 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 = '' 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 = ''; 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 = '
'; 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; };