x-editable-rails 1.0.1 → 1.0.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/lib/x-editable-rails/version.rb +1 -1
- data/vendor/assets/javascripts/editable/bootstrap-editable.js +1142 -713
- data/vendor/assets/javascripts/editable/jquery-editable-poshytip.js +353 -182
- data/vendor/assets/javascripts/editable/jqueryui-editable.js +352 -181
- data/vendor/assets/stylesheets/editable/{bootstrap-editable.css → bootstrap-editable.scss} +164 -8
- data/vendor/assets/stylesheets/editable/{jquery-editable.css → jquery-editable.scss} +11 -3
- data/vendor/assets/stylesheets/editable/{jqueryui-editable.css → jqueryui-editable.scss} +11 -3
- metadata +5 -5
| @@ -1,4 +1,4 @@ | |
| 1 | 
            -
            /*! X-editable - v1.4. | 
| 1 | 
            +
            /*! X-editable - v1.4.5 
         | 
| 2 2 | 
             
            * In-place editing with Twitter Bootstrap, jQuery UI or pure jQuery
         | 
| 3 3 | 
             
            * http://github.com/vitalets/x-editable
         | 
| 4 4 | 
             
            * Copyright (c) 2013 Vitaliy Potapov; Licensed MIT */
         | 
| @@ -65,6 +65,10 @@ Editableform is linked with one of input types, e.g. 'text', 'select' etc. | |
| 65 65 | 
             
                        //show loading state
         | 
| 66 66 | 
             
                        this.showLoading();            
         | 
| 67 67 |  | 
| 68 | 
            +
                        //flag showing is form now saving value to server. 
         | 
| 69 | 
            +
                        //It is needed to wait when closing form.
         | 
| 70 | 
            +
                        this.isSaving = false;
         | 
| 71 | 
            +
                        
         | 
| 68 72 | 
             
                        /**        
         | 
| 69 73 | 
             
                        Fired when rendering starts
         | 
| 70 74 | 
             
                        @event rendering 
         | 
| @@ -217,31 +221,38 @@ Editableform is linked with one of input types, e.g. 'text', 'select' etc. | |
| 217 221 | 
             
                            return;
         | 
| 218 222 | 
             
                        } 
         | 
| 219 223 |  | 
| 224 | 
            +
                        //convert value for submitting to server
         | 
| 225 | 
            +
                        var submitValue = this.input.value2submit(newValue);
         | 
| 226 | 
            +
                        
         | 
| 227 | 
            +
                        this.isSaving = true;
         | 
| 228 | 
            +
                        
         | 
| 220 229 | 
             
                        //sending data to server
         | 
| 221 | 
            -
                        $.when(this.save( | 
| 230 | 
            +
                        $.when(this.save(submitValue))
         | 
| 222 231 | 
             
                        .done($.proxy(function(response) {
         | 
| 232 | 
            +
                            this.isSaving = false;
         | 
| 233 | 
            +
             | 
| 223 234 | 
             
                            //run success callback
         | 
| 224 235 | 
             
                            var res = typeof this.options.success === 'function' ? this.options.success.call(this.options.scope, response, newValue) : null;
         | 
| 225 | 
            -
             | 
| 236 | 
            +
             | 
| 226 237 | 
             
                            //if success callback returns false --> keep form open and do not activate input
         | 
| 227 238 | 
             
                            if(res === false) {
         | 
| 228 239 | 
             
                                this.error(false);
         | 
| 229 240 | 
             
                                this.showForm(false);
         | 
| 230 241 | 
             
                                return;
         | 
| 231 | 
            -
                            } | 
| 232 | 
            -
             | 
| 242 | 
            +
                            }
         | 
| 243 | 
            +
             | 
| 233 244 | 
             
                            //if success callback returns string -->  keep form open, show error and activate input               
         | 
| 234 245 | 
             
                            if(typeof res === 'string') {
         | 
| 235 246 | 
             
                                this.error(res);
         | 
| 236 247 | 
             
                                this.showForm();
         | 
| 237 248 | 
             
                                return;
         | 
| 238 | 
            -
                            } | 
| 239 | 
            -
             | 
| 249 | 
            +
                            }
         | 
| 250 | 
            +
             | 
| 240 251 | 
             
                            //if success callback returns object like {newValue: <something>} --> use that value instead of submitted
         | 
| 241 252 | 
             
                            //it is usefull if you want to chnage value in url-function
         | 
| 242 253 | 
             
                            if(res && typeof res === 'object' && res.hasOwnProperty('newValue')) {
         | 
| 243 254 | 
             
                                newValue = res.newValue;
         | 
| 244 | 
            -
                            } | 
| 255 | 
            +
                            }
         | 
| 245 256 |  | 
| 246 257 | 
             
                            //clear error message
         | 
| 247 258 | 
             
                            this.error(false);   
         | 
| @@ -251,37 +262,42 @@ Editableform is linked with one of input types, e.g. 'text', 'select' etc. | |
| 251 262 | 
             
                            @event save 
         | 
| 252 263 | 
             
                            @param {Object} event event object
         | 
| 253 264 | 
             
                            @param {Object} params additional params
         | 
| 254 | 
            -
                            @param {mixed} params.newValue  | 
| 265 | 
            +
                            @param {mixed} params.newValue raw new value
         | 
| 266 | 
            +
                            @param {mixed} params.submitValue submitted value as string
         | 
| 255 267 | 
             
                            @param {Object} params.response ajax response
         | 
| 256 268 |  | 
| 257 269 | 
             
                            @example
         | 
| 258 270 | 
             
                            $('#form-div').on('save'), function(e, params){
         | 
| 259 271 | 
             
                                if(params.newValue === 'username') {...}
         | 
| 260 | 
            -
                            }); | 
| 261 | 
            -
                            **/ | 
| 262 | 
            -
                            this.$div.triggerHandler('save', {newValue: newValue, response: response});
         | 
| 272 | 
            +
                            });
         | 
| 273 | 
            +
                            **/
         | 
| 274 | 
            +
                            this.$div.triggerHandler('save', {newValue: newValue, submitValue: submitValue, response: response});
         | 
| 263 275 | 
             
                        }, this))
         | 
| 264 276 | 
             
                        .fail($.proxy(function(xhr) {
         | 
| 277 | 
            +
                            this.isSaving = false;
         | 
| 278 | 
            +
             | 
| 265 279 | 
             
                            var msg;
         | 
| 266 280 | 
             
                            if(typeof this.options.error === 'function') {
         | 
| 267 281 | 
             
                                msg = this.options.error.call(this.options.scope, xhr, newValue);
         | 
| 268 282 | 
             
                            } else {
         | 
| 269 283 | 
             
                                msg = typeof xhr === 'string' ? xhr : xhr.responseText || xhr.statusText || 'Unknown error!';
         | 
| 270 284 | 
             
                            }
         | 
| 271 | 
            -
             | 
| 285 | 
            +
             | 
| 272 286 | 
             
                            this.error(msg);
         | 
| 273 287 | 
             
                            this.showForm();
         | 
| 274 288 | 
             
                        }, this));
         | 
| 275 289 | 
             
                    },
         | 
| 276 290 |  | 
| 277 | 
            -
                    save: function( | 
| 278 | 
            -
                        //convert value for submitting to server
         | 
| 279 | 
            -
                        var submitValue = this.input.value2submit(newValue);
         | 
| 280 | 
            -
                        
         | 
| 291 | 
            +
                    save: function(submitValue) {
         | 
| 281 292 | 
             
                        //try parse composite pk defined as json string in data-pk 
         | 
| 282 293 | 
             
                        this.options.pk = $.fn.editableutils.tryParseJson(this.options.pk, true); 
         | 
| 283 294 |  | 
| 284 295 | 
             
                        var pk = (typeof this.options.pk === 'function') ? this.options.pk.call(this.options.scope) : this.options.pk,
         | 
| 296 | 
            +
                        /*
         | 
| 297 | 
            +
                          send on server in following cases:
         | 
| 298 | 
            +
                          1. url is function
         | 
| 299 | 
            +
                          2. url is string AND (pk defined OR send option = always) 
         | 
| 300 | 
            +
                        */
         | 
| 285 301 | 
             
                        send = !!(typeof this.options.url === 'function' || (this.options.url && ((this.options.send === 'always') || (this.options.send === 'auto' && pk !== null && pk !== undefined)))),
         | 
| 286 302 | 
             
                        params;
         | 
| 287 303 |  | 
| @@ -816,6 +832,27 @@ Editableform is linked with one of input types, e.g. 'text', 'select' etc. | |
| 816 832 | 
             
                           $.error('Unknown type: '+ type);
         | 
| 817 833 | 
             
                           return false; 
         | 
| 818 834 | 
             
                       }  
         | 
| 835 | 
            +
                   },
         | 
| 836 | 
            +
                   
         | 
| 837 | 
            +
                   //see http://stackoverflow.com/questions/7264899/detect-css-transitions-using-javascript-and-without-modernizr
         | 
| 838 | 
            +
                   supportsTransitions: function () {
         | 
| 839 | 
            +
                       var b = document.body || document.documentElement,
         | 
| 840 | 
            +
                           s = b.style,
         | 
| 841 | 
            +
                           p = 'transition',
         | 
| 842 | 
            +
                           v = ['Moz', 'Webkit', 'Khtml', 'O', 'ms'];
         | 
| 843 | 
            +
                           
         | 
| 844 | 
            +
                       if(typeof s[p] === 'string') {
         | 
| 845 | 
            +
                           return true; 
         | 
| 846 | 
            +
                       }
         | 
| 847 | 
            +
             | 
| 848 | 
            +
                       // Tests for vendor specific prop
         | 
| 849 | 
            +
                       p = p.charAt(0).toUpperCase() + p.substr(1);
         | 
| 850 | 
            +
                       for(var i=0; i<v.length; i++) {
         | 
| 851 | 
            +
                           if(typeof s[v[i] + p] === 'string') { 
         | 
| 852 | 
            +
                               return true; 
         | 
| 853 | 
            +
                           }
         | 
| 854 | 
            +
                       }
         | 
| 855 | 
            +
                       return false;
         | 
| 819 856 | 
             
                   }            
         | 
| 820 857 |  | 
| 821 858 | 
             
                };      
         | 
| @@ -856,6 +893,9 @@ Applied as jQuery method. | |
| 856 893 | 
             
                        this.formOptions.scope = this.$element[0]; 
         | 
| 857 894 |  | 
| 858 895 | 
             
                        this.initContainer();
         | 
| 896 | 
            +
                        
         | 
| 897 | 
            +
                        //flag to hide container, when saving value will finish
         | 
| 898 | 
            +
                        this.delayedHide = false;
         | 
| 859 899 |  | 
| 860 900 | 
             
                        //bind 'destroyed' listener to destroy container when element is removed from dom
         | 
| 861 901 | 
             
                        this.$element.on('destroyed', $.proxy(function(){
         | 
| @@ -960,7 +1000,14 @@ Applied as jQuery method. | |
| 960 1000 | 
             
                            save: $.proxy(this.save, this), //click on submit button (value changed)
         | 
| 961 1001 | 
             
                            nochange: $.proxy(function(){ this.hide('nochange'); }, this), //click on submit button (value NOT changed)                
         | 
| 962 1002 | 
             
                            cancel: $.proxy(function(){ this.hide('cancel'); }, this), //click on calcel button
         | 
| 963 | 
            -
                            show: $.proxy( | 
| 1003 | 
            +
                            show: $.proxy(function() {
         | 
| 1004 | 
            +
                                if(this.delayedHide) {
         | 
| 1005 | 
            +
                                    this.hide(this.delayedHide.reason);
         | 
| 1006 | 
            +
                                    this.delayedHide = false;
         | 
| 1007 | 
            +
                                } else {
         | 
| 1008 | 
            +
                                    this.setPosition();
         | 
| 1009 | 
            +
                                }
         | 
| 1010 | 
            +
                            }, this), //re-position container every time form is shown (occurs each time after loading state)
         | 
| 964 1011 | 
             
                            rendering: $.proxy(this.setPosition, this), //this allows to place container correctly when loading shown
         | 
| 965 1012 | 
             
                            resize: $.proxy(this.setPosition, this), //this allows to re-position container when form size is changed 
         | 
| 966 1013 | 
             
                            rendered: $.proxy(function(){
         | 
| @@ -1004,11 +1051,11 @@ Applied as jQuery method. | |
| 1004 1051 |  | 
| 1005 1052 | 
             
                        /*
         | 
| 1006 1053 | 
             
                        Currently, form is re-rendered on every show. 
         | 
| 1007 | 
            -
                        The main reason is that we dont know, what container  | 
| 1008 | 
            -
                        remove(), detach() or just hide().
         | 
| 1054 | 
            +
                        The main reason is that we dont know, what will container do with content when closed:
         | 
| 1055 | 
            +
                        remove(), detach() or just hide() - it depends on container.
         | 
| 1009 1056 |  | 
| 1010 1057 | 
             
                        Detaching form itself before hide and re-insert before show is good solution, 
         | 
| 1011 | 
            -
                        but visually it looks ugly | 
| 1058 | 
            +
                        but visually it looks ugly --> container changes size before hide.  
         | 
| 1012 1059 | 
             
                        */             
         | 
| 1013 1060 |  | 
| 1014 1061 | 
             
                        //if form already exist - delete previous data 
         | 
| @@ -1041,10 +1088,18 @@ Applied as jQuery method. | |
| 1041 1088 | 
             
                            return;
         | 
| 1042 1089 | 
             
                        }
         | 
| 1043 1090 |  | 
| 1091 | 
            +
                        //if form is saving value, schedule hide
         | 
| 1092 | 
            +
                        if(this.$form.data('editableform').isSaving) {
         | 
| 1093 | 
            +
                            this.delayedHide = {reason: reason};
         | 
| 1094 | 
            +
                            return;    
         | 
| 1095 | 
            +
                        } else {
         | 
| 1096 | 
            +
                            this.delayedHide = false;
         | 
| 1097 | 
            +
                        }
         | 
| 1098 | 
            +
             | 
| 1044 1099 | 
             
                        this.$element.removeClass('editable-open');   
         | 
| 1045 1100 | 
             
                        this.innerHide();
         | 
| 1046 | 
            -
             | 
| 1047 | 
            -
                        /** | 
| 1101 | 
            +
             | 
| 1102 | 
            +
                        /**
         | 
| 1048 1103 | 
             
                        Fired when container was hidden. It occurs on both save or cancel.  
         | 
| 1049 1104 | 
             
                        **Note:** Bootstrap popover has own `hidden` event that now cannot be separated from x-editable's one.
         | 
| 1050 1105 | 
             
                        The workaround is to check `arguments.length` that is always `2` for x-editable. 
         | 
| @@ -1058,20 +1113,20 @@ Applied as jQuery method. | |
| 1058 1113 | 
             
                                //auto-open next editable
         | 
| 1059 1114 | 
             
                                $(this).closest('tr').next().find('.editable').editable('show');
         | 
| 1060 1115 | 
             
                            } 
         | 
| 1061 | 
            -
                        }); | 
| 1062 | 
            -
                        **/ | 
| 1116 | 
            +
                        });
         | 
| 1117 | 
            +
                        **/
         | 
| 1063 1118 | 
             
                        this.$element.triggerHandler('hidden', reason || 'manual');   
         | 
| 1064 1119 | 
             
                    },
         | 
| 1065 | 
            -
             | 
| 1120 | 
            +
             | 
| 1066 1121 | 
             
                    /* internal show method. To be overwritten in child classes */
         | 
| 1067 1122 | 
             
                    innerShow: function () {
         | 
| 1068 1123 |  | 
| 1069 1124 | 
             
                    },        
         | 
| 1070 | 
            -
             | 
| 1125 | 
            +
             | 
| 1071 1126 | 
             
                    /* internal hide method. To be overwritten in child classes */
         | 
| 1072 1127 | 
             
                    innerHide: function () {
         | 
| 1073 | 
            -
             | 
| 1074 | 
            -
                    }, | 
| 1128 | 
            +
             | 
| 1129 | 
            +
                    },
         | 
| 1075 1130 |  | 
| 1076 1131 | 
             
                    /**
         | 
| 1077 1132 | 
             
                    Toggles container visibility (show / hide)
         | 
| @@ -1116,7 +1171,7 @@ Applied as jQuery method. | |
| 1116 1171 | 
             
                        **/             
         | 
| 1117 1172 | 
             
                        this.$element.triggerHandler('save', params);
         | 
| 1118 1173 |  | 
| 1119 | 
            -
                        //hide must be after trigger, as saving value may require methods  | 
| 1174 | 
            +
                        //hide must be after trigger, as saving value may require methods of plugin, applied to input
         | 
| 1120 1175 | 
             
                        this.hide('save');
         | 
| 1121 1176 | 
             
                    },
         | 
