Shuo/client/js/jquery.plugins.js

436 lines
9.3 KiB
JavaScript
Raw Normal View History

/*!
* inputHistory
* https://github.com/erming/inputHistory
*
* Copyright (c) 2014 Mattias Erming <mattias@mattiaserming.com>
* Licensed under the MIT License.
*
2014-04-28 18:05:19 +00:00
* Version 0.1.2
*/
(function($) {
$.fn.inputHistory = function(options) {
var settings = $.extend({
history: [],
submit: false,
}, options);
var self = this;
if (self.size() > 1) {
return self.each(function() {
$(this).inputHistory(options);
});
}
self.data('history', settings.history.concat(['']));
var i = 0;
self.on('keydown', function(e) {
var history = self.data('history');
var key = e.which;
switch (key) {
case 13: // Enter
if (self.val() != '') {
i = history.length;
history[i - 1] = self.val();
history.push('');
}
if (settings.submit) {
self.parents('form').eq(0).submit();
}
self.val('');
break;
case 38: // Up
case 40: // Down
history[i] = self.val();
if (key == 38 && i != 0) {
i--;
} else if (key == 40 && i < history.length - 1) {
i++;
}
self.val(history[i]);
break;
default:
return;
}
return false;
});
return this;
}
})(jQuery);
/*!
* scrollGlue
* https://github.com/erming/scrollGlue
*
* Copyright (c) 2014 Mattias Erming <mattias@mattiaserming.com>
* Licensed under the MIT License.
*
2014-05-13 22:47:35 +00:00
* Version 1.2.1
*/
(function($) {
$.fn.scrollGlue = function(options) {
var settings = $.extend({
2014-04-28 18:05:19 +00:00
disableManualScroll: false,
2014-05-03 17:21:11 +00:00
overflow: 'scroll',
scrollToBottom: true,
2014-04-28 18:05:19 +00:00
speed: 0
}, options);
2014-05-03 17:21:11 +00:00
var self = this;
if (self.size() > 1) {
return self.each(function() {
$(this).scrollGlue(options);
});
}
2014-05-03 17:21:11 +00:00
self.css('overflow-y', settings.overflow);
if (settings.scrollToBottom) {
self.scrollToBottom();
}
2014-05-03 17:21:11 +00:00
var timer;
var resizing = false;
$(window).on('resize', function() {
self.finish();
2014-05-03 17:21:11 +00:00
// This will prevent the scroll event from triggering
// while resizing the browser.
resizing = true;
clearTimeout(timer);
timer = setTimeout(function() {
resizing = false;
}, 100);
if (sticky) {
self.scrollToBottom();
}
});
2014-05-03 17:21:11 +00:00
2014-04-28 18:05:19 +00:00
var sticky = true;
self.on('scroll', function() {
2014-04-28 18:05:19 +00:00
if (settings.disableManualScroll) {
self.scrollToBottom();
2014-05-03 17:21:11 +00:00
} else if (!resizing) {
2014-04-28 18:05:19 +00:00
sticky = self.isScrollAtBottom();
}
});
self.trigger('scroll');
2014-05-03 17:21:11 +00:00
self.on('prepend append', function() {
if (sticky) {
self.scrollToBottom(settings.speed);
}
});
2014-05-03 17:21:11 +00:00
return this;
};
2014-04-24 18:11:45 +00:00
2014-05-03 17:21:11 +00:00
// Normally, these functions won't trigger any events.
// Lets override them.
var events = ['prepend', 'append'];
$.each(events, function(i, e) {
var fn = $.fn[e];
$.fn[e] = function() {
return fn.apply(this, arguments).trigger(e);
};
});
$.fn.isScrollAtBottom = function() {
if ((this.scrollTop() + this.outerHeight() + 1) >= this.prop('scrollHeight')) {
return true;
}
};
$.fn.scrollToBottom = function(speed) {
return this.each(function() {
$(this).finish().animate({scrollTop: this.scrollHeight}, speed || 0);
});
};
})(jQuery);
/*!
* tabComplete
* https://github.com/erming/tabComplete
*
* Copyright (c) 2014 Mattias Erming <mattias@mattiaserming.com>
* Licensed under the MIT License.
*
2014-05-13 22:47:35 +00:00
* Version 1.0.0-alpha2
*/
(function($) {
2014-05-13 22:47:35 +00:00
var defaults = {
after: "",
caseSensitive: false,
hint: true,
minLength: 1,
};
$.fn.tabComplete = function(args, options) {
var self = this;
2014-05-13 22:47:35 +00:00
options = $.extend(
{}, defaults, options
);
if (this.length > 1) {
return this.each(function() {
$(this).tabComplete(args, options);
});
}
2014-05-13 22:47:35 +00:00
if (options.hint) {
// Lets turn on hinting.
hint.call(self, "");
}
2014-05-13 22:47:35 +00:00
// Unbind namespace.
// This allows us to override the plugin if necessary.
this.unbind(".tabComplete");
var i = 0;
var words = [];
var last = "";
this.on("input.tabComplete", function(e) {
var input = self.val();
var word = input.split(/ |\n/).pop();
2014-05-13 22:47:35 +00:00
if (!word) {
i = 0;
2014-05-13 22:47:35 +00:00
words = [];
last = "";
} else if (typeof args === "function") {
// If the user supplies a function, invoke it
// and keep the result.
words = args(word);
} else {
// Otherwise, call the .match() function.
words = match(args, word, options.caseSensitive);
}
2014-05-13 22:47:35 +00:00
if (options.hint) {
2014-05-15 21:56:23 +00:00
if (word.length >= options.minLength && words.length) {
2014-05-13 22:47:35 +00:00
hint.call(self, words[0]);
} else {
// Clear hinting.
// This call is needed when using backspace.
hint.call(self, "");
}
}
});
this.on("keydown.tabComplete", function(e) {
var key = e.which;
if (key == 9) {
// Don't lose focus on tab click.
e.preventDefault();
// Get next match.
var word = words[i++ % words.length];
if (!word) {
return;
}
var input = self.val().trim();
last = last || input.split(/ |\n/).pop();
if (last.length < options.minLength) {
return;
}
self.val(
input.substr(0, input.lastIndexOf(last))
+ word
+ options.after
);
// Remember the word until next time.
last = word;
if (options.hint) {
// Turn off any additional hinting.
hint.call(self, "");
}
}
});
return this;
2014-05-13 22:47:35 +00:00
}
// Simple matching.
// Filter the array and return the items that begins with `word`.
function match(array, word, caseSensitive) {
return $.grep(
array,
function(w) {
if (caseSensitive) {
return !w.indexOf(word);
} else {
return !w.toLowerCase().indexOf(word.toLowerCase());
}
}
);
}
// Add input hinting.
// This works by creating a copy of the input and placing it behind
// the real input.
function hint(word) {
var input = this;
var clone = input.prev(".hint");
input.css({
backgroundColor: "transparent",
position: "relative",
});
// Lets create a clone of the input if it does
// not already exist.
if (!clone.length) {
input.wrap(
$("<div>").css({position: "relative"})
);
clone = input
.clone()
.prop("disabled", true)
.removeAttr("id name placeholder")
.addClass("hint")
.insertBefore(input);
clone.css({
position: "absolute",
});
}
var hint = "";
if (typeof word !== "undefined") {
var text = input.val();
2014-05-15 21:56:23 +00:00
hint = text + word.substr(text.split(/ |\n/).pop().length);
2014-05-13 22:47:35 +00:00
}
clone.val(hint);
}
})(jQuery);
/*!
* jQuery Cookie Plugin v1.4.0
* https://github.com/carhartl/jquery-cookie
*
* Copyright 2013 Klaus Hartl
* Released under the MIT license
*/
(function (factory) {
if (typeof define === 'function' && define.amd) {
// AMD. Register as anonymous module.
define(['jquery'], factory);
} else {
// Browser globals.
factory(jQuery);
}
}(function ($) {
var pluses = /\+/g;
function encode(s) {
return config.raw ? s : encodeURIComponent(s);
}
function decode(s) {
return config.raw ? s : decodeURIComponent(s);
}
function stringifyCookieValue(value) {
return encode(config.json ? JSON.stringify(value) : String(value));
}
function parseCookieValue(s) {
if (s.indexOf('"') === 0) {
// This is a quoted cookie as according to RFC2068, unescape...
s = s.slice(1, -1).replace(/\\"/g, '"').replace(/\\\\/g, '\\');
}
try {
// Replace server-side written pluses with spaces.
// If we can't decode the cookie, ignore it, it's unusable.
s = decodeURIComponent(s.replace(pluses, ' '));
} catch(e) {
return;
}
try {
// If we can't parse the cookie, ignore it, it's unusable.
return config.json ? JSON.parse(s) : s;
} catch(e) {}
}
function read(s, converter) {
var value = config.raw ? s : parseCookieValue(s);
return $.isFunction(converter) ? converter(value) : value;
}
var config = $.cookie = function (key, value, options) {
// Write
if (value !== undefined && !$.isFunction(value)) {
options = $.extend({}, config.defaults, options);
if (typeof options.expires === 'number') {
var days = options.expires, t = options.expires = new Date();
t.setDate(t.getDate() + days);
}
return (document.cookie = [
encode(key), '=', stringifyCookieValue(value),
options.expires ? '; expires=' + options.expires.toUTCString() : '', // use expires attribute, max-age is not supported by IE
options.path ? '; path=' + options.path : '',
options.domain ? '; domain=' + options.domain : '',
options.secure ? '; secure' : ''
].join(''));
}
// Read
var result = key ? undefined : {};
// To prevent the for loop in the first place assign an empty array
// in case there are no cookies at all. Also prevents odd result when
// calling $.cookie().
var cookies = document.cookie ? document.cookie.split('; ') : [];
for (var i = 0, l = cookies.length; i < l; i++) {
var parts = cookies[i].split('=');
var name = decode(parts.shift());
var cookie = parts.join('=');
if (key && key === name) {
// If second argument (value) is a function it's a converter...
result = read(cookie, value);
break;
}
// Prevent storing a cookie that we couldn't decode.
if (!key && (cookie = read(cookie)) !== undefined) {
result[name] = cookie;
}
}
return result;
};
config.defaults = {};
$.removeCookie = function (key, options) {
if ($.cookie(key) !== undefined) {
// Must not alter options, thus extending a fresh object...
$.cookie(key, '', $.extend({}, options, { expires: -1 }));
return true;
}
return false;
};
}));