vendor/assets/javascripts/bootstrap-editable.js in bootstrap-x-editable-rails-1.4.1 vs vendor/assets/javascripts/bootstrap-editable.js in bootstrap-x-editable-rails-1.4.3
- old
+ new
@@ -1,6 +1,6 @@
-/*! X-editable - v1.4.1
+/*! X-editable - v1.4.3
* In-place editing with Twitter Bootstrap, jQuery UI or pure jQuery
* http://github.com/vitalets/x-editable
* Copyright (c) 2013 Vitaliy Potapov; Licensed MIT */
/**
@@ -492,11 +492,11 @@
if(!response.success) return response.msg;
}
**/
success: null,
/**
- Additional options for ajax request.
+ Additional options for submit ajax request.
List of values: http://api.jquery.com/jQuery.ajax
@property ajaxOptions
@type object
@default null
@@ -709,11 +709,11 @@
result = [],
that = this;
$.each(sourceData, function(i, o) {
if(o.children) {
- result = result.concat(that.itemsByValue(value, o.children));
+ result = result.concat(that.itemsByValue(value, o.children, valueProp));
} else {
/*jslint eqeq: true*/
if(isValArray) {
if($.grep(value, function(v){ return v == (o && typeof o === 'object' ? o[valueProp] : o); }).length) {
result.push(o);
@@ -829,11 +829,12 @@
$('.editable-open').editableContainer('hide');
//todo: return focus on element
}
});
- //close containers when click outside
+ //close containers when click outside
+ //(mousedown could be better than click, it closes everything also on drag drop)
$(document).on('click.editable', function(e) {
var $target = $(e.target), i,
exclude_classes = ['.editable-container',
'.ui-datepicker-header',
'.modal-backdrop',
@@ -878,11 +879,11 @@
return this.container() ? this.container().$tip : null;
},
/* returns container object */
container: function() {
- return this.$element.data(this.containerName);
+ return this.$element.data(this.containerDataName || this.containerName);
},
call: function() {
this.$element[this.containerName].apply(this.$element, arguments);
},
@@ -907,12 +908,12 @@
@event shown
@param {Object} event event object
@example
$('#username').on('shown', function() {
- var $tip = $(this).data('editableContainer').tip();
- $tip.find('input').val('overwriting value of input..');
+ var editable = $(this).data('editable');
+ editable.input.$input.val('overwriting value of input..');
});
**/
this.$element.triggerHandler('shown');
}, this)
})
@@ -1209,13 +1210,13 @@
/**
Animation speed (inline mode)
@property anim
@type string
- @default 'fast'
+ @default false
**/
- anim: 'fast',
+ anim: false,
/**
Mode of editable, can be `popup` or `inline`
@property mode
@@ -1318,11 +1319,13 @@
doAutotext, finalize;
//name
this.options.name = this.options.name || this.$element.attr('id');
- //create input of specified type. Input will be used for converting value, not in form
+ //create input of specified type. Input needed already here to convert value for initial display (e.g. show text by id for select)
+ //also we set scope option to have access to element inside input specific callbacks (e. g. source as function)
+ this.options.scope = this.$element[0];
this.input = $.fn.editableutils.createInput(this.options);
if(!this.input) {
return;
}
@@ -1369,13 +1372,23 @@
} else {
this.$element.attr('tabindex', -1); //do not stop focus on element when toggled manually
}
//check conditions for autotext:
- //if value was generated by text or value is empty, no sense to run autotext
- doAutotext = !isValueByText && this.value !== null && this.value !== undefined;
- doAutotext &= (this.options.autotext === 'always') || (this.options.autotext === 'auto' && !this.$element.text().length);
+ switch(this.options.autotext) {
+ case 'always':
+ doAutotext = true;
+ break;
+ case 'auto':
+ //if element text is empty and value is defined and value not generated by text --> run autotext
+ doAutotext = !$.trim(this.$element.text()).length && this.value !== null && this.value !== undefined && !isValueByText;
+ break;
+ default:
+ doAutotext = false;
+ }
+
+ //depending on autotext run render() or just finilize init
$.when(doAutotext ? this.render() : true).then($.proxy(function() {
if(this.options.disabled) {
this.disable();
} else {
this.enable();
@@ -1407,10 +1420,15 @@
this.options.autotext = 'never';
//listen toggle events
this.$element.on(this.options.toggle + '.editable', selector, $.proxy(function(e){
var $target = $(e.target);
if(!$target.data('editable')) {
+ //if delegated element initially empty, we need to clear it's text (that was manually set to `empty` by user)
+ //see https://github.com/vitalets/x-editable/issues/137
+ if($target.hasClass(this.options.emptyclass)) {
+ $target.empty();
+ }
$target.editable(this.options).trigger(e);
}
}, this));
},
@@ -1528,11 +1546,11 @@
handleEmpty: function (isEmpty) {
//do not handle empty if we do not display anything
if(this.options.display === false) {
return;
}
-
+
this.isEmpty = isEmpty !== undefined ? isEmpty : $.trim(this.$element.text()) === '';
//emptytext shown only for enabled
if(!this.options.disabled) {
if (this.isEmpty) {
@@ -1749,15 +1767,18 @@
}
});
return result;
/**
- Returns current values of editable elements. If value is <code>null</code> or <code>undefined</code> it will not be returned
+ Returns current values of editable elements.
+ Note that it returns an **object** with name-value pairs, not a value itself. It allows to get data from several elements.
+ If value of some editable is `null` or `undefined` it is excluded from result object.
+
@method getValue()
@returns {Object} object of element names and values
@example
- $('#username, #fullname').editable('validate');
+ $('#username, #fullname').editable('getValue');
// possible result:
{
username: "superuser",
fullname: "John"
}
@@ -1890,28 +1911,40 @@
@type string
@default 'auto'
**/
autotext: 'auto',
/**
- Initial value of input. If not set, taken from element's text.
-
+ Initial value of input. If not set, taken from element's text.
+ Note, that if element's text is empty - text is automatically generated from value and can be customized (see `autotext` option).
+ For example, to display currency sign:
+ @example
+ <a id="price" data-type="text" data-value="100"></a>
+ <script>
+ $('#price').editable({
+ ...
+ display: function(value) {
+ $(this).text(value + '$');
+ }
+ })
+ </script>
+
@property value
@type mixed
@default element's text
**/
value: null,
/**
Callback to perform custom displaying of value in element's text.
If `null`, default input's display used.
If `false`, no displaying methods will be called, element's text will never change.
Runs under element's scope.
- _Parameters:_
+ _**Parameters:**_
* `value` current value to be displayed
* `response` server response (if display called after ajax submit), since 1.4.0
- For **inputs with source** (select, checklist) parameters are different:
+ For _inputs with source_ (select, checklist) parameters are different:
* `value` current value to be displayed
* `sourceData` array of items for current input (e.g. dropdown items)
* `response` server response (if display called after ajax submit), since 1.4.0
@@ -1954,23 +1987,27 @@
@since 1.4.1
@default editable-unsaved
**/
unsavedclass: 'editable-unsaved',
/**
- If a css selector is provided, editable will be delegated to the specified targets.
+ If selector is provided, editable will be delegated to the specified targets.
Usefull for dynamically generated DOM elements.
- **Please note**, that delegated targets can't use `emptytext` and `autotext` options,
- as they are initialized after first click.
+ **Please note**, that delegated targets can't be initialized with `emptytext` and `autotext` options,
+ as they actually become editable only after first click.
+ You should manually set class `editable-click` to these elements.
+ Also, if element originally empty you should add class `editable-empty`, set `data-value=""` and write emptytext into element:
@property selector
@type string
@since 1.4.1
@default null
@example
<div id="user">
- <a href="#" data-name="username" data-type="text" title="Username">awesome</a>
- <a href="#" data-name="group" data-type="select" data-source="/groups" data-value="1" title="Group">Operator</a>
+ <!-- empty -->
+ <a href="#" data-name="username" data-type="text" class="editable-click editable-empty" data-value="" title="Username">Empty</a>
+ <!-- non-empty -->
+ <a href="#" data-name="group" data-type="select" data-source="/groups" data-value="1" class="editable-click" title="Group">Operator</a>
</div>
<script>
$('#user').editable({
selector: 'a',
@@ -2143,11 +2180,11 @@
this.$input.addClass(this.options.inputclass);
}
},
setAttr: function(attr) {
- if (this.options[attr]) {
+ if (this.options[attr] !== undefined && this.options[attr] !== null) {
this.$input.attr(attr, this.options[attr]);
}
},
option: function(key, value) {
@@ -2170,11 +2207,14 @@
@property inputclass
@type string
@default input-medium
**/
- inputclass: 'input-medium'
+ inputclass: 'input-medium',
+ //scope for external methods (e.g. source defined as function)
+ //for internal use only
+ scope: null
};
$.extend($.fn.editabletypes, {abstractinput: AbstractInput});
}(window.jQuery));
@@ -2249,16 +2289,23 @@
this.options.source = $.fn.editableutils.tryParseJson(this.options.source, false);
} catch (e) {
error.call(this);
return;
}
+
+ var source = this.options.source;
+
+ //run source if it function
+ if ($.isFunction(source)) {
+ source = source.call(this.options.scope);
+ }
//loading from url
- if (typeof this.options.source === 'string') {
+ if (typeof source === 'string') {
//try to get from cache
if(this.options.sourceCache) {
- var cacheID = this.options.source,
+ var cacheID = source,
cache;
if (!$(document).data(cacheID)) {
$(document).data(cacheID, {});
}
@@ -2287,11 +2334,11 @@
}
}
//loading sourceData from server
$.ajax({
- url: this.options.source,
+ url: source,
type: 'get',
cache: false,
dataType: 'json',
success: $.proxy(function (data) {
if(cache) {
@@ -2322,16 +2369,12 @@
//run error callbacks for other fields
$.each(cache.err_callbacks, function () { this.call(); });
}
}, this)
});
- } else { //options as json/array/function
- if ($.isFunction(this.options.source)) {
- this.sourceData = this.makeArray(this.options.source());
- } else {
- this.sourceData = this.makeArray(this.options.source);
- }
+ } else { //options as json/array
+ this.sourceData = this.makeArray(source);
if($.isArray(this.sourceData)) {
this.doPrepend();
success.call(this);
} else {
@@ -2344,20 +2387,24 @@
if(this.options.prepend === null || this.options.prepend === undefined) {
return;
}
if(!$.isArray(this.prependData)) {
+ //run prepend if it is function (once)
+ if ($.isFunction(this.options.prepend)) {
+ this.options.prepend = this.options.prepend.call(this.options.scope);
+ }
+
//try parse json in single quotes
this.options.prepend = $.fn.editableutils.tryParseJson(this.options.prepend, true);
+
+ //convert prepend from string to object
if (typeof this.options.prepend === 'string') {
this.options.prepend = {'': this.options.prepend};
- }
- if (typeof this.options.prepend === 'function') {
- this.prependData = this.makeArray(this.options.prepend());
- } else {
- this.prependData = this.makeArray(this.options.prepend);
}
+
+ this.prependData = this.makeArray(this.options.prepend);
}
if($.isArray(this.prependData) && $.isArray(this.sourceData)) {
this.sourceData = this.prependData.concat(this.sourceData);
}
@@ -2534,12 +2581,24 @@
//render clear button
renderClear: function() {
if (this.options.clear) {
this.$clear = $('<span class="editable-clear-x"></span>');
this.$input.after(this.$clear)
- .css('padding-right', 20)
- .keyup($.proxy(this.toggleClear, this))
+ .css('padding-right', 24)
+ .keyup($.proxy(function(e) {
+ //arrows, enter, tab, etc
+ if(~$.inArray(e.keyCode, [40,38,9,13,27])) {
+ return;
+ }
+
+ clearTimeout(this.t);
+ var that = this;
+ this.t = setTimeout(function() {
+ that.toggleClear(e);
+ }, 100);
+
+ }, this))
.parent().css('position', 'relative');
this.$clear.click($.proxy(this.clear, this));
}
},
@@ -2553,23 +2612,28 @@
//workaround for plain-popup
if(delta < 3) {
delta = 3;
}
- this.$clear.css({top: delta, right: delta});
+ this.$clear.css({bottom: delta, right: delta});
}
},
//show / hide clear button
- toggleClear: function() {
+ toggleClear: function(e) {
if(!this.$clear) {
return;
}
- if(this.$input.val().length) {
+ var len = this.$input.val().length,
+ visible = this.$clear.is(':visible');
+
+ if(len && !visible) {
this.$clear.show();
- } else {
+ }
+
+ if(!len && visible) {
this.$clear.hide();
}
},
clear: function() {
@@ -2731,11 +2795,10 @@
source: [
{value: 1, text: 'Active'},
{value: 2, text: 'Blocked'},
{value: 3, text: 'Deleted'}
]
- }
});
});
</script>
**/
(function ($) {
@@ -2802,10 +2865,11 @@
});
$.fn.editabletypes.select = Select;
}(window.jQuery));
+
/**
List of checkboxes.
Internally value stored as javascript array of values.
@class checklist
@@ -2820,11 +2884,10 @@
source: [
{value: 1, text: 'option1'},
{value: 2, text: 'option2'},
{value: 3, text: 'option3'}
]
- }
});
});
</script>
**/
(function ($) {
@@ -3130,10 +3193,13 @@
You should manually include select2 distributive:
<link href="select2/select2.css" rel="stylesheet" type="text/css"></link>
<script src="select2/select2.js"></script>
+**Note:** currently `ajax` source for select2 is not supported, as it's not possible to load it in closed select2 state.
+The solution is to load source manually and assign statically.
+
@class select2
@extends abstractinput
@since 1.4.1
@final
@example
@@ -3159,25 +3225,42 @@
this.init('select2', options, Constructor.defaults);
options.select2 = options.select2 || {};
var that = this,
- mixin = {
+ mixin = { //mixin to select2 options
placeholder: options.placeholder
};
//detect whether it is multi-valued
this.isMultiple = options.select2.tags || options.select2.multiple;
- //if not `tags` mode, we need define init set data from source
+ //if not `tags` mode, we need define initSelection to set data from source
if(!options.select2.tags) {
if(options.source) {
mixin.data = options.source;
}
//this function can be defaulted in seletc2. See https://github.com/ivaynberg/select2/issues/710
mixin.initSelection = function (element, callback) {
+ //temp: try update results
+ /*
+ if(options.select2 && options.select2.ajax) {
+ console.log('attached');
+ var original = $(element).data('select2').postprocessResults;
+ console.log(original);
+ $(element).data('select2').postprocessResults = function(data, initial) {
+ console.log('postprocess');
+ // this.element.triggerHandler('loaded', [data]);
+ original.apply(this, arguments);
+ }
+
+ // $(element).on('loaded', function(){console.log('loaded');});
+ $(element).data('select2').updateResults(true);
+ }
+ */
+
var val = that.str2value(element.val()),
data = $.fn.editableutils.itemsByValue(val, mixin.data, 'id');
//for single-valued mode should not use array. Take first element instead.
if($.isArray(data) && data.length && !that.isMultiple) {
@@ -3198,30 +3281,44 @@
render: function() {
this.setClass();
//apply select2
this.$input.select2(this.options.select2);
+ //when data is loaded via ajax, we need to know when it's done
+ if('ajax' in this.options.select2) {
+ /*
+ console.log('attached');
+ var original = this.$input.data('select2').postprocessResults;
+ this.$input.data('select2').postprocessResults = function(data, initial) {
+ this.element.triggerHandler('loaded', [data]);
+ original.apply(this, arguments);
+ }
+ */
+ }
+
+
//trigger resize of editableform to re-position container in multi-valued mode
if(this.isMultiple) {
this.$input.on('change', function() {
$(this).closest('form').parent().triggerHandler('resize');
});
- }
-
- },
+ }
+ },
value2html: function(value, element) {
var text = '', data;
- if(this.$input) { //when submitting form
+ if(this.$input) { //called when submitting form and select2 already exists
data = this.$input.select2('data');
} else { //on init (autotext)
//here select2 instance not created yet and data may be even not loaded.
//we can check data/tags property of select config and if exist lookup text
if(this.options.select2.tags) {
data = value;
} else if(this.options.select2.data) {
data = $.fn.editableutils.itemsByValue(value, this.options.select2.data, 'id');
+ } else {
+ //if('ajax' in this.options.select2) {
}
}
if($.isArray(data)) {
//collect selected data and show with separator
@@ -3241,11 +3338,11 @@
html2value: function(html) {
return this.options.select2.tags ? this.str2value(html, this.options.viewseparator) : null;
},
value2input: function(value) {
- this.$input.val(value).trigger('change');
+ this.$input.val(value).trigger('change', true); //second argument needed to separate initial change from user's click (for autosubmit)
},
input2value: function() {
return this.$input.select2('val');
},
@@ -3266,11 +3363,19 @@
for (i = 0, l = val.length; i < l; i = i + 1) {
val[i] = $.trim(val[i]);
}
return val;
- }
+ },
+
+ autosubmit: function() {
+ this.$input.on('change', function(e, isInitial){
+ if(!isInitial) {
+ $(this).closest('form').submit();
+ }
+ });
+ }
});
Constructor.defaults = $.extend({}, $.fn.editabletypes.abstractinput.defaults, {
/**
@@ -3316,11 +3421,11 @@
$.fn.editabletypes.select2 = Constructor;
}(window.jQuery));
/**
-* Combodate - 1.0.1
+* Combodate - 1.0.2
* Dropdown date and time picker.
* Converts text input into dropdowns to pick day, month, year, hour, minute and second.
* Uses momentjs as datetime library http://momentjs.com.
* For i18n include corresponding file from https://github.com/timrwood/moment/tree/master/lang
*
@@ -3419,13 +3524,17 @@
/*
Initialize items of combos. Handles `firstItem` option
*/
initItems: function(key) {
- var values = [];
+ var values = [],
+ relTime;
+
if(this.options.firstItem === 'name') {
- var header = typeof moment.relativeTime[key] === 'function' ? moment.relativeTime[key](1, true, key, false) : moment.relativeTime[key];
+ //need both to suuport moment ver < 2 and >= 2
+ relTime = moment.relativeTime || moment.langData()._relativeTime;
+ var header = typeof relTime[key] === 'function' ? relTime[key](1, true, key, false) : relTime[key];
//take last entry (see momentjs lang files structure)
header = header.split(' ').reverse()[0];
values.push(['', header]);
} else if(this.options.firstItem === 'empty') {
values.push(['', '']);
@@ -3467,13 +3576,13 @@
shortNames = this.options.template.indexOf('MMM') !== -1,
twoDigit = this.options.template.indexOf('MM') !== -1;
for(i=0; i<=11; i++) {
if(longNames) {
- name = moment.months[i];
+ name = moment().month(i).format('MMMM');
} else if(shortNames) {
- name = moment.monthsShort[i];
+ name = moment().month(i).format('MMM');
} else if(twoDigit) {
name = this.leadZero(i+1);
} else {
name = i+1;
}
@@ -3715,11 +3824,11 @@
};
}(window.jQuery));
/**
Combodate input - dropdown date and time picker.
-Based on [combodate](http://vitalets.github.com/combodate) plugin. To use it you should manually include [momentjs](http://momentjs.com).
+Based on [combodate](http://vitalets.github.com/combodate) plugin (included). To use it you should manually include [momentjs](http://momentjs.com).
<script src="js/moment.min.js"></script>
Allows to input:
@@ -3763,10 +3872,13 @@
//by default viewformat equals to format
if(!this.options.viewformat) {
this.options.viewformat = this.options.format;
}
+ //try parse combodate config defined as json string in data-combodate
+ options.combodate = $.fn.editableutils.tryParseJson(options.combodate, true);
+
//overriding combodate config (as by default jQuery extend() is not recursive)
this.options.combodate = $.extend({}, Constructor.defaults.combodate, options.combodate, {
format: this.options.format,
template: this.options.template
});
@@ -4147,16 +4259,28 @@
this.$input.data('datepicker').date = null;
this.$input.find('.active').removeClass('active');
},
autosubmit: function() {
+ this.$input.on('mouseup', '.day', function(e){
+ if($(e.currentTarget).is('.old') || $(e.currentTarget).is('.new')) {
+ return;
+ }
+ var $form = $(this).closest('form');
+ setTimeout(function() {
+ $form.submit();
+ }, 200);
+ });
+ //changedate is not suitable as it triggered when showing datepicker. see #149
+ /*
this.$input.on('changeDate', function(e){
var $form = $(this).closest('form');
setTimeout(function() {
$form.submit();
}, 200);
});
+ */
}
});
Date.defaults = $.extend({}, $.fn.editabletypes.abstractinput.defaults, {
@@ -4195,16 +4319,18 @@
@property datepicker
@type object
@default {
weekStart: 0,
startView: 0,
+ minViewMode: 0,
autoclose: false
}
**/
datepicker:{
weekStart: 0,
startView: 0,
+ minViewMode: 0,
autoclose: false
},
/**
Text shown as clear date button.
If <code>false</code> clear button will not be rendered.
@@ -4289,10 +4415,11 @@
/* datepicker config */
datepicker: {
weekStart: 0,
startView: 0,
+ minViewMode: 0,
autoclose: true
}
});
$.fn.editabletypes.datefield = DateField;
@@ -4333,13 +4460,14 @@
var Datepicker = function(element, options) {
var that = this;
this.element = $(element);
this.language = options.language||this.element.data('date-language')||"en";
+ this.language = this.language in dates ? this.language : this.language.split('-')[0]; //Check if "de-DE" style date is available, if not language should fallback to 2 letter code eg "de"
this.language = this.language in dates ? this.language : "en";
this.isRTL = dates[this.language].rtl||false;
- this.format = DPGlobal.parseFormat(options.format||this.element.data('date-format')||'mm/dd/yyyy');
+ this.format = DPGlobal.parseFormat(options.format||this.element.data('date-format')||dates[this.language].format||'mm/dd/yyyy');
this.isInline = false;
this.isInput = this.element.is('input');
this.component = this.element.is('.date') ? this.element.find('.add-on') : false;
this.hasInput = this.component && this.element.find('input').length;
if(this.component && this.component.length === 0)
@@ -4351,12 +4479,12 @@
if ('forceParse' in options) {
this.forceParse = options.forceParse;
} else if ('dateForceParse' in this.element.data()) {
this.forceParse = this.element.data('date-force-parse');
}
-
+
this.picker = $(DPGlobal.template)
.appendTo(this.isInline ? this.element : 'body')
.on({
click: $.proxy(this.click, this),
mousedown: $.proxy(this.mousedown, this)
@@ -4372,11 +4500,11 @@
this.picker.find('.prev i, .next i')
.toggleClass('icon-arrow-left icon-arrow-right');
}
$(document).on('mousedown', function (e) {
// Clicked outside the datepicker, hide it
- if ($(e.target).closest('.datepicker').length === 0) {
+ if ($(e.target).closest('.datepicker.datepicker-inline, .datepicker.datepicker-dropdown').length === 0) {
that.hide();
}
});
this.autoclose = false;
@@ -4403,13 +4531,42 @@
case 'year':
this.viewMode = this.startViewMode = 1;
break;
}
+ this.minViewMode = options.minViewMode||this.element.data('date-min-view-mode')||0;
+ if (typeof this.minViewMode === 'string') {
+ switch (this.minViewMode) {
+ case 'months':
+ this.minViewMode = 1;
+ break;
+ case 'years':
+ this.minViewMode = 2;
+ break;
+ default:
+ this.minViewMode = 0;
+ break;
+ }
+ }
+
+ this.viewMode = this.startViewMode = Math.max(this.startViewMode, this.minViewMode);
+
this.todayBtn = (options.todayBtn||this.element.data('date-today-btn')||false);
this.todayHighlight = (options.todayHighlight||this.element.data('date-today-highlight')||false);
+ this.calendarWeeks = false;
+ if ('calendarWeeks' in options) {
+ this.calendarWeeks = options.calendarWeeks;
+ } else if ('dateCalendarWeeks' in this.element.data()) {
+ this.calendarWeeks = this.element.data('date-calendar-weeks');
+ }
+ if (this.calendarWeeks)
+ this.picker.find('tfoot th.today')
+ .attr('colspan', function(i, val){
+ return parseInt(val) + 1;
+ });
+
this.weekStart = ((options.weekStart||this.element.data('date-weekstart')||dates[this.language].weekStart||0) % 7);
this.weekEnd = ((this.weekStart + 6) % 7);
this.startDate = -Infinity;
this.endDate = Infinity;
this.daysOfWeekDisabled = [];
@@ -4495,10 +4652,11 @@
});
},
hide: function(e){
if(this.isInline) return;
+ if (!this.picker.is(':visible')) return;
this.picker.hide();
$(window).off('resize', this.place);
this.viewMode = this.startViewMode;
this.showMode();
if (!this.isInput) {
@@ -4616,31 +4774,28 @@
this.date = DPGlobal.parseDate(date, this.format, this.language);
if(fromArgs) this.setValue();
- var oldViewDate = this.viewDate;
if (this.date < this.startDate) {
this.viewDate = new Date(this.startDate);
} else if (this.date > this.endDate) {
this.viewDate = new Date(this.endDate);
} else {
this.viewDate = new Date(this.date);
}
-
- if (oldViewDate && oldViewDate.getTime() != this.viewDate.getTime()){
- this.element.trigger({
- type: 'changeDate',
- date: this.viewDate
- });
- }
this.fill();
},
fillDow: function(){
var dowCnt = this.weekStart,
html = '<tr>';
+ if(this.calendarWeeks){
+ var cell = '<th class="cw"> </th>';
+ html += cell;
+ this.picker.find('.datepicker-days thead tr:first-child').prepend(cell);
+ }
while (dowCnt < this.weekStart + 7) {
html += '<th class="dow">'+dates[this.language].daysMin[(dowCnt++)%7]+'</th>';
}
html += '</tr>';
this.picker.find('.datepicker-days thead').append(html);
@@ -4663,11 +4818,11 @@
startMonth = this.startDate !== -Infinity ? this.startDate.getUTCMonth() : -Infinity,
endYear = this.endDate !== Infinity ? this.endDate.getUTCFullYear() : Infinity,
endMonth = this.endDate !== Infinity ? this.endDate.getUTCMonth() : Infinity,
currentDate = this.date && this.date.valueOf(),
today = new Date();
- this.picker.find('.datepicker-days thead th:eq(1)')
+ this.picker.find('.datepicker-days thead th.switch')
.text(dates[this.language].months[month]+' '+year);
this.picker.find('tfoot th.today')
.text(dates[this.language].today)
.toggle(this.todayBtn !== false);
this.updateNavArrows();
@@ -4682,10 +4837,25 @@
var html = [];
var clsName;
while(prevMonth.valueOf() < nextMonth) {
if (prevMonth.getUTCDay() == this.weekStart) {
html.push('<tr>');
+ if(this.calendarWeeks){
+ // ISO 8601: First week contains first thursday.
+ // ISO also states week starts on Monday, but we can be more abstract here.
+ var
+ // Start of current week: based on weekstart/current date
+ ws = new Date(+prevMonth + (this.weekStart - prevMonth.getUTCDay() - 7) % 7 * 864e5),
+ // Thursday of this week
+ th = new Date(+ws + (7 + 4 - ws.getUTCDay()) % 7 * 864e5),
+ // First Thursday of year, year from thursday
+ yth = new Date(+(yth = UTCDate(th.getUTCFullYear(), 0, 1)) + (7 + 4 - yth.getUTCDay())%7*864e5),
+ // Calendar week: ms between thursdays, div ms per day, div 7 days
+ calWeek = (th - yth) / 864e5 / 7 + 1;
+ html.push('<td class="cw">'+ calWeek +'</td>');
+
+ }
}
clsName = '';
if (prevMonth.getUTCFullYear() < year || (prevMonth.getUTCFullYear() == year && prevMonth.getUTCMonth() < month)) {
clsName += ' old';
} else if (prevMonth.getUTCFullYear() > year || (prevMonth.getUTCFullYear() == year && prevMonth.getUTCMonth() > month)) {
@@ -4817,23 +4987,33 @@
break;
case 'span':
if (!target.is('.disabled')) {
this.viewDate.setUTCDate(1);
if (target.is('.month')) {
+ var day = 1;
var month = target.parent().find('span').index(target);
+ var year = this.viewDate.getUTCFullYear();
this.viewDate.setUTCMonth(month);
this.element.trigger({
type: 'changeMonth',
date: this.viewDate
});
+ if ( this.minViewMode == 1 ) {
+ this._setDate(UTCDate(year, month, day,0,0,0,0));
+ }
} else {
var year = parseInt(target.text(), 10)||0;
+ var day = 1;
+ var month = 0;
this.viewDate.setUTCFullYear(year);
this.element.trigger({
type: 'changeYear',
date: this.viewDate
});
+ if ( this.minViewMode == 2 ) {
+ this._setDate(UTCDate(year, month, day,0,0,0,0));
+ }
}
this.showMode(-1);
this.fill();
}
break;
@@ -5026,11 +5206,11 @@
}
},
showMode: function(dir) {
if (dir) {
- this.viewMode = Math.max(0, Math.min(2, this.viewMode + dir));
+ this.viewMode = Math.max(this.minViewMode, Math.min(2, this.viewMode + dir));
}
/*
vitalets: fixing bug of very special conditions:
jquery 1.7.1 + webkit + show inline datepicker in bootstrap popover.
Method show() does not set display css correctly and datepicker is not shown.
@@ -5286,11 +5466,10 @@
source: [
{value: 'gb', text: 'Great Britain'},
{value: 'us', text: 'United States'},
{value: 'ru', text: 'Russia'}
]
- }
});
});
</script>
**/
(function ($) {
@@ -5318,12 +5497,15 @@
this.options.typeahead.source = this.sourceData;
//apply typeahead
this.$input.typeahead(this.options.typeahead);
- //attach own render method
- this.$input.data('typeahead').render = $.proxy(this.typeaheadRender, this.$input.data('typeahead'));
+ //patch some methods in typeahead
+ var ta = this.$input.data('typeahead');
+ ta.render = $.proxy(this.typeaheadRender, ta);
+ ta.select = $.proxy(this.typeaheadSelect, ta);
+ ta.move = $.proxy(this.typeaheadMove, ta);
this.renderClear();
this.setClass();
this.setAttr('placeholder');
},
@@ -5398,11 +5580,11 @@
/*
Typeahead option methods used as defaults
*/
- /*jshint eqeqeq:false, curly: false, laxcomma: true*/
+ /*jshint eqeqeq:false, curly: false, laxcomma: true, asi: true*/
matcher: function (item) {
return $.fn.typeahead.Constructor.prototype.matcher.call(this, item.text);
},
sorter: function (items) {
var beginswith = []
@@ -5422,22 +5604,21 @@
},
highlighter: function (item) {
return $.fn.typeahead.Constructor.prototype.highlighter.call(this, item.text);
},
updater: function (item) {
- item = this.$menu.find('.active').data('item');
this.$element.data('value', item.value);
return item.text;
},
/*
Overwrite typeahead's render method to store objects.
There are a lot of disscussion in bootstrap repo on this point and still no result.
See https://github.com/twitter/bootstrap/issues/5967
- This function just store item in via jQuery data() method instead of attr('data-value')
+ This function just store item via jQuery data() method instead of attr('data-value')
*/
typeaheadRender: function (items) {
var that = this;
items = $(items).map(function (i, item) {
@@ -5445,14 +5626,60 @@
i = $(that.options.item).data('item', item);
i.find('a').html(that.highlighter(item));
return i[0];
});
- items.first().addClass('active');
+ //add option to disable autoselect of first line
+ //see https://github.com/twitter/bootstrap/pull/4164
+ if (this.options.autoSelect) {
+ items.first().addClass('active');
+ }
this.$menu.html(items);
return this;
+ },
+
+ //add option to disable autoselect of first line
+ //see https://github.com/twitter/bootstrap/pull/4164
+ typeaheadSelect: function () {
+ var val = this.$menu.find('.active').data('item')
+ if(this.options.autoSelect || val){
+ this.$element
+ .val(this.updater(val))
+ .change()
+ }
+ return this.hide()
+ },
+
+ /*
+ if autoSelect = false and nothing matched we need extra press onEnter that is not convinient.
+ This patch fixes it.
+ */
+ typeaheadMove: function (e) {
+ if (!this.shown) return
+
+ switch(e.keyCode) {
+ case 9: // tab
+ case 13: // enter
+ case 27: // escape
+ if (!this.$menu.find('.active').length) return
+ e.preventDefault()
+ break
+
+ case 38: // up arrow
+ e.preventDefault()
+ this.prev()
+ break
+
+ case 40: // down arrow
+ e.preventDefault()
+ this.next()
+ break
+ }
+
+ e.stopPropagation()
}
- /*jshint eqeqeq: true, curly: true, laxcomma: false*/
+
+ /*jshint eqeqeq: true, curly: true, laxcomma: false, asi: false*/
});
Constructor.defaults = $.extend({}, $.fn.editabletypes.list.defaults, {
/**
\ No newline at end of file