2018-02-10 15:18:01 +00:00
|
|
|
import ngModule from '../../module';
|
|
|
|
import Component from '../../lib/component';
|
|
|
|
import getModifiedData from '../../lib/modified';
|
|
|
|
import copyObject from '../../lib/copy';
|
|
|
|
import isEqual from '../../lib/equals';
|
|
|
|
import isFullEmpty from '../../lib/full-empty';
|
2017-02-06 17:01:04 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Component that checks for changes on a specific model property and
|
|
|
|
* asks the user to save or discard it when the state changes.
|
|
|
|
* Also it can save the data to the server when the @url and @idField
|
|
|
|
* properties are provided.
|
|
|
|
*/
|
|
|
|
export default class Watcher extends Component {
|
2018-05-25 15:25:35 +00:00
|
|
|
constructor($element, $scope, $state, $transitions, $http, vnApp, $translate, $attrs, $q) {
|
2017-02-06 17:01:04 +00:00
|
|
|
super($element);
|
2017-02-07 11:58:25 +00:00
|
|
|
this.$scope = $scope;
|
2017-02-06 17:01:04 +00:00
|
|
|
this.$state = $state;
|
|
|
|
this.$http = $http;
|
2017-02-07 16:38:30 +00:00
|
|
|
this.$translate = $translate;
|
2018-01-09 12:49:15 +00:00
|
|
|
this.$attrs = $attrs;
|
2017-06-05 07:01:21 +00:00
|
|
|
this.vnApp = vnApp;
|
2018-05-25 15:25:35 +00:00
|
|
|
this.$q = $q;
|
2017-02-06 17:01:04 +00:00
|
|
|
|
|
|
|
this.state = null;
|
|
|
|
this.deregisterCallback = $transitions.onStart({},
|
2017-03-07 11:37:59 +00:00
|
|
|
transition => this.callback(transition));
|
2017-10-05 07:31:28 +00:00
|
|
|
this.updateOriginalData();
|
2017-02-06 17:01:04 +00:00
|
|
|
}
|
2017-10-04 11:27:39 +00:00
|
|
|
|
2017-02-07 11:58:25 +00:00
|
|
|
$onInit() {
|
2017-09-28 09:29:01 +00:00
|
|
|
if (this.get && this.url) {
|
2017-02-07 11:58:25 +00:00
|
|
|
this.fetchData();
|
2017-09-28 09:29:01 +00:00
|
|
|
} else if (this.get && !this.url) {
|
|
|
|
throw new Error('Error: Parameter url ommitted');
|
2017-02-07 11:58:25 +00:00
|
|
|
}
|
|
|
|
}
|
2017-10-04 11:27:39 +00:00
|
|
|
|
2017-02-07 11:58:25 +00:00
|
|
|
$onChanges(changes) {
|
2017-03-07 11:37:59 +00:00
|
|
|
if (this.data) {
|
2017-10-05 07:31:28 +00:00
|
|
|
this.updateOriginalData();
|
2017-02-07 11:58:25 +00:00
|
|
|
}
|
|
|
|
}
|
2017-10-04 11:27:39 +00:00
|
|
|
|
2017-02-07 11:58:25 +00:00
|
|
|
$onDestroy() {
|
|
|
|
this.deregisterCallback();
|
|
|
|
}
|
2017-10-04 11:27:39 +00:00
|
|
|
|
2017-02-07 11:58:25 +00:00
|
|
|
fetchData() {
|
|
|
|
let id = this.data[this.idField];
|
2018-02-08 12:13:42 +00:00
|
|
|
// return new Promise((resolve, reject) => {
|
2017-09-28 09:29:01 +00:00
|
|
|
this.$http.get(`${this.url}/${id}`).then(
|
|
|
|
json => {
|
2017-10-05 07:20:40 +00:00
|
|
|
this.data = copyObject(json.data);
|
2017-10-05 07:31:28 +00:00
|
|
|
this.updateOriginalData();
|
2017-09-28 09:29:01 +00:00
|
|
|
}
|
|
|
|
// json => reject(json)
|
|
|
|
);
|
2018-02-08 12:13:42 +00:00
|
|
|
// });
|
2017-02-07 11:58:25 +00:00
|
|
|
}
|
2018-02-08 12:13:42 +00:00
|
|
|
|
2017-03-07 16:11:56 +00:00
|
|
|
/**
|
|
|
|
* Submits the data and goes back in the history.
|
2018-02-08 12:13:42 +00:00
|
|
|
*
|
|
|
|
* @return {Promise} The request promise
|
2017-03-07 16:11:56 +00:00
|
|
|
*/
|
|
|
|
submitBack() {
|
2018-05-25 15:25:35 +00:00
|
|
|
return this.submit().then(res => {
|
|
|
|
this.window.history.back();
|
|
|
|
return res;
|
|
|
|
});
|
2017-03-07 16:11:56 +00:00
|
|
|
}
|
2018-02-08 12:13:42 +00:00
|
|
|
|
2017-06-03 11:01:47 +00:00
|
|
|
/**
|
|
|
|
* Submits the data and goes another state.
|
|
|
|
*
|
|
|
|
* @param {String} state The state name
|
2018-02-08 12:13:42 +00:00
|
|
|
* @param {Object} params The request params
|
|
|
|
* @return {Promise} The request promise
|
2017-06-03 11:01:47 +00:00
|
|
|
*/
|
2017-12-04 07:17:29 +00:00
|
|
|
submitGo(state, params) {
|
2018-05-25 15:25:35 +00:00
|
|
|
return this.submit().then(res => {
|
|
|
|
this.$state.go(state, params || {});
|
|
|
|
return res;
|
|
|
|
});
|
2017-06-03 11:01:47 +00:00
|
|
|
}
|
2018-02-08 12:13:42 +00:00
|
|
|
|
2017-03-07 16:11:56 +00:00
|
|
|
/**
|
|
|
|
* Submits the data to the server.
|
|
|
|
*
|
|
|
|
* @return {Promise} The http request promise
|
|
|
|
*/
|
2017-02-06 17:01:04 +00:00
|
|
|
submit() {
|
2018-05-25 15:25:35 +00:00
|
|
|
return this.realSubmit().then(res => {
|
|
|
|
this.vnApp.showMessage(this.$translate.instant('Data saved!'));
|
|
|
|
return res;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
errorHandler(err) {
|
|
|
|
this.vnApp.showError(err.message);
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
realSubmit() {
|
2017-03-07 16:11:56 +00:00
|
|
|
if (this.form) {
|
|
|
|
this.form.$setSubmitted();
|
|
|
|
|
|
|
|
if (!this.form.$valid)
|
2018-05-25 15:25:35 +00:00
|
|
|
return this.invalidForm();
|
2017-02-06 17:01:04 +00:00
|
|
|
}
|
2018-05-25 15:25:35 +00:00
|
|
|
if (!this.dataChanged())
|
|
|
|
return this.noChanges();
|
|
|
|
|
2018-02-21 10:09:27 +00:00
|
|
|
let isPost = (this.$attrs.save && this.$attrs.save.toLowerCase() === 'post');
|
|
|
|
let changedData = isPost
|
2018-05-24 14:09:32 +00:00
|
|
|
? this.data
|
2018-02-08 11:57:57 +00:00
|
|
|
: getModifiedData(this.data, this.orgData);
|
2017-02-21 10:36:43 +00:00
|
|
|
|
2018-02-01 12:28:45 +00:00
|
|
|
if (this.save && this.save.accept) {
|
2018-05-25 15:25:35 +00:00
|
|
|
this.save.model = changedData;
|
|
|
|
return this.$q((resolve, reject) => {
|
2017-02-21 10:36:43 +00:00
|
|
|
this.save.accept().then(
|
|
|
|
json => this.writeData({data: json}, resolve),
|
|
|
|
json => reject(json)
|
|
|
|
);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
// XXX: Alternative when mgCrud is not used
|
2017-02-06 17:01:04 +00:00
|
|
|
|
2018-02-01 12:28:45 +00:00
|
|
|
let id = this.idField ? this.orgData[this.idField] : null;
|
2017-02-06 17:01:04 +00:00
|
|
|
|
2017-03-07 11:37:59 +00:00
|
|
|
if (id) {
|
2018-05-25 15:25:35 +00:00
|
|
|
return this.$q((resolve, reject) => {
|
2017-06-03 11:01:47 +00:00
|
|
|
this.$http.patch(`${this.url}/${id}`, changedData).then(
|
2017-02-07 16:02:17 +00:00
|
|
|
json => this.writeData(json, resolve),
|
2018-05-25 15:25:35 +00:00
|
|
|
reject
|
2017-02-07 11:58:25 +00:00
|
|
|
);
|
|
|
|
});
|
2017-02-06 17:01:04 +00:00
|
|
|
}
|
2017-03-07 11:37:59 +00:00
|
|
|
|
2018-05-25 15:25:35 +00:00
|
|
|
return this.$q((resolve, reject) => {
|
2018-02-01 12:28:45 +00:00
|
|
|
this.$http.post(this.url, changedData).then(
|
2017-03-07 11:37:59 +00:00
|
|
|
json => this.writeData(json, resolve),
|
2018-05-25 15:25:35 +00:00
|
|
|
reject
|
2017-03-07 11:37:59 +00:00
|
|
|
);
|
|
|
|
});
|
2017-02-06 17:01:04 +00:00
|
|
|
}
|
2017-10-04 11:27:39 +00:00
|
|
|
|
2018-05-25 15:25:35 +00:00
|
|
|
noChanges() {
|
|
|
|
let message = this.$translate.instant('No changes to save');
|
|
|
|
let p = this.$q.reject(new Error(message));
|
|
|
|
console.log(p);
|
|
|
|
return p;
|
2017-02-07 16:02:17 +00:00
|
|
|
}
|
2017-10-04 11:27:39 +00:00
|
|
|
|
2018-05-25 15:25:35 +00:00
|
|
|
invalidForm() {
|
|
|
|
let message = this.$translate.instant('Some fields are invalid');
|
|
|
|
return this.$q.reject(new Error(message));
|
2017-02-21 10:36:43 +00:00
|
|
|
}
|
2017-10-04 11:27:39 +00:00
|
|
|
|
2018-05-25 15:25:35 +00:00
|
|
|
writeData(json, resolve) {
|
|
|
|
Object.assign(this.data, json.data);
|
|
|
|
this.updateOriginalData();
|
|
|
|
resolve(json);
|
2017-02-21 10:36:43 +00:00
|
|
|
}
|
2017-10-04 11:27:39 +00:00
|
|
|
|
2017-10-05 07:31:28 +00:00
|
|
|
updateOriginalData() {
|
|
|
|
this.orgData = this.copyInNewObject(this.data);
|
2018-02-06 08:33:37 +00:00
|
|
|
if (this.form && this.form.$dirty)
|
|
|
|
this.form.$setPristine();
|
2017-02-06 17:01:04 +00:00
|
|
|
}
|
2017-10-04 11:27:39 +00:00
|
|
|
|
2017-10-05 07:31:28 +00:00
|
|
|
copyInNewObject(data) {
|
|
|
|
let newCopy = {};
|
|
|
|
if (data && typeof data === 'object') {
|
2017-07-13 12:55:07 +00:00
|
|
|
Object.keys(data).forEach(
|
|
|
|
val => {
|
2017-12-21 13:09:16 +00:00
|
|
|
if (!isFullEmpty(data[val])) {
|
2017-10-05 07:20:40 +00:00
|
|
|
if (typeof data[val] === 'object') {
|
2017-10-05 07:31:28 +00:00
|
|
|
newCopy[val] = this.copyInNewObject(data[val]);
|
2017-10-05 07:20:40 +00:00
|
|
|
} else {
|
2017-10-05 07:31:28 +00:00
|
|
|
newCopy[val] = data[val];
|
2017-10-05 07:20:40 +00:00
|
|
|
}
|
|
|
|
}
|
2017-07-13 12:55:07 +00:00
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
2017-10-05 07:31:28 +00:00
|
|
|
return newCopy;
|
2017-07-13 12:55:07 +00:00
|
|
|
}
|
2017-10-04 11:27:39 +00:00
|
|
|
|
2017-02-06 17:01:04 +00:00
|
|
|
callback(transition) {
|
2017-09-21 10:24:14 +00:00
|
|
|
let dataChanged = this.dataChanged();
|
|
|
|
if (!this.state && dataChanged) {
|
2017-02-06 17:01:04 +00:00
|
|
|
this.state = transition.to().name;
|
2017-02-07 16:38:30 +00:00
|
|
|
this.$scope.confirm.show();
|
2017-02-06 17:01:04 +00:00
|
|
|
return false;
|
|
|
|
}
|
2017-03-07 11:37:59 +00:00
|
|
|
|
2017-02-06 17:01:04 +00:00
|
|
|
return true;
|
|
|
|
}
|
2017-10-04 11:27:39 +00:00
|
|
|
|
2017-02-06 17:01:04 +00:00
|
|
|
dataChanged() {
|
2017-10-05 07:31:28 +00:00
|
|
|
let newData = this.copyInNewObject(this.data);
|
2018-04-06 16:40:24 +00:00
|
|
|
if (this.form && this.form.$dirty) return !isEqual(newData, this.orgData);
|
2017-10-05 07:51:27 +00:00
|
|
|
return !isEqual(newData, this.orgData);
|
2017-02-06 17:01:04 +00:00
|
|
|
}
|
2017-10-04 11:27:39 +00:00
|
|
|
|
2017-02-06 17:01:04 +00:00
|
|
|
onConfirmResponse(response) {
|
2017-03-07 11:37:59 +00:00
|
|
|
if (response === 'ACCEPT') {
|
2018-02-27 07:09:51 +00:00
|
|
|
if (this.data)
|
|
|
|
Object.assign(this.data, this.orgData);
|
2017-02-06 17:01:04 +00:00
|
|
|
this.$state.go(this.state);
|
2017-03-07 11:37:59 +00:00
|
|
|
} else {
|
2017-02-06 17:01:04 +00:00
|
|
|
this.state = null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-05-25 15:25:35 +00:00
|
|
|
Watcher.$inject = ['$element', '$scope', '$state', '$transitions', '$http', 'vnApp', '$translate', '$attrs', '$q'];
|
2017-02-06 17:01:04 +00:00
|
|
|
|
2018-02-10 15:18:01 +00:00
|
|
|
ngModule.component('vnWatcher', {
|
2017-05-31 08:28:39 +00:00
|
|
|
template: require('./watcher.html'),
|
2017-02-06 17:01:04 +00:00
|
|
|
bindings: {
|
|
|
|
url: '@?',
|
|
|
|
idField: '@?',
|
2017-02-07 11:58:25 +00:00
|
|
|
data: '<',
|
2017-02-21 10:36:43 +00:00
|
|
|
form: '<',
|
|
|
|
save: '<',
|
2017-02-07 11:58:25 +00:00
|
|
|
get: '=?'
|
2017-02-06 17:01:04 +00:00
|
|
|
},
|
|
|
|
controller: Watcher
|
|
|
|
});
|