Path: blob/main/src/resources/formats/revealjs/plugins/chalkboard/plugin.js
12923 views
/*****************************************************************1** Author: Asvin Goel, [email protected]2**3** A plugin for reveal.js adding a chalkboard.4**5** Version: 2.3.36**7** License: MIT license (see LICENSE.md)8**9** Credits:10** Chalkboard effect by Mohamed Moustafa https://github.com/mmoustafa/Chalkboard11** Multi color support initially added by Kurt Rinnert https://github.com/rinnert12** Compatibility with reveal.js v4 by Hakim El Hattab https://github.com/hakimel13******************************************************************/1415"use strict";1617window.RevealChalkboard = window.RevealChalkboard || {18id: 'RevealChalkboard',19init: function ( deck ) {20initChalkboard.call(this, deck );21},22configure: function ( config ) {23configure( config );24},25toggleNotesCanvas: function () {26toggleNotesCanvas();27},28toggleChalkboard: function () {29toggleChalkboard();30},31colorIndex: function () {32colorIndex();33},34colorNext: function () {35colorNext();36},37colorPrev: function () {38colorPrev();39},40clear: function () {41clear();42},43reset: function () {44reset();45},46resetAll: function () {47resetAll();48},49updateStorage: function () {50updateStorage();51},52getData: function () {53return getData();54},55download: function () {56download();57},58};5960function scriptPath() {61// obtain plugin path from the script element62var src;63if ( document.currentScript ) {64src = document.currentScript.src;65} else {66var sel = document.querySelector( 'script[src$="/chalkboard/plugin.js"]' )67if ( sel ) {68src = sel.src;69}70}71var path = ( src === undefined ) ? "" : src.slice( 0, src.lastIndexOf( "/" ) + 1 );72//console.log("Path: " + path);73return path;74}75var path = scriptPath();7677const initChalkboard = function ( Reveal ) {78//console.warn(path);79/* Feature detection for passive event handling*/80var passiveSupported = false;8182try {83window.addEventListener( 'test', null, Object.defineProperty( {}, 'passive', {84get: function () {85passiveSupported = true;86}87} ) );88} catch ( err ) {}899091/*****************************************************************92** Configuration93******************************************************************/94var background, pens, draw, color;95var grid = false;96var boardmarkerWidth = 3;97var chalkWidth = 7;98var chalkEffect = 1.0;99var rememberColor = [ true, false ];100var eraser = {101src: path + 'img/sponge.png',102radius: 20103};104var boardmarkers = [ {105color: 'rgba(100,100,100,1)',106cursor: 'url(' + path + 'img/boardmarker-black.png), auto'107},108{109color: 'rgba(30,144,255, 1)',110cursor: 'url(' + path + 'img/boardmarker-blue.png), auto'111},112{113color: 'rgba(220,20,60,1)',114cursor: 'url(' + path + 'img/boardmarker-red.png), auto'115},116{117color: 'rgba(50,205,50,1)',118cursor: 'url(' + path + 'img/boardmarker-green.png), auto'119},120{121color: 'rgba(255,140,0,1)',122cursor: 'url(' + path + 'img/boardmarker-orange.png), auto'123},124{125color: 'rgba(150,0,20150,1)',126cursor: 'url(' + path + 'img/boardmarker-purple.png), auto'127},128{129color: 'rgba(255,220,0,1)',130cursor: 'url(' + path + 'img/boardmarker-yellow.png), auto'131}132];133var chalks = [ {134color: 'rgba(255,255,255,0.5)',135cursor: 'url(' + path + 'img/chalk-white.png), auto'136},137{138color: 'rgba(96, 154, 244, 0.5)',139cursor: 'url(' + path + 'img/chalk-blue.png), auto'140},141{142color: 'rgba(237, 20, 28, 0.5)',143cursor: 'url(' + path + 'img/chalk-red.png), auto'144},145{146color: 'rgba(20, 237, 28, 0.5)',147cursor: 'url(' + path + 'img/chalk-green.png), auto'148},149{150color: 'rgba(220, 133, 41, 0.5)',151cursor: 'url(' + path + 'img/chalk-orange.png), auto'152},153{154color: 'rgba(220,0,220,0.5)',155cursor: 'url(' + path + 'img/chalk-purple.png), auto'156},157{158color: 'rgba(255,220,0,0.5)',159cursor: 'url(' + path + 'img/chalk-yellow.png), auto'160}161];162163var sponge = {164cursor: 'url(' + path + 'img/sponge.png), auto'165}166167168var keyBindings = {169toggleNotesCanvas: {170keyCode: 67,171key: 'C',172description: 'Toggle notes canvas'173},174toggleChalkboard: {175keyCode: 66,176key: 'B',177description: 'Toggle chalkboard'178},179clear: {180keyCode: 46,181key: 'DEL',182description: 'Clear drawings on slide'183},184/*185reset: {186keyCode: 173,187key: '-',188description: 'Reset drawings on slide'189},190*/191resetAll: {192keyCode: 8,193key: 'BACKSPACE',194description: 'Reset all drawings'195},196colorNext: {197keyCode: 88,198key: 'X',199description: 'Next color'200},201colorPrev: {202keyCode: 89,203key: 'Y',204description: 'Previous color'205},206download: {207keyCode: 68,208key: 'D',209description: 'Download drawings'210}211};212213214var theme = 'chalkboard';215var color = [ 0, 0 ];216var toggleChalkboardButton = false;217var toggleNotesButton = false;218var colorButtons = true;219var boardHandle = true;220var transition = 800;221222var readOnly = false;223var messageType = 'broadcast';224225var config = configure( Reveal.getConfig().chalkboard || {} );226if ( config.keyBindings ) {227for ( var key in config.keyBindings ) {228keyBindings[ key ] = config.keyBindings[ key ];229};230}231232function configure( config ) {233234if ( config.boardmarkerWidth || config.penWidth ) boardmarkerWidth = config.boardmarkerWidth || config.penWidth;235if ( config.chalkWidth ) chalkWidth = config.chalkWidth;236if ( config.chalkEffect ) chalkEffect = config.chalkEffect;237if ( config.rememberColor ) rememberColor = config.rememberColor;238if ( config.eraser ) eraser = config.eraser;239if ( config.boardmarkers ) boardmarkers = config.boardmarkers;240if ( config.chalks ) chalks = config.chalks;241242if ( config.theme ) theme = config.theme;243switch ( theme ) {244case 'whiteboard':245background = [ 'rgba(127,127,127,.1)', path + 'img/whiteboard.png' ];246draw = [ drawWithBoardmarker, drawWithBoardmarker ];247pens = [ boardmarkers, boardmarkers ];248grid = {249color: 'rgb(127,127,255,0.1)',250distance: 40,251width: 2252};253break;254case 'chalkboard':255default:256background = [ 'rgba(127,127,127,.1)', path + 'img/blackboard.png' ];257draw = [ drawWithBoardmarker, drawWithChalk ];258pens = [ boardmarkers, chalks ];259grid = {260color: 'rgb(50,50,10,0.5)',261distance: 80,262width: 2263};264}265266if ( config.background ) background = config.background;267if ( config.grid != undefined ) grid = config.grid;268269if ( config.toggleChalkboardButton != undefined ) toggleChalkboardButton = config.toggleChalkboardButton;270if ( config.toggleNotesButton != undefined ) toggleNotesButton = config.toggleNotesButton;271if ( config.colorButtons != undefined ) colorButtons = config.colorButtons;272if ( config.boardHandle != undefined ) boardHandle = config.boardHandle;273if ( config.transition ) transition = config.transition;274275if ( config.readOnly != undefined ) readOnly = config.readOnly;276if ( config.messageType ) messageType = config.messageType;277278if ( drawingCanvas && ( config.theme || config.background || config.grid ) ) {279var canvas = document.getElementById( drawingCanvas[ 1 ].id );280canvas.style.background = 'url("' + background[ 1 ] + '") repeat';281clearCanvas( 1 );282drawGrid();283}284285return config;286}287/*****************************************************************288** Setup289******************************************************************/290291function whenReady( callback ) {292// wait for markdown to be parsed and code to be highlighted293if ( !document.querySelector( 'section[data-markdown]:not([data-markdown-parsed])' )294&& !document.querySelector( '[data-load]:not([data-loaded])')295&& !document.querySelector( 'code[data-line-numbers*="|"]')296) {297callback();298} else {299console.log( "Wait for external sources to be loaded and code to be highlighted" );300setTimeout( whenReady, 500, callback )301}302}303304function whenLoaded( callback ) {305// wait for drawings to be loaded and markdown to be parsed306if ( loaded !== null ) {307callback();308} else {309console.log( "Wait for drawings to be loaded" );310setTimeout( whenLoaded, 500, callback )311}312}313314var drawingCanvas = [ {315id: 'notescanvas'316}, {317id: 'chalkboard'318} ];319setupDrawingCanvas( 0 );320setupDrawingCanvas( 1 );321322var mode = 0; // 0: notes canvas, 1: chalkboard323var board = 0; // board index (only for chalkboard)324325var mouseX = 0;326var mouseY = 0;327var lastX = null;328var lastY = null;329330var drawing = false;331var erasing = false;332333var slideStart = Date.now();334var slideIndices = {335h: 0,336v: 0337};338339var timeouts = [340[],341[]342];343var slidechangeTimeout = null;344var updateStorageTimeout = null;345var playback = false;346347function changeCursor( element, tool ) {348element.style.cursor = tool.cursor;349var palette = document.querySelector('.palette[data-mode="' + mode + '"]');350if ( palette ) {351palette.style.cursor = tool.cursor;352}353}354355function createPalette( colors, length ) {356if ( length === true || length > colors.length ) {357length = colors.length;358}359var palette = document.createElement( 'div' );360palette.classList.add( 'palette' );361var list = document.createElement( 'ul' );362// color pickers363for ( var i = 0; i < length; i++ ) {364var colorButton = document.createElement( 'li' );365colorButton.setAttribute( 'data-color', i );366colorButton.innerHTML = '<i class="fa fa-square"></i>';367colorButton.style.color = colors[ i ].color;368colorButton.addEventListener( 'click', function ( e ) {369var element = e.target;370while ( !element.hasAttribute( 'data-color' ) ) {371element = element.parentElement;372}373colorIndex( parseInt( element.getAttribute( 'data-color' ) ) );374} );375colorButton.addEventListener( 'touchstart', function ( e ) {376var element = e.target;377while ( !element.hasAttribute( 'data-color' ) ) {378element = element.parentElement;379}380colorIndex( parseInt( element.getAttribute( 'data-color' ) ) );381} );382list.appendChild( colorButton );383}384// eraser385var eraserButton = document.createElement( 'li' );386eraserButton.setAttribute( 'data-eraser', 'true' );387var spongeImg = document.createElement( 'img' );388spongeImg.src = eraser.src;389spongeImg.height = "24";390spongeImg.width = "24";391spongeImg.style.marginTop = '10px';392spongeImg.style.marginRight = '0';393spongeImg.style.marginBottom = '0';394spongeImg.style.marginLeft = '0';395eraserButton.appendChild(spongeImg);396eraserButton.addEventListener( 'click', function ( e ) {397colorIndex( -1 );398} );399eraserButton.addEventListener( 'touchstart', function ( e ) {400colorIndex( -1 );401} );402list.appendChild( eraserButton );403404palette.appendChild( list );405return palette;406};407408function switchBoard( boardIdx ) {409selectBoard( boardIdx, true );410// broadcast411var message = new CustomEvent( messageType );412message.content = {413sender: 'chalkboard-plugin',414type: 'selectboard',415timestamp: Date.now() - slideStart,416mode,417board418};419document.dispatchEvent( message );420}421422function setupDrawingCanvas( id ) {423var container = document.createElement( 'div' );424container.id = drawingCanvas[ id ].id;425container.classList.add( 'overlay' );426container.setAttribute( 'data-prevent-swipe', 'true' );427container.oncontextmenu = function () {428return false;429}430431changeCursor( container, pens[ id ][ color[ id ] ] );432433drawingCanvas[ id ].width = window.innerWidth;434drawingCanvas[ id ].height = window.innerHeight;435drawingCanvas[ id ].scale = 1;436drawingCanvas[ id ].xOffset = 0;437drawingCanvas[ id ].yOffset = 0;438439if ( id == "0" ) {440container.style.background = 'rgba(0,0,0,0)';441container.style.zIndex = 24;442container.style.opacity = 1;443container.style.visibility = 'visible';444container.style.pointerEvents = 'none';445container.style['backdrop-filter'] = 'none';446container.style['-webkit-backdrop-filter'] = 'none';447448var slides = document.querySelector( '.slides' );449var aspectRatio = Reveal.getConfig().width / Reveal.getConfig().height;450if ( drawingCanvas[ id ].width > drawingCanvas[ id ].height * aspectRatio ) {451drawingCanvas[ id ].xOffset = ( drawingCanvas[ id ].width - drawingCanvas[ id ].height * aspectRatio ) / 2;452} else if ( drawingCanvas[ id ].height > drawingCanvas[ id ].width / aspectRatio ) {453drawingCanvas[ id ].yOffset = ( drawingCanvas[ id ].height - drawingCanvas[ id ].width / aspectRatio ) / 2;454}455456if ( colorButtons ) {457var palette = createPalette( boardmarkers, colorButtons );458palette.dataset.mode = id;459palette.style.visibility = 'hidden'; // only show palette in drawing mode460container.appendChild( palette );461}462} else {463container.style.background = 'url("' + background[ id ] + '") repeat';464container.style.zIndex = 26;465container.style.opacity = 0;466container.style.visibility = 'hidden';467468if ( colorButtons ) {469var palette = createPalette( chalks, colorButtons );470palette.dataset.mode = id;471container.appendChild( palette );472}473if ( boardHandle ) {474var handle = document.createElement( 'div' );475handle.classList.add( 'boardhandle' );476handle.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>';477handle.querySelector( '#previousboard' ).addEventListener( 'click', function ( e ) {478e.preventDefault();479switchBoard( board - 1 );480} );481handle.querySelector( '#nextboard' ).addEventListener( 'click', function ( e ) {482e.preventDefault();483switchBoard( board + 1 );484} );485handle.querySelector( '#previousboard' ).addEventListener( 'touchstart', function ( e ) {486e.preventDefault();487switchBoard( board - 1 );488} );489handle.querySelector( '#nextboard' ).addEventListener( 'touchstart', function ( e ) {490e.preventDefault();491switchBoard( board + 1 );492} );493494container.appendChild( handle );495}496}497498var canvas = document.createElement( 'canvas' );499canvas.width = drawingCanvas[ id ].width;500canvas.height = drawingCanvas[ id ].height;501canvas.setAttribute( 'data-chalkboard', id );502changeCursor( canvas, pens[ id ][ color[ id ] ] );503container.appendChild( canvas );504drawingCanvas[ id ].canvas = canvas;505506drawingCanvas[ id ].context = canvas.getContext( '2d' );507508setupCanvasEvents( container );509510document.querySelector( '.reveal' ).appendChild( container );511drawingCanvas[ id ].container = container;512}513514515/*****************************************************************516** Storage517******************************************************************/518519var storage = [ {520width: Reveal.getConfig().width,521height: Reveal.getConfig().height,522data: []523},524{525width: Reveal.getConfig().width,526height: Reveal.getConfig().height,527data: []528}529];530531var loaded = null;532533if ( config.storage ) {534// Get chalkboard drawings from session storage535loaded = initStorage( sessionStorage.getItem( config.storage ) );536}537538if ( !loaded && config.src != null ) {539// Get chalkboard drawings from the given file540loadData( config.src );541}542543/**544* Initialize storage.545*/546function initStorage( json ) {547var success = false;548try {549var data = JSON.parse( json );550for ( var id = 0; id < data.length; id++ ) {551if ( drawingCanvas[ id ].width != data[ id ].width || drawingCanvas[ id ].height != data[ id ].height ) {552drawingCanvas[ id ].scale = Math.min( drawingCanvas[ id ].width / data[ id ].width, drawingCanvas[ id ].height / data[ id ].height );553drawingCanvas[ id ].xOffset = ( drawingCanvas[ id ].width - data[ id ].width * drawingCanvas[ id ].scale ) / 2;554drawingCanvas[ id ].yOffset = ( drawingCanvas[ id ].height - data[ id ].height * drawingCanvas[ id ].scale ) / 2;555}556if ( config.readOnly ) {557drawingCanvas[ id ].container.style.cursor = 'default';558drawingCanvas[ id ].canvas.style.cursor = 'default';559}560}561success = true;562storage = data;563} catch ( err ) {564console.warn( "Cannot initialise storage!" );565}566return success;567}568569570/**571* Load data.572*/573function loadData( filename ) {574var xhr = new XMLHttpRequest();575xhr.onload = function () {576if ( xhr.readyState === 4 && xhr.status != 404 ) {577loaded = initStorage( xhr.responseText );578updateStorage();579console.log( "Drawings loaded from file" );580} else {581config.readOnly = undefined;582readOnly = undefined;583console.warn( 'Failed to get file ' + filename + '. ReadyState: ' + xhr.readyState + ', Status: ' + xhr.status );584loaded = false;585}586};587588xhr.open( 'GET', filename, true );589try {590xhr.send();591} catch ( error ) {592config.readOnly = undefined;593readOnly = undefined;594console.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 );595loaded = false;596}597}598599600function storageChanged( now ) {601if ( !now ) {602// create or update timer603if ( updateStorageTimeout ) {604clearTimeout( updateStorageTimeout );605}606updateStorageTimeout = setTimeout( storageChanged, 1000, true);607}608else {609// console.log("Update storage", updateStorageTimeout, Date.now());610updateStorage();611updateStorageTimeout = null;612}613}614615function updateStorage() {616var json = JSON.stringify( storage )617if ( config.storage ) {618sessionStorage.setItem( config.storage, json )619}620return json;621}622623function recordEvent( event ) {624//console.log(event);625event.time = Date.now() - slideStart;626if ( mode == 1 ) event.board = board;627var slideData = getSlideData();628var i = slideData.events.length;629while ( i > 0 && event.time < slideData.events[ i - 1 ].time ) {630i--;631}632slideData.events.splice( i, 0, event );633slideData.duration = Math.max( slideData.duration, Date.now() - slideStart ) + 1;634635storageChanged();636}637638/**639* Get data as json string.640*/641function getData() {642// cleanup slide data without events643for ( var id = 0; id < 2; id++ ) {644for ( var i = storage[ id ].data.length - 1; i >= 0; i-- ) {645if ( storage[ id ].data[ i ].events.length == 0 ) {646storage[ id ].data.splice( i, 1 );647}648}649}650651return updateStorage();652}653654/**655* Download data.656*/657function downloadData() {658var a = document.createElement( 'a' );659document.body.appendChild( a );660try {661a.download = 'chalkboard.json';662var blob = new Blob( [ getData() ], {663type: 'application/json'664} );665a.href = window.URL.createObjectURL( blob );666} catch ( error ) {667// https://stackoverflow.com/a/6234804668// escape data for proper handling of quotes and line breaks669// in case malicious user gets a chance to craft the exception message670error = String(error)671.replace(/&/g, "&")672.replace(/</g, "<")673.replace(/>/g, ">")674.replace(/"/g, """)675.replace(/'/g, "'");676a.innerHTML += ' (' + error + ')';677}678a.click();679document.body.removeChild( a );680}681682/**683* Returns data object for the slide with the given indices.684*/685function getSlideData( indices, id ) {686if ( id == undefined ) id = mode;687if ( !indices ) indices = slideIndices;688var data;689for ( var i = 0; i < storage[ id ].data.length; i++ ) {690if ( storage[ id ].data[ i ].slide.h === indices.h && storage[ id ].data[ i ].slide.v === indices.v && storage[ id ].data[ i ].slide.f === indices.f ) {691data = storage[ id ].data[ i ];692return data;693}694}695var page = Number( Reveal.getCurrentSlide().getAttribute('data-pdf-page-number') );696//console.log( indices, Reveal.getCurrentSlide() );697storage[ id ].data.push( {698slide: indices,699page,700events: [],701duration: 0702} );703data = storage[ id ].data[ storage[ id ].data.length - 1 ];704return data;705}706707/**708* Returns maximum duration of slide playback for both modes709*/710function getSlideDuration( indices ) {711if ( !indices ) indices = slideIndices;712var duration = 0;713for ( var id = 0; id < 2; id++ ) {714for ( var i = 0; i < storage[ id ].data.length; i++ ) {715if ( storage[ id ].data[ i ].slide.h === indices.h && storage[ id ].data[ i ].slide.v === indices.v && storage[ id ].data[ i ].slide.f === indices.f ) {716duration = Math.max( duration, storage[ id ].data[ i ].duration );717break;718}719}720}721//console.log( duration );722return duration;723}724725/*****************************************************************726727******************************************************************/728var printMode = ( /print-pdf/gi ).test( window.location.search );729//console.log("createPrintout" + printMode)730731function addPageNumbers() {732// determine page number for printouts with fragments serialised733var slides = Reveal.getSlides();734var page = 0;735for ( var i=0; i < slides.length; i++) {736slides[i].setAttribute('data-pdf-page-number',page.toString());737// add number of fragments without fragment indices738var count = slides[i].querySelectorAll('.fragment:not([data-fragment-index])').length;739var fragments = slides[i].querySelectorAll('.fragment[data-fragment-index]');740for ( var j=0; j < fragments.length; j++) {741// increasenumber of fragments by highest fragment index (which start at 0)742if ( Number(fragments[j].getAttribute('data-fragment-index')) + 1 > count ) {743count = Number(fragments[j].getAttribute('data-fragment-index')) + 1;744}745}746page += count + 1;747}748}749750function createPrintout() {751//console.warn(Reveal.getTotalSlides(),Reveal.getSlidesElement());752if ( storage[ 1 ].data.length == 0 ) return;753console.log( 'Create printout(s) for ' + storage[ 1 ].data.length + " slides" );754drawingCanvas[ 0 ].container.style.opacity = 0; // do not print notes canvas755drawingCanvas[ 0 ].container.style.visibility = 'hidden';756757var patImg = new Image();758patImg.onload = function () {759var slides = Reveal.getSlides();760//console.log(slides);761for ( var i = storage[ 1 ].data.length - 1; i >= 0; i-- ) {762console.log( 'Create printout for slide ' + storage[ 1 ].data[ i ].slide.h + '.' + storage[ 1 ].data[ i ].slide.v );763var slideData = getSlideData( storage[ 1 ].data[ i ].slide, 1 );764var drawings = createDrawings( slideData, patImg );765addDrawings( slides[storage[ 1 ].data[ i ].page], drawings );766767}768// Reveal.sync();769};770patImg.src = background[ 1 ];771}772773774function cloneCanvas( oldCanvas ) {775//create a new canvas776var newCanvas = document.createElement( 'canvas' );777var context = newCanvas.getContext( '2d' );778//set dimensions779newCanvas.width = oldCanvas.width;780newCanvas.height = oldCanvas.height;781//apply the old canvas to the new one782context.drawImage( oldCanvas, 0, 0 );783//return the new canvas784return newCanvas;785}786787function getCanvas( template, container, board ) {788var idx = container.findIndex( element => element.board === board );789if ( idx === -1 ) {790var canvas = cloneCanvas( template );791if ( !container.length ) {792idx = 0;793container.push( {794board,795canvas796} );797} else if ( board < container[ 0 ].board ) {798idx = 0;799container.unshift( {800board,801canvas802} );803} else if ( board > container[ container.length - 1 ].board ) {804idx = container.length;805container.push( {806board,807canvas808} );809}810}811812return container[ idx ].canvas;813}814815function createDrawings( slideData, patImg ) {816var width = Reveal.getConfig().width;817var height = Reveal.getConfig().height;818var scale = 1;819var xOffset = 0;820var yOffset = 0;821if ( width != storage[ 1 ].width || height != storage[ 1 ].height ) {822scale = Math.min( width / storage[ 1 ].width, height / storage[ 1 ].height );823xOffset = ( width - storage[ 1 ].width * scale ) / 2;824yOffset = ( height - storage[ 1 ].height * scale ) / 2;825}826mode = 1;827board = 0;828// console.log( 'Create printout(s) for slide ', slideData );829830var drawings = [];831var template = document.createElement( 'canvas' );832template.width = width;833template.height = height;834835var imgCtx = template.getContext( '2d' );836imgCtx.fillStyle = imgCtx.createPattern( patImg, 'repeat' );837imgCtx.rect( 0, 0, width, height );838imgCtx.fill();839840for ( var j = 0; j < slideData.events.length; j++ ) {841switch ( slideData.events[ j ].type ) {842case 'draw':843draw[ 1 ]( getCanvas( template, drawings, board ).getContext( '2d' ),844xOffset + slideData.events[ j ].x1 * scale,845yOffset + slideData.events[ j ].y1 * scale,846xOffset + slideData.events[ j ].x2 * scale,847yOffset + slideData.events[ j ].y2 * scale,848yOffset + slideData.events[ j ].color849);850break;851case 'erase':852eraseWithSponge( getCanvas( template, drawings, board ).getContext( '2d' ),853xOffset + slideData.events[ j ].x * scale,854yOffset + slideData.events[ j ].y * scale855);856break;857case 'selectboard':858selectBoard( slideData.events[ j ].board );859break;860case 'clear':861getCanvas( template, drawings, board ).getContext( '2d' ).clearRect( 0, 0, width, height );862getCanvas( template, drawings, board ).getContext( '2d' ).fill();863break;864default:865break;866}867}868869drawings = drawings.sort( ( a, b ) => a.board > b.board && 1 || -1 );870871mode = 0;872873return drawings;874}875876function addDrawings( slide, drawings ) {877var parent = slide.parentElement.parentElement;878var nextSlide = slide.parentElement.nextElementSibling;879880for ( var i = 0; i < drawings.length; i++ ) {881var newPDFPage = document.createElement( 'div' );882newPDFPage.classList.add( 'pdf-page' );883newPDFPage.style.height = Reveal.getConfig().height;884newPDFPage.append( drawings[ i ].canvas );885//console.log("Add drawing", newPDFPage);886if ( nextSlide != null ) {887parent.insertBefore( newPDFPage, nextSlide );888} else {889parent.append( newPDFPage );890}891}892}893894/*****************************************************************895** Drawings896******************************************************************/897898function drawWithBoardmarker( context, fromX, fromY, toX, toY, colorIdx ) {899if ( colorIdx == undefined ) colorIdx = color[ mode ];900context.lineWidth = boardmarkerWidth;901context.lineCap = 'round';902context.strokeStyle = boardmarkers[ colorIdx ].color;903context.beginPath();904context.moveTo( fromX, fromY );905context.lineTo( toX, toY );906context.stroke();907}908909function drawWithChalk( context, fromX, fromY, toX, toY, colorIdx ) {910if ( colorIdx == undefined ) colorIdx = color[ mode ];911var brushDiameter = chalkWidth;912context.lineWidth = brushDiameter;913context.lineCap = 'round';914context.fillStyle = chalks[ colorIdx ].color; // 'rgba(255,255,255,0.5)';915context.strokeStyle = chalks[ colorIdx ].color;916917var opacity = 1.0;918context.strokeStyle = context.strokeStyle.replace( /[\d\.]+\)$/g, opacity + ')' );919context.beginPath();920context.moveTo( fromX, fromY );921context.lineTo( toX, toY );922context.stroke();923// Chalk Effect924var length = Math.round( Math.sqrt( Math.pow( toX - fromX, 2 ) + Math.pow( toY - fromY, 2 ) ) / ( 5 / brushDiameter ) );925var xUnit = ( toX - fromX ) / length;926var yUnit = ( toY - fromY ) / length;927for ( var i = 0; i < length; i++ ) {928if ( chalkEffect > ( Math.random() * 0.9 ) ) {929var xCurrent = fromX + ( i * xUnit );930var yCurrent = fromY + ( i * yUnit );931var xRandom = xCurrent + ( Math.random() - 0.5 ) * brushDiameter * 1.2;932var yRandom = yCurrent + ( Math.random() - 0.5 ) * brushDiameter * 1.2;933context.clearRect( xRandom, yRandom, Math.random() * 2 + 2, Math.random() + 1 );934}935}936}937938function eraseWithSponge( context, x, y ) {939context.save();940context.beginPath();941context.arc( x + eraser.radius, y + eraser.radius, eraser.radius, 0, 2 * Math.PI, false );942context.clip();943context.clearRect( x - 1, y - 1, eraser.radius * 2 + 2, eraser.radius * 2 + 2 );944context.restore();945if ( mode == 1 && grid ) {946redrawGrid( x + eraser.radius, y + eraser.radius, eraser.radius );947}948}949950951/**952* Show an overlay for the chalkboard.953*/954function showChalkboard() {955//console.log("showChalkboard");956drawingCanvas[ 1 ].container.style.opacity = 1;957drawingCanvas[ 1 ].container.style.visibility = 'visible';958mode = 1;959}960961962/**963* Closes open chalkboard.964*/965function closeChalkboard() {966drawingCanvas[ 1 ].container.style.opacity = 0;967drawingCanvas[ 1 ].container.style.visibility = 'hidden';968lastX = null;969lastY = null;970mode = 0;971}972973/**974* Clear current canvas.975*/976function clearCanvas( id ) {977if ( id == 0 ) clearTimeout( slidechangeTimeout );978drawingCanvas[ id ].context.clearRect( 0, 0, drawingCanvas[ id ].width, drawingCanvas[ id ].height );979if ( id == 1 && grid ) drawGrid();980}981982/**983* Draw grid on background984*/985function drawGrid() {986var context = drawingCanvas[ 1 ].context;987988drawingCanvas[ 1 ].scale = Math.min( drawingCanvas[ 1 ].width / storage[ 1 ].width, drawingCanvas[ 1 ].height / storage[ 1 ].height );989drawingCanvas[ 1 ].xOffset = ( drawingCanvas[ 1 ].width - storage[ 1 ].width * drawingCanvas[ 1 ].scale ) / 2;990drawingCanvas[ 1 ].yOffset = ( drawingCanvas[ 1 ].height - storage[ 1 ].height * drawingCanvas[ 1 ].scale ) / 2;991992var scale = drawingCanvas[ 1 ].scale;993var xOffset = drawingCanvas[ 1 ].xOffset;994var yOffset = drawingCanvas[ 1 ].yOffset;995996var distance = grid.distance * scale;997998var fromX = drawingCanvas[ 1 ].width / 2 - distance / 2 - Math.floor( ( drawingCanvas[ 1 ].width - distance ) / 2 / distance ) * distance;999for ( var x = fromX; x < drawingCanvas[ 1 ].width; x += distance ) {1000context.beginPath();1001context.lineWidth = grid.width * scale;1002context.lineCap = 'round';1003context.fillStyle = grid.color;1004context.strokeStyle = grid.color;1005context.moveTo( x, 0 );1006context.lineTo( x, drawingCanvas[ 1 ].height );1007context.stroke();1008}1009var fromY = drawingCanvas[ 1 ].height / 2 - distance / 2 - Math.floor( ( drawingCanvas[ 1 ].height - distance ) / 2 / distance ) * distance;10101011for ( var y = fromY; y < drawingCanvas[ 1 ].height; y += distance ) {1012context.beginPath();1013context.lineWidth = grid.width * scale;1014context.lineCap = 'round';1015context.fillStyle = grid.color;1016context.strokeStyle = grid.color;1017context.moveTo( 0, y );1018context.lineTo( drawingCanvas[ 1 ].width, y );1019context.stroke();1020}1021}10221023function redrawGrid( centerX, centerY, diameter ) {1024var context = drawingCanvas[ 1 ].context;10251026drawingCanvas[ 1 ].scale = Math.min( drawingCanvas[ 1 ].width / storage[ 1 ].width, drawingCanvas[ 1 ].height / storage[ 1 ].height );1027drawingCanvas[ 1 ].xOffset = ( drawingCanvas[ 1 ].width - storage[ 1 ].width * drawingCanvas[ 1 ].scale ) / 2;1028drawingCanvas[ 1 ].yOffset = ( drawingCanvas[ 1 ].height - storage[ 1 ].height * drawingCanvas[ 1 ].scale ) / 2;10291030var scale = drawingCanvas[ 1 ].scale;1031var xOffset = drawingCanvas[ 1 ].xOffset;1032var yOffset = drawingCanvas[ 1 ].yOffset;10331034var distance = grid.distance * scale;10351036var fromX = drawingCanvas[ 1 ].width / 2 - distance / 2 - Math.floor( ( drawingCanvas[ 1 ].width - distance ) / 2 / distance ) * distance;10371038for ( var x = fromX + distance * Math.ceil( ( centerX - diameter - fromX ) / distance ); x <= fromX + distance * Math.floor( ( centerX + diameter - fromX ) / distance ); x += distance ) {1039context.beginPath();1040context.lineWidth = grid.width * scale;1041context.lineCap = 'round';1042context.fillStyle = grid.color;1043context.strokeStyle = grid.color;1044context.moveTo( x, centerY - Math.sqrt( diameter * diameter - ( centerX - x ) * ( centerX - x ) ) );1045context.lineTo( x, centerY + Math.sqrt( diameter * diameter - ( centerX - x ) * ( centerX - x ) ) );1046context.stroke();1047}1048var fromY = drawingCanvas[ 1 ].height / 2 - distance / 2 - Math.floor( ( drawingCanvas[ 1 ].height - distance ) / 2 / distance ) * distance;1049for ( var y = fromY + distance * Math.ceil( ( centerY - diameter - fromY ) / distance ); y <= fromY + distance * Math.floor( ( centerY + diameter - fromY ) / distance ); y += distance ) {1050context.beginPath();1051context.lineWidth = grid.width * scale;1052context.lineCap = 'round';1053context.fillStyle = grid.color;1054context.strokeStyle = grid.color;1055context.moveTo( centerX - Math.sqrt( diameter * diameter - ( centerY - y ) * ( centerY - y ) ), y );1056context.lineTo( centerX + Math.sqrt( diameter * diameter - ( centerY - y ) * ( centerY - y ) ), y );1057context.stroke();1058}1059}10601061/**1062* Set the color1063*/1064function setColor( index, record ) {1065// protect against out of bounds (this could happen when1066// replaying events recorded with different color settings).1067if ( index >= pens[ mode ].length ) index = 0;10681069color[ mode ] = index;10701071if ( color[ mode ] < 0 ) {1072// use eraser1073changeCursor( drawingCanvas[ mode ].canvas, sponge );1074}1075else {1076changeCursor( drawingCanvas[ mode ].canvas, pens[ mode ][ color[ mode ] ] );1077}1078}10791080/**1081* Set the board1082*/1083function selectBoard( boardIdx, record ) {1084//console.log("Set board",boardIdx);1085if ( board == boardIdx ) return;10861087board = boardIdx;1088redrawChalkboard( boardIdx );1089if ( record ) {1090recordEvent( { type: 'selectboard' } );1091}1092}10931094function redrawChalkboard( boardIdx ) {1095clearCanvas( 1 );1096var slideData = getSlideData( slideIndices, 1 );1097var index = 0;1098var play = ( boardIdx == 0 );1099while ( index < slideData.events.length && slideData.events[ index ].time < Date.now() - slideStart ) {1100if ( boardIdx == slideData.events[ index ].board ) {1101playEvent( 1, slideData.events[ index ], Date.now() - slideStart );1102}11031104index++;1105}1106}110711081109/**1110* Forward cycle color1111*/1112function cycleColorNext() {1113color[ mode ] = ( color[ mode ] + 1 ) % pens[ mode ].length;1114return color[ mode ];1115}11161117/**1118* Backward cycle color1119*/1120function cycleColorPrev() {1121color[ mode ] = ( color[ mode ] + ( pens[ mode ].length - 1 ) ) % pens[ mode ].length;1122return color[ mode ];1123}11241125/*****************************************************************1126** Broadcast1127******************************************************************/11281129var eventQueue = [];11301131document.addEventListener( 'received', function ( message ) {1132if ( message.content && message.content.sender == 'chalkboard-plugin' ) {1133// add message to queue1134eventQueue.push( message );1135console.log( JSON.stringify( message ) );1136}1137if ( eventQueue.length == 1 ) processQueue();1138} );11391140function processQueue() {1141// take first message from queue1142var message = eventQueue.shift();11431144// synchronize time with seminar host1145slideStart = Date.now() - message.content.timestamp;1146// set status1147if ( mode < message.content.mode ) {1148// open chalkboard1149showChalkboard();1150} else if ( mode > message.content.mode ) {1151// close chalkboard1152closeChalkboard();1153}1154if ( board != message.content.board ) {1155board = message.content.board;1156redrawChalkboard( board );1157};11581159switch ( message.content.type ) {1160case 'showChalkboard':1161showChalkboard();1162break;1163case 'closeChalkboard':1164closeChalkboard();1165break;1166case 'erase':1167erasePoint( message.content.x, message.content.y );1168break;1169case 'draw':1170drawSegment( message.content.fromX, message.content.fromY, message.content.toX, message.content.toY, message.content.color );1171break;1172case 'clear':1173clearSlide();1174break;1175case 'selectboard':1176selectBoard( message.content.board, true );1177break;1178case 'resetSlide':1179resetSlideDrawings();1180break;1181case 'init':1182storage = message.content.storage;1183for ( var id = 0; id < 2; id++ ) {1184drawingCanvas[ id ].scale = Math.min( drawingCanvas[ id ].width / storage[ id ].width, drawingCanvas[ id ].height / storage[ id ].height );1185drawingCanvas[ id ].xOffset = ( drawingCanvas[ id ].width - storage[ id ].width * drawingCanvas[ id ].scale ) / 2;1186drawingCanvas[ id ].yOffset = ( drawingCanvas[ id ].height - storage[ id ].height * drawingCanvas[ id ].scale ) / 2;1187}1188clearCanvas( 0 );1189clearCanvas( 1 );1190if ( !playback ) {1191slidechangeTimeout = setTimeout( startPlayback, transition, getSlideDuration(), 0 );1192}1193if ( mode == 1 && message.content.mode == 0 ) {1194setTimeout( closeChalkboard, transition + 50 );1195}1196if ( mode == 0 && message.content.mode == 1 ) {1197setTimeout( showChalkboard, transition + 50 );1198}1199mode = message.content.mode;1200board = message.content.board;1201break;1202default:1203break;1204}12051206// continue with next message if queued1207if ( eventQueue.length > 0 ) {1208processQueue();1209} else {1210storageChanged();1211}1212}12131214document.addEventListener( 'welcome', function ( user ) {1215// broadcast storage1216var message = new CustomEvent( messageType );1217message.content = {1218sender: 'chalkboard-plugin',1219recipient: user.id,1220type: 'init',1221timestamp: Date.now() - slideStart,1222storage: storage,1223mode,1224board1225};1226document.dispatchEvent( message );1227} );12281229/*****************************************************************1230** Playback1231******************************************************************/12321233document.addEventListener( 'seekplayback', function ( event ) {1234//console.log('event seekplayback ' + event.timestamp);1235stopPlayback();1236if ( !playback || event.timestamp == 0 ) {1237// in other cases startplayback fires after seeked1238startPlayback( event.timestamp );1239}1240//console.log('seeked');1241} );124212431244document.addEventListener( 'startplayback', function ( event ) {1245//console.log('event startplayback ' + event.timestamp);1246stopPlayback();1247playback = true;1248startPlayback( event.timestamp );1249} );12501251document.addEventListener( 'stopplayback', function ( event ) {1252//console.log('event stopplayback ' + (Date.now() - slideStart) );1253playback = false;1254stopPlayback();1255} );12561257document.addEventListener( 'startrecording', function ( event ) {1258//console.log('event startrecording ' + event.timestamp);1259startRecording();1260} );126112621263function startRecording() {1264resetSlide( true );1265slideStart = Date.now();1266}12671268function startPlayback( timestamp, finalMode ) {1269//console.log("playback " + timestamp );1270slideStart = Date.now() - timestamp;1271closeChalkboard();1272mode = 0;1273board = 0;1274for ( var id = 0; id < 2; id++ ) {1275clearCanvas( id );1276var slideData = getSlideData( slideIndices, id );1277//console.log( timestamp +" / " + JSON.stringify(slideData));1278var index = 0;1279while ( index < slideData.events.length && slideData.events[ index ].time < ( Date.now() - slideStart ) ) {1280playEvent( id, slideData.events[ index ], timestamp );1281index++;1282}12831284while ( playback && index < slideData.events.length ) {1285timeouts[ id ].push( setTimeout( playEvent, slideData.events[ index ].time - ( Date.now() - slideStart ), id, slideData.events[ index ], timestamp ) );1286index++;1287}1288}1289//console.log("Mode: " + finalMode + "/" + mode );1290if ( finalMode != undefined ) {1291mode = finalMode;1292}1293if ( mode == 1 ) showChalkboard();1294//console.log("playback (ok)");12951296};12971298function stopPlayback() {1299//console.log("stopPlayback");1300//console.log("Timeouts: " + timeouts[0].length + "/"+ timeouts[1].length);1301for ( var id = 0; id < 2; id++ ) {1302for ( var i = 0; i < timeouts[ id ].length; i++ ) {1303clearTimeout( timeouts[ id ][ i ] );1304}1305timeouts[ id ] = [];1306}1307};13081309function playEvent( id, event, timestamp ) {1310//console.log( timestamp +" / " + JSON.stringify(event));1311//console.log( id + ": " + timestamp +" / " + event.time +" / " + event.type +" / " + mode );1312switch ( event.type ) {1313case 'open':1314if ( timestamp <= event.time ) {1315showChalkboard();1316} else {1317mode = 1;1318}13191320break;1321case 'close':1322if ( timestamp < event.time ) {1323closeChalkboard();1324} else {1325mode = 0;1326}1327break;1328case 'clear':1329clearCanvas( id );1330break;1331case 'selectboard':1332selectBoard( event.board );1333break;1334case 'draw':1335drawLine( id, event, timestamp );1336break;1337case 'erase':1338eraseCircle( id, event, timestamp );1339break;1340}1341};13421343function drawLine( id, event, timestamp ) {1344var ctx = drawingCanvas[ id ].context;1345var scale = drawingCanvas[ id ].scale;1346var xOffset = drawingCanvas[ id ].xOffset;1347var yOffset = drawingCanvas[ id ].yOffset;1348draw[ id ]( ctx, xOffset + event.x1 * scale, yOffset + event.y1 * scale, xOffset + event.x2 * scale, yOffset + event.y2 * scale, event.color );1349};13501351function eraseCircle( id, event, timestamp ) {1352var ctx = drawingCanvas[ id ].context;1353var scale = drawingCanvas[ id ].scale;1354var xOffset = drawingCanvas[ id ].xOffset;1355var yOffset = drawingCanvas[ id ].yOffset;13561357eraseWithSponge( ctx, xOffset + event.x * scale, yOffset + event.y * scale );1358};13591360function startErasing( x, y ) {1361drawing = false;1362erasing = true;1363erasePoint( x, y );1364}13651366function erasePoint( x, y ) {1367var ctx = drawingCanvas[ mode ].context;1368var scale = drawingCanvas[ mode ].scale;1369var xOffset = drawingCanvas[ mode ].xOffset;1370var yOffset = drawingCanvas[ mode ].yOffset;13711372recordEvent( {1373type: 'erase',1374x,1375y1376} );13771378if (1379x * scale + xOffset > 0 &&1380y * scale + yOffset > 0 &&1381x * scale + xOffset < drawingCanvas[ mode ].width &&1382y * scale + yOffset < drawingCanvas[ mode ].height1383) {1384eraseWithSponge( ctx, x * scale + xOffset, y * scale + yOffset );1385}1386}13871388function stopErasing() {1389erasing = false;1390}13911392function startDrawing( x, y ) {1393drawing = true;13941395var ctx = drawingCanvas[ mode ].context;1396var scale = drawingCanvas[ mode ].scale;1397var xOffset = drawingCanvas[ mode ].xOffset;1398var yOffset = drawingCanvas[ mode ].yOffset;1399lastX = x * scale + xOffset;1400lastY = y * scale + yOffset;1401}14021403function drawSegment( fromX, fromY, toX, toY, colorIdx ) {1404var ctx = drawingCanvas[ mode ].context;1405var scale = drawingCanvas[ mode ].scale;1406var xOffset = drawingCanvas[ mode ].xOffset;1407var yOffset = drawingCanvas[ mode ].yOffset;14081409recordEvent( {1410type: 'draw',1411color: colorIdx,1412x1: fromX,1413y1: fromY,1414x2: toX,1415y2: toY1416} );14171418if (1419fromX * scale + xOffset > 0 &&1420fromY * scale + yOffset > 0 &&1421fromX * scale + xOffset < drawingCanvas[ mode ].width &&1422fromY * scale + yOffset < drawingCanvas[ mode ].height &&1423toX * scale + xOffset > 0 &&1424toY * scale + yOffset > 0 &&1425toX * scale + xOffset < drawingCanvas[ mode ].width &&1426toY * scale + yOffset < drawingCanvas[ mode ].height1427) {1428draw[ mode ]( ctx, fromX * scale + xOffset, fromY * scale + yOffset, toX * scale + xOffset, toY * scale + yOffset, colorIdx );1429}1430}14311432function stopDrawing() {1433drawing = false;1434}143514361437/*****************************************************************1438** User interface1439******************************************************************/14401441function setupCanvasEvents( canvas ) {1442// TODO: check all touchevents1443canvas.addEventListener( 'touchstart', function ( evt ) {1444evt.preventDefault();1445//console.log("Touch start");1446if ( !readOnly && evt.target.getAttribute( 'data-chalkboard' ) == mode ) {1447var scale = drawingCanvas[ mode ].scale;1448var xOffset = drawingCanvas[ mode ].xOffset;1449var yOffset = drawingCanvas[ mode ].yOffset;14501451var touch = evt.touches[ 0 ];1452mouseX = touch.pageX;1453mouseY = touch.pageY;1454if ( color[ mode ] < 0 ) {1455startErasing( ( mouseX - xOffset ) / scale, ( mouseY - yOffset ) / scale);1456}1457else {1458startDrawing( ( mouseX - xOffset ) / scale, ( mouseY - yOffset ) / scale );1459}1460}1461}, passiveSupported ? {1462passive: false1463} : false );14641465canvas.addEventListener( 'touchmove', function ( evt ) {1466evt.preventDefault();1467//console.log("Touch move");1468if ( drawing || erasing ) {1469var scale = drawingCanvas[ mode ].scale;1470var xOffset = drawingCanvas[ mode ].xOffset;1471var yOffset = drawingCanvas[ mode ].yOffset;14721473var touch = evt.touches[ 0 ];1474mouseX = touch.pageX;1475mouseY = touch.pageY;14761477if ( drawing ) {1478drawSegment( ( lastX - xOffset ) / scale, ( lastY - yOffset ) / scale, ( mouseX - xOffset ) / scale, ( mouseY - yOffset ) / scale, color[ mode ] );1479// broadcast1480var message = new CustomEvent( messageType );1481message.content = {1482sender: 'chalkboard-plugin',1483type: 'draw',1484timestamp: Date.now() - slideStart,1485mode,1486board,1487fromX: ( lastX - xOffset ) / scale,1488fromY: ( lastY - yOffset ) / scale,1489toX: ( mouseX - xOffset ) / scale,1490toY: ( mouseY - yOffset ) / scale,1491color: color[ mode ]1492};1493document.dispatchEvent( message );14941495lastX = mouseX;1496lastY = mouseY;1497} else {1498erasePoint( ( mouseX - xOffset ) / scale, ( mouseY - yOffset ) / scale );1499// broadcast1500var message = new CustomEvent( messageType );1501message.content = {1502sender: 'chalkboard-plugin',1503type: 'erase',1504timestamp: Date.now() - slideStart,1505mode,1506board,1507x: ( mouseX - xOffset ) / scale,1508y: ( mouseY - yOffset ) / scale1509};1510document.dispatchEvent( message );1511}15121513}1514}, false );151515161517canvas.addEventListener( 'touchend', function ( evt ) {1518evt.preventDefault();1519stopDrawing();1520stopErasing();1521}, false );15221523canvas.addEventListener( 'mousedown', function ( evt ) {1524evt.preventDefault();1525if ( !readOnly && evt.target.getAttribute( 'data-chalkboard' ) == mode ) {1526//console.log( "mousedown: " + evt.button );1527var scale = drawingCanvas[ mode ].scale;1528var xOffset = drawingCanvas[ mode ].xOffset;1529var yOffset = drawingCanvas[ mode ].yOffset;15301531mouseX = evt.pageX;1532mouseY = evt.pageY;15331534if ( color[ mode ] < 0 || evt.button == 2 || evt.button == 1 ) {1535if ( color[ mode ] >= 0 ) {1536// show sponge1537changeCursor( drawingCanvas[ mode ].canvas, sponge );1538}1539startErasing( ( mouseX - xOffset ) / scale, ( mouseY - yOffset ) / scale );1540// broadcast1541var message = new CustomEvent( messageType );1542message.content = {1543sender: 'chalkboard-plugin',1544type: 'erase',1545timestamp: Date.now() - slideStart,1546mode,1547board,1548x: ( mouseX - xOffset ) / scale,1549y: ( mouseY - yOffset ) / scale1550};1551document.dispatchEvent( message );1552} else {1553startDrawing( ( mouseX - xOffset ) / scale, ( mouseY - yOffset ) / scale );1554}1555}1556} );15571558canvas.addEventListener( 'mousemove', function ( evt ) {1559evt.preventDefault();1560//console.log("Mouse move");15611562var scale = drawingCanvas[ mode ].scale;1563var xOffset = drawingCanvas[ mode ].xOffset;1564var yOffset = drawingCanvas[ mode ].yOffset;15651566mouseX = evt.pageX;1567mouseY = evt.pageY;15681569if ( drawing || erasing ) {1570var scale = drawingCanvas[ mode ].scale;1571var xOffset = drawingCanvas[ mode ].xOffset;1572var yOffset = drawingCanvas[ mode ].yOffset;15731574mouseX = evt.pageX;1575mouseY = evt.pageY;15761577if ( drawing ) {1578drawSegment( ( lastX - xOffset ) / scale, ( lastY - yOffset ) / scale, ( mouseX - xOffset ) / scale, ( mouseY - yOffset ) / scale, color[ mode ] );1579// broadcast1580var message = new CustomEvent( messageType );1581message.content = {1582sender: 'chalkboard-plugin',1583type: 'draw',1584timestamp: Date.now() - slideStart,1585mode,1586board,1587fromX: ( lastX - xOffset ) / scale,1588fromY: ( lastY - yOffset ) / scale,1589toX: ( mouseX - xOffset ) / scale,1590toY: ( mouseY - yOffset ) / scale,1591color: color[ mode ]1592};1593document.dispatchEvent( message );15941595lastX = mouseX;1596lastY = mouseY;1597} else {1598erasePoint( ( mouseX - xOffset ) / scale, ( mouseY - yOffset ) / scale );1599// broadcast1600var message = new CustomEvent( messageType );1601message.content = {1602sender: 'chalkboard-plugin',1603type: 'erase',1604timestamp: Date.now() - slideStart,1605mode,1606board,1607x: ( mouseX - xOffset ) / scale,1608y: ( mouseY - yOffset ) / scale1609};1610document.dispatchEvent( message );1611}16121613}1614} );161516161617canvas.addEventListener( 'mouseup', function ( evt ) {1618evt.preventDefault();1619if ( color[ mode ] >= 0 ) {1620changeCursor( drawingCanvas[ mode ].canvas, pens[ mode ][ color[ mode ] ] );1621}1622if ( drawing || erasing ) {1623stopDrawing();1624stopErasing();1625}1626} );1627}16281629function resize() {1630//console.log("resize");1631// Resize the canvas and draw everything again1632var timestamp = Date.now() - slideStart;1633if ( !playback ) {1634timestamp = getSlideDuration();1635}16361637//console.log( drawingCanvas[0].scale + "/" + drawingCanvas[0].xOffset + "/" +drawingCanvas[0].yOffset );1638for ( var id = 0; id < 2; id++ ) {1639drawingCanvas[ id ].width = window.innerWidth;1640drawingCanvas[ id ].height = window.innerHeight;1641drawingCanvas[ id ].canvas.width = drawingCanvas[ id ].width;1642drawingCanvas[ id ].canvas.height = drawingCanvas[ id ].height;1643drawingCanvas[ id ].context.canvas.width = drawingCanvas[ id ].width;1644drawingCanvas[ id ].context.canvas.height = drawingCanvas[ id ].height;16451646drawingCanvas[ id ].scale = Math.min( drawingCanvas[ id ].width / storage[ id ].width, drawingCanvas[ id ].height / storage[ id ].height );1647drawingCanvas[ id ].xOffset = ( drawingCanvas[ id ].width - storage[ id ].width * drawingCanvas[ id ].scale ) / 2;1648drawingCanvas[ id ].yOffset = ( drawingCanvas[ id ].height - storage[ id ].height * drawingCanvas[ id ].scale ) / 2;1649//console.log( drawingCanvas[id].scale + "/" + drawingCanvas[id].xOffset + "/" +drawingCanvas[id].yOffset );1650}1651//console.log( window.innerWidth + "/" + window.innerHeight);1652startPlayback( timestamp, mode, true );1653}16541655Reveal.addEventListener( 'pdf-ready', function ( evt ) {1656// console.log( "Create printouts when ready" );1657whenLoaded( createPrintout );1658});16591660Reveal.addEventListener( 'ready', function ( evt ) {1661//console.log('ready');1662if ( !printMode ) {1663window.addEventListener( 'resize', resize );16641665slideStart = Date.now() - getSlideDuration();1666slideIndices = Reveal.getIndices();1667if ( !playback ) {1668startPlayback( getSlideDuration(), 0 );1669}1670if ( Reveal.isAutoSliding() ) {1671var event = new CustomEvent( 'startplayback' );1672event.timestamp = 0;1673document.dispatchEvent( event );1674}1675updateStorage();1676whenReady( addPageNumbers );1677}1678} );1679Reveal.addEventListener( 'slidechanged', function ( evt ) {1680// clearTimeout( slidechangeTimeout );1681//console.log('slidechanged');1682if ( !printMode ) {1683slideStart = Date.now() - getSlideDuration();1684slideIndices = Reveal.getIndices();1685closeChalkboard();1686board = 0;1687clearCanvas( 0 );1688clearCanvas( 1 );1689if ( !playback ) {1690slidechangeTimeout = setTimeout( startPlayback, transition, getSlideDuration(), 0 );1691}1692if ( Reveal.isAutoSliding() ) {1693var event = new CustomEvent( 'startplayback' );1694event.timestamp = 0;1695document.dispatchEvent( event );1696}1697}1698} );1699Reveal.addEventListener( 'fragmentshown', function ( evt ) {1700// clearTimeout( slidechangeTimeout );1701//console.log('fragmentshown');1702if ( !printMode ) {1703slideStart = Date.now() - getSlideDuration();1704slideIndices = Reveal.getIndices();1705closeChalkboard();1706board = 0;1707clearCanvas( 0 );1708clearCanvas( 1 );1709if ( Reveal.isAutoSliding() ) {1710var event = new CustomEvent( 'startplayback' );1711event.timestamp = 0;1712document.dispatchEvent( event );1713} else if ( !playback ) {1714startPlayback( getSlideDuration(), 0 );1715// closeChalkboard();1716}1717}1718} );1719Reveal.addEventListener( 'fragmenthidden', function ( evt ) {1720// clearTimeout( slidechangeTimeout );1721//console.log('fragmenthidden');1722if ( !printMode ) {1723slideStart = Date.now() - getSlideDuration();1724slideIndices = Reveal.getIndices();1725closeChalkboard();1726board = 0;1727clearCanvas( 0 );1728clearCanvas( 1 );1729if ( Reveal.isAutoSliding() ) {1730document.dispatchEvent( new CustomEvent( 'stopplayback' ) );1731} else if ( !playback ) {1732startPlayback( getSlideDuration() );1733closeChalkboard();1734}1735}1736} );17371738Reveal.addEventListener( 'autoslideresumed', function ( evt ) {1739//console.log('autoslideresumed');1740var event = new CustomEvent( 'startplayback' );1741event.timestamp = 0;1742document.dispatchEvent( event );1743} );1744Reveal.addEventListener( 'autoslidepaused', function ( evt ) {1745//console.log('autoslidepaused');1746document.dispatchEvent( new CustomEvent( 'stopplayback' ) );17471748// advance to end of slide1749// closeChalkboard();1750startPlayback( getSlideDuration(), 0 );1751} );17521753function toggleNotesCanvas() {1754if ( !readOnly ) {1755if ( mode == 1 ) {1756toggleChalkboard();1757notescanvas.style.background = background[ 0 ]; //'rgba(255,0,0,0.5)';1758notescanvas.style.pointerEvents = 'auto';1759}1760else {1761if ( notescanvas.style.pointerEvents != 'none' ) {1762// hide notes canvas1763if ( colorButtons ) {1764notescanvas.querySelector( '.palette' ).style.visibility = 'hidden';1765}1766notescanvas.style.background = 'rgba(0,0,0,0)';1767notescanvas.style.pointerEvents = 'none';1768}1769else {1770// show notes canvas1771if ( colorButtons ) {1772notescanvas.querySelector( '.palette' ).style.visibility = 'visible';1773}1774notescanvas.style.background = background[ 0 ]; //'rgba(255,0,0,0.5)';1775notescanvas.style.pointerEvents = 'auto';17761777var idx = 0;1778if ( color[ mode ] ) {1779idx = color[ mode ];1780}17811782setColor( idx, true );1783}1784}1785}1786};17871788function toggleChalkboard() {1789//console.log("toggleChalkboard " + mode);1790if ( mode == 1 ) {1791if ( !readOnly ) {1792recordEvent( { type: 'close' } );1793}1794closeChalkboard();17951796// broadcast1797var message = new CustomEvent( messageType );1798message.content = {1799sender: 'chalkboard-plugin',1800type: 'closeChalkboard',1801timestamp: Date.now() - slideStart,1802mode: 0,1803board1804};1805document.dispatchEvent( message );180618071808} else {1809showChalkboard();1810if ( !readOnly ) {1811recordEvent( { type: 'open' } );1812// broadcast1813var message = new CustomEvent( messageType );1814message.content = {1815sender: 'chalkboard-plugin',1816type: 'showChalkboard',1817timestamp: Date.now() - slideStart,1818mode: 1,1819board1820};1821document.dispatchEvent( message );18221823var idx = 0;18241825if ( rememberColor[ mode ] ) {1826idx = color[ mode ];1827}18281829setColor( idx, true );1830}1831}1832};18331834function clearSlide() {1835recordEvent( { type: 'clear' } );1836clearCanvas( mode );1837}18381839function clear() {1840if ( !readOnly ) {1841clearSlide();1842// broadcast1843var message = new CustomEvent( messageType );1844message.content = {1845sender: 'chalkboard-plugin',1846type: 'clear',1847timestamp: Date.now() - slideStart,1848mode,1849board1850};1851document.dispatchEvent( message );1852}1853};18541855function colorIndex( idx ) {1856if ( !readOnly ) {1857setColor( idx, true );1858}1859}18601861function colorNext() {1862if ( !readOnly ) {1863let idx = cycleColorNext();1864setColor( idx, true );1865}1866}18671868function colorPrev() {1869if ( !readOnly ) {1870let idx = cycleColorPrev();1871setColor( idx, true );1872}1873}18741875function resetSlideDrawings() {1876slideStart = Date.now();1877closeChalkboard();18781879clearCanvas( 0 );1880clearCanvas( 1 );18811882mode = 1;1883var slideData = getSlideData();1884slideData.duration = 0;1885slideData.events = [];1886mode = 0;1887var slideData = getSlideData();1888slideData.duration = 0;1889slideData.events = [];18901891updateStorage();1892}18931894function resetSlide( force ) {1895var ok = force || confirm( "Please confirm to delete chalkboard drawings on this slide!" );1896if ( ok ) {1897//console.log("resetSlide ");1898stopPlayback();1899resetSlideDrawings();1900// broadcast1901var message = new CustomEvent( messageType );1902message.content = {1903sender: 'chalkboard-plugin',1904type: 'resetSlide',1905timestamp: Date.now() - slideStart,1906mode,1907board1908};1909document.dispatchEvent( message );1910}1911};19121913function resetStorage( force ) {1914var ok = force || confirm( "Please confirm to delete all chalkboard drawings!" );1915if ( ok ) {1916stopPlayback();1917slideStart = Date.now();1918clearCanvas( 0 );1919clearCanvas( 1 );1920if ( mode == 1 ) {1921closeChalkboard();1922}19231924storage = [ {1925width: Reveal.getConfig().width,1926height: Reveal.getConfig().height,1927data: []1928},1929{1930width: Reveal.getConfig().width,1931height: Reveal.getConfig().height,1932data: []1933}1934];19351936if ( config.storage ) {1937sessionStorage.setItem( config.storage, null )1938}1939// broadcast1940var message = new CustomEvent( messageType );1941message.content = {1942sender: 'chalkboard-plugin',1943type: 'init',1944timestamp: Date.now() - slideStart,1945storage,1946mode,1947board1948};1949document.dispatchEvent( message );1950}1951};19521953this.toggleNotesCanvas = toggleNotesCanvas;1954this.toggleChalkboard = toggleChalkboard;1955this.colorIndex = colorIndex;1956this.colorNext = colorNext;1957this.colorPrev = colorPrev;1958this.clear = clear;1959this.reset = resetSlide;1960this.resetAll = resetStorage;1961this.download = downloadData;1962this.updateStorage = updateStorage;1963this.getData = getData;1964this.configure = configure;196519661967for ( var key in keyBindings ) {1968if ( keyBindings[ key ] ) {1969Reveal.addKeyBinding( keyBindings[ key ], RevealChalkboard[ key ] );1970}1971};19721973return this;1974};197519761977