2017-10-30 02:31:14 +03:00
|
|
|
/* global Marionette, Handlebars */
|
|
|
|
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
* @copyright Copyright (c) 2017, Daniel Calviño Sánchez (danxuliu@gmail.com)
|
|
|
|
*
|
|
|
|
* @license GNU AGPL version 3 or any later version
|
|
|
|
*
|
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU Affero General Public License as
|
|
|
|
* published by the Free Software Foundation, either version 3 of the
|
|
|
|
* License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU Affero General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Affero General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
(function(OCA, Marionette, Handlebars) {
|
|
|
|
'use strict';
|
|
|
|
|
|
|
|
OCA.SpreedMe = OCA.SpreedMe || {};
|
|
|
|
OCA.SpreedMe.Views = OCA.SpreedMe.Views || {};
|
|
|
|
|
|
|
|
var TEMPLATE =
|
|
|
|
'<div class="label-wrapper">' +
|
|
|
|
' <{{labelTagName}} class="label">{{text}}</{{labelTagName}}>' +
|
|
|
|
' {{#if editionEnabled}}' +
|
2018-03-23 19:05:31 +03:00
|
|
|
' <div class="edit-button"><span class="icon button icon-rename" {{#if buttonTitle}} title="{{buttonTitle}}" {{/if}}></span></div>' +
|
2017-10-30 02:31:14 +03:00
|
|
|
' {{/if}}' +
|
|
|
|
'</div>' +
|
|
|
|
'{{#if editionEnabled}}' +
|
|
|
|
' <div class="input-wrapper hidden-important">' +
|
2018-03-23 19:05:31 +03:00
|
|
|
' <input class="username" {{#if inputMaxLength}} maxlength="{{inputMaxLength}}" {{/if}} type="text" value="{{inputValue}}" {{#if inputPlaceholder}} placeholder="{{inputPlaceholder}}" {{/if}}>'+
|
|
|
|
' <input type="submit" value="" class="icon icon-confirm confirm-button"></div>'+
|
2017-10-30 02:31:14 +03:00
|
|
|
' </div>' +
|
|
|
|
'{{/if}}';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* View for an editable text label.
|
|
|
|
*
|
|
|
|
* In its main state, an EditableTextLabel shows text in a label (an HTML
|
|
|
|
* element that can contain a line of text, like "<h1>" or "<p>"). The text
|
|
|
|
* comes from an attribute in a Backbone model and is automatically updated
|
|
|
|
* when the attribute changes.
|
|
|
|
*
|
|
|
|
* It also provides an edition state in which a text input field replaces
|
|
|
|
* the label, making possible to edit and save the attribute of the model.
|
|
|
|
* The EditableTextLabel can be make read-only by calling
|
|
|
|
* "disableEdition()", or read-write by calling "enableEdition()".
|
|
|
|
*
|
|
|
|
* The EditableTextLabel works on a single attribute of a model; they must
|
|
|
|
* be set in the constructor using the "model" and "modelAttribute" options
|
|
|
|
* (the first is the Backbone model to get the attribute from, the second is
|
|
|
|
* the name of the attribute). The "modelSaveOptions" option can be set if
|
|
|
|
* needed to control the options passed to "Model.save", and
|
2017-10-30 08:27:01 +03:00
|
|
|
* "extraClassNames", "labelTagName", "labelPlaceholder", "inputMaxLength",
|
|
|
|
* "inputPlaceholder" and "buttonTitle" can be used to customize some
|
|
|
|
* elements of the view.
|
2018-01-10 02:18:16 +03:00
|
|
|
*
|
|
|
|
* After initialization, and once the view has been rendered, the
|
|
|
|
* "modelAttribute" and "labelPlaceholder" options can be updated using the
|
|
|
|
* "setModelAttribute" and "setLabelPlaceholder" methods.
|
2017-10-30 02:31:14 +03:00
|
|
|
*/
|
|
|
|
var EditableTextLabel = Marionette.View.extend({
|
|
|
|
|
|
|
|
className: function() {
|
|
|
|
return 'editable-text-label' + (this.getOption('extraClassNames')? ' ' + this.getOption('extraClassNames') : '');
|
|
|
|
},
|
|
|
|
|
|
|
|
labelTagName: 'p',
|
|
|
|
|
|
|
|
buttonTitle: t('spreed', 'Edit'),
|
|
|
|
|
|
|
|
ui: {
|
|
|
|
labelWrapper: '.label-wrapper',
|
|
|
|
label: '.label',
|
|
|
|
editButton: '.edit-button',
|
|
|
|
inputWrapper: '.input-wrapper',
|
2018-03-23 19:05:31 +03:00
|
|
|
input: 'input.username',
|
2017-10-30 02:31:14 +03:00
|
|
|
confirmButton: '.confirm-button',
|
|
|
|
},
|
|
|
|
|
|
|
|
events: {
|
|
|
|
'click @ui.editButton': 'showInput',
|
|
|
|
'keyup @ui.input': 'handleInputKeyUp',
|
|
|
|
'click @ui.confirmButton': 'confirmEdit',
|
|
|
|
},
|
|
|
|
|
|
|
|
modelEvents: function() {
|
|
|
|
var modelEvents = {};
|
|
|
|
modelEvents['change:' + this.modelAttribute] = 'updateText';
|
|
|
|
|
|
|
|
return modelEvents;
|
|
|
|
},
|
|
|
|
|
|
|
|
template: Handlebars.compile(TEMPLATE),
|
|
|
|
|
|
|
|
templateContext: function() {
|
|
|
|
return {
|
2017-10-30 08:27:01 +03:00
|
|
|
text: this._getText(),
|
2017-10-30 02:31:14 +03:00
|
|
|
|
|
|
|
editionEnabled: this._editionEnabled,
|
|
|
|
|
|
|
|
labelTagName: this.getOption('labelTagName'),
|
|
|
|
inputMaxLength: this.getOption('inputMaxLength'),
|
2017-10-30 08:27:01 +03:00
|
|
|
// The text of the label is not used as input value as it could
|
|
|
|
// contain a placeholder text.
|
|
|
|
inputValue: this.model.get(this.modelAttribute),
|
2017-10-30 02:31:14 +03:00
|
|
|
inputPlaceholder: this.getOption('inputPlaceholder'),
|
|
|
|
buttonTitle: this.getOption('buttonTitle')
|
|
|
|
};
|
|
|
|
},
|
|
|
|
|
|
|
|
initialize: function(options) {
|
2018-01-10 02:18:16 +03:00
|
|
|
this.mergeOptions(options, ['model', 'modelAttribute', 'modelSaveOptions', 'labelPlaceholder']);
|
2017-10-30 02:31:14 +03:00
|
|
|
|
|
|
|
this._editionEnabled = true;
|
|
|
|
},
|
|
|
|
|
2018-01-10 02:18:16 +03:00
|
|
|
setModelAttribute: function(modelAttribute) {
|
|
|
|
if (this.modelAttribute === modelAttribute) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
var modelEvents = _.result(this, 'modelEvents');
|
|
|
|
this.unbindEvents(this.model, modelEvents);
|
|
|
|
|
|
|
|
this.modelAttribute = modelAttribute;
|
|
|
|
|
|
|
|
modelEvents = _.result(this, 'modelEvents');
|
|
|
|
this.bindEvents(this.model, modelEvents);
|
|
|
|
|
|
|
|
this.updateText();
|
|
|
|
this.hideInput();
|
|
|
|
},
|
|
|
|
|
|
|
|
setLabelPlaceholder: function(labelPlaceholder) {
|
|
|
|
if (this.labelPlaceholder === labelPlaceholder) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.labelPlaceholder = labelPlaceholder;
|
|
|
|
|
|
|
|
this.updateText();
|
|
|
|
},
|
|
|
|
|
2017-10-30 02:31:14 +03:00
|
|
|
enableEdition: function() {
|
|
|
|
if (this._editionEnabled) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
this._editionEnabled = true;
|
|
|
|
|
|
|
|
this.render();
|
|
|
|
},
|
|
|
|
|
|
|
|
disableEdition: function() {
|
|
|
|
if (!this._editionEnabled) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
this._editionEnabled = false;
|
|
|
|
|
|
|
|
this.render();
|
|
|
|
},
|
|
|
|
|
2017-10-30 08:27:01 +03:00
|
|
|
_getText: function() {
|
2018-01-10 02:18:16 +03:00
|
|
|
return this.model.get(this.modelAttribute) || this.labelPlaceholder || '';
|
2017-10-30 08:27:01 +03:00
|
|
|
},
|
|
|
|
|
2017-10-30 02:31:14 +03:00
|
|
|
updateText: function() {
|
2017-10-30 08:27:01 +03:00
|
|
|
this.getUI('label').text(this._getText());
|
2017-10-30 02:31:14 +03:00
|
|
|
},
|
|
|
|
|
|
|
|
showInput: function() {
|
2018-01-10 01:28:58 +03:00
|
|
|
this.getUI('input').val(this.model.get(this.modelAttribute));
|
|
|
|
|
2017-10-30 02:31:14 +03:00
|
|
|
this.getUI('inputWrapper').removeClass('hidden-important');
|
|
|
|
this.getUI('labelWrapper').addClass('hidden-important');
|
|
|
|
|
|
|
|
this.getUI('input').focus();
|
|
|
|
},
|
|
|
|
|
|
|
|
hideInput: function() {
|
|
|
|
this.getUI('labelWrapper').removeClass('hidden-important');
|
|
|
|
this.getUI('inputWrapper').addClass('hidden-important');
|
|
|
|
},
|
|
|
|
|
|
|
|
handleInputKeyUp: function(event) {
|
|
|
|
if (event.keyCode === 13) {
|
|
|
|
// Enter
|
|
|
|
this.confirmEdit();
|
|
|
|
} else if (event.keyCode === 27) {
|
|
|
|
// ESC
|
2018-01-10 01:28:58 +03:00
|
|
|
this.hideInput();
|
2017-10-30 02:31:14 +03:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
confirmEdit: function() {
|
|
|
|
var newText = this.getUI('input').val().trim();
|
|
|
|
|
|
|
|
if (newText === this.model.get(this.modelAttribute)) {
|
|
|
|
this.hideInput();
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-02-27 22:03:56 +03:00
|
|
|
// TODO This should show the error message instead of just hiding
|
|
|
|
// the input without changes.
|
|
|
|
var hideInputOnValidationError = function(/*model, error*/) {
|
|
|
|
this.hideInput();
|
|
|
|
}.bind(this);
|
|
|
|
this.model.listenToOnce(this.model, 'invalid', hideInputOnValidationError);
|
|
|
|
|
2017-10-30 02:31:14 +03:00
|
|
|
var options = _.clone(this.modelSaveOptions || {});
|
|
|
|
options.success = _.bind(function() {
|
2019-02-27 22:03:56 +03:00
|
|
|
this.model.stopListening(this.model, 'invalid', hideInputOnValidationError);
|
|
|
|
|
2017-10-30 02:31:14 +03:00
|
|
|
this.hideInput();
|
|
|
|
|
2018-01-15 17:43:27 +03:00
|
|
|
if (this.modelSaveOptions && _.isFunction(this.modelSaveOptions.success)) {
|
2017-10-30 02:31:14 +03:00
|
|
|
this.modelSaveOptions.success.apply(this, arguments);
|
|
|
|
}
|
|
|
|
}, this);
|
2019-02-20 19:48:05 +03:00
|
|
|
options.error = _.bind(function() {
|
2019-02-27 22:03:56 +03:00
|
|
|
this.model.stopListening(this.model, 'invalid', hideInputOnValidationError);
|
|
|
|
|
2019-02-20 19:48:05 +03:00
|
|
|
this.hideInput();
|
|
|
|
}, this);
|
2017-10-30 02:31:14 +03:00
|
|
|
|
|
|
|
this.model.save(this.modelAttribute, newText, options);
|
|
|
|
},
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
OCA.SpreedMe.Views.EditableTextLabel = EditableTextLabel;
|
|
|
|
|
|
|
|
})(OCA, Marionette, Handlebars);
|