diff options
Diffstat (limited to 'present/js/controllers/jumptoslide.js')
| -rw-r--r-- | present/js/controllers/jumptoslide.js | 170 | 
1 files changed, 170 insertions, 0 deletions
| diff --git a/present/js/controllers/jumptoslide.js b/present/js/controllers/jumptoslide.js new file mode 100644 index 0000000..cf2de99 --- /dev/null +++ b/present/js/controllers/jumptoslide.js @@ -0,0 +1,170 @@ +/** + * Makes it possible to jump to a slide by entering its + * slide number or id. + */ +export default class JumpToSlide { + +	constructor( Reveal ) { + +		this.Reveal = Reveal; + +		this.onInput = this.onInput.bind( this ); +		this.onBlur = this.onBlur.bind( this ); +		this.onKeyDown = this.onKeyDown.bind( this ); + +	} + +	render() { + +		this.element = document.createElement( 'div' ); +		this.element.className = 'jump-to-slide'; + +    this.jumpInput = document.createElement( 'input' ); +    this.jumpInput.type = 'text'; +    this.jumpInput.className = 'jump-to-slide-input'; +    this.jumpInput.placeholder = 'Jump to slide'; +		this.jumpInput.addEventListener( 'input', this.onInput ); +		this.jumpInput.addEventListener( 'keydown', this.onKeyDown ); +		this.jumpInput.addEventListener( 'blur', this.onBlur ); + +    this.element.appendChild( this.jumpInput ); + +	} + +	show() { + +		this.indicesOnShow = this.Reveal.getIndices(); + +		this.Reveal.getRevealElement().appendChild( this.element ); +		this.jumpInput.focus(); + +	} + +	hide() { + +		if( this.isVisible() ) { +			this.element.remove(); +			this.jumpInput.value = ''; + +			clearTimeout( this.jumpTimeout ); +			delete this.jumpTimeout; +		} + +	} + +	isVisible() { + +		return !!this.element.parentNode; + +	} + +	/** +	 * Parses the current input and jumps to the given slide. +	 */ +	jump() { + +		clearTimeout( this.jumpTimeout ); +		delete this.jumpTimeout; + +		const query = this.jumpInput.value.trim( '' ); +		let indices = this.Reveal.location.getIndicesFromHash( query, { oneBasedIndex: true } ); + +		// If no valid index was found and the input query is a +		// string, fall back on a simple search +		if( !indices && /\S+/i.test( query ) && query.length > 1 ) { +			indices = this.search( query ); +		} + +		if( indices && query !== '' ) { +			this.Reveal.slide( indices.h, indices.v, indices.f ); +			return true; +		} +		else { +			this.Reveal.slide( this.indicesOnShow.h, this.indicesOnShow.v, this.indicesOnShow.f ); +			return false; +		} + +	} + +	jumpAfter( delay ) { + +		clearTimeout( this.jumpTimeout ); +		this.jumpTimeout = setTimeout( () => this.jump(), delay ); + +	} + +	/** +	 * A lofi search that looks for the given query in all +	 * of our slides and returns the first match. +	 */ +	search( query ) { + +		const regex = new RegExp( '\\b' + query.trim() + '\\b', 'i' ); + +		const slide = this.Reveal.getSlides().find( ( slide ) => { +			return regex.test( slide.innerText ); +		} ); + +		if( slide ) { +			return this.Reveal.getIndices( slide ); +		} +		else { +			return null; +		} + +	} + +	/** +	 * Reverts back to the slide we were on when jump to slide was +	 * invoked. +	 */ +	cancel() { + +		this.Reveal.slide( this.indicesOnShow.h, this.indicesOnShow.v, this.indicesOnShow.f ); +		this.hide(); + +	} + +	confirm() { + +		this.jump(); +		this.hide(); + +	} + +	destroy() { + +		this.jumpInput.removeEventListener( 'input', this.onInput ); +		this.jumpInput.removeEventListener( 'keydown', this.onKeyDown ); +		this.jumpInput.removeEventListener( 'blur', this.onBlur ); + +		this.element.remove(); + +	} + +	onKeyDown( event ) { + +		if( event.keyCode === 13 ) { +			this.confirm(); +		} +		else if( event.keyCode === 27 ) { +			this.cancel(); + +			event.stopImmediatePropagation(); +		} + +	} + +	onInput( event ) { + +		this.jumpAfter( 200 ); + +	} + +	onBlur() { + +		setTimeout( () => this.hide(), 1 ); + +	} + +}
\ No newline at end of file | 
