Вход Регистрация
Файл: framework/web/js/source/jquery.autocomplete.js
Строк: 758
<?php
/*
 * jQuery Autocomplete plugin 1.1
 *
 * Modified for Yii Framework:
 * - Renamed "autocomplete" to "legacyautocomplete".
 * - Fixed IE8 problems (mario.ffranco).
 *
 * Copyright (c) 2009 Jörn Zaefferer
 *
 * Dual licensed under the MIT and GPL licenses:
 *   http://www.opensource.org/licenses/mit-license.php
 *   http://www.gnu.org/licenses/gpl.html
 *
 * Revision: $Id: jquery.autocomplete.js 15 2009-08-22 10:30:27Z joern.zaefferer $
 */

;(function($) {
    
$.fn.
extend({
    
legacyautocomplete: function(urlOrDataoptions) {
        var 
isUrl typeof urlOrData == "string";
        
options = $.extend({}, $.Autocompleter.defaults, {
            
urlisUrl urlOrData null,
            
dataisUrl null urlOrData,
            
delayisUrl ? $.Autocompleter.defaults.delay 10,
            
maxoptions && !options.scroll 10 150
        
}, options);
        
        
// if highlight is set to false, replace it with a do-nothing function
        
options.highlight options.highlight || function(value) { return value; };
        
        
// if the formatMatch option is not specified, then use formatItem for backwards compatibility
        
options.formatMatch options.formatMatch || options.formatItem;
        
        return 
this.each(function() {
            new $.
Autocompleter(thisoptions);
        });
    },
    
result: function(handler) {
        return 
this.bind("result"handler);
    },
    
search: function(handler) {
        return 
this.trigger("search", [handler]);
    },
    
flushCache: function() {
        return 
this.trigger("flushCache");
    },
    
setOptions: function(options){
        return 
this.trigger("setOptions", [options]);
    },
    
unautocomplete: function() {
        return 
this.trigger("unautocomplete");
    }
});

$.
Autocompleter = function(inputoptions) {

    var 
KEY = {
        
UP38,
        
DOWN40,
        
DEL46,
        
TAB9,
        RETURN: 
13,
        
ESC27,
        
COMMA188,
        
PAGEUP33,
        
PAGEDOWN34,
        
BACKSPACE8
    
};

    
// Create $ object for input element
    
var $input = $(input).attr("autocomplete""off").addClass(options.inputClass);

    var 
timeout;
    var 
previousValue "";
    var 
cache = $.Autocompleter.Cache(options);
    var 
hasFocus 0;
    var 
lastKeyPressCode;
    var 
config = {
        
mouseDownOnSelectfalse
    
};
    var 
select = $.Autocompleter.Select(optionsinputselectCurrentconfig);
    
    var 
blockSubmit;
    
    
// prevent form submit in opera when selecting with return key
    
$.browser.opera && $(input.form).bind("submit.autocomplete", function() {
        if (
blockSubmit) {
            
blockSubmit false;
            return 
false;
        }
    });
    
    
// only opera doesn't trigger keydown multiple times while pressed, others don't work with keypress at all
    
$input.bind(($.browser.opera "keypress" "keydown") + ".autocomplete", function(event) {
        
// a keypress means the input has focus
        // avoids issue where input had focus before the autocomplete was applied
        
hasFocus 1;
        
// track last key pressed
        
lastKeyPressCode event.keyCode;
        switch(
event.keyCode) {
        
            case 
KEY.UP:
                
event.preventDefault();
                if ( 
select.visible() ) {
                    
select.prev();
                } else {
                    
onChange(0true);
                }
                break;
                
            case 
KEY.DOWN:
                
event.preventDefault();
                if ( 
select.visible() ) {
                    
select.next();
                } else {
                    
onChange(0true);
                }
                break;
                
            case 
KEY.PAGEUP:
                
event.preventDefault();
                if ( 
select.visible() ) {
                    
select.pageUp();
                } else {
                    
onChange(0true);
                }
                break;
                
            case 
KEY.PAGEDOWN:
                
event.preventDefault();
                if ( 
select.visible() ) {
                    
select.pageDown();
                } else {
                    
onChange(0true);
                }
                break;
            
            
// matches also semicolon
            
case options.multiple && $.trim(options.multipleSeparator) == "," && KEY.COMMA:
            case 
KEY.TAB:
            case 
KEY.RETURN:
                if( 
selectCurrent() ) {
                    
// stop default to prevent a form submit, Opera needs special handling
                    
event.preventDefault();
                    
blockSubmit true;
                    return 
false;
                }
                break;
                
            case 
KEY.ESC:
                
select.hide();
                break;
                
            default:
                
clearTimeout(timeout);
                
timeout setTimeout(onChangeoptions.delay);
                break;
        }
    }).
focus(function(){
        
// track whether the field has focus, we shouldn't process any
        // results if the field no longer has focus
        
hasFocus++;
    }).
blur(function() {
        
hasFocus 0;
        if (!
config.mouseDownOnSelect) {
            
hideResults();
        }
    }).
click(function() {
        
// show select when clicking in a focused field
        
if ( hasFocus++ > && !select.visible() ) {
            
onChange(0true);
        }
    }).
bind("search", function() {
        
// TODO why not just specifying both arguments?
        
var fn = (arguments.length 1) ? arguments[1] : null;
        function 
findValueCallback(qdata) {
            var 
result;
            if( 
data && data.length ) {
                for (var 
i=0data.lengthi++) {
                    if( 
data[i].result.toLowerCase() == q.toLowerCase() ) {
                        
result data[i];
                        break;
                    }
                }
            }
            if( 
typeof fn == "function" ) fn(result);
            else 
$input.trigger("result"result && [result.dataresult.value]);
        }
        $.
each(trimWords($input.val()), function(ivalue) {
            
request(valuefindValueCallbackfindValueCallback);
        });
    }).
bind("flushCache", function() {
        
cache.flush();
    }).
bind("setOptions", function() {
        $.
extend(optionsarguments[1]);
        
// if we've updated the data, repopulate
        
if ( "data" in arguments[1] )
            
cache.populate();
    }).
bind("unautocomplete", function() {
        
select.unbind();
        
$input.unbind();
        $(
input.form).unbind(".autocomplete");
    });
    
    
    function 
selectCurrent() {
        var 
selected select.selected();
        if( !
selected )
            return 
false;
        
        var 
selected.result;
        
previousValue v;
        
        if ( 
options.multiple ) {
            var 
words trimWords($input.val());
            if ( 
words.length ) {
                var 
seperator options.multipleSeparator.length;
                var 
cursorAt = $(input).selection().start;
                var 
wordAtprogress 0;
                $.
each(words, function(iword) {
                    
progress += word.length;
                    if (
cursorAt <= progress) {
                        
wordAt i;
                        
// Following return caused IE8 to set cursor to the start of the line.
                        // return false;
                    
}
                    
progress += seperator;
                });
                
words[wordAt] = v;
                
// TODO this should set the cursor to the right position, but it gets overriden somewhere
                //$.Autocompleter.Selection(input, progress + seperator, progress + seperator);
                
words.joinoptions.multipleSeparator );
            }
            
+= options.multipleSeparator;
        }
        
        
$input.val(v);
        
hideResultsNow();
        
$input.trigger("result", [selected.dataselected.value]);
        return 
true;
    }
    
    function 
onChange(crapskipPrevCheck) {
        if( 
lastKeyPressCode == KEY.DEL ) {
            
select.hide();
            return;
        }
        
        var 
currentValue $input.val();
        
        if ( !
skipPrevCheck && currentValue == previousValue )
            return;
        
        
previousValue currentValue;
        
        
currentValue lastWord(currentValue);
        if ( 
currentValue.length >= options.minChars) {
            
$input.addClass(options.loadingClass);
            if (!
options.matchCase)
                
currentValue currentValue.toLowerCase();
            
request(currentValuereceiveDatahideResultsNow);
        } else {
            
stopLoading();
            
select.hide();
        }
    };
    
    function 
trimWords(value) {
        if (!
value)
            return [
""];
        if (!
options.multiple)
            return [$.
trim(value)];
        return $.
map(value.split(options.multipleSeparator), function(word) {
            return $.
trim(value).length ? $.trim(word) : null;
        });
    }
    
    function 
lastWord(value) {
        if ( !
options.multiple )
            return 
value;
        var 
words trimWords(value);
        if (
words.length == 1
            return 
words[0];
        var 
cursorAt = $(input).selection().start;
        if (
cursorAt == value.length) {
            
words trimWords(value)
        } else {
            
words trimWords(value.replace(value.substring(cursorAt), ""));
        }
        return 
words[words.length 1];
    }
    
    
// fills in the input box w/the first match (assumed to be the best match)
    // q: the term entered
    // sValue: the first matching result
    
function autoFill(qsValue){
        
// autofill in the complete box w/the first match as long as the user hasn't entered in more data
        // if the last user key pressed was backspace, don't autofill
        
if( options.autoFill && (lastWord($input.val()).toLowerCase() == q.toLowerCase()) && lastKeyPressCode != KEY.BACKSPACE ) {
            
// fill in the value (keep the case the user has typed)
            
$input.val($input.val() + sValue.substring(lastWord(previousValue).length));
            
// select the portion of the value not typed by the user (so the next character will erase)
            
$(input).selection(previousValue.lengthpreviousValue.length sValue.length);
        }
    };

    function 
hideResults() {
        
clearTimeout(timeout);
        
timeout setTimeout(hideResultsNow200);
    };

    function 
hideResultsNow() {
        var 
wasVisible select.visible();
        
select.hide();
        
clearTimeout(timeout);
        
stopLoading();
        if (
options.mustMatch) {
            
// call search and run callback
            
$input.search(
                function (
result){
                    
// if no value found, clear the input box
                    
if( !result ) {
                        if (
options.multiple) {
                            var 
words trimWords($input.val()).slice(0, -1);
                            
$input.valwords.join(options.multipleSeparator) + (words.length options.multipleSeparator "") );
                        }
                        else {
                            
$input.val"" );
                            
$input.trigger("result"null);
                        }
                    }
                }
            );
        }
    };

    function 
receiveData(qdata) {
        if ( 
data && data.length && hasFocus ) {
            
stopLoading();
            
select.display(dataq);
            
autoFill(qdata[0].value);
            
select.show();
        } else {
            
hideResultsNow();
        }
    };

    function 
request(termsuccessfailure) {
        if (!
options.matchCase)
            
term term.toLowerCase();
        var 
data cache.load(term);
        
// recieve the cached data
        
if (data && data.length) {
            
success(termdata);
        
// if an AJAX url has been supplied, try loading the data now
        
} else if( (typeof options.url == "string") && (options.url.length 0) ){
            
            var 
extraParams = {
                
timestamp: +new Date()
            };
            $.
each(options.extraParams, function(keyparam) {
                
extraParams[key] = typeof param == "function" param() : param;
            });
            
            $.
ajax({
                
// try to leverage ajaxQueue plugin to abort previous requests
                
mode"abort",
                
// limit abortion to this input
                
port"autocomplete" input.name,
                
dataTypeoptions.dataType,
                
urloptions.url,
                
data: $.extend({
                    
qlastWord(term),
                    
limitoptions.max
                
}, extraParams),
                
success: function(data) {
                    var 
parsed options.parse && options.parse(data) || parse(data);
                    
cache.add(termparsed);
                    
success(termparsed);
                }
            });
        } else {
            
// if we have a failure, we need to empty the list -- this prevents the the [TAB] key from selecting the last successful match
            
select.emptyList();
            
failure(term);
        }
    };
    
    function 
parse(data) {
        var 
parsed = [];
        var 
rows data.split("n");
        for (var 
i=0rows.lengthi++) {
            var 
row = $.trim(rows[i]);
            if (
row) {
                
row row.split("|");
                
parsed[parsed.length] = {
                    
datarow,
                    
valuerow[0],
                    
resultoptions.formatResult && options.formatResult(rowrow[0]) || row[0]
                };
            }
        }
        return 
parsed;
    };

    function 
stopLoading() {
        
$input.removeClass(options.loadingClass);
    };

};

