1/*
2 * Handles finding a text string anywhere in the slides and showing the next occurrence to the user
3 * by navigatating to that slide and highlighting it.
4 *
5 * By Jon Snyder <snyder.jon@gmail.com>, February 2013
6 */
7
8var RevealSearch = (function() {
9
10	var matchedSlides;
11	var currentMatchedIndex;
12	var searchboxDirty;
13	var myHilitor;
14
15// Original JavaScript code by Chirp Internet: www.chirp.com.au
16// Please acknowledge use of this code by including this header.
17// 2/2013 jon: modified regex to display any match, not restricted to word boundaries.
18
19function Hilitor(id, tag)
20{
21
22	var targetNode = document.getElementById(id) || document.body;
23	var hiliteTag = tag || "EM";
24	var skipTags = new RegExp("^(?:" + hiliteTag + "|SCRIPT|FORM)$");
25	var colors = ["#ff6", "#a0ffff", "#9f9", "#f99", "#f6f"];
26	var wordColor = [];
27	var colorIdx = 0;
28	var matchRegex = "";
29	var matchingSlides = [];
30
31	this.setRegex = function(input)
32	{
33		input = input.replace(/^[^\w]+|[^\w]+$/g, "").replace(/[^\w'-]+/g, "|");
34		matchRegex = new RegExp("(" + input + ")","i");
35	}
36
37	this.getRegex = function()
38	{
39		return matchRegex.toString().replace(/^\/\\b\(|\)\\b\/i$/g, "").replace(/\|/g, " ");
40	}
41
42	// recursively apply word highlighting
43	this.hiliteWords = function(node)
44	{
45		if(node == undefined || !node) return;
46		if(!matchRegex) return;
47		if(skipTags.test(node.nodeName)) return;
48
49		if(node.hasChildNodes()) {
50			for(var i=0; i < node.childNodes.length; i++)
51				this.hiliteWords(node.childNodes[i]);
52		}
53		if(node.nodeType == 3) { // NODE_TEXT
54			if((nv = node.nodeValue) && (regs = matchRegex.exec(nv))) {
55				//find the slide's section element and save it in our list of matching slides
56				var secnode = node;
57				while (secnode != null && secnode.nodeName != 'SECTION') {
58					secnode = secnode.parentNode;
59				}
60
61				var slideIndex = Reveal.getIndices(secnode);
62				var slidelen = matchingSlides.length;
63				var alreadyAdded = false;
64				for (var i=0; i < slidelen; i++) {
65					if ( (matchingSlides[i].h === slideIndex.h) && (matchingSlides[i].v === slideIndex.v) ) {
66						alreadyAdded = true;
67					}
68				}
69				if (! alreadyAdded) {
70					matchingSlides.push(slideIndex);
71				}
72
73				if(!wordColor[regs[0].toLowerCase()]) {
74					wordColor[regs[0].toLowerCase()] = colors[colorIdx++ % colors.length];
75				}
76
77				var match = document.createElement(hiliteTag);
78				match.appendChild(document.createTextNode(regs[0]));
79				match.style.backgroundColor = wordColor[regs[0].toLowerCase()];
80				match.style.fontStyle = "inherit";
81				match.style.color = "#000";
82
83				var after = node.splitText(regs.index);
84				after.nodeValue = after.nodeValue.substring(regs[0].length);
85				node.parentNode.insertBefore(match, after);
86			}
87		}
88	};
89
90	// remove highlighting
91	this.remove = function()
92	{
93		var arr = document.getElementsByTagName(hiliteTag);
94		while(arr.length && (el = arr[0])) {
95			el.parentNode.replaceChild(el.firstChild, el);
96		}
97	};
98
99	// start highlighting at target node
100	this.apply = function(input)
101	{
102		if(input == undefined || !input) return;
103		this.remove();
104		this.setRegex(input);
105		this.hiliteWords(targetNode);
106		return matchingSlides;
107	};
108
109}
110
111	function openSearch() {
112		//ensure the search term input dialog is visible and has focus:
113		var inputboxdiv = document.getElementById("searchinputdiv");
114		var inputbox = document.getElementById("searchinput");
115		inputboxdiv.style.display = "inline";
116		inputbox.focus();
117		inputbox.select();
118	}
119
120	function closeSearch() {
121		var inputboxdiv = document.getElementById("searchinputdiv");
122		inputboxdiv.style.display = "none";
123		if(myHilitor) myHilitor.remove();
124	}
125
126	function toggleSearch() {
127		var inputboxdiv = document.getElementById("searchinputdiv");
128		if (inputboxdiv.style.display !== "inline") {
129			openSearch();
130		}
131		else {
132			closeSearch();
133		}
134	}
135
136	function doSearch() {
137		//if there's been a change in the search term, perform a new search:
138		if (searchboxDirty) {
139			var searchstring = document.getElementById("searchinput").value;
140
141			if (searchstring === '') {
142				if(myHilitor) myHilitor.remove();
143				matchedSlides = null;
144			}
145			else {
146				//find the keyword amongst the slides
147				myHilitor = new Hilitor("slidecontent");
148				matchedSlides = myHilitor.apply(searchstring);
149				currentMatchedIndex = 0;
150			}
151		}
152
153		if (matchedSlides) {
154			//navigate to the next slide that has the keyword, wrapping to the first if necessary
155			if (matchedSlides.length && (matchedSlides.length <= currentMatchedIndex)) {
156				currentMatchedIndex = 0;
157			}
158			if (matchedSlides.length > currentMatchedIndex) {
159				Reveal.slide(matchedSlides[currentMatchedIndex].h, matchedSlides[currentMatchedIndex].v);
160				currentMatchedIndex++;
161			}
162		}
163	}
164
165	var dom = {};
166	dom.wrapper = document.querySelector( '.reveal' );
167
168	if( !dom.wrapper.querySelector( '.searchbox' ) ) {
169			var searchElement = document.createElement( 'div' );
170			searchElement.id = "searchinputdiv";
171			searchElement.classList.add( 'searchdiv' );
172			searchElement.style.position = 'absolute';
173			searchElement.style.top = '10px';
174			searchElement.style.right = '10px';
175			searchElement.style.zIndex = 10;
176			//embedded base64 search icon Designed by Sketchdock - http://www.sketchdock.com/:
177			searchElement.innerHTML = '<span><input type="search" id="searchinput" class="searchinput" style="vertical-align: top;"/><img src="" id="searchbutton" class="searchicon" style="vertical-align: top; margin-top: -1px;"/></span>';
178			dom.wrapper.appendChild( searchElement );
179	}
180
181	document.getElementById( 'searchbutton' ).addEventListener( 'click', function(event) {
182		doSearch();
183	}, false );
184
185	document.getElementById( 'searchinput' ).addEventListener( 'keyup', function( event ) {
186		switch (event.keyCode) {
187			case 13:
188				event.preventDefault();
189				doSearch();
190				searchboxDirty = false;
191				break;
192			default:
193				searchboxDirty = true;
194		}
195	}, false );
196
197	document.addEventListener( 'keydown', function( event ) {
198		if( event.key == "F" && (event.ctrlKey || event.metaKey) ) { //Control+Shift+f
199			event.preventDefault();
200			toggleSearch();
201		}
202	}, false );
203	if( window.Reveal ) Reveal.registerKeyboardShortcut( 'CTRL + Shift + F', 'Search' );
204	closeSearch();
205	return { open: openSearch };
206})();
207