/*
 * Copyright (c) 2009 Tom Coote (http://www.tomcoote.co.uk)
 * This is licensed under GPL (http://www.opensource.org/licenses/gpl-license.php) licenses.
 * Adjusted for Alysis by info.nl
 */

(function($){

	$.fn.jsonSuggest = function(searchData, settings,toonAlleUrl) {
		var defaults = {  
			minCharacters: 1,
			maxResults: undefined,
			wildCard: "*",
			caseSensitive: false,
			notCharacter: "!",
			maxHeight: 350,
			highlightMatches: false,
			onSelect: undefined,
			ajaxResults: false,
			width: undefined
			};  
		settings = $.extend(defaults, settings);  
        

		return this.each(function() {
			
			function regexEscape(txt, omit) {
				var specials = ['/', '.', '*', '+', '?', '|',
								'(', ')', '[', ']', '{', '}', '\\'];
				
				if (omit) {
					for (var i=0; i < specials.length; i++) {
						if (specials[i] === omit) { specials.splice(i,1); }
					}
				}
				
				var escapePatt = new RegExp('(\\' + specials.join('|\\') + ')', 'g');
				return txt.replace(escapePatt, '\\$1');
			}
			
			var obj = $(this),
				wildCardPatt = new RegExp(regexEscape(settings.wildCard || ''),'g'),
				results = $('<div/>'),
				currentSelection, pageX, pageY;
			
			// When an item has been selected then update the input box,
			// hide the results again and if set, call the onSelect function
			function selectResultItem(item) {

				
				if (typeof settings.onSelect === 'function') {
					settings.onSelect(item);
				}				
                obj.val(item.text);
				$(results).html('').hide();
                
			}

			// Used to get rid of the hover class on all result item elements in the
			// current set of results and add it only to the given element. We also
			// need to set the current selection to the given element here.
			 function setHoverClass(el) {
				$('li', results).removeClass('hoover');
				$(el).addClass('hoover');
				
				currentSelection = el;
			}
			
			// Build the results HTML based on an array of objects that matched
			// the search criteria, highlight the matches if feature is turned on in
			// the settings.
			function buildResults(resultObjects, sFilterTxt) {
				sFilterTxt = "(" + sFilterTxt + ")";
			
				var bOddRow = true, i, iFound = 0,
					filterPatt = settings.caseSensitive ? new RegExp(sFilterTxt, "g") : new RegExp(sFilterTxt, "ig");
					
				$(results).html('').hide();
                                
				
                var ul = $('<ul class="dd-suggestions"></ul>').show();
				
				for (i = 0; i < resultObjects.length; i += 1) {
                	
					var item = $('<li />'),
						text = resultObjects[i].text;
                        url = resultObjects[i].link;                       
						
					if (settings.highlightMatches === true) {
						text = text.replace(filterPatt, "<strong>$1</strong>");
					}
					
					$(item).append('<a href=\"' + url +'\" >' + text + '</a>');
					
					if (typeof resultObjects[i].extra === 'string') {
						$(item).append('<a href=\"' + url +'\" class="extra">' + resultObjects[i].extra + '</a>');
					}
					
					//if (typeof resultObjects[i].image === 'string') {
					//	$(item).prepend('<img src="' + resultObjects[i].image + '" />').
					//		append('<br style="clear:both;" />');
					//}
					
					 $(item).addClass('resultItem').
						//addClass((bOddRow) ? 'odd' : 'even').
						click(function(n) { return function() {
							selectResultItem(resultObjects[n]);						
						};}(i)).
						mouseover(function(el) { return function() { 
							setHoverClass(el); 
						};}(item)); 
					
					$(ul).append(item);
                    
				    
                    //bOddRow = !bOddRow;
					
					iFound += 1;
					if (typeof settings.maxResults === 'number' && iFound >= settings.maxResults) {
						break;
					}
                    //Last one is 'Toon alle' link
                    if((i == resultObjects.length-1) || (resultObjects.length == 0)){
                        var item = $('<li />'),
						text = 'Toon alle';
                        url = toonAlleUrl; 
                        $(item).append('<a href=\"' + url +'\" class=\"toonalle\" >' + text + '</a>');
                        					 $(item).addClass('resultItem').
						addClass('show-all').	
                            click(function(n) { return function() {
                             obj.val('Toon alle');
				            $(results).html('').hide();
							window.location = url;						
						    };}(i)).						
                        mouseover(function(el) { return function() { 
							setHoverClass(el); 
						};}(item)); 
					    $(ul).append(item);
                    }
				}
                if(resultObjects.length == 0){
                    var item = $('<li />'),
                    text = 'Toon alle';
                    url = toonAlleUrl; 
                    $(item).append('<a href=\"' + url +'\" class=\"toonalle\" >' + text + '</a>');
                    $(item).addClass('resultItem').
                    addClass('show-all').	
                        click(function(n) { return function() {
                            obj.val('Toon alle');
				            $(results).html('').hide();                        
							window.location = url;							
						};}(i)).                    
                        mouseover(function(el) { return function() { 
                        setHoverClass(el); 
                        };}(item)); 
                    $(ul).append(item);
                }
                
                
                $(results).prepend(ul);                
                              
         //       if(toonAlleUrl.length > 0){
         //          $(results).append('<p class=show-all><a href='+ toonAlleUrl + '>Toon allex</a></p>');
         //       }
         //       if(!(toonAlleUrl.length > 0)){
         //           $(results).append('<p class=show-all><a href=http://www.info.nl>Toon alley</a></p>');
         //       }  
                
                
				if ($('ul', results).length > 0) {               
                
					currentSelection = undefined;
					//$(results).show().css('height', 'auto');
					$(results).show();
                    
					//if ($(results).height() > settings.maxHeight) {
					//	$(results).css({'overflow': 'auto', 'height': settings.maxHeight + 'px'});
					//}
				}
			}
			
			// Prepare the search string based on the settings for this plugin,
			// run it against each item in the searchData and display any 
			// results on the page allowing selection by the user.
			function runSuggest(e) {	
				if (this.value.length < settings.minCharacters) {
					$(results).html('').hide();
					return false;
				}
				
				var resultObjects = [],
					sFilterTxt = (!settings.wildCard) ? regexEscape(this.value) : regexEscape(this.value, settings.wildCard).replace(wildCardPatt, '.*'),
					bMatch = true, 
					filterPatt, i;
						
				if (settings.notCharacter && sFilterTxt.indexOf(settings.notCharacter) === 0) {
					sFilterTxt = sFilterTxt.substr(settings.notCharacter.length,sFilterTxt.length);
					if (sFilterTxt.length > 0) { bMatch = false; }
				}
				sFilterTxt = sFilterTxt || '.*';
				sFilterTxt = settings.wildCard ? '^' + sFilterTxt : sFilterTxt;
				filterPatt = settings.caseSensitive ? new RegExp(sFilterTxt) : new RegExp(sFilterTxt,"i");
				
				// Get the results from the correct place. If settings.ajaxResults then results are retrieved from
				// an external function each time they are needed else they are retrieved from the data
				// given on contruction.
				if (settings.ajaxResults === true) {
					resultObjects = searchData(this.value, 	settings.wildCard, 
															settings.caseSensitive,
															settings.notCharacter);
															
					if (typeof resultObjects === 'string') {
						resultObjects = JSON.parse(resultObjects);
					}
				}
				else {
                    
					// Look for the required match against each single search data item. When the not
					// character is used we are looking for a false match. 
					for (i = 0; i < searchData.length; i += 1) {
						if (filterPatt.test(searchData[i].text) === bMatch) {
							resultObjects.push(searchData[i]);
						}
					}
				}
				
				buildResults(resultObjects, sFilterTxt);
			}
			
			// To call specific actions based on the keys pressed in the input
			// box. Special keys are up, down and return. All other keys
			// act as normal.
			function keyListener(e) {
                
				switch (e.keyCode) {
					case 13: // return key
                        
						$(currentSelection).triggerHandler('click');
                        
					
						return false;
					case 40: // down key              
                             
						if (typeof currentSelection === 'undefined') {                            
							currentSelection = $('li.resultItem:first', results).get(0);
						}
						else {
                        
							currentSelection = $(currentSelection).next().get(0);
						}
						
						setHoverClass(currentSelection);
						if (currentSelection) {
							$(results).scrollTop(currentSelection.offsetTop);
						}
						
						return false;
					case 38: // up key
						if (typeof currentSelection === 'undefined') {
							currentSelection = $('li.resultItem::last', results).get(0);

						}
						else {
							currentSelection = $(currentSelection).prev().get(0);
						}
						
						setHoverClass(currentSelection);
						if (currentSelection) {
							$(results).scrollTop(currentSelection.offsetTop);
						}
						
						return false;
					default:
						runSuggest.apply(this, [e]);
				}
			}
			
			// Prepare the input box to show suggest results by adding in the events
			// that will initiate the search and placing the element on the page
			// that will show the results.
			$(results).addClass('dd-quicksearch').hide();
				
			obj.after(results).
				keyup(keyListener).
				blur(function(e) {
					// We need to make sure we don't hide the result set
					// if the input blur event is called because of clicking on
					// a result item.
					var resPos = $(results).offset();
					resPos.bottom = resPos.top + $(results).height();
					resPos.right = resPos.left + $(results).width();
					
					if (pageY < resPos.top || pageY > resPos.bottom || pageX < resPos.left || pageX > resPos.right) {
						$(results).hide();
					}
				}).
				focus(function(e) {
					//$(results).css({
						//'top': (obj.position().top + obj.height() + 5) + 'px',
						//'left': obj.position().left + 'px'
					//});
				
					if ($('ul', results).length > 0) {
						$(results).show();
					}
				}).
				attr('autocomplete', 'off');
			$().mousemove(function(e) {
				pageX = e.pageX;
				pageY = e.pageY;
			});
			
			// Opera doesn't seem to assign a keyCode for the down
			// key on the keyup event. why?
			if ($.browser.opera) {
				obj.keydown(function(e) {
					if (e.keyCode === 40) { // up key
						return keyListener(e);
					}
				});
			}
			
			// Escape the not character if present so that it doesn't act in the regular expression
			settings.notCharacter = regexEscape(settings.notCharacter || '');
			
			// We need to get the javascript array type data from the searchData setting.
			// Setting can either be a string, already an array or a function that returns one
			// of those things. We only get this data if it isn't being provided using ajax on
			// each search
			if (!settings.ajaxResults) {
				if (typeof searchData === 'function') {
					searchData = searchData();
				}
				if (typeof searchData === 'string') {
					searchData = JSON.parse(searchData);
				}
			}
		});
	};

})(jQuery);