$.
Autocompleter.defaults = {
    
inputClass"ac_input",
    
resultsClass"ac_results",
    
loadingClass"ac_loading",
    
minChars1,
    
delay400,
    
matchCasefalse,
    
matchSubsettrue,
    
matchContainsfalse,
    
cacheLength10,
    
max100,
    
mustMatchfalse,
    
extraParams: {},
    
selectFirsttrue,
    
formatItem: function(row) { return row[0]; },
    
formatMatchnull,
    
autoFillfalse,
    
width0,
    
multiplefalse,
    
multipleSeparator", ",
    
highlight: function(valueterm) {
        return 
value.replace(new RegExp("(?![^&;]+;)(?!<[^<>]*)(" term.replace(/([^$()[]{}*.+?|\])/gi"\$1") + ")(?![^<>]*>)(?![^&;]+;)""gi"), "<strong>$1</strong>");
    },
    
scrolltrue,
    
scrollHeight180
};

$.
Autocompleter.Cache = function(options) {

    var 
data = {};
    var 
length 0;
    
    function 
matchSubset(ssub) {
        if (!
options.matchCase
            
s.toLowerCase();
        var 
s.indexOf(sub);
        if (
options.matchContains == "word"){
            
s.toLowerCase().search("\b" sub.toLowerCase());
        }
        if (
== -1) return false;
        return 
== || options.matchContains;
    };
    
    function 
add(qvalue) {
        if (
length options.cacheLength){
            
flush();
        }
        if (!
data[q]){ 
            
length++;
        }
        
data[q] = value;
    }
    
    function 
populate(){
        if( !
options.data ) return false;
        
// track the matches
        
var stMatchSets = {},
            
nullData 0;

        
// no url was specified, we need to adjust the cache length to make sure it fits the local data store
        
if( !options.url options.cacheLength 1;
        
        
// track all options for minChars = 0
        
stMatchSets[""] = [];
        
        
// loop through the array and create a lookup structure
        
for ( var 0ol options.data.lengtholi++ ) {
            var 
rawValue options.data[i];
            
// if rawValue is a string, make an array otherwise just reference the array
            
rawValue = (typeof rawValue == "string") ? [rawValue] : rawValue;
            
            var 
value options.formatMatch(rawValuei+1options.data.length);
            if ( 
value === false )
                continue;
                
            var 
firstChar value.charAt(0).toLowerCase();
            
// if no lookup array for this character exists, look it up now
            
if( !stMatchSets[firstChar] ) 
                
stMatchSets[firstChar] = [];

            
// if the match is a string
            
var row = {
                
valuevalue,
                
datarawValue,
                
resultoptions.formatResult && options.formatResult(rawValue) || value
            
};
            
            
// push the current match into the set list
            
stMatchSets[firstChar].push(row);

            
// keep track of minChars zero items
            
if ( nullData++ < options.max ) {
                
stMatchSets[""].push(row);
            }
        };

        
// add the data items to the cache
        
$.each(stMatchSets, function(ivalue) {
            
// increase the cache size
            
options.cacheLength++;
            
// add to the cache
            
add(ivalue);
        });
    }
    
    
// populate any existing data
    
setTimeout(populate25);
    
    function 
flush(){
        
data = {};
        
length 0;
    }
    
    return {
        
flushflush,
        
addadd,
        
populatepopulate,
        
load: function(q) {
            if (!
options.cacheLength || !length)
                return 
null;
            
/* 
             * if dealing w/local data and matchContains than we must make sure
             * to loop through all the data collections looking for matches
             */
            
if( !options.url && options.matchContains ){
                
// track all matches
                
var csub = [];
                
// loop through all the data grids for matches
                
for( var k in data ){
                    
// don't search through the stMatchSets[""] (minChars: 0) cache
                    // this prevents duplicates
                    
if( k.length ){
                        var 
data[k];
                        $.
each(c, function(ix) {
                            
// if we've got a match, add it to the array
                            
if (matchSubset(x.valueq)) {
                                
csub.push(x);
                            }
                        });
                    }
                }                
                return 
csub;
            } else 
            
// if the exact item exists, use it
            
if (data[q]){
                return 
data[q];
            } else
            if (
options.matchSubset) {
                for (var 
q.length 1>= options.minCharsi--) {
                    var 
data[q.substr(0i)];
                    if (
c) {
                        var 
csub = [];
                        $.
each(c, function(ix) {
                            if (
matchSubset(x.valueq)) {
                                
csub[csub.length] = x;
                            }
                        });
                        return 
csub;
                    }
                }
            }
            return 
null;
        }
    };
};

$.
Autocompleter.Select = function (optionsinputselectconfig) {
    var 
CLASSES = {
        
ACTIVE"ac_over"
    
};
    
    var 
listItems,
        
active = -1,
        
data,
        
term "",
        
needsInit true,
        
element,
        list;
    
    
// Create results
    
function init() {
        if (!
needsInit)
            return;
        
element = $("<div/>")
        .
hide()
        .
addClass(options.resultsClass)
        .
css("position""absolute")
        .
appendTo(document.body);
    
        list = $(
"<ul/>").appendTo(element).mouseover( function(event) {
            if(
target(event).nodeName && target(event).nodeName.toUpperCase() == 'LI') {
                
active = $("li", list).removeClass(CLASSES.ACTIVE).index(target(event));
                $(
target(event)).addClass(CLASSES.ACTIVE);            
            }
        }).
click(function(event) {
            $(
target(event)).addClass(CLASSES.ACTIVE);
            
select();
            
// TODO provide option to avoid setting focus again after selection? useful for cleanup-on-focus
            
input.focus();
            return 
false;
        }).
mousedown(function() {
            
config.mouseDownOnSelect true;
        }).
mouseup(function() {
            
config.mouseDownOnSelect false;
        });
        
        if( 
options.width )
            
element.css("width"options.width);
            
        
needsInit false;
    } 
    
    function 
target(event) {
        var 
element event.target;
        while(
element && element.tagName != "LI")
            
element element.parentNode;
        
// more fun with IE, sometimes event.target is empty, just ignore it then
        
if(!element)
            return [];
        return 
element;
    }

    function 
moveSelect(step) {
        
listItems.slice(activeactive 1).removeClass(CLASSES.ACTIVE);
        
movePosition(step);
        var 
activeItem listItems.slice(activeactive 1).addClass(CLASSES.ACTIVE);
        if(
options.scroll) {
            var 
offset 0;
            
listItems.slice(0active).each(function() {
                
offset += this.offsetHeight;
            });
            if((
offset activeItem[0].offsetHeight - list.scrollTop()) > list[0].clientHeight) {
                list.
scrollTop(offset activeItem[0].offsetHeight - list.innerHeight());
            } else if(
offset < list.scrollTop()) {
                list.
scrollTop(offset);
            }
        }
    };
    
    function 
movePosition(step) {
        
active += step;
        if (
active 0) {
            
active listItems.size() - 1;
        } else if (
active >= listItems.size()) {
            
active 0;
        }
    }
    
    function 
limitNumberOfItems(available) {
        return 
options.max && options.max available
            
options.max
            
available;
    }
    
    function 
fillList() {
        list.empty();
        var 
max limitNumberOfItems(data.length);
        for (var 
i=0maxi++) {
            if (!
data[i])
                continue;
            var 
formatted options.formatItem(data[i].datai+1maxdata[i].valueterm);
            if ( 
formatted === false )
                continue;
            var 
li = $("<li/>").htmloptions.highlight(formattedterm) ).addClass(i%== "ac_even" "ac_odd").appendTo(list)[0];
            $.
data(li"ac_data"data[i]);
        }
        
listItems = list.find("li");
        if ( 
options.selectFirst ) {
            
listItems.slice(01).addClass(CLASSES.ACTIVE);
            
active 0;
        }
        
// apply bgiframe if available
        
if ( $.fn.bgiframe )
            list.
bgiframe();
    }
    
    return {
        
display: function(dq) {
            
init();
            
data d;
            
term q;
            
fillList();
        },
        
next: function() {
            
moveSelect(1);
        },
        
prev: function() {
            
moveSelect(-1);
        },
        
pageUp: function() {
            if (
active != && active 0) {
                
moveSelect( -active );
            } else {
                
moveSelect(-8);
            }
        },
        
pageDown: function() {
            if (
active != listItems.size() - && active listItems.size()) {
                
moveSelectlistItems.size() - active );
            } else {
                
moveSelect(8);
            }
        },
        
hide: function() {
            
element && element.hide();
            
listItems && listItems.removeClass(CLASSES.ACTIVE);
            
active = -1;
        },
        
visible : function() {
            return 
element && element.is(":visible");
        },
        
current: function() {
            return 
this.visible() && (listItems.filter("." CLASSES.ACTIVE)[0] || options.selectFirst && listItems[0]);
        },
        
show: function() {
            var 
offset = $(input).offset();
            
element.css({
                
widthtypeof options.width == "string" || options.width options.width : $(input).width(),
                
topoffset.top input.offsetHeight,
                
leftoffset.left
            
}).show();
            if(
options.scroll) {
                list.
scrollTop(0);
                list.
css({
                    
maxHeightoptions.scrollHeight,
                    
overflow'auto'
                
});
                
                if($.
browser.msie && typeof document.body.style.maxHeight === "undefined") {
                    var 
listHeight 0;
                    
listItems.each(function() {
                        
listHeight += this.offsetHeight;
                    });
                    var 
scrollbarsVisible listHeight options.scrollHeight;
                    list.
css('height'scrollbarsVisible options.scrollHeight listHeight );
                    if (!
scrollbarsVisible) {
                        
// IE doesn't recalculate width when scrollbar disappears
                        
listItems.width( list.width() - parseInt(listItems.css("padding-left")) - parseInt(listItems.css("padding-right")) );
                    }
                }
                
            }
        },
        
selected: function() {
            var 
selected listItems && listItems.filter("." CLASSES.ACTIVE).removeClass(CLASSES.ACTIVE);
            return 
selected && selected.length && $.data(selected[0], "ac_data");
        },
        
emptyList: function (){
            list && list.empty();
        },
        
unbind: function() {
            
element && element.remove();
        }
    };
};