| 1122 1177 |  | 
| @@ -1276,7 +1331,7 @@ Applied as jQuery method. | |
| 1276 1331 | 
             
                    onblur: 'cancel',
         | 
| 1277 1332 |  | 
| 1278 1333 | 
             
                    /**
         | 
| 1279 | 
            -
                    Animation speed (inline mode)
         | 
| 1334 | 
            +
                    Animation speed (inline mode only)
         | 
| 1280 1335 | 
             
                    @property anim 
         | 
| 1281 1336 | 
             
                    @type string
         | 
| 1282 1337 | 
             
                    @default false
         | 
| @@ -1380,6 +1435,11 @@ Makes editable any HTML element on the page. Applied as jQuery method. | |
| 1380 1435 | 
             
                    } else {
         | 
| 1381 1436 | 
             
                        this.init();
         | 
| 1382 1437 | 
             
                    }
         | 
| 1438 | 
            +
                    
         | 
| 1439 | 
            +
                    //check for transition support
         | 
| 1440 | 
            +
                    if(this.options.highlight && !$.fn.editableutils.supportsTransitions()) {
         | 
| 1441 | 
            +
                        this.options.highlight = false;
         | 
| 1442 | 
            +
                    }
         | 
| 1383 1443 | 
             
                };
         | 
| 1384 1444 |  | 
| 1385 1445 | 
             
                Editable.prototype = {
         | 
| @@ -1424,25 +1484,33 @@ Makes editable any HTML element on the page. Applied as jQuery method. | |
| 1424 1484 | 
             
                        if(this.options.toggle !== 'manual') {
         | 
| 1425 1485 | 
             
                            this.$element.addClass('editable-click');
         | 
| 1426 1486 | 
             
                            this.$element.on(this.options.toggle + '.editable', $.proxy(function(e){
         | 
| 1427 | 
            -
                                //prevent following link
         | 
| 1428 | 
            -
                                 | 
| 1487 | 
            +
                                //prevent following link if editable enabled
         | 
| 1488 | 
            +
                                if(!this.options.disabled) {
         | 
| 1489 | 
            +
                                    e.preventDefault();
         | 
| 1490 | 
            +
                                }
         | 
| 1429 1491 |  | 
| 1430 1492 | 
             
                                //stop propagation not required because in document click handler it checks event target
         | 
| 1431 1493 | 
             
                                //e.stopPropagation();
         | 
| 1432 1494 |  | 
| 1433 1495 | 
             
                                if(this.options.toggle === 'mouseenter') {
         | 
| 1434 1496 | 
             
                                    //for hover only show container
         | 
| 1435 | 
            -
                                    this.show(); | 
| 1497 | 
            +
                                    this.show();
         | 
| 1436 1498 | 
             
                                } else {
         | 
| 1437 1499 | 
             
                                    //when toggle='click' we should not close all other containers as they will be closed automatically in document click listener
         | 
| 1438 1500 | 
             
                                    var closeAll = (this.options.toggle !== 'click');
         | 
| 1439 1501 | 
             
                                    this.toggle(closeAll);
         | 
| 1440 | 
            -
                                } | 
| 1502 | 
            +
                                }
         | 
| 1441 1503 | 
             
                            }, this));
         | 
| 1442 1504 | 
             
                        } else {
         | 
| 1443 1505 | 
             
                            this.$element.attr('tabindex', -1); //do not stop focus on element when toggled manually
         | 
| 1444 1506 | 
             
                        }
         | 
| 1445 1507 |  | 
| 1508 | 
            +
                        //if display is function it's far more convinient to have autotext = always to render correctly on init
         | 
| 1509 | 
            +
                        //see https://github.com/vitalets/x-editable-yii/issues/34
         | 
| 1510 | 
            +
                        if(typeof this.options.display === 'function') {
         | 
| 1511 | 
            +
                            this.options.autotext = 'always';
         | 
| 1512 | 
            +
                        }
         | 
| 1513 | 
            +
                        
         | 
| 1446 1514 | 
             
                        //check conditions for autotext:
         | 
| 1447 1515 | 
             
                        switch(this.options.autotext) {
         | 
| 1448 1516 | 
             
                          case 'always':
         | 
| @@ -1621,12 +1689,29 @@ Makes editable any HTML element on the page. Applied as jQuery method. | |
| 1621 1689 | 
             
                            return;
         | 
| 1622 1690 | 
             
                        }
         | 
| 1623 1691 |  | 
| 1624 | 
            -
                         | 
| 1692 | 
            +
                        /* 
         | 
| 1693 | 
            +
                        isEmpty may be set directly as param of method.
         | 
| 1694 | 
            +
                        It is required when we enable/disable field and can't rely on content 
         | 
| 1695 | 
            +
                        as node content is text: "Empty" that is not empty %)
         | 
| 1696 | 
            +
                        */
         | 
| 1697 | 
            +
                        if(isEmpty !== undefined) { 
         | 
| 1698 | 
            +
                            this.isEmpty = isEmpty;
         | 
| 1699 | 
            +
                        } else {
         | 
| 1700 | 
            +
                            //detect empty
         | 
| 1701 | 
            +
                            if($.trim(this.$element.html()) === '') { 
         | 
| 1702 | 
            +
                                this.isEmpty = true;
         | 
| 1703 | 
            +
                            } else if($.trim(this.$element.text()) !== '') {
         | 
| 1704 | 
            +
                                this.isEmpty = false;
         | 
| 1705 | 
            +
                            } else {
         | 
| 1706 | 
            +
                                //e.g. '<img>'
         | 
| 1707 | 
            +
                                this.isEmpty = !this.$element.height() || !this.$element.width();
         | 
| 1708 | 
            +
                            }
         | 
| 1709 | 
            +
                        }           
         | 
| 1625 1710 |  | 
| 1626 1711 | 
             
                        //emptytext shown only for enabled
         | 
| 1627 1712 | 
             
                        if(!this.options.disabled) {
         | 
| 1628 1713 | 
             
                            if (this.isEmpty) {
         | 
| 1629 | 
            -
                                this.$element. | 
| 1714 | 
            +
                                this.$element.html(this.options.emptytext);
         | 
| 1630 1715 | 
             
                                if(this.options.emptyclass) {
         | 
| 1631 1716 | 
             
                                    this.$element.addClass(this.options.emptyclass);
         | 
| 1632 1717 | 
             
                                }
         | 
| @@ -1721,6 +1806,21 @@ Makes editable any HTML element on the page. Applied as jQuery method. | |
| 1721 1806 | 
             
                            }
         | 
| 1722 1807 | 
             
                        }
         | 
| 1723 1808 |  | 
| 1809 | 
            +
                        //highlight when saving
         | 
| 1810 | 
            +
                        if(this.options.highlight) {
         | 
| 1811 | 
            +
                            var $e = this.$element,
         | 
| 1812 | 
            +
                                $bgColor = $e.css('background-color');
         | 
| 1813 | 
            +
                                
         | 
| 1814 | 
            +
                            $e.css('background-color', this.options.highlight);
         | 
| 1815 | 
            +
                            setTimeout(function(){
         | 
| 1816 | 
            +
                                $e.css('background-color', $bgColor);
         | 
| 1817 | 
            +
                                $e.addClass('editable-bg-transition');
         | 
| 1818 | 
            +
                                setTimeout(function(){
         | 
| 1819 | 
            +
                                   $e.removeClass('editable-bg-transition');  
         | 
| 1820 | 
            +
                                }, 1700);
         | 
| 1821 | 
            +
                            }, 0);
         | 
| 1822 | 
            +
                        }
         | 
| 1823 | 
            +
                        
         | 
| 1724 1824 | 
             
                        //set new value
         | 
| 1725 1825 | 
             
                        this.setValue(params.newValue, false, params.response);
         | 
| 1726 1826 |  | 
| @@ -1787,6 +1887,8 @@ Makes editable any HTML element on the page. Applied as jQuery method. | |
| 1787 1887 | 
             
                        if(this.container) {
         | 
| 1788 1888 | 
             
                           this.container.destroy(); 
         | 
| 1789 1889 | 
             
                        }
         | 
| 1890 | 
            +
                        
         | 
