Skip to content
Snippets Groups Projects
CoursewareCourseManager.vue 14 KiB
Newer Older
<template>
    <div class="cw-course-manager">
        <courseware-tabs class="cw-course-manager-tabs">
            <courseware-tab :name="$gettext('Diese Courseware')" :selected="true">
                <courseware-manager-element
                    type="current"
                    :currentElement="currentElement"
                    @selectElement="setCurrentId"
Ron Lucke's avatar
Ron Lucke committed
                    @reloadElement="reloadElements"
                />
            </courseware-tab>
            <courseware-tab :name="$gettext('Export')">
                <button
                    class="button"
                    @click.prevent="doExportCourseware"
                    :class="{
                        disabled: exportRunning,
                    }"
                >
                    <translate>Alles exportieren</translate>
                </button>
Ron Lucke's avatar
Ron Lucke committed
                <courseware-companion-box v-show="exportRunning" :msgCompanion="$gettext('Export läuft, bitte haben sie einen Moment Geduld...')" mood="pointing"/>
                <div v-if="exportRunning" class="cw-import-zip">
                    <header>{{exportState}}:</header>
                    <div class="progress-bar-wrapper">
                        <div class="progress-bar" role="progressbar" :style="{width: exportProgress + '%'}" :aria-valuenow="exportProgress" aria-valuemin="0" aria-valuemax="100">{{ exportProgress }}%</div>
                    </div>
                </div>
            </courseware-tab>
        </courseware-tabs>

        <courseware-tabs class="cw-course-manager-tabs">
            <courseware-tab :name="$gettext('FAQ')">
                <courseware-collapsible-box :open="true" :title="$gettext('Wie finde ich die gewünschte Stelle?')">
                    <p><translate>
                        Wählen Sie auf der linken Seite "Diese Courseware" aus.
                        Beim laden der Seite ist dies immer gewählt. Die Überschrift
                        gibt an welche Seite Sie grade ausgewählt haben. Darunter befinden
                        sich die Abschnitte der Seite und innerhalb dieser dessen Blöcke.
                        Möchten Sie eine Seite die unterhalb der gewählten liegt bearbeiten,
                        können Sie diese über die Schaltflächen im Bereich "Seiten" wählen.
                        Über der Überschrift wird eine Navigation eingeblendet, mit dieser können
                        Sie beliebig weit hoch in der Hierarchie springen.
                    </translate></p>
                </courseware-collapsible-box>
                <courseware-collapsible-box :title="$gettext('Wie sortiere ich Objekte?')">
                    <p><translate>
                        Seiten, Abschnitte und Blöcke lassen sich in ihrer Reihenfolge sortieren.
                        Hierzu wählen Sie auf der linken Seite unter "Diese Courseware" die Schaltfläche "Seiten sortieren",
                        "Abschnitte sortieren" oder "Blöcke sortieren".
                        An den Objekten werden Pfeile angezeigt, mit diesen können die Objekte an die gewünschte
                        Position gebracht werden. Um die neue Sortierung zu speichern wählen Sie "Sortieren beenden".
                        Sie können die Änderungen auch rückgängig machen indem Sie "Sortieren abbrechen" wählen.
                    </translate></p>
                </courseware-collapsible-box>
                <courseware-collapsible-box :title="$gettext('Wie verschiebe ich Objekte?')">
                    <p><translate>
                        Seiten, Abschnitte und Blöcke lassen sich verschieben.
                        Hierzu wählen Sie auf der linken Seite unter "Diese Courseware" die Schaltfläche
                        "Seite an diese Stelle einfügen", "Abschnitt an diese Stelle einfügen" oder
                        "Block an diese Stelle einfügen". Wählen Sie dann auf der rechten Seite unter
                        "Verschieben" das Objekt aus das Sie verschieben möchten. Verschiebbare Objekte
                        erkennen Sie an den zwei nach links zeigenden gelben Pfeilen.
                    </translate></p>
                </courseware-collapsible-box>
                <courseware-collapsible-box :title="$gettext('Wie kopiere ich Objekte?')">
                    <p><translate>
                        Seiten, Abschnitte und Blöcke lassen sich aus einer anderen Veranstaltung und Ihren
                        eigenen Inhalten kopieren.
                        Hierzu wählen Sie auf der linken Seite unter "Diese Courseware" die Schaltfläche
                        "Seite an diese Stelle einfügen", "Abschnitt an diese Stelle einfügen" oder
                        "Block an diese Stelle einfügen". Wählen Sie dann auf der rechten Seite unter
                        "Kopieren" erst die Veranstaltung aus der Sie kopieren möchten oder Ihre eigenen
                        Inhalte. Wählen sie dann das Objekt aus das Sie kopieren möchten. Kopierbare Objekte
                        erkennen Sie an den zwei nach links zeigenden gelben Pfeilen.
                    </translate></p>
                </courseware-collapsible-box>
            </courseware-tab>
            <courseware-tab name="Verschieben" :selected="true">
                <courseware-manager-element
                type="self"
                :currentElement="selfElement"
                :moveSelfPossible="moveSelfPossible"
                :moveSelfChildPossible="moveSelfChildPossible"
                @selectElement="setSelfId"
                @reloadElement="reloadElements"
                />
            </courseware-tab>

            <courseware-tab :name="$gettext('Kopieren')">
                <courseware-manager-copy-selector @loadSelf="reloadElements"/>
            </courseware-tab>

            <courseware-tab :name="$gettext('Importieren')">
Ron Lucke's avatar
Ron Lucke committed
                <courseware-companion-box v-show="!importRunning && importDone" :msgCompanion="$gettext('Import erfolgreich!')" mood="special"/>
                <courseware-companion-box v-show="importRunning" :msgCompanion="$gettext('Import läuft. Bitte verlassen Sie die Seite nicht bis der Import abgeschlossen wurde.')" mood="pointing"/>
Ron Lucke's avatar
Ron Lucke committed
                    v-show="!importRunning"
                    class="button"
                    @click.prevent="chooseFile"
                >
Ron Lucke's avatar
Ron Lucke committed
                    <translate>Importdatei auswählen</translate>
Ron Lucke's avatar
Ron Lucke committed
                <div v-if="importZip" class="cw-import-zip">
                    <header>{{ importZip.name }}</header>
                    <p><translate>Größe</translate>: {{ getFileSizeText(importZip.size) }}</p>
Ron Lucke's avatar
Ron Lucke committed
                <div v-if="importRunning" class="cw-import-zip">
                    <header><translate>Importiere Dateien</translate>:</header>
                    <div class="progress-bar-wrapper">
                        <div class="progress-bar" role="progressbar" :style="{width: importFilesProgress + '%'}" :aria-valuenow="importFilesProgress" aria-valuemin="0" aria-valuemax="100">{{ importFilesProgress }}%</div>
                    </div>
                    {{ importFilesState }}
                </div>
Ron Lucke's avatar
Ron Lucke committed
                <div v-if="fileImportDone && importRunning" class="cw-import-zip">
                    <header><translate>Importiere Elemente</translate>:</header>
                    <div class="progress-bar-wrapper">
                        <div class="progress-bar" role="progressbar" :style="{width: importStructuresProgress + '%'}" :aria-valuenow="importStructuresProgress" aria-valuemin="0" aria-valuemax="100">{{ importStructuresProgress }}%</div>
                    </div>
                    {{ importStructuresState }}
Ron Lucke's avatar
Ron Lucke committed
                    v-show="importZip && !importRunning"
                    class="button"
                    @click.prevent="doImportCourseware"
                >
                    <translate>Alles importieren</translate>
                </button>

                <input ref="importFile" type="file" accept=".zip" @change="setImport" style="visibility: hidden" />
            </courseware-tab>
        </courseware-tabs>
    </div>
</template>
<script>
import CoursewareTabs from './CoursewareTabs.vue';
import CoursewareTab from './CoursewareTab.vue';
import CoursewareCollapsibleBox from './CoursewareCollapsibleBox.vue';
import CoursewareManagerElement from './CoursewareManagerElement.vue';
import CoursewareManagerCopySelector from './CoursewareManagerCopySelector.vue';
Ron Lucke's avatar
Ron Lucke committed
import CoursewareCompanionBox from './CoursewareCompanionBox.vue';
import CoursewareImport from '@/vue/mixins/courseware/import.js';
import CoursewareExport from '@/vue/mixins/courseware/export.js';
import { mapActions, mapGetters } from 'vuex';

import JSZip from 'jszip';
import FileSaver from 'file-saver';

