Commit 002cc923 authored by Phil Hughes's avatar Phil Hughes

Added pending tabs to IDE

Pending tabs are normal tabs that are opened from the right sidebar.
They are opened in diff mode and when changed to edit mode they get
closed
& the actual file gets opened.
parent b3fb82a9
<script> <script>
import { mapActions } from 'vuex'; import { mapActions } from 'vuex';
import icon from '~/vue_shared/components/icon.vue'; import icon from '~/vue_shared/components/icon.vue';
import router from '../../ide_router'; import router from '../../ide_router';
export default { export default {
components: { components: {
icon, icon,
},
props: {
file: {
type: Object,
required: true,
}, },
props: { },
file: { computed: {
type: Object, iconName() {
required: true, return this.file.tempFile ? 'file-addition' : 'file-modified';
},
}, },
computed: { iconClass() {
iconName() { return `multi-file-${this.file.tempFile ? 'addition' : 'modified'} append-right-8`;
return this.file.tempFile ? 'file-addition' : 'file-modified';
},
iconClass() {
return `multi-file-${this.file.tempFile ? 'addition' : 'modified'} append-right-8`;
},
}, },
methods: { },
...mapActions([ methods: {
'discardFileChanges', ...mapActions(['discardFileChanges', 'updateViewer', 'openPendingTab']),
'updateViewer', openFileInEditor(file) {
]), return this.updateViewer('diff').then(() => {
openFileInEditor(file) { this.openPendingTab(file);
this.updateViewer('diff'); router.push(`/project/${file.projectId}/tree/master/`);
});
router.push(`/project${file.url}`);
},
}, },
}; },
};
</script> </script>
<template> <template>
......
<script> <script>
import { mapState, mapGetters } from 'vuex'; import { mapState, mapGetters } from 'vuex';
import ideSidebar from './ide_side_bar.vue'; import ideSidebar from './ide_side_bar.vue';
import ideContextbar from './ide_context_bar.vue'; import ideContextbar from './ide_context_bar.vue';
import repoTabs from './repo_tabs.vue'; import repoTabs from './repo_tabs.vue';
import repoFileButtons from './repo_file_buttons.vue'; import repoFileButtons from './repo_file_buttons.vue';
import ideStatusBar from './ide_status_bar.vue'; import ideStatusBar from './ide_status_bar.vue';
import repoEditor from './repo_editor.vue'; import repoEditor from './repo_editor.vue';
export default { export default {
components: { components: {
ideSidebar, ideSidebar,
ideContextbar, ideContextbar,
repoTabs, repoTabs,
repoFileButtons, repoFileButtons,
ideStatusBar, ideStatusBar,
repoEditor, repoEditor,
},
props: {
emptyStateSvgPath: {
type: String,
required: true,
}, },
props: { noChangesStateSvgPath: {
emptyStateSvgPath: { type: String,
type: String, required: true,
required: true,
},
noChangesStateSvgPath: {
type: String,
required: true,
},
committedStateSvgPath: {
type: String,
required: true,
},
}, },
computed: { committedStateSvgPath: {
...mapState(['changedFiles', 'openFiles', 'viewer']), type: String,
...mapGetters(['activeFile', 'hasChanges']), required: true,
}, },
mounted() { },
const returnValue = 'Are you sure you want to lose unsaved changes?'; computed: {
window.onbeforeunload = e => { ...mapState(['changedFiles', 'openFiles', 'viewer']),
if (!this.changedFiles.length) return undefined; ...mapGetters(['activeFile', 'hasChanges', 'tabs']),
},
mounted() {
const returnValue = 'Are you sure you want to lose unsaved changes?';
window.onbeforeunload = e => {
if (!this.changedFiles.length) return undefined;
Object.assign(e, { Object.assign(e, {
returnValue, returnValue,
}); });
return returnValue; return returnValue;
}; };
}, },
}; };
</script> </script>
<template> <template>
...@@ -60,7 +60,7 @@ ...@@ -60,7 +60,7 @@
v-if="activeFile" v-if="activeFile"
> >
<repo-tabs <repo-tabs
:files="openFiles" :files="tabs"
:viewer="viewer" :viewer="viewer"
:has-changes="hasChanges" :has-changes="hasChanges"
/> />
......
<script> <script>
import { mapActions } from 'vuex'; import { mapActions } from 'vuex';
import fileIcon from '~/vue_shared/components/file_icon.vue'; import fileIcon from '~/vue_shared/components/file_icon.vue';
import icon from '~/vue_shared/components/icon.vue'; import icon from '~/vue_shared/components/icon.vue';
import fileStatusIcon from './repo_file_status_icon.vue'; import fileStatusIcon from './repo_file_status_icon.vue';
import changedFileIcon from './changed_file_icon.vue'; import changedFileIcon from './changed_file_icon.vue';
export default { export default {
components: { components: {
fileStatusIcon, fileStatusIcon,
fileIcon, fileIcon,
icon, icon,
changedFileIcon, changedFileIcon,
},
props: {
tab: {
type: Object,
required: true,
}, },
props: { },
tab: { data() {
type: Object, return {
required: true, tabMouseOver: false,
}, };
},
computed: {
closeLabel() {
if (this.tab.changed || this.tab.tempFile) {
return `${this.tab.name} changed`;
}
return `Close ${this.tab.name}`;
}, },
data() { showChangedIcon() {
return { return this.tab.changed ? !this.tabMouseOver : false;
tabMouseOver: false,
};
},
computed: {
closeLabel() {
if (this.tab.changed || this.tab.tempFile) {
return `${this.tab.name} changed`;
}
return `Close ${this.tab.name}`;
},
showChangedIcon() {
return this.tab.changed ? !this.tabMouseOver : false;
},
}, },
},
methods: { methods: {
...mapActions([ ...mapActions(['closeFile']),
'closeFile', clickFile(tab) {
]), this.$router.push(`/project${tab.url}`);
clickFile(tab) { },
this.$router.push(`/project${tab.url}`); mouseOverTab() {
}, if (this.tab.changed) {
mouseOverTab() { this.tabMouseOver = true;
if (this.tab.changed) { }
this.tabMouseOver = true; },
} mouseOutTab() {
}, if (this.tab.changed) {
mouseOutTab() { this.tabMouseOver = false;
if (this.tab.changed) { }
this.tabMouseOver = false;
}
},
}, },
}; },
};
</script> </script>
<template> <template>
...@@ -66,7 +64,7 @@ ...@@ -66,7 +64,7 @@
<button <button
type="button" type="button"
class="multi-file-tab-close" class="multi-file-tab-close"
@click.stop.prevent="closeFile(tab.path)" @click.stop.prevent="closeFile(tab)"
:aria-label="closeLabel" :aria-label="closeLabel"
> >
<icon <icon
...@@ -82,7 +80,10 @@ ...@@ -82,7 +80,10 @@
<div <div
class="multi-file-tab" class="multi-file-tab"
:class="{active : tab.active }" :class="{
active: tab.active,
pending: tab.pending
}"
:title="tab.url" :title="tab.url"
> >
<file-icon <file-icon
......
<script> <script>
import { mapActions } from 'vuex'; import { mapActions } from 'vuex';
import RepoTab from './repo_tab.vue'; import RepoTab from './repo_tab.vue';
import EditorMode from './editor_mode_dropdown.vue'; import EditorMode from './editor_mode_dropdown.vue';
export default { export default {
components: { components: {
RepoTab, RepoTab,
EditorMode, EditorMode,
},
props: {
files: {
type: Array,
required: true,
}, },
props: { viewer: {
files: { type: String,
type: Array, required: true,
required: true,
},
viewer: {
type: String,
required: true,
},
hasChanges: {
type: Boolean,
required: true,
},
}, },
data() { hasChanges: {
return { type: Boolean,
showShadow: false, required: true,
};
}, },
updated() { },
if (!this.$refs.tabsScroller) return; data() {
return {
showShadow: false,
};
},
updated() {
if (!this.$refs.tabsScroller) return;
this.showShadow = this.showShadow = this.$refs.tabsScroller.scrollWidth > this.$refs.tabsScroller.offsetWidth;
this.$refs.tabsScroller.scrollWidth > this.$refs.tabsScroller.offsetWidth; },
}, methods: {
methods: { ...mapActions(['updateViewer']),
...mapActions(['updateViewer']), },
}, };
};
</script> </script>
<template> <template>
...@@ -47,7 +46,7 @@ ...@@ -47,7 +46,7 @@
> >
<repo-tab <repo-tab
v-for="tab in files" v-for="tab in files"
:key="tab.key" :key="`${tab.key}${tab.pending ? '-pending' : ''}`"
:tab="tab" :tab="tab"
/> />
</ul> </ul>
......
...@@ -6,8 +6,7 @@ import FilesDecoratorWorker from './workers/files_decorator_worker'; ...@@ -6,8 +6,7 @@ import FilesDecoratorWorker from './workers/files_decorator_worker';
export const redirectToUrl = (_, url) => visitUrl(url); export const redirectToUrl = (_, url) => visitUrl(url);
export const setInitialData = ({ commit }, data) => export const setInitialData = ({ commit }, data) => commit(types.SET_INITIAL_DATA, data);
commit(types.SET_INITIAL_DATA, data);
export const discardAllChanges = ({ state, commit, dispatch }) => { export const discardAllChanges = ({ state, commit, dispatch }) => {
state.changedFiles.forEach(file => { state.changedFiles.forEach(file => {
...@@ -43,14 +42,11 @@ export const createTempEntry = ( ...@@ -43,14 +42,11 @@ export const createTempEntry = (
) => ) =>
new Promise(resolve => { new Promise(resolve => {
const worker = new FilesDecoratorWorker(); const worker = new FilesDecoratorWorker();
const fullName = const fullName = name.slice(-1) !== '/' && type === 'tree' ? `${name}/` : name;
name.slice(-1) !== '/' && type === 'tree' ? `${name}/` : name;
if (state.entries[name]) { if (state.entries[name]) {
flash( flash(
`The name "${name `The name "${name.split('/').pop()}" is already taken in this directory.`,
.split('/')
.pop()}" is already taken in this directory.`,
'alert', 'alert',
document, document,
null, null,
......
...@@ -6,21 +6,26 @@ import * as types from '../mutation_types'; ...@@ -6,21 +6,26 @@ import * as types from '../mutation_types';
import router from '../../ide_router'; import router from '../../ide_router';
import { setPageTitle } from '../utils'; import { setPageTitle } from '../utils';
export const closeFile = ({ commit, state, getters, dispatch }, path) => { export const closeFile = ({ commit, state, getters, dispatch }, file) => {
const indexOfClosedFile = state.openFiles.findIndex(f => f.path === path); const path = file.path;
const file = state.entries[path];
const fileWasActive = file.active; if (file.pending) {
commit(types.REMOVE_PENDING_TAB, file);
} else {
const indexOfClosedFile = state.openFiles.findIndex(f => f.path === path);
const fileWasActive = file.active;
commit(types.TOGGLE_FILE_OPEN, path); commit(types.TOGGLE_FILE_OPEN, path);
commit(types.SET_FILE_ACTIVE, { path, active: false }); commit(types.SET_FILE_ACTIVE, { path, active: false });
if (state.openFiles.length > 0 && fileWasActive) { if (state.openFiles.length > 0 && fileWasActive) {
const nextIndexToOpen = indexOfClosedFile === 0 ? 0 : indexOfClosedFile - 1; const nextIndexToOpen = indexOfClosedFile === 0 ? 0 : indexOfClosedFile - 1;
const nextFileToOpen = state.entries[state.openFiles[nextIndexToOpen].path]; const nextFileToOpen = state.entries[state.openFiles[nextIndexToOpen].path];
router.push(`/project${nextFileToOpen.url}`); router.push(`/project${nextFileToOpen.url}`);
} else if (!state.openFiles.length) { } else if (!state.openFiles.length) {
router.push(`/project/${file.projectId}/tree/${file.branchId}/`); router.push(`/project/${file.projectId}/tree/${file.branchId}/`);
}
} }
eventHub.$emit(`editor.update.model.dispose.${file.path}`); eventHub.$emit(`editor.update.model.dispose.${file.path}`);
...@@ -66,14 +71,7 @@ export const getFileData = ({ state, commit, dispatch }, file) => { ...@@ -66,14 +71,7 @@ export const getFileData = ({ state, commit, dispatch }, file) => {
}) })
.catch(() => { .catch(() => {
commit(types.TOGGLE_LOADING, { entry: file }); commit(types.TOGGLE_LOADING, { entry: file });
flash( flash('Error loading file data. Please try again.', 'alert', document, null, false, true);
'Error loading file data. Please try again.',
'alert',
document,
null,
false,
true,
);
}); });
}; };
...@@ -84,14 +82,7 @@ export const getRawFileData = ({ commit, dispatch }, file) => ...@@ -84,14 +82,7 @@ export const getRawFileData = ({ commit, dispatch }, file) =>
commit(types.SET_FILE_RAW_DATA, { file, raw }); commit(types.SET_FILE_RAW_DATA, { file, raw });
}) })
.catch(() => .catch(() =>
flash( flash('Error loading file content. Please try again.', 'alert', document, null, false, true),
'Error loading file content. Please try again.',
'alert',
document,
null,
false,
true,
),
); );
export const changeFileContent = ({ state, commit }, { path, content }) => { export const changeFileContent = ({ state, commit }, { path, content }) => {
...@@ -119,10 +110,7 @@ export const setFileEOL = ({ getters, commit }, { eol }) => { ...@@ -119,10 +110,7 @@ export const setFileEOL = ({ getters, commit }, { eol }) => {
} }
}; };
export const setEditorPosition = ( export const setEditorPosition = ({ getters, commit }, { editorRow, editorColumn }) => {
{ getters, commit },
{ editorRow, editorColumn },
) => {
if (getters.activeFile) { if (getters.activeFile) {
commit(types.SET_FILE_POSITION, { commit(types.SET_FILE_POSITION, {
file: getters.activeFile, file: getters.activeFile,
...@@ -144,3 +132,7 @@ export const discardFileChanges = ({ state, commit }, path) => { ...@@ -144,3 +132,7 @@ export const discardFileChanges = ({ state, commit }, path) => {
eventHub.$emit(`editor.update.model.content.${file.path}`, file.raw); eventHub.$emit(`editor.update.model.content.${file.path}`, file.raw);
}; };
export const openPendingTab = ({ commit }, file) => {
commit(types.ADD_PENDING_TAB, file);
};
export const activeFile = state => export const tabs = state => state.openFiles.concat(state.pendingTabs);
state.openFiles.find(file => file.active) || null;
export const activeFile = state => tabs(state).find(file => file.active) || null;
export const addedFiles = state => state.changedFiles.filter(f => f.tempFile); export const addedFiles = state => state.changedFiles.filter(f => f.tempFile);
export const modifiedFiles = state => export const modifiedFiles = state => state.changedFiles.filter(f => !f.tempFile);
state.changedFiles.filter(f => !f.tempFile);
export const projectsWithTrees = state => export const projectsWithTrees = state =>
Object.keys(state.projects).map(projectId => { Object.keys(state.projects).map(projectId => {
......
...@@ -41,3 +41,6 @@ export const SET_ENTRIES = 'SET_ENTRIES'; ...@@ -41,3 +41,6 @@ export const SET_ENTRIES = 'SET_ENTRIES';
export const CREATE_TMP_ENTRY = 'CREATE_TMP_ENTRY'; export const CREATE_TMP_ENTRY = 'CREATE_TMP_ENTRY';
export const UPDATE_VIEWER = 'UPDATE_VIEWER'; export const UPDATE_VIEWER = 'UPDATE_VIEWER';
export const UPDATE_DELAY_VIEWER_CHANGE = 'UPDATE_DELAY_VIEWER_CHANGE'; export const UPDATE_DELAY_VIEWER_CHANGE = 'UPDATE_DELAY_VIEWER_CHANGE';
export const ADD_PENDING_TAB = 'ADD_PENDING_TAB';
export const REMOVE_PENDING_TAB = 'REMOVE_PENDING_TAB';
...@@ -80,4 +80,18 @@ export default { ...@@ -80,4 +80,18 @@ export default {
changed, changed,
}); });
}, },
[types.ADD_PENDING_TAB](state, file) {
Object.assign(state, {
pendingTabs: state.pendingTabs.concat({
...file,
active: true,
pending: true,
}),
});
},
[types.REMOVE_PENDING_TAB](state, file) {
Object.assign(state, {
pendingTabs: state.pendingTabs.filter(f => f.path !== file.path),
});
},
}; };
...@@ -16,4 +16,5 @@ export default () => ({ ...@@ -16,4 +16,5 @@ export default () => ({
entries: {}, entries: {},
viewer: 'editor', viewer: 'editor',
delayViewerUpdated: false, delayViewerUpdated: false,
pendingTabs: [],
}); });
...@@ -177,6 +177,10 @@ ...@@ -177,6 +177,10 @@
background-color: $white-light; background-color: $white-light;
border-bottom-color: $white-light; border-bottom-color: $white-light;
} }
&.pending {
font-style: italic;
}
} }
.multi-file-tab-close { .multi-file-tab-close {
...@@ -720,9 +724,7 @@ ...@@ -720,9 +724,7 @@
} }
.ide-view { .ide-view {
height: calc( height: calc(100vh - #{$header-height + $performance-bar-height + $flash-height});
100vh - #{$header-height + $performance-bar-height + $flash-height}
);
} }
} }
} }
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment