feat: exercise form and submission list

This commit is contained in:
Jannat Patel
2025-06-20 19:59:10 +05:30
parent 9bb4c45a23
commit 88a2b69980
26 changed files with 938 additions and 51 deletions

View File

@@ -0,0 +1,206 @@
<template>
<Dialog v-model="show" :options="{ size: '2xl' }">
<template #body-title>
<div class="text-xl font-semibold text-ink-gray-9">
{{
props.exerciseID === 'new'
? __('Create Programming Exercise')
: __('Edit Programming Exercise')
}}
</div>
</template>
<template #body-content>
<div class="space-y-4">
<FormControl
v-model="exercise.title"
:label="__('Title')"
:required="true"
/>
<FormControl
v-model="exercise.language"
:label="__('Language')"
type="select"
:options="languageOptions"
:required="true"
/>
<div>
<div class="text-xs text-ink-gray-5 mb-2">
{{ __('Problem Statement') }}
<span class="text-ink-red-3">*</span>
</div>
<TextEditor
:content="exercise.problem_statement"
@change="(val: string) => (exercise.problem_statement = val)"
:editable="true"
:fixedMenu="true"
editorClass="prose-sm max-w-none border-b border-x bg-surface-gray-2 rounded-b-md py-1 px-2 min-h-[7rem] max-h-[18rem] overflow-y-auto"
/>
</div>
<ChildTable
v-model="exercise.test_cases"
:columns="testCaseColumns"
:required="true"
:addable="true"
:deletable="true"
:editable="true"
:placeholder="__('Add Test Case')"
/>
</div>
</template>
<template #actions="{ close }">
<div class="flex justify-end space-x-2">
<router-link
:to="{
name: 'ProgrammingExerciseSubmission',
params: {
exerciseID: props.exerciseID,
submissionID: 'new',
},
}"
>
<Button>
{{ __('Test this Exercise') }}
</Button>
</router-link>
<router-link
:to="{
name: 'ProgrammingExerciseSubmissions',
params: { exerciseID: props.exerciseID },
}"
>
<Button>
{{ __('Check Submission') }}
</Button>
</router-link>
<Button variant="solid" @click="saveExercise(close)">
{{ __('Save') }}
</Button>
</div>
</template>
</Dialog>
</template>
<script setup lang="ts">
import {
Button,
createListResource,
Dialog,
FormControl,
TextEditor,
toast,
} from 'frappe-ui'
import { computed, ref, watch } from 'vue'
import { ProgrammingExercise, TestCase } from '@/types/programming-exercise'
import ChildTable from '@/components/Controls/ChildTable.vue'
const show = defineModel()
const exercises = defineModel<{
data: ProgrammingExercise[]
reload: () => void
hasNextPage: boolean
next: () => void
setValue: {
submit: (
data: ProgrammingExercise,
options?: { onSuccess?: () => void }
) => void
}
}>('exercises')
const exercise = ref<ProgrammingExercise>({
title: '',
language: 'Python',
problem_statement: '',
test_cases: [],
})
const languageOptions = [
{ label: 'Python', value: 'Python' },
{ label: 'JavaScript', value: 'JavaScript' },
{ label: 'Java', value: 'Java' },
{ label: 'C++', value: 'C++' },
]
const props = withDefaults(
defineProps<{
exerciseID: string
}>(),
{
exerciseID: 'new',
}
)
watch(
() => props.exerciseID,
() => {
if (props.exerciseID != 'new') {
setExerciseData()
fetchTestCases()
}
}
)
const setExerciseData = () => {
exercises.value?.data.forEach((ex) => {
if (ex.name === props.exerciseID) {
exercise.value = { ...ex }
}
})
}
const testCases = createListResource({
doctype: 'LMS Test Case',
fields: ['input', 'expected_output', 'name'],
cache: ['testCases', props.exerciseID],
parent: 'LMS Programming Exercise',
onSuccess(data: TestCase[]) {
exercise.value.test_cases = data
},
})
const fetchTestCases = () => {
testCases.update({
filters: {
parent: props.exerciseID,
},
})
testCases.reload()
}
const saveExercise = (close: () => void) => {
if (props.exerciseID == 'new') createNewExercise(close)
else updateExercise(close)
}
const createNewExercise = (close: () => void) => {
exercises.value.insert.submit(
{
...exercise.value,
},
{
onSuccess() {
close()
exercises.value.reload()
toast.success(__('Programming Exercise created successfully'))
},
}
)
}
const updateExercise = (close: () => void) => {
exercises.value.setValue.submit(
{
name: props.exerciseID,
...exercise.value,
},
{
onSuccess() {
close()
exercises.value.reload()
toast.success(__('Programming Exercise updated successfully'))
},
}
)
}
const testCaseColumns = computed(() => {
return ['Input', 'Expected Output']
})
</script>