export default {
    name: 'courseware-course-manager',
    components: {
        CoursewareTabs,
        CoursewareTab,
        CoursewareCollapsibleBox,
        CoursewareManagerElement,
        CoursewareManagerCopySelector,
Ron Lucke's avatar
Ron Lucke committed
        CoursewareCompanionBox,
    },

    mixins: [CoursewareImport, CoursewareExport],

    data() {
        return {
            exportRunning: false,
            importRunning: false,
            importZip: null,
            currentElement: {},
            currentId: null,
            selfElement: {},
            selfId: null,
            zip: null
        };
    },

    computed: {
        ...mapGetters({
            courseware: 'courseware',
            structuralElementById: 'courseware-structural-elements/byId',
Ron Lucke's avatar
Ron Lucke committed
            importFilesState: 'importFilesState',
            importFilesProgress: 'importFilesProgress',
            importStructuresState: 'importStructuresState',
            importStructuresProgress: 'importStructuresProgress',
            exportState: 'exportState',
            exportProgress: 'exportProgress'
        }),
        moveSelfPossible() {
            if (this.selfElement.relationships === undefined) {
                return false
            } else if (this.selfElement.relationships.parent.data === null) {
                return false;
            } else if (this.currentElement.id === this.selfElement.relationships.parent.data.id) {
                return false;
            } else if (this.currentId === this.selfId) {
                return false;
            } else {
                return true;
            }
        },
        moveSelfChildPossible() {
            return this.currentId !== this.selfId;
        },
Ron Lucke's avatar
Ron Lucke committed
        fileImportDone() {
            return this.importFilesProgress === 100;
        },
        importDone() {
            return this.importFilesProgress === 100 && this.importStructuresProgress === 100;
        }
    },

    methods: {
        ...mapActions({
            loadCoursewareStructure: 'loadCoursewareStructure',
            createStructuralElement: 'createStructuralElement',
            updateStructuralElement: 'updateStructuralElement',
            deleteStructuralElement: 'deleteStructuralElement',
            loadStructuralElement: 'loadStructuralElement',
            lockObject: 'lockObject',
            unlockObject: 'unlockObject',
            addBookmark: 'addBookmark',
            companionInfo: 'companionInfo',
Ron Lucke's avatar
Ron Lucke committed
            setImportFilesProgress: 'setImportFilesProgress',
            setImportStructuresProgress: 'setImportStructuresProgress',
        }),
        async reloadElements() {
            await this.setCurrentId(this.currentId);
            await this.setSelfId(this.selfId);
        },
        async setCurrentId(target) {
            this.currentId = target;
            await this.loadStructuralElement(this.currentId);
            this.initCurrent();
        },
Ron Lucke's avatar
Ron Lucke committed
        initCurrent() {
            this.currentElement = _.cloneDeep(this.structuralElementById({ id: this.currentId }));
        },
        async setSelfId(target) {
            this.selfId = target;
            await this.loadStructuralElement(this.selfId);
            this.initSelf();
        },
        initSelf() {
Ron Lucke's avatar
Ron Lucke committed
            this.selfElement = _.cloneDeep(this.structuralElementById({ id: this.selfId }));
        },

        async doExportCourseware() {
            if (this.exportRunning) {
                return false;
            }

            this.exportRunning = true;

            await this.loadCoursewareStructure();
Ron Lucke's avatar
Ron Lucke committed
            await this.sendExportZip(
                this.courseware.relationships.root.data.id,
                {withChildren: true}
            );

            this.exportRunning = false;
        },

Ron Lucke's avatar
Ron Lucke committed
        setImport(event) {
            this.importZip = event.target.files[0];
        },

        async doImportCourseware() {
            if (this.importZip === null) {
                return false;
            }

            this.importRunning = true;

            let view = this;

            view.zip = new JSZip();

            await view.zip.loadAsync(this.importZip).then(async function () {
                let data = await view.zip.file('courseware.json').async('string');
                let courseware = JSON.parse(data);

                let data_files = await view.zip.file('files.json').async('string');
                let files = JSON.parse(data_files);

                await view.loadCoursewareStructure();
                let parent_id = view.courseware.relationships.root.data.id;

                await view.importCourseware(courseware, parent_id, files);
            });

            this.importZip = null;
            this.importRunning = false;
        },

        chooseFile() {
            this.$refs.importFile.click();
Ron Lucke's avatar
Ron Lucke committed
            this.setImportFilesProgress(0);
            this.setImportStructuresProgress(0);
        },
        getFileSizeText(size) {
            if (size / 1024 < 1000) {
                return (size / 1024).toFixed(2) + ' kB';
            } else {
                return (size / 1048576).toFixed(2) + ' MB';
            }
        }
    },
    watch: {
        courseware(newValue, oldValue) {
            let currentId = newValue.relationships.root.data.id;
            this.setCurrentId(currentId);
            this.setSelfId(currentId);
        },
    },
};
</script>