diff --git a/kolibri/plugins/coach/assets/src/constants/index.js b/kolibri/plugins/coach/assets/src/constants/index.js index b38f0f7c75..6cdd0045e3 100644 --- a/kolibri/plugins/coach/assets/src/constants/index.js +++ b/kolibri/plugins/coach/assets/src/constants/index.js @@ -25,6 +25,7 @@ export const PageNames = { EXAM_CREATION_PREVIEW: 'EXAM_CREATION_PREVIEW', EXAM_CREATION_SEARCH: 'EXAM_CREATION_SEARCH', EXAM_CREATION_QUESTION_SELECTION: 'EXAM_CREATION_QUESTION_SELECTION', + EXAM_SIDE_PANEL: 'EXAM_SIDE_PANEL', EXAM_PREVIEW: 'EXAM_PREVIEW', EXAM_REPORT: 'EXAM_REPORT', EXAM_REPORT_DETAIL: 'EXAM_REPORT_DETAIL', diff --git a/kolibri/plugins/coach/assets/src/routes/planExamRoutes.js b/kolibri/plugins/coach/assets/src/routes/planExamRoutes.js index 5d2f629c50..ff95e2c181 100644 --- a/kolibri/plugins/coach/assets/src/routes/planExamRoutes.js +++ b/kolibri/plugins/coach/assets/src/routes/planExamRoutes.js @@ -14,6 +14,10 @@ import { import CreatePracticeQuizPage from '../views/plan/CreateExamPage/CreatePracticeQuizPage.vue'; import CreateExamPage from '../views/plan/CreateExamPage'; import CreateExamPreview from '../views/plan/CreateExamPage/CreateExamPreview.vue'; +import SectionSidePanel from '../views/plan/CreateExamPage/SectionSidePanel.vue'; +import SectionEditor from '../views/plan/CreateExamPage/SectionEditor.vue'; +import ResourceSelection from '../views/plan/CreateExamPage/ResourceSelection.vue'; +import ReplaceQuestions from '../views/plan/CreateExamPage/ReplaceQuestions.vue'; import PlanQuizPreviewPage from '../views/plan/PlanQuizPreviewPage'; import CoachExamsPage from '../views/plan/CoachExamsPage'; import { showExamsPage } from '../modules/examsRoot/handlers'; @@ -37,26 +41,30 @@ export default [ name: PageNames.EXAM_CREATION_ROOT, path: '/:classId/plan/quizzes/new', component: CreateExamPage, - }, - { - name: PageNames.QUIZ_SECTION_EDITOR, - path: '/:classId/plan/quizzes/new/:section_id/edit', - component: CreateExamPage, - }, - { - name: PageNames.QUIZ_REPLACE_QUESTIONS, - path: '/:classId/plan/quizzes/new/:section_id/replace-questions', - component: CreateExamPage, - }, - { - name: PageNames.QUIZ_SELECT_RESOURCES, - path: '/:classId/plan/quizzes/new/:section_id/select-resources/:topic_id?', - component: CreateExamPage, - }, - { - name: PageNames.BOOK_MARKED_RESOURCES, - path: '/:classId/plan/quizzes/new/:section_id/select-resources/bookmarks', - component: CreateExamPage, + children: [ + { + name: PageNames.EXAM_SIDE_PANEL, + path: ':section_id', + component: SectionSidePanel, + children: [ + { + name: PageNames.QUIZ_SECTION_EDITOR, + path: 'edit', + component: SectionEditor, + }, + { + name: PageNames.QUIZ_REPLACE_QUESTIONS, + path: 'replace-questions', + component: ReplaceQuestions, + }, + { + name: PageNames.QUIZ_SELECT_RESOURCES, + path: 'select-resources/:topic_id?', + component: ResourceSelection, + }, + ], + }, + ], }, { name: PageNames.EXAM_CREATION_PRACTICE_QUIZ, diff --git a/kolibri/plugins/coach/assets/src/views/plan/CreateExamPage/ConfirmCancellationModal.vue b/kolibri/plugins/coach/assets/src/views/plan/CreateExamPage/ConfirmCancellationModal.vue index 38497c9b06..e60e3640bb 100644 --- a/kolibri/plugins/coach/assets/src/views/plan/CreateExamPage/ConfirmCancellationModal.vue +++ b/kolibri/plugins/coach/assets/src/views/plan/CreateExamPage/ConfirmCancellationModal.vue @@ -48,16 +48,9 @@ export default { name: 'ConfirmCancellationModal', mixins: [commonCoreStrings], - props: { - closePanelRoute: { - type: Object, - required: true, - }, - }, methods: { handleContinueAction() { this.$emit('continue'); - this.$router.replace(this.closePanelRoute); }, closeModal() { this.$emit('cancel'); diff --git a/kolibri/plugins/coach/assets/src/views/plan/CreateExamPage/CreateQuizSection.vue b/kolibri/plugins/coach/assets/src/views/plan/CreateExamPage/CreateQuizSection.vue index f3edc5cbba..0e028cca79 100644 --- a/kolibri/plugins/coach/assets/src/views/plan/CreateExamPage/CreateQuizSection.vue +++ b/kolibri/plugins/coach/assets/src/views/plan/CreateExamPage/CreateQuizSection.vue @@ -313,8 +313,6 @@ - - @@ -334,7 +332,7 @@ import Draggable from 'kolibri.coreVue.components.Draggable'; import { injectQuizCreation } from '../../../composables/useQuizCreation'; import commonCoach from '../../common'; - import SectionSidePanel from './SectionSidePanel'; + import { PageNames } from '../../../constants'; import TabsWithOverflow from './TabsWithOverflow'; import AccordionContainer from './AccordionContainer'; import AccordionItem from './AccordionItem'; @@ -351,7 +349,6 @@ DragSortWidget, DragHandle, TabsWithOverflow, - SectionSidePanel, }, mixins: [commonCoreStrings, commonCoach], setup() { @@ -509,13 +506,15 @@ methods: { handleReplaceSelection() { const section_id = get(this.activeSection).section_id; - this.$router.replace({ path: 'new/' + section_id + '/replace-questions' }); + const route = this.$router.getRoute(PageNames.QUIZ_REPLACE_QUESTIONS, { section_id }); + this.$router.push(route); }, handleActiveSectionAction(opt) { const section_id = this.activeSection.section_id; + const editRoute = this.$router.getRoute(PageNames.QUIZ_SECTION_EDITOR, { section_id }); switch (opt.label) { case this.editSectionLabel$(): - this.$router.replace({ path: 'new/' + section_id + '/edit' }); + this.$router.push(editRoute); break; case this.deleteSectionLabel$(): this.removeSection(this.activeSection.section_id); @@ -573,7 +572,8 @@ set(this.dragActive, true); }, openSelectResources(section_id) { - this.$router.replace({ path: 'new/' + section_id + '/select-resources' }); + const route = this.$router.getRoute(PageNames.QUIZ_SELECT_RESOURCES, { section_id }); + this.$router.push(route); }, }, }; diff --git a/kolibri/plugins/coach/assets/src/views/plan/CreateExamPage/ResourceSelection.vue b/kolibri/plugins/coach/assets/src/views/plan/CreateExamPage/ResourceSelection.vue index d11589743b..80315507f7 100644 --- a/kolibri/plugins/coach/assets/src/views/plan/CreateExamPage/ResourceSelection.vue +++ b/kolibri/plugins/coach/assets/src/views/plan/CreateExamPage/ResourceSelection.vue @@ -105,13 +105,18 @@ style="float: right;" :text="coreString('saveChangesAction')" :primary="true" - :disabled="!hasTopicId() && !showBookmarks" + :disabled="!workingPoolHasChanged" @click="saveSelectedResource" /> + @@ -133,6 +138,7 @@ import { injectQuizCreation } from '../../../composables/useQuizCreation'; import LessonsSearchBox from '../LessonResourceSelectionPage/SearchTools/LessonsSearchBox.vue'; import ContentCardList from './../LessonResourceSelectionPage/ContentCardList.vue'; + import ConfirmCancellationModal from './ConfirmCancellationModal.vue'; import ResourceSelectionBreadcrumbs from './../LessonResourceSelectionPage/SearchTools/ResourceSelectionBreadcrumbs.vue'; export default { @@ -142,9 +148,10 @@ BookmarkIcon, LessonsSearchBox, ResourceSelectionBreadcrumbs, + ConfirmCancellationModal, }, mixins: [commonCoreStrings], - setup() { + setup(_, context) { const store = getCurrentInstance().proxy.$store; const route = computed(() => store.state.route); const topicId = computed(() => route.value.params.topic_id); @@ -154,6 +161,9 @@ const showBookmarks = computed(() => route.value.query.showBookmarks); const searchQuery = computed(() => route.value.query.search); const { updateSection, activeResourcePool, selectAllQuestions } = injectQuizCreation(); + const showConfirmationModal = ref(false); + + const prevRoute = ref({ name: PageNames.EXAM_CREATION_ROOT }); const { sectionSettings$, @@ -377,6 +387,7 @@ // call this annotateTopicsWithDescendantCounts method to ensure that the channels are // annotated with their num_assessments and those without assessments are filtered out annotateTopicsWithDescendantCounts(resources.value.map(c => c.id)).then(() => { + channels.value = resources.value; _loading.value = false; }); }); @@ -408,6 +419,10 @@ return searchResults.value; } + if (!topicId.value) { + return channels.value; + } + return resources.value; }); @@ -432,16 +447,36 @@ return fetchMoreQuizResources(); } + function handleCancelClose() { + showConfirmationModal.value = false; + } + + function handleConfirmClose() { + context.emit('closePanel'); + } + + const workingPoolHasChanged = computed(() => { + return ( + workingResourcePool.value.length != activeResourcePool.value.length || + !isEqual(workingResourcePool.value.sort(), activeResourcePool.value.sort()) + ); + }); + return { selectAllChecked, selectAllIndeterminate, showSelectAll, handleSelectAll, toggleSelected, + prevRoute, + workingPoolHasChanged, + handleConfirmClose, + handleCancelClose, topic, topicId, contentList, resources, + showConfirmationModal, hasCheckbox, loading, hasMore, @@ -464,18 +499,13 @@ updateSection, selectAllQuestions, workingResourcePool, + activeResourcePool, addToWorkingResourcePool, removeFromWorkingResourcePool, showBookmarks, selectedResourcesInformation$, }; }, - props: { - closePanelRoute: { - type: Object, - required: true, - }, - }, computed: { isTopicIdSet() { return this.$route.params.topic_id; @@ -506,6 +536,19 @@ this.bookmarksCount = newVal.length; }, }, + beforeRouteEnter(_, from, next) { + next(vm => { + vm.prevRoute = from; + }); + }, + beforeRouteLeave(_, __, next) { + if (!this.showConfirmationModal && this.workingPoolHasChanged) { + this.showConfirmationModal = true; + next(false); + } else { + next(); + } + }, methods: { showTopicSizeWarningCard(content) { return !this.hasCheckbox(content) && content.kind === ContentNodeKinds.TOPIC; @@ -547,9 +590,6 @@ topicsLink(topicId) { return this.topicListingLink({ ...this.$route.params, topicId }); }, - hasTopicId() { - return Boolean(this.$route.params.topic_id); - }, saveSelectedResource() { this.updateSection({ section_id: this.$route.params.section_id, @@ -559,7 +599,7 @@ //Also reset workingResourcePool this.resetWorkingResourcePool(); - this.$router.replace(this.closePanelRoute); + this.$router.replace(this.prevRoute); }, selectionMetadata(content) { if (content.kind === ContentNodeKinds.TOPIC) { diff --git a/kolibri/plugins/coach/assets/src/views/plan/CreateExamPage/SectionEditor.vue b/kolibri/plugins/coach/assets/src/views/plan/CreateExamPage/SectionEditor.vue index f3c76ca92a..57042cc5d6 100644 --- a/kolibri/plugins/coach/assets/src/views/plan/CreateExamPage/SectionEditor.vue +++ b/kolibri/plugins/coach/assets/src/views/plan/CreateExamPage/SectionEditor.vue @@ -15,7 +15,7 @@ :layout4="{ span: 2 }" > @@ -29,7 +29,7 @@
@@ -43,8 +43,8 @@ icon="minus" aria-hidden="true" class="number-btn" - :disabled="numberOfQuestions === 1" - @click="numberOfQuestions -= 1" + :disabled="question_count === 1" + @click="question_count -= 1" />
@@ -62,7 +62,7 @@ + @@ -222,23 +227,69 @@