$.fn.
selection = function(startend) {
    if (
start !== undefined) {
        return 
this.each(function() {
            if( 
this.createTextRange ){
                var 
selRange this.createTextRange();
                if (
end === undefined || start == end) {
                    
selRange.move("character"start);
                    
selRange.select();
                } else {
                    
selRange.collapse(true);
                    
selRange.moveStart("character"start);
                    
selRange.moveEnd("character"end);
                    
selRange.select();
                }
            } else if( 
this.setSelectionRange ){
                
this.setSelectionRange(startend);
            } else if( 
this.selectionStart ){
                
this.selectionStart start;
                
this.selectionEnd end;
            }
        });
    }
    var 
field this[0];
    if ( 
field.createTextRange ) {
        var 
range document.selection.createRange(),
            
orig field.value,
            
teststring "<->",
            
textLength range.text.length;
        
range.text teststring;
        var 
caretAt field.value.indexOf(teststring);
        
field.value orig;
        
this.selection(caretAtcaretAt textLength);
        return {
            
startcaretAt,
            
endcaretAt textLength
        
}
    } else if( 
field.selectionStart !== undefined ){
        return {
            
startfield.selectionStart,
            
endfield.selectionEnd
        
}
    }
};

})(
jQuery);
?>
Онлайн: 0
Реклама