
Polymer 2 deprecates the 'fire' method for legacy elements. So let's use gerrit's core fire method which is almost exactly the same. Change-Id: I59aebd29a89d26d9cb39e63e9a41afa9756b942f
244 lines
6.6 KiB
JavaScript
244 lines
6.6 KiB
JavaScript
/**
|
|
* @license
|
|
* Copyright (C) 2017 The Android Open Source Project
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
(function() {
|
|
'use strict';
|
|
|
|
const RESTORED_MESSAGE = 'Content restored from a previous edit.';
|
|
const SAVING_MESSAGE = 'Saving changes...';
|
|
const SAVED_MESSAGE = 'All changes saved';
|
|
const SAVE_FAILED_MSG = 'Failed to save changes';
|
|
|
|
const STORAGE_DEBOUNCE_INTERVAL_MS = 100;
|
|
|
|
Polymer({
|
|
is: 'gr-editor-view',
|
|
_legacyUndefinedCheck: true,
|
|
|
|
/**
|
|
* Fired when the title of the page should change.
|
|
*
|
|
* @event title-change
|
|
*/
|
|
|
|
/**
|
|
* Fired to notify the user of
|
|
*
|
|
* @event show-alert
|
|
*/
|
|
|
|
properties: {
|
|
/**
|
|
* URL params passed from the router.
|
|
*/
|
|
params: {
|
|
type: Object,
|
|
observer: '_paramsChanged',
|
|
},
|
|
|
|
_change: Object,
|
|
_changeEditDetail: Object,
|
|
_changeNum: String,
|
|
_patchNum: String,
|
|
_path: String,
|
|
_type: String,
|
|
_content: String,
|
|
_newContent: String,
|
|
_saving: {
|
|
type: Boolean,
|
|
value: false,
|
|
},
|
|
_successfulSave: {
|
|
type: Boolean,
|
|
value: false,
|
|
},
|
|
_saveDisabled: {
|
|
type: Boolean,
|
|
value: true,
|
|
computed: '_computeSaveDisabled(_content, _newContent, _saving)',
|
|
},
|
|
_prefs: Object,
|
|
},
|
|
|
|
behaviors: [
|
|
Gerrit.FireBehavior,
|
|
Gerrit.KeyboardShortcutBehavior,
|
|
Gerrit.PatchSetBehavior,
|
|
Gerrit.PathListBehavior,
|
|
],
|
|
|
|
listeners: {
|
|
'content-change': '_handleContentChange',
|
|
},
|
|
|
|
keyBindings: {
|
|
'ctrl+s meta+s': '_handleSaveShortcut',
|
|
},
|
|
|
|
attached() {
|
|
this._getEditPrefs().then(prefs => { this._prefs = prefs; });
|
|
},
|
|
|
|
get storageKey() {
|
|
return `c${this._changeNum}_ps${this._patchNum}_${this._path}`;
|
|
},
|
|
|
|
_getLoggedIn() {
|
|
return this.$.restAPI.getLoggedIn();
|
|
},
|
|
|
|
_getEditPrefs() {
|
|
return this.$.restAPI.getEditPreferences();
|
|
},
|
|
|
|
_paramsChanged(value) {
|
|
if (value.view !== Gerrit.Nav.View.EDIT) {
|
|
return;
|
|
}
|
|
|
|
this._changeNum = value.changeNum;
|
|
this._path = value.path;
|
|
this._patchNum = value.patchNum || this.EDIT_NAME;
|
|
|
|
// NOTE: This may be called before attachment (e.g. while parentElement is
|
|
// null). Fire title-change in an async so that, if attachment to the DOM
|
|
// has been queued, the event can bubble up to the handler in gr-app.
|
|
this.async(() => {
|
|
const title = `Editing ${this.computeTruncatedPath(this._path)}`;
|
|
this.fire('title-change', {title});
|
|
});
|
|
|
|
const promises = [];
|
|
|
|
promises.push(this._getChangeDetail(this._changeNum));
|
|
promises.push(
|
|
this._getFileData(this._changeNum, this._path, this._patchNum));
|
|
return Promise.all(promises);
|
|
},
|
|
|
|
_getChangeDetail(changeNum) {
|
|
return this.$.restAPI.getDiffChangeDetail(changeNum).then(change => {
|
|
this._change = change;
|
|
});
|
|
},
|
|
|
|
_handlePathChanged(e) {
|
|
const path = e.detail;
|
|
if (path === this._path) {
|
|
return Promise.resolve();
|
|
}
|
|
return this.$.restAPI.renameFileInChangeEdit(this._changeNum,
|
|
this._path, path).then(res => {
|
|
if (!res.ok) { return; }
|
|
|
|
this._successfulSave = true;
|
|
this._viewEditInChangeView();
|
|
});
|
|
},
|
|
|
|
_viewEditInChangeView() {
|
|
const patch = this._successfulSave ? this.EDIT_NAME : this._patchNum;
|
|
Gerrit.Nav.navigateToChange(this._change, patch, null,
|
|
patch !== this.EDIT_NAME);
|
|
},
|
|
|
|
_getFileData(changeNum, path, patchNum) {
|
|
const storedContent =
|
|
this.$.storage.getEditableContentItem(this.storageKey);
|
|
|
|
return this.$.restAPI.getFileContent(changeNum, path, patchNum)
|
|
.then(res => {
|
|
if (storedContent && storedContent.message &&
|
|
storedContent.message !== res.content) {
|
|
this.dispatchEvent(new CustomEvent('show-alert', {
|
|
detail: {message: RESTORED_MESSAGE},
|
|
bubbles: true,
|
|
composed: true,
|
|
}));
|
|
|
|
this._newContent = storedContent.message;
|
|
} else {
|
|
this._newContent = res.content || '';
|
|
}
|
|
this._content = res.content || '';
|
|
|
|
// A non-ok response may result if the file does not yet exist.
|
|
// The `type` field of the response is only valid when the file
|
|
// already exists.
|
|
if (res.ok && res.type) {
|
|
this._type = res.type;
|
|
} else {
|
|
this._type = '';
|
|
}
|
|
});
|
|
},
|
|
|
|
_saveEdit() {
|
|
this._saving = true;
|
|
this._showAlert(SAVING_MESSAGE);
|
|
this.$.storage.eraseEditableContentItem(this.storageKey);
|
|
return this.$.restAPI.saveChangeEdit(this._changeNum, this._path,
|
|
this._newContent).then(res => {
|
|
this._saving = false;
|
|
this._showAlert(res.ok ? SAVED_MESSAGE : SAVE_FAILED_MSG);
|
|
if (!res.ok) { return; }
|
|
|
|
this._content = this._newContent;
|
|
this._successfulSave = true;
|
|
});
|
|
},
|
|
|
|
_showAlert(message) {
|
|
this.dispatchEvent(new CustomEvent('show-alert', {
|
|
detail: {message},
|
|
bubbles: true,
|
|
composed: true,
|
|
}));
|
|
},
|
|
|
|
_computeSaveDisabled(content, newContent, saving) {
|
|
if (saving) {
|
|
return true;
|
|
}
|
|
return content === newContent;
|
|
},
|
|
|
|
_handleCloseTap() {
|
|
// TODO(kaspern): Add a confirm dialog if there are unsaved changes.
|
|
this._viewEditInChangeView();
|
|
},
|
|
|
|
_handleContentChange(e) {
|
|
this.debounce('store', () => {
|
|
const content = e.detail.value;
|
|
if (content) {
|
|
this.set('_newContent', e.detail.value);
|
|
this.$.storage.setEditableContentItem(this.storageKey, content);
|
|
} else {
|
|
this.$.storage.eraseEditableContentItem(this.storageKey);
|
|
}
|
|
}, STORAGE_DEBOUNCE_INTERVAL_MS);
|
|
},
|
|
|
|
_handleSaveShortcut(e) {
|
|
e.preventDefault();
|
|
if (!this._saveDisabled) {
|
|
this._saveEdit();
|
|
}
|
|
},
|
|
});
|
|
})();
|