| 1891 | 
            +
                        this.input.destroy();
         | 
| 1790 1892 |  | 
| 1791 1893 | 
             
                        if(this.options.toggle !== 'manual') {
         | 
| 1792 1894 | 
             
                            this.$element.removeClass('editable-click');
         | 
| @@ -1844,28 +1946,37 @@ Makes editable any HTML element on the page. Applied as jQuery method. | |
| 1844 1946 | 
             
                        /**
         | 
| 1845 1947 | 
             
                        Returns current values of editable elements.   
         | 
| 1846 1948 | 
             
                        Note that it returns an **object** with name-value pairs, not a value itself. It allows to get data from several elements.    
         | 
| 1847 | 
            -
                        If value of some editable is `null` or `undefined` it is excluded from result object. | 
| 1949 | 
            +
                        If value of some editable is `null` or `undefined` it is excluded from result object.
         | 
| 1950 | 
            +
                        When param `isSingle` is set to **true** - it is supposed you have single element and will return value of editable instead of object.   
         | 
| 1848 1951 |  | 
| 1849 1952 | 
             
                        @method getValue()
         | 
| 1953 | 
            +
                        @param {bool} isSingle whether to return just value of single element
         | 
| 1850 1954 | 
             
                        @returns {Object} object of element names and values
         | 
| 1851 1955 | 
             
                        @example
         | 
| 1852 1956 | 
             
                        $('#username, #fullname').editable('getValue');
         | 
| 1853 | 
            -
                        // | 
| 1957 | 
            +
                        //result:
         | 
| 1854 1958 | 
             
                        {
         | 
| 1855 1959 | 
             
                        username: "superuser",
         | 
| 1856 1960 | 
             
                        fullname: "John"
         | 
| 1857 1961 | 
             
                        }
         | 
| 1962 | 
            +
                        //isSingle = true
         | 
| 1963 | 
            +
                        $('#username').editable('getValue', true);
         | 
| 1964 | 
            +
                        //result "superuser" 
         | 
| 1858 1965 | 
             
                        **/
         | 
| 1859 1966 | 
             
                        case 'getValue':
         | 
| 1860 | 
            -
                             | 
| 1861 | 
            -
                                 | 
| 1862 | 
            -
             | 
| 1863 | 
            -
             | 
| 1864 | 
            -
             | 
| 1865 | 
            -
             | 
| 1967 | 
            +
                            if(arguments.length === 2 && arguments[1] === true) { //isSingle = true
         | 
| 1968 | 
            +
                                result = this.eq(0).data(datakey).value;
         | 
| 1969 | 
            +
                            } else {
         | 
| 1970 | 
            +
                                this.each(function () {
         | 
| 1971 | 
            +
                                    var $this = $(this), data = $this.data(datakey);
         | 
| 1972 | 
            +
                                    if (data && data.value !== undefined && data.value !== null) {
         | 
| 1973 | 
            +
                                        result[data.options.name] = data.input.value2submit(data.value);
         | 
| 1974 | 
            +
                                    }
         | 
| 1975 | 
            +
                                });
         | 
| 1976 | 
            +
                            }
         | 
| 1866 1977 | 
             
                        return result;
         | 
| 1867 1978 |  | 
| 1868 | 
            -
                        /** | 
| 1979 | 
            +
                        /**
         | 
| 1869 1980 | 
             
                        This method collects values from several editable elements and submit them all to server.   
         | 
| 1870 1981 | 
             
                        Internally it runs client-side validation for all fields and submits only in case of success.  
         | 
| 1871 1982 | 
             
                        See <a href="#newrecord">creating new records</a> for details.
         | 
| @@ -2089,7 +2200,16 @@ Makes editable any HTML element on the page. Applied as jQuery method. | |
| 2089 2200 | 
             
                    });
         | 
| 2090 2201 | 
             
                    </script>
         | 
| 2091 2202 | 
             
                    **/         
         | 
| 2092 | 
            -
                    selector: null | 
| 2203 | 
            +
                    selector: null,
         | 
| 2204 | 
            +
                    /**
         | 
| 2205 | 
            +
                    Color used to highlight element after update. Implemented via CSS3 transition, works in modern browsers.
         | 
| 2206 | 
            +
                    
         | 
| 2207 | 
            +
                    @property highlight 
         | 
| 2208 | 
            +
                    @type string|boolean
         | 
| 2209 | 
            +
                    @since 1.4.5        
         | 
| 2210 | 
            +
                    @default #FFFF80 
         | 
| 2211 | 
            +
                    **/
         | 
| 2212 | 
            +
                    highlight: '#FFFF80'        
         | 
| 2093 2213 | 
             
                };
         | 
| 2094 2214 |  | 
| 2095 2215 | 
             
            }(window.jQuery));
         | 
| @@ -2103,23 +2223,23 @@ To create your own input you can inherit from this class. | |
| 2103 2223 | 
             
            **/
         | 