View File

@@ -0,0 +1,53 @@
<template>
<Dialog
v-model="show"
:options="{
title: __('Add a programming exercise to your lesson'),
size: 'xl',
actions: [
{
label: __('Save'),
variant: 'solid',
onClick: () => {
saveExercise()
},
},
],
}"
>
<template #body-content>
<div class="text-base">
<Link
v-model="exercise"
doctype="LMS Programming Exercise"
:label="__('Select a Programming Exercise')"
/>
</div>
</template>
</Dialog>
</template>
<script setup lang="ts">
import { Dialog } from 'frappe-ui'
import { onMounted, nextTick, ref } from 'vue'
import Link from '@/components/Controls/Link.vue'
const show = ref(false)
const exercise = ref(null)
const props = defineProps({
onSave: {
type: Function,
required: true,
},
})
onMounted(async () => {
await nextTick()
show.value = true
})
const saveExercise = () => {
props.onSave(exercise.value)
show.value = false
}
</script>

View File

@@ -1,5 +1,6 @@
<template>
<header
v-if="!fromLesson"
class="sticky flex items-center justify-between top-0 z-10 border-b bg-surface-white px-3 py-2.5 sm:px-5"
>
<Breadcrumbs :items="breadcrumbs" />
@@ -135,6 +136,7 @@ const boilerplate = ref<string>(
)
const { brand } = sessionStore()
const router = useRouter()
const fromLesson = ref(false)
const props = withDefaults(
defineProps<{
@@ -154,6 +156,9 @@ onMounted(() => {
if (!code.value) {
code.value = boilerplate.value
}
if (new URLSearchParams(window.location.search).get('fromLesson')) {
fromLesson.value = true
}
})
const exercise = createDocumentResource({
@@ -261,8 +266,6 @@ const execute = async (stdin = '') => {
code: code.value,
files: [{ filename: 'stdin', contents: stdin }],
onMessage: (msg: any) => {
console.log(msg)
if (msg.msgtype === 'write' && msg.file === 'stdout') {
outputChunks.push(msg.data)
}

View File

@@ -0,0 +1,249 @@
<template>
<header
class="sticky flex items-center justify-between top-0 z-10 border-b bg-surface-white px-3 py-2.5 sm:px-5"
>
<Breadcrumbs :items="breadcrumbs" />
</header>
<div class="p-6">
<div class="flex items-center justify-between space-x-32 mb-5">
<div class="text-xl font-semibold text-ink-gray-7">
{{
submissions.data?.length
? __('{0} Submissions').format(submissions.data.length)
: __('No Submissions')
}}
</div>
<div
v-if="submissions.data?.length"
class="grid grid-cols-3 gap-5 flex-1"
>
<Link
doctype="LMS Programming Exercise"
v-model="filters.exercise"
:placeholder="__('Filter by Exercise')"
/>
<Link
doctype="User"
v-model="filters.member"
:placeholder="__('Filter by Member')"
/>
<FormControl
v-model="filters.status"
type="select"
:options="[
{ label: __(''), value: '' },
{ label: __('Passed'), value: 'Passed' },
{ label: __('Failed'), value: 'Failed' },
]"
:placeholder="__('Filter by Status')"
/>
</div>
</div>
<ListView
v-if="submissions.loading || submissions.data?.length"
:columns="submissionColumns"
:rows="submissions.data"
rowKey="name"
:options="{
selectable: false,
}"
>
<ListHeader
class="mb-2 grid items-center space-x-4 rounded bg-surface-gray-2 p-2"
>
<ListHeaderItem
:item="item"
v-for="item in submissionColumns"
:key="item.key"
>
<template #prefix="{ item }">
<FeatherIcon :name="item.icon?.toString()" class="h-4 w-4" />
</template>
</ListHeaderItem>
</ListHeader>
<ListRows>
<router-link
v-for="row in submissions.data"
:to="{
name: 'ProgrammingExerciseSubmission',
params: {
exerciseID: row.exercise,
submissionID: row.name,
},
}"
>
<ListRow :row="row">
<template #default="{ column, item }">
<ListRowItem :item="row[column.key]" :align="column.align">
<template #prefix>
<div v-if="column.key == 'member_name'">
<Avatar
class="flex items-center"
:image="row['member_image']"
:label="item"
size="sm"
/>
</div>
</template>
<div v-if="column.key == 'status'">
<Badge
:theme="row[column.key] === 'Passed' ? 'green' : 'red'"
>
{{ row[column.key] }}
</Badge>
</div>
<div v-else>
{{ row[column.key] }}
</div>
</ListRowItem>
</template>
</ListRow>
</router-link>
</ListRows>
</ListView>
<EmptyState v-else type="Programming Exercise Submissions" />
<div
v-if="submissions.data && submissions.hasNextPage"
class="flex justify-center my-5"
>
<Button @click="submissions.next()">
{{ __('Load More') }}
</Button>
</div>
</div>
</template>
<script setup lang="ts">
import {
Avatar,
Badge,
Breadcrumbs,
Button,
createListResource,
FeatherIcon,
FormControl,
ListView,
ListHeader,
ListHeaderItem,
ListRows,
ListRow,
ListRowItem,
usePageMeta,
} from 'frappe-ui'
import { computed, inject, ref, watch } from 'vue'
import { sessionStore } from '@/stores/session'
import type {
ProgrammingExerciseSubmission,
Filters,
} from '@/pages/ProgrammingExercises/types'
import { useRouter } from 'vue-router'
import Link from '@/components/Controls/Link.vue'
import EmptyState from '@/components/EmptyState.vue'
const { brand } = sessionStore()
const dayjs = inject('$dayjs') as any
const filters = ref<Filters>({
exercise: '',
member: '',
status: '',
})
const router = useRouter()
const submissions = createListResource({
doctype: 'LMS Programming Exercise Submission',
fields: [
'name',
'exercise',
'exercise_title',
'member_name',
'member_image',
'status',
'modified',
],
auto: true,
transform(data: ProgrammingExercise[]) {
return data.map((submission: ProgrammingExerciseSubmission) => {
return {
...submission,
modified: dayjs(submission.modified).fromNow(),
}
})
},
})
watch(filters.value, () => {
let filtersToApply: Record<string, any> = {}
const filterFields = ['exercise', 'member', 'status']
filterFields.forEach((field) => {
if (filters.value[field as keyof Filters]) {
filtersToApply[field] = filters.value[field as keyof Filters]
router.push({
query: {
...router.currentRoute.value.query,
[field]: filters.value[field as keyof Filters],
},
})
} else {
delete filtersToApply[field]
const query = { ...router.currentRoute.value.query }
delete query[field]
router.push({
query,
})
}
})
submissions.update({
filters: {
...filtersToApply,
},
})
submissions.reload()
})
const submissionColumns = computed(() => {
return [
{
label: __('Member'),
key: 'member_name',
width: '20%',
icon: 'user',
},
{
label: __('Exercise'),
key: 'exercise_title',
width: '40%',
icon: 'code',
},
{
label: __('Status'),
key: 'status',
width: '20%',
icon: 'check-circle',
},
{
label: __('Modified'),
key: 'modified',
width: '20%',
icon: 'clock',
},
]
})
const breadcrumbs = computed(() => {
return [
{
label: __('Programming Exercise Submissions'),
route: {
name: 'ProgrammingExerciseSubmissions',
},
},
]
})
usePageMeta(() => {
return {
title: __('Programming Exercises'),
icon: brand.favicon,
}
})
</script>

View File

@@ -0,0 +1,140 @@
<template>
<header
class="sticky flex items-center justify-between top-0 z-10 border-b bg-surface-white px-3 py-2.5 sm:px-5"
>
<Breadcrumbs :items="breadcrumbs" />
<Button
v-if="!readOnlyMode"
variant="solid"
@click="
() => {
exerciseID = 'new'
showForm = true
}
"
>
<template #prefix>
<Plus class="h-4 w-4 stroke-1.5" />
</template>
{{ __('New') }}
</Button>
</header>
<div class="md:w-4/5 md:mx-auto p-5">
<div class="flex items-center justify-between mb-5">
<div v-if="exerciseCount" class="text-lg font-semibold text-ink-gray-9">
{{ __('{0} Exercises').format(exerciseCount) }}
</div>
<div
v-if="exercises.data?.length || exerciseCount > 0"
class="grid grid-cols-2 gap-5"
>
<!-- <FormControl
v-model="titleFilter"
:placeholder="__('Search by title')"
/>
<FormControl
v-model="typeFilter"
type="select"
:options="assignmentTypes"
:placeholder="__('Type')"
/> -->
</div>
</div>
<div
v-if="exercises.data?.length"
class="grid grid-cols-1 md:grid-cols-3 gap-4"
>
<div
v-for="exercise in exercises.data"
:key="exercise.name"
@click="
() => {
exerciseID = exercise.name
showForm = true
}
"
class="flex flex-col border rounded-md p-3 h-full hover:border-outline-gray-3 space-y-2"
>
<div class="text-lg font-semibold text-ink-gray-9">
{{ exercise.title }}
</div>
<div class="text-sm text-ink-gray-7">
{{ exercise.language }}
</div>
</div>
</div>
<EmptyState v-else type="Programming Exercises" />
<div
v-if="exercises.data && exercises.hasNextPage"
class="flex justify-center my-5"
>
<Button @click="exercises.next()">
{{ __('Load More') }}
</Button>
</div>
</div>
<ProgrammingExerciseForm
v-model="showForm"
:exerciseID="exerciseID"
v-model:exercises="exercises"
/>
</template>
<script setup lang="ts">
import { computed, onMounted, ref } from 'vue'
import {
Breadcrumbs,
Button,
call,
createListResource,
usePageMeta,
} from 'frappe-ui'
import { Plus } from 'lucide-vue-next'
import { sessionStore } from '@/stores/session'
import ProgrammingExerciseForm from '@/pages/ProgrammingExercises/ProgrammingExerciseForm.vue'
const exerciseCount = ref<number>(0)
const readOnlyMode = window.read_only_mode
const { brand } = sessionStore()
const showForm = ref<boolean>(false)
const exerciseID = ref<string | null>('new')
onMounted(() => {
getExerciseCount()
})
const getExerciseCount = () => {
call('frappe.client.get_count', {
doctype: 'LMS Programming Exercise',
})
.then((count: number) => {
exerciseCount.value = count
})
.catch((error: any) => {
console.error('Error fetching exercise count:', error)
})
}
const exercises = createListResource({
doctype: 'LMS Programming Exercise',
cache: ['programmingExercises'],
fields: ['name', 'title', 'language', 'problem_statement'],
auto: true,
})
usePageMeta(() => {
return {
title: __('Programming Exercises'),
icon: brand.favicon,
}
})
const breadcrumbs = computed(() => {
return [
{
label: __('Programming Exercises'),
route: { name: 'ProgrammingExercises' },
},
]
})
</script>

View File

@@ -0,0 +1,22 @@
interface ProgrammingExercise {
name: string;
title: string;
language: 'Python' | 'JavaScript';
test_cases_count: number;
problem_statement: string;
test_cases: [TestCase];
}
interface TestCase {
name: string;
input: string;
expected_output: string;
output: string;
status: 'Passed' | 'Failed';
}
type Filters = {
exercise?: string,
member?: string,
status?: string
}