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