| 2104 2224 | 
             
            (function ($) {
         | 
| 2105 2225 | 
             
                "use strict";
         | 
| 2106 | 
            -
             | 
| 2226 | 
            +
             | 
| 2107 2227 | 
             
                //types
         | 
| 2108 2228 | 
             
                $.fn.editabletypes = {};
         | 
| 2109 | 
            -
             | 
| 2229 | 
            +
             | 
| 2110 2230 | 
             
                var AbstractInput = function () { };
         | 
| 2111 2231 |  | 
| 2112 2232 | 
             
                AbstractInput.prototype = {
         | 
| 2113 2233 | 
             
                   /**
         | 
| 2114 2234 | 
             
                    Initializes input
         | 
| 2115 | 
            -
             | 
| 2235 | 
            +
             | 
| 2116 2236 | 
             
                    @method init() 
         | 
| 2117 2237 | 
             
                    **/
         | 
| 2118 2238 | 
             
                   init: function(type, options, defaults) {
         | 
| 2119 2239 | 
             
                       this.type = type;
         | 
| 2120 2240 | 
             
                       this.options = $.extend({}, defaults, options);
         | 
| 2121 2241 | 
             
                   },
         | 
| 2122 | 
            -
             | 
| 2242 | 
            +
             | 
| 2123 2243 | 
             
                   /*
         | 
| 2124 2244 | 
             
                   this method called before render to init $tpl that is inserted in DOM
         | 
| 2125 2245 | 
             
                   */
         | 
| @@ -2133,107 +2253,107 @@ To create your own input you can inherit from this class. | |
| 2133 2253 | 
             
                   /**
         | 
| 2134 2254 | 
             
                    Renders input from tpl. Can return jQuery deferred object.
         | 
| 2135 2255 | 
             
                    Can be overwritten in child objects
         | 
| 2136 | 
            -
             | 
| 2137 | 
            -
                    @method render() | 
| 2138 | 
            -
                   **/ | 
| 2256 | 
            +
             | 
| 2257 | 
            +
                    @method render()
         | 
| 2258 | 
            +
                   **/
         | 
| 2139 2259 | 
             
                   render: function() {
         | 
| 2140 2260 |  | 
| 2141 2261 | 
             
                   }, 
         | 
| 2142 2262 |  | 
| 2143 2263 | 
             
                   /**
         | 
| 2144 2264 | 
             
                    Sets element's html by value. 
         | 
| 2145 | 
            -
             | 
| 2146 | 
            -
                    @method value2html(value, element) | 
| 2265 | 
            +
             | 
| 2266 | 
            +
                    @method value2html(value, element)
         | 
| 2147 2267 | 
             
                    @param {mixed} value
         | 
| 2148 2268 | 
             
                    @param {DOMElement} element
         | 
| 2149 | 
            -
                   **/ | 
| 2269 | 
            +
                   **/
         | 
| 2150 2270 | 
             
                   value2html: function(value, element) {
         | 
| 2151 | 
            -
                       $(element).text(value);
         | 
| 2271 | 
            +
                       $(element).text($.trim(value));
         | 
| 2152 2272 | 
             
                   },
         | 
| 2153 | 
            -
             | 
| 2273 | 
            +
             | 
| 2154 2274 | 
             
                   /**
         | 
| 2155 2275 | 
             
                    Converts element's html to value
         | 
| 2156 | 
            -
             | 
| 2157 | 
            -
                    @method html2value(html) | 
| 2276 | 
            +
             | 
| 2277 | 
            +
                    @method html2value(html)
         | 
| 2158 2278 | 
             
                    @param {string} html
         | 
| 2159 2279 | 
             
                    @returns {mixed}
         | 
| 2160 | 
            -
                   **/ | 
| 2280 | 
            +
                   **/
         | 
| 2161 2281 | 
             
                   html2value: function(html) {
         | 
| 2162 2282 | 
             
                       return $('<div>').html(html).text();
         | 
| 2163 2283 | 
             
                   },
         | 
| 2164 | 
            -
             | 
| 2284 | 
            +
             | 
| 2165 2285 | 
             
                   /**
         | 
| 2166 2286 | 
             
                    Converts value to string (for internal compare). For submitting to server used value2submit().
         | 
| 2167 | 
            -
             | 
| 2287 | 
            +
             | 
| 2168 2288 | 
             
                    @method value2str(value) 
         | 
| 2169 2289 | 
             
                    @param {mixed} value
         | 
| 2170 2290 | 
             
                    @returns {string}
         | 
| 2171 | 
            -
                   **/ | 
| 2291 | 
            +
                   **/
         | 
| 2172 2292 | 
             
                   value2str: function(value) {
         | 
| 2173 2293 | 
             
                       return value;
         | 
| 2174 2294 | 
             
                   }, 
         | 
| 2175 | 
            -
             | 
| 2295 | 
            +
             | 
| 2176 2296 | 
             
                   /**
         | 
| 2177 2297 | 
             
                    Converts string received from server into value. Usually from `data-value` attribute.
         | 
| 2178 | 
            -
             | 
| 2179 | 
            -
                    @method str2value(str) | 
| 2298 | 
            +
             | 
| 2299 | 
            +
                    @method str2value(str)
         | 
| 2180 2300 | 
             
                    @param {string} str
         | 
| 2181 2301 | 
             
                    @returns {mixed}
         | 
| 2182 | 
            -
                   **/ | 
| 2302 | 
            +
                   **/
         | 
| 2183 2303 | 
             
                   str2value: function(str) {
         | 
| 2184 2304 | 
             
                       return str;
         | 
| 2185 2305 | 
             
                   }, 
         | 
| 2186 2306 |  | 
| 2187 2307 | 
             
                   /**
         | 
| 2188 2308 | 
             
                    Converts value for submitting to server. Result can be string or object.
         | 
| 2189 | 
            -
             | 
| 2309 | 
            +
             | 
| 2190 2310 | 
             
                    @method value2submit(value) 
         | 
| 2191 2311 | 
             
                    @param {mixed} value
         | 
| 2192 2312 | 
             
                    @returns {mixed}
         | 
| 2193 | 
            -
                   **/ | 
| 2313 | 
            +
                   **/
         | 
| 2194 2314 | 
             
                   value2submit: function(value) {
         | 
| 2195 2315 | 
             
                       return value;
         | 
| 2196 | 
            -
                   }, | 
| 2197 | 
            -
             | 
| 2316 | 
            +
                   },
         | 
| 2317 | 
            +
             | 
| 2198 2318 | 
             
                   /**
         | 
| 2199 2319 | 
             
                    Sets value of input.
         | 
| 2200 | 
            -
             | 
| 2320 | 
            +
             | 
| 2201 2321 | 
             
                    @method value2input(value) 
         | 
| 2202 2322 | 
             
                    @param {mixed} value
         | 
| 2203 | 
            -
                   **/ | 
| 2323 | 
            +
                   **/
         | 
| 2204 2324 | 
             
                   value2input: function(value) {
         | 
| 2205 2325 | 
             
                       this.$input.val(value);
         | 
| 2206 2326 | 
             
                   },
         | 
| 2207 | 
            -
             | 
| 2327 | 
            +
             | 
| 2208 2328 | 
             
                   /**
         | 
| 2209 2329 | 
             
                    Returns value of input. Value can be object (e.g. datepicker)
         | 
| 2210 | 
            -
             | 
| 2330 | 
            +
             | 
| 2211 2331 | 
             
                    @method input2value() 
         | 
| 2212 | 
            -
                   **/ | 
| 2332 | 
            +
                   **/
         | 
| 2213 2333 | 
             
                   input2value: function() { 
         | 
| 2214 2334 | 
             
                       return this.$input.val();
         | 
| 2215 2335 | 
             
                   }, 
         | 
| 2216 2336 |  | 
| 2217 2337 | 
             
                   /**
         | 
| 2218 2338 | 
             
                    Activates input. For text it sets focus.
         | 
| 2219 | 
            -
             | 
| 2339 | 
            +
             | 
| 2220 2340 | 
             
                    @method activate() 
         | 
| 2221 | 
            -
                   **/ | 
| 2341 | 
            +
                   **/
         | 
| 2222 2342 | 
             
                   activate: function() {
         | 
| 2223 2343 | 
             
                       if(this.$input.is(':visible')) {
         | 
| 2224 2344 | 
             
                           this.$input.focus();
         | 
| 2225 2345 | 
             
                       }
         | 
| 2226 2346 | 
             
                   },
         | 
| 2227 | 
            -
             | 
| 2347 | 
            +
             | 
| 2228 2348 | 
             
                   /**
         | 
| 2229 2349 | 
             
                    Creates input.
         | 
| 2230 | 
            -
             | 
| 2350 | 
            +
             | 
| 2231 2351 | 
             
                    @method clear() 
         | 
| 2232 2352 | 
             
                   **/        
         | 
| 2233 2353 | 
             
                   clear: function() {
         | 
| 2234 2354 | 
             
                       this.$input.val(null);
         | 
| 2235 2355 | 
             
                   },
         | 
| 2236 | 
            -
             | 
| 2356 | 
            +
             | 
| 2237 2357 | 
             
                   /**
         | 
| 2238 2358 | 
             
                    method to escape html.
         | 
| 2239 2359 | 
             
                   **/
         | 
| @@ -2243,18 +2363,24 @@ To create your own input you can inherit from this class. | |
| 2243 2363 |  | 
| 2244 2364 | 
             
                   /**
         | 
| 2245 2365 | 
             
                    attach handler to automatically submit form when value changed (useful when buttons not shown)
         | 
| 2246 | 
            -
                   **/ | 
| 2366 | 
            +
                   **/
         | 
| 2247 2367 | 
             
                   autosubmit: function() {
         | 
| 2248 2368 |  | 
| 2249 2369 | 
             
                   },
         | 
| 2250 2370 |  | 
| 2371 | 
            +
                   /**
         | 
| 2372 | 
            +
                   Additional actions when destroying element 
         | 
| 2373 | 
            +
                   **/
         | 
| 2374 | 
            +
                    destroy: function() {
         | 
| 2375 | 
            +
                   },
         | 
| 2376 | 
            +
             | 
| 2251 2377 | 
             
                   // -------- helper functions --------
         | 
| 2252 2378 | 
             
                   setClass: function() {
         | 
| 2253 2379 | 
             
                       if(this.options.inputclass) {
         | 
| 2254 2380 | 
             
                           this.$input.addClass(this.options.inputclass); 
         | 
| 2255 2381 | 
             
                       } 
         | 
| 2256 2382 | 
             
                   },
         | 
| 2257 | 
            -
             | 
| 2383 | 
            +
             | 
| 2258 2384 | 
             
                   setAttr: function(attr) {
         | 
| 2259 2385 | 
             
                       if (this.options[attr] !== undefined && this.options[attr] !== null) {
         | 
| 2260 2386 | 
             
                           this.$input.attr(attr, this.options[attr]);
         | 
| @@ -2356,30 +2482,33 @@ List - abstract class for inputs that have source option loaded from js array or | |
| 2356 2482 | 
             
                    // ------------- additional functions ------------
         | 
| 2357 2483 |  | 
| 2358 2484 | 
             
                    onSourceReady: function (success, error) {
         | 
| 2485 | 
            +
                        //run source if it function
         | 
| 2486 | 
            +
                        var source;
         | 
| 2487 | 
            +
                        if ($.isFunction(this.options.source)) {
         | 
| 2488 | 
            +
                            source = this.options.source.call(this.options.scope);
         | 
| 2489 | 
            +
                            this.sourceData = null;
         | 
| 2490 | 
            +
                            //note: if function returns the same source as URL - sourceData will be taken from cahce and no extra request performed
         | 
| 2491 | 
            +
                        } else {
         | 
| 2492 | 
            +
                            source = this.options.source;
         | 
| 2493 | 
            +
                        }            
         | 
| 2494 | 
            +
                        
         | 
| 2359 2495 | 
             
                        //if allready loaded just call success
         | 
| 2360 | 
            -
                        if($.isArray(this.sourceData)) {
         | 
| 2496 | 
            +
                        if(this.options.sourceCache && $.isArray(this.sourceData)) {
         | 
| 2361 2497 | 
             
                            success.call(this);
         | 
| 2362 2498 | 
             
                            return; 
         | 
| 2363 2499 | 
             
                        }
         | 
| 2364 2500 |  | 
| 2365 | 
            -
                        // | 
| 2501 | 
            +
                        //try parse json in single quotes (for double quotes jquery does automatically)
         | 
| 2366 2502 | 
             
                        try {
         | 
| 2367 | 
            -
                             | 
| 2503 | 
            +
                            source = $.fn.editableutils.tryParseJson(source, false);
         | 
| 2368 2504 | 
             
                        } catch (e) {
         | 
| 2369 2505 | 
             
                            error.call(this);
         | 
| 2370 2506 | 
             
                            return;
         | 
| 2371 2507 | 
             
                        }
         | 
| 2372 | 
            -
                        
         | 
| 2373 | 
            -
                        var source = this.options.source;
         | 
| 2374 | 
            -
                        
         | 
| 2375 | 
            -
                        //run source if it function
         | 
| 2376 | 
            -
                        if ($.isFunction(source)) {
         | 
| 2377 | 
            -
                            source = source.call(this.options.scope);
         | 
| 2378 | 
            -
                        }
         | 
| 2379 2508 |  | 
| 2380 2509 | 
             
                        //loading from url
         | 
| 2381 2510 | 
             
                        if (typeof source === 'string') {
         | 
| 2382 | 
            -
                            //try to get from cache
         | 
| 2511 | 
            +
                            //try to get sourceData from cache
         | 
| 2383 2512 | 
             
                            if(this.options.sourceCache) {
         | 
| 2384 2513 | 
             
                                var cacheID = source,
         | 
| 2385 2514 | 
             
                                cache;
         | 
| @@ -3299,25 +3428,29 @@ Range (inherit from number) | |
| 3299 3428 | 
             
            }(window.jQuery));
         | 
| 3300 3429 | 
             
            /**
         | 
| 3301 3430 | 
             
            Select2 input. Based on amazing work of Igor Vaynberg https://github.com/ivaynberg/select2.  
         | 
| 3302 | 
            -
            Please see [original docs](http://ivaynberg.github.com/select2) for detailed description and options.  
         | 
| 3303 | 
            -
             | 
| 3431 | 
            +
            Please see [original select2 docs](http://ivaynberg.github.com/select2) for detailed description and options.  
         | 
| 3432 | 
            +
            Compatible **select2 version is 3.4.1**!   
         | 
| 3433 | 
            +
            You should manually download and include select2 distributive:  
         | 
| 3304 3434 |  | 
| 3305 3435 | 
             
                <link href="select2/select2.css" rel="stylesheet" type="text/css"></link>  
         | 
| 3306 3436 | 
             
                <script src="select2/select2.js"></script>  
         | 
| 3307 3437 |  | 
| 3308 | 
            -
             | 
| 3438 | 
            +
            To make it **bootstrap-styled** you can use css from [here](https://github.com/t0m/select2-bootstrap-css): 
         | 
| 3309 3439 |  | 
| 3310 3440 | 
             
                <link href="select2-bootstrap.css" rel="stylesheet" type="text/css"></link>    
         | 
| 3311 3441 |  | 
| 3312 | 
            -
            **Note:** currently ` | 
| 3313 | 
            -
             | 
| 3442 | 
            +
            **Note:** currently `autotext` feature does not work for select2 with `ajax` remote source.    
         | 
| 3443 | 
            +
            You need initially put both `data-value` and element's text youself:    
         | 
| 3444 | 
            +
             | 
| 3445 | 
            +
                <a href="#" data-type="select2" data-value="1">Text1</a>
         | 
| 3446 | 
            +
                
         | 
| 3314 3447 |  | 
| 3315 3448 | 
             
            @class select2
         | 
| 3316 3449 | 
             
            @extends abstractinput
         | 
| 3317 3450 | 
             
            @since 1.4.1
         | 
| 3318 3451 | 
             
            @final
         | 
| 3319 3452 | 
             
            @example
         | 
| 3320 | 
            -
            <a href="#" id="country" data-type="select2" data-pk="1" data-value="ru" data-url="/post" data- | 
| 3453 | 
            +
            <a href="#" id="country" data-type="select2" data-pk="1" data-value="ru" data-url="/post" data-title="Select country"></a>
         | 
| 3321 3454 | 
             
            <script>
         | 
| 3322 3455 | 
             
            $(function(){
         | 
| 3323 3456 | 
             
                $('#country').editable({
         | 
| @@ -3338,56 +3471,47 @@ $(function(){ | |
| 3338 3471 |  | 
| 3339 3472 | 
             
                var Constructor = function (options) {
         | 
| 3340 3473 | 
             
                    this.init('select2', options, Constructor.defaults);
         | 
| 3341 | 
            -
             | 
| 3474 | 
            +
             | 
| 3342 3475 | 
             
                    options.select2 = options.select2 || {};
         | 
| 3476 | 
            +
             | 
| 3477 | 
            +
                    this.sourceData = null;
         | 
| 3343 3478 |  | 
| 3344 | 
            -
                     | 
| 3345 | 
            -
             | 
| 3346 | 
            -
             | 
| 3347 | 
            -
             | 
| 3348 | 
            -
                   
         | 
| 3349 | 
            -
                   //detect whether it is multi-valued
         | 
| 3350 | 
            -
                   this.isMultiple = options.select2.tags || options.select2.multiple;
         | 
| 3479 | 
            +
                    //placeholder
         | 
| 3480 | 
            +
                    if(options.placeholder) {
         | 
| 3481 | 
            +
                        options.select2.placeholder = options.placeholder;
         | 
| 3482 | 
            +
                    }
         | 
| 3351 3483 |  | 
| 3352 | 
            -
             | 
| 3353 | 
            -
             | 
| 3354 | 
            -
                         | 
| 3355 | 
            -
             | 
| 3356 | 
            -
                         | 
| 3484 | 
            +
                    //if not `tags` mode, use source
         | 
| 3485 | 
            +
                    if(!options.select2.tags && options.source) {
         | 
| 3486 | 
            +
                        var source = options.source;
         | 
| 3487 | 
            +
                        //if source is function, call it (once!)
         | 
| 3488 | 
            +
                        if ($.isFunction(options.source)) {
         | 
| 3489 | 
            +
                            source = options.source.call(options.scope);
         | 
| 3490 | 
            +
                        }               
         | 
| 3357 3491 |  | 
| 3358 | 
            -
                         | 
| 3359 | 
            -
             | 
| 3360 | 
            -
                            // | 
| 3361 | 
            -
                             | 
| 3362 | 
            -
             | 
| 3363 | 
            -
                              console.log('attached');
         | 
| 3364 | 
            -
                              var original =  $(element).data('select2').postprocessResults;
         | 
| 3365 | 
            -
                              console.log(original);
         | 
| 3366 | 
            -
                              $(element).data('select2').postprocessResults = function(data, initial) {
         | 
| 3367 | 
            -
                                console.log('postprocess');
         | 
| 3368 | 
            -
                               // this.element.triggerHandler('loaded', [data]);
         | 
| 3369 | 
            -
                                original.apply(this, arguments);  
         | 
| 3370 | 
            -
                              }                  
         | 
| 3371 | 
            -
             | 
| 3372 | 
            -
                           //   $(element).on('loaded', function(){console.log('loaded');});
         | 
| 3373 | 
            -
                              $(element).data('select2').updateResults(true);
         | 
| 3492 | 
            +
                        if (typeof source === 'string') {
         | 
| 3493 | 
            +
                            options.select2.ajax = options.select2.ajax || {};
         | 
| 3494 | 
            +
                            //some default ajax params
         | 
| 3495 | 
            +
                            if(!options.select2.ajax.data) {
         | 
| 3496 | 
            +
                                options.select2.ajax.data = function(term) {return { query:term };};
         | 
| 3374 3497 | 
             
                            }
         | 
| 3375 | 
            -
                             | 
| 3376 | 
            -
             | 
| 3377 | 
            -
                            var val = that.str2value(element.val()),
         | 
| 3378 | 
            -
                                data = $.fn.editableutils.itemsByValue(val, mixin.data, 'id');
         | 
| 3379 | 
            -
                            
         | 
| 3380 | 
            -
                            //for single-valued mode should not use array. Take first element instead.
         | 
| 3381 | 
            -
                            if($.isArray(data) && data.length && !that.isMultiple) {
         | 
| 3382 | 
            -
                               data = data[0]; 
         | 
| 3498 | 
            +
                            if(!options.select2.ajax.results) {
         | 
| 3499 | 
            +
                                options.select2.ajax.results = function(data) { return {results:data };};
         | 
| 3383 3500 | 
             
                            }
         | 
| 3384 | 
            -
             | 
| 3385 | 
            -
             | 
| 3386 | 
            -
             | 
| 3387 | 
            -
             | 
| 3501 | 
            +
                            options.select2.ajax.url = source;
         | 
| 3502 | 
            +
                        } else {
         | 
| 3503 | 
            +
                            //check format and convert x-editable format to select2 format (if needed)
         | 
| 3504 | 
            +
                            this.sourceData = this.convertSource(source);
         | 
| 3505 | 
            +
                            options.select2.data = this.sourceData;
         | 
| 3506 | 
            +
                        }
         | 
| 3507 | 
            +
                    } 
         | 
| 3388 3508 |  | 
| 3389 3509 | 
             
                    //overriding objects in config (as by default jQuery extend() is not recursive)
         | 
| 3390 | 
            -
                    this.options.select2 = $.extend({}, Constructor.defaults.select2,  | 
| 3510 | 
            +
                    this.options.select2 = $.extend({}, Constructor.defaults.select2, options.select2);
         | 
| 3511 | 
            +
                    
         | 
| 3512 | 
            +
                    //detect whether it is multi-valued
         | 
| 3513 | 
            +
                    this.isMultiple = this.options.select2.tags || this.options.select2.multiple;
         | 
| 3514 | 
            +
                    this.isRemote = ('ajax' in this.options.select2);         
         | 
| 3391 3515 | 
             
                };
         | 
| 3392 3516 |  | 
| 3393 3517 | 
             
                $.fn.editableutils.inherit(Constructor, $.fn.editabletypes.abstractinput);
         | 
| @@ -3395,21 +3519,17 @@ $(function(){ | |
| 3395 3519 | 
             
                $.extend(Constructor.prototype, {
         | 
| 3396 3520 | 
             
                    render: function() {
         | 
| 3397 3521 | 
             
                        this.setClass();
         | 
| 3522 | 
            +
                        
         | 
| 3398 3523 | 
             
                        //apply select2
         | 
| 3399 3524 | 
             
                        this.$input.select2(this.options.select2);
         | 
| 3400 3525 |  | 
| 3401 | 
            -
                        //when data is loaded via ajax, we need to know when it's done
         | 
| 3402 | 
            -
                        if( | 
| 3403 | 
            -
             | 
| 3404 | 
            -
             | 
| 3405 | 
            -
             | 
| 3406 | 
            -
             | 
| 3407 | 
            -
                              this.element.triggerHandler('loaded', [data]);
         | 
| 3408 | 
            -
                              original.apply(this, arguments);  
         | 
| 3409 | 
            -
                          }
         | 
| 3410 | 
            -
                          */
         | 
| 3526 | 
            +
                        //when data is loaded via ajax, we need to know when it's done to populate listData
         | 
| 3527 | 
            +
                        if(this.isRemote) {
         | 
| 3528 | 
            +
                            //listen to loaded event to populate data
         | 
| 3529 | 
            +
                            this.$input.on('select2-loaded', $.proxy(function(e) {
         | 
| 3530 | 
            +
                                this.sourceData = e.items.results;
         | 
| 3531 | 
            +
                            }, this));
         | 
| 3411 3532 | 
             
                        }
         | 
| 3412 | 
            -
                                     
         | 
| 3413 3533 |  | 
| 3414 3534 | 
             
                        //trigger resize of editableform to re-position container in multi-valued mode           
         | 
| 3415 3535 | 
             
                        if(this.isMultiple) {
         | 
| @@ -3421,20 +3541,16 @@ $(function(){ | |
| 3421 3541 |  | 
| 3422 3542 | 
             
                   value2html: function(value, element) {
         | 
| 3423 3543 | 
             
                       var text = '', data;
         | 
| 3424 | 
            -
                        | 
| 3425 | 
            -
             | 
| 3426 | 
            -
             | 
| 3427 | 
            -
             | 
| 3428 | 
            -
             | 
| 3429 | 
            -
             | 
| 3430 | 
            -
             | 
| 3431 | 
            -
                           } else if(this.options.select2.data) {
         | 
| 3432 | 
            -
                               data = $.fn.editableutils.itemsByValue(value, this.options.select2.data, 'id');   
         | 
| 3433 | 
            -
                           } else {
         | 
| 3434 | 
            -
                               //if('ajax' in this.options.select2) {
         | 
| 3435 | 
            -
                           }
         | 
| 3544 | 
            +
                       
         | 
| 3545 | 
            +
                       if(this.options.select2.tags) { //in tags mode just assign value
         | 
| 3546 | 
            +
                          data = value; 
         | 
| 3547 | 
            +
                       } else if(this.sourceData) {
         | 
| 3548 | 
            +
                          data = $.fn.editableutils.itemsByValue(value, this.sourceData, 'id'); 
         | 
| 3549 | 
            +
                       } else {
         | 
| 3550 | 
            +
                          //can not get list of possible values (e.g. autotext for select2 with ajax source) 
         | 
| 3436 3551 | 
             
                       }
         | 
| 3437 3552 |  | 
| 3553 | 
            +
                       //data may be array (when multiple values allowed)          
         | 
| 3438 3554 | 
             
                       if($.isArray(data)) {
         | 
| 3439 3555 | 
             
                           //collect selected data and show with separator
         | 
| 3440 3556 | 
             
                           text = [];
         | 
| @@ -3455,7 +3571,26 @@ $(function(){ | |
| 3455 3571 | 
             
                   }, 
         | 
| 3456 3572 |  | 
| 3457 3573 | 
             
                   value2input: function(value) {
         | 
| 3458 | 
            -
                        | 
| 3574 | 
            +
                       //for remote source .val() is not working, need to look in sourceData 
         | 
| 3575 | 
            +
                       if(this.isRemote) {
         | 
| 3576 | 
            +
                           //todo: check value for array
         | 
| 3577 | 
            +
                           var item, items;
         | 
| 3578 | 
            +
                           //if sourceData loaded, use it to get text for display
         | 
| 3579 | 
            +
                           if(this.sourceData) {
         | 
| 3580 | 
            +
                               items = $.fn.editableutils.itemsByValue(value, this.sourceData, 'id');
         | 
| 3581 | 
            +
                               if(items.length) {
         | 
| 3582 | 
            +
                                   item = items[0];
         | 
| 3583 | 
            +
                               } 
         | 
| 3584 | 
            +
                           } 
         | 
| 3585 | 
            +
                           //if item not found by sourceData, use element text (e.g. for the first show)
         | 
| 3586 | 
            +
                           if(!item) {   
         | 
| 3587 | 
            +
                               item = {id: value, text: $(this.options.scope).text()};
         | 
| 3588 | 
            +
                           } 
         | 
| 3589 | 
            +
                           //select2('data', ...) allows to set both id and text --> usefull for initial show when items are not loaded   
         | 
| 3590 | 
            +
                           this.$input.select2('data', item).trigger('change', true); //second argument needed to separate initial change from user's click (for autosubmit)
         | 
| 3591 | 
            +
                       } else {
         | 
| 3592 | 
            +
                           this.$input.val(value).trigger('change', true); //second argument needed to separate initial change from user's click (for autosubmit)
         | 
| 3593 | 
            +
                       }
         | 
| 3459 3594 | 
             
                   },
         | 
| 3460 3595 |  | 
| 3461 3596 | 
             
                   input2value: function() { 
         | 
| @@ -3488,6 +3623,22 @@ $(function(){ | |
| 3488 3623 | 
             
                              $(this).closest('form').submit();
         | 
| 3489 3624 | 
             
                            }
         | 
| 3490 3625 | 
             
                        });
         | 
| 3626 | 
            +
                    },
         | 
| 3627 | 
            +
                    
         | 
| 3628 | 
            +
                    /*
         | 
| 3629 | 
            +
                    Converts source from x-editable format: {value: 1, text: "1"} to
         | 
| 3630 | 
            +
                    select2 format: {id: 1, text: "1"}
         | 
| 3631 | 
            +
                    */
         | 
| 3632 | 
            +
                    convertSource: function(source) {
         | 
| 3633 | 
            +
                        if($.isArray(source) && source.length && source[0].value !== undefined) {
         | 
| 3634 | 
            +
                            for(var i = 0; i<source.length; i++) {
         | 
| 3635 | 
            +
                                if(source[i].value !== undefined) {
         | 
| 3636 | 
            +
                                    source[i].id = source[i].value;
         | 
| 3637 | 
            +
                                    delete source[i].value;
         | 
| 3638 | 
            +
                                }
         | 
| 3639 | 
            +
                            }
         | 
| 3640 | 
            +
                        }
         | 
| 3641 | 
            +
                        return source;            
         | 
| 3491 3642 | 
             
                    }               
         | 
| 3492 3643 |  | 
| 3493 3644 | 
             
                });      
         | 
| @@ -3539,12 +3690,22 @@ $(function(){ | |
| 3539 3690 | 
             
            }(window.jQuery));
         | 
| 3540 3691 |  | 
| 3541 3692 | 
             
            /**
         | 
| 3542 | 
            -
            * Combodate - 1.0. | 
| 3693 | 
            +
            * Combodate - 1.0.4
         | 
| 3543 3694 | 
             
            * Dropdown date and time picker.
         | 
| 3544 3695 | 
             
            * Converts text input into dropdowns to pick day, month, year, hour, minute and second.
         | 
| 3545 3696 | 
             
            * Uses momentjs as datetime library http://momentjs.com.
         | 
| 3546 3697 | 
             
            * For i18n include corresponding file from https://github.com/timrwood/moment/tree/master/lang 
         | 
| 3547 3698 | 
             
            *
         | 
| 3699 | 
            +
            * Confusion at noon and midnight - see http://en.wikipedia.org/wiki/12-hour_clock#Confusion_at_noon_and_midnight
         | 
| 3700 | 
            +
            * In combodate: 
         | 
| 3701 | 
            +
            * 12:00 pm --> 12:00 (24-h format, midday)
         | 
| 3702 | 
            +
            * 12:00 am --> 00:00 (24-h format, midnight, start of day)
         | 
| 3703 | 
            +
            * 
         | 
| 3704 | 
            +
            * Differs from momentjs parse rules:
         | 
| 3705 | 
            +
            * 00:00 pm, 12:00 pm --> 12:00 (24-h format, day not change)
         | 
| 3706 | 
            +
            * 00:00 am, 12:00 am --> 00:00 (24-h format, day not change)
         | 
| 3707 | 
            +
            * 
         | 
| 3708 | 
            +
            * 
         | 
| 3548 3709 | 
             
            * Author: Vitaliy Potapov
         | 
| 3549 3710 | 
             
            * Project page: http://github.com/vitalets/combodate
         | 
| 3550 3711 | 
             
            * Copyright (c) 2012 Vitaliy Potapov. Released under MIT License.
         | 
| @@ -3694,9 +3855,10 @@ $(function(){ | |
| 3694 3855 |  | 
| 3695 3856 | 
             
                        for(i=0; i<=11; i++) {
         | 
| 3696 3857 | 
             
                            if(longNames) {
         | 
| 3697 | 
            -
                                 | 
| 3858 | 
            +
                                //see https://github.com/timrwood/momentjs.com/pull/36
         | 
| 3859 | 
            +
                                name = moment().date(1).month(i).format('MMMM');
         | 
| 3698 3860 | 
             
                            } else if(shortNames) {
         | 
| 3699 | 
            -
                                name = moment().month(i).format('MMM');
         | 
| 3861 | 
            +
                                name = moment().date(1).month(i).format('MMM');
         | 
| 3700 3862 | 
             
                            } else if(twoDigit) {
         | 
| 3701 3863 | 
             
                                name = this.leadZero(i+1);
         | 
| 3702 3864 | 
             
                            } else {
         | 
| @@ -3732,9 +3894,10 @@ $(function(){ | |
| 3732 3894 | 
             
                            h12 = this.options.template.indexOf('h') !== -1,
         | 
| 3733 3895 | 
             
                            h24 = this.options.template.indexOf('H') !== -1,
         | 
| 3734 3896 | 
             
                            twoDigit = this.options.template.toLowerCase().indexOf('hh') !== -1,
         | 
| 3897 | 
            +
                            min = h12 ? 1 : 0, 
         | 
| 3735 3898 | 
             
                            max = h12 ? 12 : 23;
         | 
| 3736 3899 |  | 
| 3737 | 
            -
                        for(i= | 
| 3900 | 
            +
                        for(i=min; i<=max; i++) {
         | 
| 3738 3901 | 
             
                            name = twoDigit ? this.leadZero(i) : i;
         | 
| 3739 3902 | 
             
                            items.push([i, name]);
         | 
| 3740 3903 | 
             
                        } 
         | 
| @@ -3783,7 +3946,7 @@ $(function(){ | |
| 3783 3946 | 
             
                    },                                       
         | 
| 3784 3947 |  | 
| 3785 3948 | 
             
                    /*
         | 
| 3786 | 
            -
                     Returns current date value. 
         | 
| 3949 | 
            +
                     Returns current date value from combos. 
         | 
| 3787 3950 | 
             
                     If format not specified - `options.format` used.
         | 
| 3788 3951 | 
             
                     If format = `null` - Moment object returned.
         | 
| 3789 3952 | 
             
                    */
         | 
| @@ -3812,12 +3975,14 @@ $(function(){ | |
| 3812 3975 | 
             
                           return '';
         | 
| 3813 3976 | 
             
                        }
         | 
| 3814 3977 |  | 
| 3815 | 
            -
                        //convert hours  | 
| 3978 | 
            +
                        //convert hours 12h --> 24h 
         | 
| 3816 3979 | 
             
                        if(this.$ampm) {
         | 
| 3817 | 
            -
             | 
| 3818 | 
            -
             | 
| 3819 | 
            -
             | 
| 3820 | 
            -
             | 
| 3980 | 
            +
                            //12:00 pm --> 12:00 (24-h format, midday), 12:00 am --> 00:00 (24-h format, midnight, start of day)
         | 
| 3981 | 
            +
                            if(values.hour === 12) {
         | 
| 3982 | 
            +
                                values.hour = this.$ampm.val() === 'am' ? 0 : 12;                    
         | 
| 3983 | 
            +
                            } else {
         | 
| 3984 | 
            +
                                values.hour = this.$ampm.val() === 'am' ? values.hour : values.hour+12;
         | 
| 3985 | 
            +
                            }
         | 
| 3821 3986 | 
             
                        }    
         | 
| 3822 3987 |  | 
| 3823 3988 | 
             
                        dt = moment([values.year, values.month, values.day, values.hour, values.minute, values.second]);
         | 
| @@ -3868,11 +4033,17 @@ $(function(){ | |
| 3868 4033 | 
             
                             });
         | 
| 3869 4034 |  | 
| 3870 4035 | 
             
                           if(this.$ampm) {
         | 
| 3871 | 
            -
                                | 
| 3872 | 
            -
             | 
| 4036 | 
            +
                               //12:00 pm --> 12:00 (24-h format, midday), 12:00 am --> 00:00 (24-h format, midnight, start of day)
         | 
| 4037 | 
            +
                               if(values.hour >= 12) {
         | 
| 3873 4038 | 
             
                                   values.ampm = 'pm';
         | 
| 4039 | 
            +
                                   if(values.hour > 12) {
         | 
| 4040 | 
            +
                                       values.hour -= 12;
         | 
| 4041 | 
            +
                                   }
         | 
| 3874 4042 | 
             
                               } else {
         | 
| 3875 | 
            -
                                   values.ampm = 'am'; | 
| 4043 | 
            +
                                   values.ampm = 'am';
         | 
| 4044 | 
            +
                                   if(values.hour === 0) {
         | 
| 4045 | 
            +
                                       values.hour = 12;
         | 
| 4046 | 
            +
                                   }
         | 
| 3876 4047 | 
             
                               } 
         | 
| 3877 4048 | 
             
                           }
         | 
| 3878 4049 |  |