Compare commits

..

29 Commits

Author SHA1 Message Date
Jannat Patel
699c821edd fix: spacing and widths 2024-08-05 11:12:34 +05:30
Jannat Patel
6820dfc820 fix: make question attachments public 2024-08-01 13:02:20 +05:30
Jannat Patel
e0855a2c1b fix: quiz question population issue 2024-07-31 22:30:29 +05:30
Jannat Patel
6a0b37a4d4 chore: changed release branch to develop 2024-07-31 12:43:04 +05:30
Jannat Patel
f7fd6916e2 chore: release notes action 2024-07-31 12:15:24 +05:30
Jannat Patel
30e61f4b7c Merge pull request #947 from pateljannat/semantic-release
chore: semantic release action
2024-07-31 11:45:42 +05:30
Jannat Patel
48b37d58d8 chore: semantic release action 2024-07-31 11:10:47 +05:30
Jannat Patel
e96f18df7c Merge pull request #946 from pateljannat/issues-27
fix: quiz shuffle issue
2024-07-30 18:04:26 +05:30
Jannat Patel
7d15527831 fix: quiz shuffle issue 2024-07-30 16:58:09 +05:30
Jannat Patel
794c0e760b Merge pull request #945 from pateljannat/issues-26
fix: eval request issues
2024-07-30 15:26:46 +05:30
Jannat Patel
e46a60d00a chore: linters 2024-07-30 15:05:24 +05:30
Jannat Patel
819aac70fd fix: linters 2024-07-30 14:54:21 +05:30
Jannat Patel
ed7db2d7c5 fix: eval request issues 2024-07-30 14:17:17 +05:30
Jannat Patel
98a56f9117 Merge pull request #936 from pateljannat/image-pasting
fix: batch filters on desk
2024-07-19 12:19:34 +05:30
Jannat Patel
cbc4b8c59d chore: removed unused plugin 2024-07-19 12:07:35 +05:30
Jannat Patel
69d266e018 feat: image pasting 2024-07-19 11:55:36 +05:30
Jannat Patel
4bc3ac1665 fix: batch filters on desk 2024-07-18 21:29:40 +05:30
Jannat Patel
e0de9d70de Merge pull request #933 from pateljannat/issues-25
fix: misc issues
2024-07-17 11:40:35 +05:30
Jannat Patel
493bab8163 fix: multiline tags 2024-07-17 11:29:14 +05:30
Jannat Patel
25a2d82e82 fix: misc issues 2024-07-16 16:11:24 +05:30
Jannat Patel
0183677494 Merge pull request #932 from pateljannat/batch-categories
feat: batch categories filter
2024-07-15 14:45:57 +05:30
Jannat Patel
7ae9244896 feat: batch categories filter 2024-07-15 13:45:38 +05:30
Jannat Patel
15330cb41d Merge pull request #928 from pateljannat/completion-certificate
feat: completion certificate
2024-07-12 20:36:06 +05:30
Jannat Patel
166996d77a chore: removed unnecessary lines 2024-07-12 20:17:42 +05:30
Jannat Patel
4943e0e902 chore: fixed linters 2024-07-12 19:58:52 +05:30
Jannat Patel
1db6a8bfda chore: fixed linters 2024-07-12 19:56:01 +05:30
Jannat Patel
57f43b256a feat: enable certification from course form 2024-07-12 19:46:45 +05:30
Jannat Patel
23b2e8d682 feat: generate certificate from course page 2024-07-12 15:56:50 +05:30
Jannat Patel
6e1d62340f feat: completion certificate 2024-07-11 18:12:15 +05:30
41 changed files with 946 additions and 577 deletions

32
.github/workflows/on_release.yml vendored Normal file
View File

@@ -0,0 +1,32 @@
name: Generate Semantic Release
on:
workflow_dispatch:
push:
branches:
- main
jobs:
release:
name: Release
runs-on: ubuntu-latest
steps:
- name: Checkout Entire Repository
uses: actions/checkout@v4
with:
fetch-depth: 0
persist-credentials: false
- name: Setup Node.js
uses: actions/setup-node@v4
with:
node-version: 20
- name: Setup dependencies
run: |
npm install @semantic-release/git @semantic-release/exec --no-save
- name: Create Release
env:
GH_TOKEN: ${{ secrets.RELEASE_TOKEN }}
GITHUB_TOKEN: ${{ secrets.RELEASE_TOKEN }}
GIT_AUTHOR_NAME: "Frappe PR Bot"
GIT_AUTHOR_EMAIL: "developers@frappe.io"
GIT_COMMITTER_NAME: "Frappe PR Bot"
GIT_COMMITTER_EMAIL: "developers@frappe.io"
run: npx semantic-release

39
.github/workflows/release_notes.yml vendored Normal file
View File

@@ -0,0 +1,39 @@
# This action:
#
# 1. Generates release notes using github API.
# 2. Strips unnecessary info like chore/style etc from notes.
# 3. Updates release info.
name: 'Release Notes'
on:
workflow_dispatch:
inputs:
tag_name:
description: 'Tag of release like v2.0.0'
required: true
type: string
release:
types: [released]
permissions:
contents: read
jobs:
regen-notes:
name: 'Regenerate release notes'
runs-on: ubuntu-latest
steps:
- name: Update notes
run: |
NEW_NOTES=$(gh api --method POST -H "Accept: application/vnd.github+json" /repos/frappe/lms/releases/generate-notes -f tag_name=$RELEASE_TAG \
| jq -r '.body' \
| sed -E '/^\* (chore|ci|test|docs|style)/d' \
| sed -E 's/by @mergify //'
)
RELEASE_ID=$(gh api -H "Accept: application/vnd.github+json" /repos/frappe/lms/releases/tags/$RELEASE_TAG | jq -r '.id')
gh api --method PATCH -H "Accept: application/vnd.github+json" /repos/frappe/lms/releases/$RELEASE_ID -f body="$NEW_NOTES"
env:
GH_TOKEN: ${{ secrets.RELEASE_TOKEN }}
RELEASE_TAG: ${{ github.event.inputs.tag_name || github.event.release.tag_name }}

21
.releaserc Normal file
View File

@@ -0,0 +1,21 @@
{
"branches": ["develop"],
"plugins": [
"@semantic-release/commit-analyzer", {
"preset": "angular"
},
"@semantic-release/release-notes-generator",
[
"@semantic-release/exec", {
"prepareCmd": 'sed -ir "s/[0-9]*\.[0-9]*\.[0-9]*/${nextRelease.version}/" lms/__init__.py'
}
],
[
"@semantic-release/git", {
"assets": ["lms/__init__.py"],
"message": "chore(release): Bumped to Version ${nextRelease.version}"
}
],
"@semantic-release/github"
]
}

Submodule frappe-ui deleted from aa44431c18

View File

@@ -14,10 +14,10 @@
"@editorjs/editorjs": "^2.29.0", "@editorjs/editorjs": "^2.29.0",
"@editorjs/embed": "^2.7.0", "@editorjs/embed": "^2.7.0",
"@editorjs/header": "^2.8.1", "@editorjs/header": "^2.8.1",
"@editorjs/image": "^2.9.0",
"@editorjs/inline-code": "^1.5.0", "@editorjs/inline-code": "^1.5.0",
"@editorjs/nested-list": "^1.4.2", "@editorjs/nested-list": "^1.4.2",
"@editorjs/paragraph": "^2.11.3", "@editorjs/paragraph": "^2.11.3",
"@editorjs/simple-image": "^1.6.0",
"chart.js": "^4.4.1", "chart.js": "^4.4.1",
"dayjs": "^1.11.6", "dayjs": "^1.11.6",
"feather-icons": "^4.28.0", "feather-icons": "^4.28.0",

View File

@@ -18,7 +18,7 @@
</div> </div>
<div <div
v-if="sidebarSettings.data?.web_pages?.length || isModerator" v-if="sidebarSettings.data?.web_pages?.length || isModerator"
class="pt-1" class="mt-4"
> >
<div <div
class="flex items-center justify-between pr-2 cursor-pointer" class="flex items-center justify-between pr-2 cursor-pointer"
@@ -27,7 +27,7 @@
> >
<div <div
v-if="!isSidebarCollapsed" v-if="!isSidebarCollapsed"
class="flex items-center text-sm font-medium text-gray-600" class="flex items-center text-sm text-gray-600 my-1"
> >
<span class="grid h-5 w-6 flex-shrink-0 place-items-center"> <span class="grid h-5 w-6 flex-shrink-0 place-items-center">
<ChevronRight <ChevronRight
@@ -36,7 +36,7 @@
/> />
</span> </span>
<span class="ml-2"> <span class="ml-2">
{{ __('Web Pages') }} {{ __('More') }}
</span> </span>
</div> </div>
<Button v-if="isModerator" variant="ghost" @click="openPageModal()"> <Button v-if="isModerator" variant="ghost" @click="openPageModal()">

View File

@@ -3,6 +3,9 @@
class="flex flex-col shadow hover:bg-gray-100 rounded-md p-4 h-full" class="flex flex-col shadow hover:bg-gray-100 rounded-md p-4 h-full"
style="min-height: 150px" style="min-height: 150px"
> >
<div class="text-lg leading-5 font-semibold mb-2">
{{ batch.title }}
</div>
<Badge <Badge
v-if="batch.seat_count && batch.seats_left > 0" v-if="batch.seat_count && batch.seats_left > 0"
theme="green" theme="green"
@@ -19,38 +22,38 @@
> >
{{ __('Sold Out') }} {{ __('Sold Out') }}
</Badge> </Badge>
<div class="text-xl font-semibold mb-1"> <div class="short-introduction text-sm text-gray-700">
{{ batch.title }}
</div>
<div class="short-introduction">
{{ batch.description }} {{ batch.description }}
</div> </div>
<div class="flex flex-col space-y-2 mt-auto"> <div v-if="batch.amount" class="font-semibold mb-4">
<div v-if="batch.amount" class="font-semibold text-lg">
{{ batch.price }} {{ batch.price }}
</div> </div>
<div class="flex items-center"> <div class="flex flex-col space-y-2 mt-auto">
<BookOpen class="h-4 w-4 stroke-1.5 mr-2 text-gray-700" />
<span> {{ batch.courses.length }} {{ __('Courses') }} </span>
</div>
<DateRange <DateRange
:startDate="batch.start_date" :startDate="batch.start_date"
:endDate="batch.end_date" :endDate="batch.end_date"
class="mb-3" class="text-sm text-gray-700"
/> />
<div class="flex items-center"> <div class="flex items-center text-sm text-gray-700">
<Clock class="h-4 w-4 stroke-1.5 mr-2 text-gray-700" /> <Clock class="h-4 w-4 stroke-1.5 mr-2 text-gray-700" />
<span> <span>
{{ formatTime(batch.start_time) }} - {{ formatTime(batch.end_time) }} {{ formatTime(batch.start_time) }} - {{ formatTime(batch.end_time) }}
</span> </span>
</div> </div>
<div v-if="batch.timezone" class="flex items-center"> <div
<Globe class="h-4 w-4 stroke-1.5 mr-2 text-gray-700" /> v-if="batch.timezone"
class="flex items-center text-sm text-gray-700"
>
<Globe class="h-4 w-4 stroke-1.5 mr-2 text-gray-600" />
<span> <span>
{{ batch.timezone }} {{ batch.timezone }}
</span> </span>
</div> </div>
<div v-if="batch.instructors?.length" class="flex avatar-group overlap"> </div>
<div
v-if="batch.instructors?.length"
class="flex avatar-group overlap mt-4"
>
<div <div
class="h-6 mr-1" class="h-6 mr-1"
:class="{ 'avatar-group overlap': batch.instructors.length > 1 }" :class="{ 'avatar-group overlap': batch.instructors.length > 1 }"
@@ -63,7 +66,6 @@
<CourseInstructors :instructors="batch.instructors" /> <CourseInstructors :instructors="batch.instructors" />
</div> </div>
</div> </div>
</div>
</template> </template>
<script setup> <script setup>
import { Badge } from 'frappe-ui' import { Badge } from 'frappe-ui'
@@ -88,7 +90,7 @@ const props = defineProps({
text-overflow: ellipsis; text-overflow: ellipsis;
width: 100%; width: 100%;
overflow: hidden; overflow: hidden;
margin: 0.25rem 0 1.25rem; margin: 0.25rem 0 1rem;
line-height: 1.5; line-height: 1.5;
} }

View File

@@ -2,28 +2,23 @@
<div <div
v-if="course.title" v-if="course.title"
class="flex flex-col h-full rounded-md shadow-md text-base overflow-auto" class="flex flex-col h-full rounded-md shadow-md text-base overflow-auto"
style="min-height: 320px" style="min-height: 350px"
> >
<div <div
class="course-image" class="course-image"
:class="{ 'default-image': !course.image }" :class="{ 'default-image': !course.image }"
:style="{ backgroundImage: 'url(\'' + encodeURI(course.image) + '\')' }" :style="{ backgroundImage: 'url(\'' + encodeURI(course.image) + '\')' }"
> >
<div class="flex relative top-4 left-4 w-fit flex-wrap"> <div
<Badge class="flex items-center flex-wrap space-y-1 space-x-1 relative top-4 px-2 w-fit"
v-if="course.featured"
variant="subtle"
theme="green"
size="md"
class="mr-2"
> >
<Badge v-if="course.featured" variant="subtle" theme="green" size="md">
{{ __('Featured') }} {{ __('Featured') }}
</Badge> </Badge>
<Badge <Badge
variant="outline" variant="outline"
theme="gray" theme="gray"
size="md" size="md"
class="mr-2"
v-for="tag in course.tags" v-for="tag in course.tags"
> >
{{ tag }} {{ tag }}
@@ -77,7 +72,7 @@
{{ course.title }} {{ course.title }}
</div> </div>
<div class="short-introduction"> <div class="short-introduction text-gray-700 text-sm">
{{ course.short_introduction }} {{ course.short_introduction }}
</div> </div>

View File

@@ -63,6 +63,15 @@
{{ __('Start Learning') }} {{ __('Start Learning') }}
</span> </span>
</Button> </Button>
<Button
v-if="canGetCertificate"
@click="fetchCertificate()"
variant="subtle"
class="w-full mt-2"
size="md"
>
{{ __('Get Certificate') }}
</Button>
<router-link <router-link
v-if="user?.data?.is_moderator || is_instructor()" v-if="user?.data?.is_moderator || is_instructor()"
:to="{ :to="{
@@ -136,7 +145,7 @@ function enrollStudent() {
}) })
setTimeout(() => { setTimeout(() => {
window.location.href = `/login?redirect-to=${window.location.pathname}` window.location.href = `/login?redirect-to=${window.location.pathname}`
}, 3000) }, 2000)
} else { } else {
const enrollStudentResource = createResource({ const enrollStudentResource = createResource({
url: 'lms.lms.doctype.lms_enrollment.lms_enrollment.create_membership', url: 'lms.lms.doctype.lms_enrollment.lms_enrollment.create_membership',
@@ -174,4 +183,39 @@ const is_instructor = () => {
}) })
return user_is_instructor return user_is_instructor
} }
const canGetCertificate = computed(() => {
if (
props.course.data?.enable_certification &&
props.course.data?.membership?.progress == 100
) {
return true
}
return false
})
const certificate = createResource({
url: 'lms.lms.doctype.lms_certificate.lms_certificate.create_certificate',
makeParams(values) {
return {
course: values.course,
}
},
onSuccess(data) {
console.log(data)
window.open(
`/api/method/frappe.utils.print_format.download_pdf?doctype=LMS+Certificate&name=${
data.name
}&format=${encodeURIComponent(data.template)}`,
'_blank'
)
},
})
const fetchCertificate = () => {
certificate.submit({
course: props.course.data?.name,
member: user.data?.name,
})
}
</script> </script>

View File

@@ -130,11 +130,14 @@ function submitEvaluation(close) {
close() close()
}, },
onError(err) { onError(err) {
let message = err.messages?.[0] || err
let unavailabilityMessage = message.includes('unavailable')
createToast({ createToast({
title: 'Error', title: unavailabilityMessage ? 'Evaluator is Unavailable' : 'Error',
text: err.messages?.[0] || err, text: message,
icon: 'x', icon: unavailabilityMessage ? 'alert-circle' : 'x',
iconClasses: 'bg-red-600 text-white rounded-md p-px', iconClasses: 'bg-yellow-600 text-white rounded-md p-px',
position: 'top-center', position: 'top-center',
timeout: 10, timeout: 10,
}) })

View File

@@ -3,9 +3,7 @@
<div class="bg-blue-100 py-2 px-2 mb-4 rounded-md text-sm text-blue-800"> <div class="bg-blue-100 py-2 px-2 mb-4 rounded-md text-sm text-blue-800">
<div class="leading-relaxed"> <div class="leading-relaxed">
{{ {{
__('This quiz consists of {0} questions.').format( __('This quiz consists of {0} questions.').format(questions.length)
quiz.data.questions.length
)
}} }}
</div> </div>
<div v-if="quiz.data.passing_percentage" class="leading-relaxed"> <div v-if="quiz.data.passing_percentage" class="leading-relaxed">
@@ -59,7 +57,7 @@
</div> </div>
</div> </div>
<div v-else-if="!quizSubmission.data"> <div v-else-if="!quizSubmission.data">
<div v-for="(question, qtidx) in quiz.data.questions"> <div v-for="(question, qtidx) in questions">
<div <div
v-if="qtidx == activeQuestion - 1 && questionDetails.data" v-if="qtidx == activeQuestion - 1 && questionDetails.data"
class="border rounded-md p-5" class="border rounded-md p-5"
@@ -166,7 +164,7 @@
{{ {{
__('Question {0} of {1}').format( __('Question {0} of {1}').format(
activeQuestion, activeQuestion,
quiz.data.questions.length questions.length
) )
}} }}
</div> </div>
@@ -179,7 +177,7 @@
</span> </span>
</Button> </Button>
<Button <Button
v-else-if="activeQuestion != quiz.data.questions.length" v-else-if="activeQuestion != questions.length"
@click="nextQuetion()" @click="nextQuetion()"
> >
<span> <span>
@@ -250,6 +248,7 @@ const activeQuestion = ref(0)
const currentQuestion = ref('') const currentQuestion = ref('')
const selectedOptions = reactive([0, 0, 0, 0]) const selectedOptions = reactive([0, 0, 0, 0])
const showAnswers = reactive([]) const showAnswers = reactive([])
let questions = reactive([])
const possibleAnswer = ref(null) const possibleAnswer = ref(null)
const props = defineProps({ const props = defineProps({
@@ -270,15 +269,30 @@ const quiz = createResource({
cache: ['quiz', props.quizName], cache: ['quiz', props.quizName],
auto: true, auto: true,
onSuccess(data) { onSuccess(data) {
if (data.shuffle_questions) { populateQuestions()
data.questions = data.questions.sort(() => Math.random() - 0.5)
}
if (data.limit_questions_to) {
data.questions = data.questions.slice(0, data.limit_questions_to)
}
}, },
}) })
const populateQuestions = () => {
let data = quiz.data
if (data.shuffle_questions) {
questions = shuffleArray(data.questions)
if (data.limit_questions_to) {
questions = questions.slice(0, data.limit_questions_to)
}
} else {
questions = data.questions
}
}
const shuffleArray = (array) => {
for (let i = array.length - 1; i > 0; i--) {
const j = Math.floor(Math.random() * (i + 1))
;[array[i], array[j]] = [array[j], array[i]]
}
return array
}
const attempts = createResource({ const attempts = createResource({
url: 'frappe.client.get_list', url: 'frappe.client.get_list',
makeParams(values) { makeParams(values) {
@@ -310,7 +324,7 @@ const attempts = createResource({
watch( watch(
() => quiz.data, () => quiz.data,
() => { () => {
if (quiz.data) { if (quiz.data && quiz.data.max_attempts) {
attempts.reload() attempts.reload()
resetQuiz() resetQuiz()
} }
@@ -464,7 +478,7 @@ const submitQuiz = () => {
const createSubmission = () => { const createSubmission = () => {
quizSubmission.reload().then(() => { quizSubmission.reload().then(() => {
attempts.reload() if (quiz.data && quiz.data.max_attempts) attempts.reload()
}) })
} }
@@ -473,6 +487,7 @@ const resetQuiz = () => {
selectedOptions.splice(0, selectedOptions.length, ...[0, 0, 0, 0]) selectedOptions.splice(0, selectedOptions.length, ...[0, 0, 0, 0])
showAnswers.length = 0 showAnswers.length = 0
quizSubmission.reset() quizSubmission.reset()
populateQuestions()
} }
const getSubmissionColumns = () => { const getSubmissionColumns = () => {

View File

@@ -1,10 +1,14 @@
<template> <template>
<div ref="videoContainer" class="video-block group relative"> <div ref="videoContainer" class="video-block group relative">
<video @timeupdate="updateTime" @ended="videoEnded" class="rounded-lg"> <video
@timeupdate="updateTime"
@ended="videoEnded"
class="rounded-lg border border-gray-100"
>
<source :src="fileURL" :type="type" /> <source :src="fileURL" :type="type" />
</video> </video>
<div <div
class="flex items-center space-x-2 bg-gray-200 rounded-lg p-0.5 absolute bottom-3 w-[98%] left-0 right-0 mx-auto" class="flex items-center space-x-2 bg-gray-200 rounded-md p-0.5 absolute bottom-3 w-[98%] left-0 right-0 mx-auto"
> >
<Button variant="ghost"> <Button variant="ghost">
<template #icon> <template #icon>

View File

@@ -7,7 +7,15 @@
class="h-7" class="h-7"
:items="[{ label: __('All Batches'), route: { name: 'Batches' } }]" :items="[{ label: __('All Batches'), route: { name: 'Batches' } }]"
/> />
<div class="flex"> <div class="flex space-x-2">
<div class="w-40">
<Select
v-if="categories.data?.length"
v-model="currentCategory"
:options="categories.data"
:placeholder="__('Filter')"
/>
</div>
<router-link <router-link
v-if="user.data?.is_moderator" v-if="user.data?.is_moderator"
:to="{ :to="{
@@ -33,7 +41,7 @@
</div> </div>
<Tabs <Tabs
v-model="tabIndex" v-model="tabIndex"
:tabs="tabs" :tabs="makeTabs"
tablistClass="overflow-x-visible flex-wrap !gap-3 md:flex-nowrap" tablistClass="overflow-x-visible flex-wrap !gap-3 md:flex-nowrap"
> >
<template #tab="{ tab, selected }"> <template #tab="{ tab, selected }">
@@ -87,13 +95,29 @@
</div> </div>
</template> </template>
<script setup> <script setup>
import { createListResource, Breadcrumbs, Button, Tabs, Badge } from 'frappe-ui' import {
createListResource,
createResource,
Breadcrumbs,
Button,
Tabs,
Badge,
Select,
} from 'frappe-ui'
import { Plus } from 'lucide-vue-next' import { Plus } from 'lucide-vue-next'
import BatchCard from '@/components/BatchCard.vue' import BatchCard from '@/components/BatchCard.vue'
import { inject, ref, computed } from 'vue' import { inject, ref, computed, onMounted, watch } from 'vue'
import { updateDocumentTitle } from '@/utils' import { updateDocumentTitle } from '@/utils'
const user = inject('$user') const user = inject('$user')
const currentCategory = ref(null)
onMounted(() => {
let queries = new URLSearchParams(location.search)
if (queries.has('category')) {
currentCategory.value = queries.get('category')
}
})
const batches = createListResource({ const batches = createListResource({
doctype: 'LMS Batch', doctype: 'LMS Batch',
@@ -102,35 +126,76 @@ const batches = createListResource({
auto: true, auto: true,
}) })
const tabIndex = ref(0) const categories = createResource({
const tabs = [ url: 'lms.lms.api.get_categories',
{ makeParams() {
label: 'Upcoming', return {
batches: computed(() => batches.data?.upcoming || []), doctype: 'LMS Batch',
count: computed(() => batches.data?.upcoming?.length), filters: {
published: 1,
}, },
] }
},
cache: ['batchCategories'],
auto: true,
transform(data) {
data.unshift({
label: '',
value: null,
})
},
})
const tabIndex = ref(0)
let tabs
const makeTabs = computed(() => {
tabs = []
addToTabs('Upcoming')
if (user.data?.is_moderator) { if (user.data?.is_moderator) {
tabs.push({ addToTabs('Archived')
label: 'Archived', addToTabs('Private')
batches: computed(() => batches.data?.archived),
count: computed(() => batches.data?.archived?.length),
})
tabs.push({
label: 'Private',
batches: computed(() => batches.data?.private),
count: computed(() => batches.data?.private?.length),
})
} }
if (user.data) { if (user.data) {
addToTabs('Enrolled')
}
return tabs
})
const getBatches = (type) => {
if (currentCategory.value && currentCategory.value != '') {
return batches.data[type].filter(
(batch) => batch.category == currentCategory.value
)
}
return batches.data[type]
}
const addToTabs = (label) => {
let batches = getBatches(label.toLowerCase().split(' ').join('_'))
tabs.push({ tabs.push({
label: 'Enrolled', label,
batches: computed(() => batches.data?.enrolled), batches: computed(() => batches),
count: computed(() => batches.data?.enrolled?.length), count: computed(() => batches.length),
}) })
} }
watch(
() => currentCategory.value,
() => {
let queries = new URLSearchParams(location.search)
if (currentCategory.value) {
queries.set('category', currentCategory.value)
} else {
queries.delete('category')
}
history.pushState(null, '', `${location.pathname}?${queries.toString()}`)
}
)
const pageMeta = computed(() => { const pageMeta = computed(() => {
return { return {
title: 'Batches', title: 'Batches',

View File

@@ -18,7 +18,10 @@
</header> </header>
<div class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-4 m-5"> <div class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-4 m-5">
<div v-if="participants.data" v-for="participant in participants.data"> <div
v-if="participants.data?.length"
v-for="participant in participantsList"
>
<router-link <router-link
:to="{ :to="{
name: 'Profile', name: 'Profile',
@@ -59,12 +62,7 @@ const searchQuery = ref('')
const participants = createResource({ const participants = createResource({
url: 'lms.lms.api.get_certified_participants', url: 'lms.lms.api.get_certified_participants',
method: 'GET', method: 'GET',
debounce: 300, cache: 'certified-participants',
makeParams(values) {
return {
search_query: searchQuery.value,
}
},
auto: true, auto: true,
}) })
@@ -79,5 +77,16 @@ const pageMeta = computed(() => {
} }
}) })
const participantsList = computed(() => {
if (searchQuery.value) {
return participants.data.filter((participant) => {
return participant.full_name
.toLowerCase()
.includes(searchQuery.value.toLowerCase())
})
}
return participants.data
})
updateDocumentTitle(pageMeta) updateDocumentTitle(pageMeta)
</script> </script>

View File

@@ -134,29 +134,30 @@ let tabs
const makeTabs = computed(() => { const makeTabs = computed(() => {
tabs = [] tabs = []
addToTabs('Live', getCourses('live')) addToTabs('Live')
addToTabs('New', getCourses('new')) addToTabs('New')
addToTabs('Upcoming', getCourses('upcoming')) addToTabs('Upcoming')
if (user.data) { if (user.data) {
addToTabs('Enrolled', getCourses('enrolled')) addToTabs('Enrolled')
if ( if (
user.data.is_moderator || user.data.is_moderator ||
user.data.is_instructor || user.data.is_instructor ||
courses.data?.created?.length courses.data?.created?.length
) { ) {
addToTabs('Created', getCourses('created')) addToTabs('Created')
} }
if (user.data.is_moderator) { if (user.data.is_moderator) {
addToTabs('Under Review', getCourses('under_review')) addToTabs('Under Review')
} }
} }
return tabs return tabs
}) })
const addToTabs = (label, courses) => { const addToTabs = (label) => {
let courses = getCourses(label.toLowerCase().split(' ').join('_'))
tabs.push({ tabs.push({
label, label,
courses: computed(() => courses), courses: computed(() => courses),
@@ -166,8 +167,12 @@ const addToTabs = (label, courses) => {
const getCourses = (type) => { const getCourses = (type) => {
if (searchQuery.value) { if (searchQuery.value) {
return courses.data[type].filter((course) => let query = searchQuery.value.toLowerCase()
course.title.toLowerCase().includes(searchQuery.value.toLowerCase()) return courses.data[type].filter(
(course) =>
course.title.toLowerCase().includes(query) ||
course.short_introduction.toLowerCase().includes(query) ||
course.tags.filter((tag) => tag.toLowerCase().includes(query)).length
) )
} }
return courses.data[type] return courses.data[type]

View File

@@ -119,7 +119,7 @@
<div class="text-lg font-semibold mt-5 mb-4"> <div class="text-lg font-semibold mt-5 mb-4">
{{ __('Settings') }} {{ __('Settings') }}
</div> </div>
<div class="grid grid-cols-2 gap-10 mb-4"> <div class="grid grid-cols-3 gap-10 mb-4">
<div <div
v-if="user.data?.is_moderator" v-if="user.data?.is_moderator"
class="flex flex-col space-y-3" class="flex flex-col space-y-3"
@@ -147,11 +147,18 @@
v-model="course.featured" v-model="course.featured"
:label="__('Featured')" :label="__('Featured')"
/> />
</div>
<div class="flex flex-col space-y-3">
<FormControl <FormControl
type="checkbox" type="checkbox"
v-model="course.disable_self_learning" v-model="course.disable_self_learning"
:label="__('Disable Self Enrollment')" :label="__('Disable Self Enrollment')"
/> />
<FormControl
type="checkbox"
v-model="course.enable_certification"
:label="__('Completion Certificate')"
/>
</div> </div>
</div> </div>
</div> </div>
@@ -244,6 +251,7 @@ const course = reactive({
featured: false, featured: false,
upcoming: false, upcoming: false,
disable_self_learning: false, disable_self_learning: false,
enable_certification: false,
paid_course: false, paid_course: false,
course_price: '', course_price: '',
currency: '', currency: '',
@@ -337,6 +345,7 @@ const courseResource = createResource({
'disable_self_learning', 'disable_self_learning',
'paid_course', 'paid_course',
'featured', 'featured',
'enable_certification',
] ]
for (let idx in checkboxes) { for (let idx in checkboxes) {
let key = checkboxes[idx] let key = checkboxes[idx]

View File

@@ -43,7 +43,7 @@
<div <div
v-show="openInstructorEditor" v-show="openInstructorEditor"
id="instructor-notes" id="instructor-notes"
class="ProseMirror prose prose-table:table-fixed prose-td:p-2 prose-th:p-2 prose-td:border prose-th:border prose-td:border-gray-300 prose-th:border-gray-300 prose-td:relative prose-th:relative prose-th:bg-gray-100 prose-sm max-w-none !whitespace-normal mt-6 py-3" class="ProseMirror prose prose-table:table-fixed prose-td:p-2 prose-th:p-2 prose-td:border prose-th:border prose-td:border-gray-300 prose-th:border-gray-300 prose-td:relative prose-th:relative prose-th:bg-gray-100 prose-sm max-w-none !whitespace-normal py-3"
></div> ></div>
</div> </div>
</div> </div>
@@ -54,7 +54,7 @@
</label> </label>
<div <div
id="content" id="content"
class="ProseMirror prose prose-table:table-fixed prose-td:p-2 prose-th:p-2 prose-td:border prose-th:border prose-td:border-gray-300 prose-th:border-gray-300 prose-td:relative prose-th:relative prose-th:bg-gray-100 prose-sm max-w-none !whitespace-normal mt-6 py-3" class="ProseMirror prose prose-table:table-fixed prose-td:p-2 prose-th:p-2 prose-td:border prose-th:border prose-td:border-gray-300 prose-th:border-gray-300 prose-td:relative prose-th:relative prose-th:bg-gray-100 prose-sm max-w-none !whitespace-normal py-3"
></div> ></div>
</div> </div>
</div> </div>
@@ -439,7 +439,8 @@ const pageMeta = computed(() => {
updateDocumentTitle(pageMeta) updateDocumentTitle(pageMeta)
</script> </script>
<style> <style>
.embed-tool__caption { .embed-tool__caption,
.cdx-simple-image__caption {
display: none; display: none;
} }

View File

@@ -50,9 +50,9 @@
</Button> </Button>
</div> </div>
</header> </header>
<div v-if="job.data" class="w-3/4 mx-auto"> <div v-if="job.data" class="max-w-3xl mx-auto">
<div class="p-4"> <div class="p-4">
<div class="flex mb-4"> <div class="flex mb-10">
<img <img
:src="job.data.company_logo" :src="job.data.company_logo"
class="w-16 h-16 rounded-lg object-contain mr-4" class="w-16 h-16 rounded-lg object-contain mr-4"
@@ -62,8 +62,9 @@
<div class="text-2xl font-semibold mb-4"> <div class="text-2xl font-semibold mb-4">
{{ job.data.job_title }} {{ job.data.job_title }}
</div> </div>
<div class="grid grid-cols-3 gap-8"> <div
<div class="grid grid-cols-1 gap-2"> class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-x-10 gap-y-2 md:gap-y-4"
>
<div class="flex items-center space-x-2"> <div class="flex items-center space-x-2">
<Building2 class="h-4 w-4 stroke-1.5" /> <Building2 class="h-4 w-4 stroke-1.5" />
<span>{{ job.data.company_name }}</span> <span>{{ job.data.company_name }}</span>
@@ -72,20 +73,16 @@
<MapPin class="h-4 w-4 stroke-1.5" /> <MapPin class="h-4 w-4 stroke-1.5" />
<span>{{ job.data.location }}</span> <span>{{ job.data.location }}</span>
</div> </div>
</div>
<div class="grid grid-cols-1 gap-2">
<div class="flex items-center space-x-2"> <div class="flex items-center space-x-2">
<ClipboardType class="h-4 w-4 stroke-1.5" /> <ClipboardType class="h-4 w-4 stroke-1.5" />
<span>{{ job.data.type }}</span> <span>{{ job.data.type }}</span>
</div> </div>
<div class="flex items-center space-x-2"> <div class="flex items-center space-x-2">
<CalendarDays class="h-4 w-4 stroke-1.5" /> <CalendarDays class="h-4 w-4 stroke-1.5" />
<span>{{ <span>
dayjs(job.data.creation).format('DD MMM YYYY') {{ dayjs(job.data.creation).format('DD MMM YYYY') }}
}}</span> </span>
</div> </div>
</div>
<div class="grid grid-cols-1 h-fit">
<div <div
v-if="applicationCount.data" v-if="applicationCount.data"
class="flex items-center space-x-2" class="flex items-center space-x-2"
@@ -99,7 +96,6 @@
</div> </div>
</div> </div>
</div> </div>
</div>
<p <p
v-html="job.data.description" v-html="job.data.description"
class="ProseMirror prose prose-table:table-fixed prose-td:p-2 prose-th:p-2 prose-td:border prose-th:border prose-td:border-gray-300 prose-th:border-gray-300 prose-td:relative prose-th:relative prose-th:bg-gray-100 prose-sm max-w-none !whitespace-normal mt-6" class="ProseMirror prose prose-table:table-fixed prose-td:p-2 prose-th:p-2 prose-td:border prose-th:border prose-td:border-gray-300 prose-th:border-gray-300 prose-td:relative prose-th:relative prose-th:bg-gray-100 prose-sm max-w-none !whitespace-normal mt-6"

View File

@@ -90,6 +90,17 @@
</span> </span>
</Button> </Button>
</router-link> </router-link>
<router-link
v-else
:to="{
name: 'CourseDetail',
params: { courseName: courseName },
}"
>
<Button>
{{ __('Back to Course') }}
</Button>
</router-link>
</div> </div>
</div> </div>
@@ -160,12 +171,12 @@
{{ lesson.data.course_title }} {{ lesson.data.course_title }}
</div> </div>
<div v-if="user && lesson.data.membership" class="text-sm mt-3"> <div v-if="user && lesson.data.membership" class="text-sm mt-3">
{{ Math.ceil(lesson.data.membership.progress) }}% completed {{ Math.ceil(lessonProgress) }}% {{ __('completed') }}
</div> </div>
<ProgressBar <ProgressBar
v-if="user && lesson.data.membership" v-if="user && lesson.data.membership"
:progress="lesson.data.membership.progress" :progress="lessonProgress"
/> />
</div> </div>
<CourseOutline <CourseOutline
@@ -179,7 +190,7 @@
</template> </template>
<script setup> <script setup>
import { createResource, Breadcrumbs, Button } from 'frappe-ui' import { createResource, Breadcrumbs, Button } from 'frappe-ui'
import { computed, watch, inject, ref } from 'vue' import { computed, watch, inject, ref, onMounted, onBeforeUnmount } from 'vue'
import CourseOutline from '@/components/CourseOutline.vue' import CourseOutline from '@/components/CourseOutline.vue'
import UserAvatar from '@/components/UserAvatar.vue' import UserAvatar from '@/components/UserAvatar.vue'
import { useRoute } from 'vue-router' import { useRoute } from 'vue-router'
@@ -196,6 +207,9 @@ const route = useRoute()
const allowDiscussions = ref(false) const allowDiscussions = ref(false)
const editor = ref(null) const editor = ref(null)
const instructorEditor = ref(null) const instructorEditor = ref(null)
const lessonProgress = ref(0)
const timer = ref(0)
let timerInterval
const props = defineProps({ const props = defineProps({
courseName: { courseName: {
@@ -212,6 +226,10 @@ const props = defineProps({
}, },
}) })
onMounted(() => {
startTimer()
})
const lesson = createResource({ const lesson = createResource({
url: 'lms.lms.utils.get_lesson', url: 'lms.lms.utils.get_lesson',
cache: ['lesson', props.courseName, props.chapterNumber, props.lessonNumber], cache: ['lesson', props.courseName, props.chapterNumber, props.lessonNumber],
@@ -224,7 +242,7 @@ const lesson = createResource({
}, },
auto: true, auto: true,
onSuccess(data) { onSuccess(data) {
markProgress(data) lessonProgress.value = data.membership?.progress
if (data.content) editor.value = renderEditor('editor', data.content) if (data.content) editor.value = renderEditor('editor', data.content)
if (data.instructor_content?.blocks?.length) if (data.instructor_content?.blocks?.length)
instructorEditor.value = renderEditor( instructorEditor.value = renderEditor(
@@ -256,8 +274,10 @@ const renderEditor = (holder, content) => {
}) })
} }
const markProgress = (data) => { const markProgress = () => {
if (user.data && !data.progress) progress.submit() if (user.data && !lesson.data?.progress) {
progress.submit()
}
} }
const progress = createResource({ const progress = createResource({
@@ -268,6 +288,9 @@ const progress = createResource({
course: props.courseName, course: props.courseName,
} }
}, },
onSuccess(data) {
lessonProgress.value = data
},
}) })
const breadcrumbs = computed(() => { const breadcrumbs = computed(() => {
@@ -304,10 +327,27 @@ watch(
chapter: newChapterNumber, chapter: newChapterNumber,
lesson: newLessonNumber, lesson: newLessonNumber,
}) })
clearInterval(timerInterval)
timer.value = 0
startTimer()
} }
} }
) )
const startTimer = () => {
timerInterval = setInterval(() => {
timer.value++
if (timer.value == 30) {
clearInterval(timerInterval)
markProgress()
}
}, 1000)
}
onBeforeUnmount(() => {
clearInterval(timerInterval)
})
const checkIfDiscussionsAllowed = () => { const checkIfDiscussionsAllowed = () => {
let quizPresent = false let quizPresent = false
JSON.parse(lesson.data?.content)?.blocks?.forEach((block) => { JSON.parse(lesson.data?.content)?.blocks?.forEach((block) => {

View File

@@ -5,7 +5,7 @@ import router from '@/router'
import { ref, computed } from 'vue' import { ref, computed } from 'vue'
export const sessionStore = defineStore('lms-session', () => { export const sessionStore = defineStore('lms-session', () => {
let { userResource } = usersStore() let { userResource, allUsers } = usersStore()
function sessionUser() { function sessionUser() {
let cookies = new URLSearchParams(document.cookie.split('; ').join('&')) let cookies = new URLSearchParams(document.cookie.split('; ').join('&'))
@@ -17,6 +17,9 @@ export const sessionStore = defineStore('lms-session', () => {
} }
let user = ref(sessionUser()) let user = ref(sessionUser())
if (user) {
allUsers.reload()
}
const isLoggedIn = computed(() => !!user.value) const isLoggedIn = computed(() => !!user.value)
const login = createResource({ const login = createResource({

View File

@@ -15,7 +15,6 @@ export const usersStore = defineStore('lms-users', () => {
const allUsers = createResource({ const allUsers = createResource({
url: 'lms.lms.api.get_all_users', url: 'lms.lms.api.get_all_users',
cache: ['allUsers'], cache: ['allUsers'],
auto: true,
}) })
return { return {

View File

@@ -10,6 +10,7 @@ import InlineCode from '@editorjs/inline-code'
import { watch } from 'vue' import { watch } from 'vue'
import dayjs from '@/utils/dayjs' import dayjs from '@/utils/dayjs'
import Embed from '@editorjs/embed' import Embed from '@editorjs/embed'
import SimpleImage from '@editorjs/simple-image'
export function createToast(options) { export function createToast(options) {
toast({ toast({
@@ -79,15 +80,18 @@ export function getFileSize(file_size) {
return value return value
} }
export function showToast(title, text, icon) { export function showToast(title, text, icon, iconClasses = null) {
if (!iconClasses) {
iconClasses =
icon == 'check'
? 'bg-green-600 text-white rounded-md p-px'
: 'bg-red-600 text-white rounded-md p-px'
}
createToast({ createToast({
title: title, title: title,
text: htmlToText(text), text: htmlToText(text),
icon: icon, icon: icon,
iconClasses: iconClasses: iconClasses,
icon == 'check'
? 'bg-green-600 text-white rounded-md p-px'
: 'bg-red-600 text-white rounded-md p-px',
position: icon == 'check' ? 'bottom-right' : 'top-center', position: icon == 'check' ? 'bottom-right' : 'top-center',
timeout: 5, timeout: 5,
}) })
@@ -133,6 +137,7 @@ export function getEditorTools() {
header: Header, header: Header,
quiz: Quiz, quiz: Quiz,
upload: Upload, upload: Upload,
image: SimpleImage,
paragraph: { paragraph: {
class: Paragraph, class: Paragraph,
inlineToolbar: true, inlineToolbar: true,
@@ -164,10 +169,68 @@ export function getEditorTools() {
inlineToolbar: false, inlineToolbar: false,
config: { config: {
services: { services: {
youtube: true, youtube: {
regex: /(?:https?:\/\/)?(?:www\.)?(?:(?:youtu\.be\/)|(?:youtube\.com)\/(?:v\/|u\/\w\/|embed\/|watch))(?:(?:\?v=)?([^#&?=]*))?((?:[?&]\w*=\w*)*)/,
embedUrl:
'https://www.youtube.com/embed/<%= remote_id %>',
html: '<iframe style="width:100%; height: 30rem;" frameborder="0" allowfullscreen></iframe>',
height: 320,
width: 580,
id: ([id, params]) => {
if (!params && id) {
return id
}
const paramsMap = {
start: 'start',
end: 'end',
t: 'start',
// eslint-disable-next-line camelcase
time_continue: 'start',
list: 'list',
}
let newParams = params
.slice(1)
.split('&')
.map((param) => {
const [name, value] = param.split('=')
if (!id && name === 'v') {
id = value
return null
}
if (!paramsMap[name]) {
return null
}
if (
value === 'LL' ||
value.startsWith('RDMM') ||
value.startsWith('FL')
) {
return null
}
return `${paramsMap[name]}=${value}`
})
.filter((param) => !!param)
return id + '?' + newParams.join('&')
},
},
vimeo: true, vimeo: true,
codepen: true, codepen: true,
aparat: true, aparat: {
regex: /(?:http[s]?:\/\/)?(?:www.)?aparat\.com\/v\/([^\/\?\&]+)\/?/,
embedUrl:
'https://www.aparat.com/video/video/embed/videohash/<%= remote_id %>/vt/frame',
html: '<iframe style="margin: 0 auto; width: 100%; height: 25rem;" frameborder="0" scrolling="no" allowtransparency="true"></iframe>',
height: 300,
width: 600,
},
github: true, github: true,
slides: { slides: {
regex: /https:\/\/docs\.google\.com\/presentation\/d\/e\/([A-Za-z0-9_-]+)\/pub/, regex: /https:\/\/docs\.google\.com\/presentation\/d\/e\/([A-Za-z0-9_-]+)\/pub/,

View File

@@ -68,12 +68,12 @@
dependencies: dependencies:
"@codexteam/icons" "^0.0.5" "@codexteam/icons" "^0.0.5"
"@editorjs/image@^2.9.0": "@editorjs/image@^2.9.2":
version "2.9.0" version "2.9.2"
resolved "https://registry.yarnpkg.com/@editorjs/image/-/image-2.9.0.tgz#0c83252d569a0dc3af14c3f7d16b6df033b9c37b" resolved "https://registry.yarnpkg.com/@editorjs/image/-/image-2.9.2.tgz#c8bea65a578fab65a1a75df1223b4fd8f06b57d5"
integrity sha512-xItihKJFiWJ06SMtLWQZvzHv4LRPNAFZYaHAXesBFzXvWwUrtVaVMcNSf0eNnw3InrPO3Po1vZRRgpsT+Ya3Bg== integrity sha512-n09sMieGW8cksoeflpplzvbmFH2bdVzVTWbnidPWAHaeU467HRteoXU9yfGBB7+eeHZLnmCulQ2dr6ae+G2niw==
dependencies: dependencies:
"@codexteam/icons" "^0.0.6" "@codexteam/icons" "^0.3.0"
"@editorjs/inline-code@^1.5.0": "@editorjs/inline-code@^1.5.0":
version "1.5.0" version "1.5.0"
@@ -96,6 +96,13 @@
dependencies: dependencies:
"@codexteam/icons" "^0.0.4" "@codexteam/icons" "^0.0.4"
"@editorjs/simple-image@^1.6.0":
version "1.6.0"
resolved "https://registry.yarnpkg.com/@editorjs/simple-image/-/simple-image-1.6.0.tgz#711c3900e17845331d6667cf0fe91793a5557f84"
integrity sha512-WvdGfQPlozwZd3PXQrJnRXk6gEYbv1U2vRupYJ6lTd3/UsLInXYUX5jSFcnGB5ZMH3bd0JDZfcb4d4Sv1/1big==
dependencies:
"@codexteam/icons" "^0.0.6"
"@esbuild/aix-ppc64@0.20.2": "@esbuild/aix-ppc64@0.20.2":
version "0.20.2" version "0.20.2"
resolved "https://registry.yarnpkg.com/@esbuild/aix-ppc64/-/aix-ppc64-0.20.2.tgz#a70f4ac11c6a1dfc18b8bbb13284155d933b9537" resolved "https://registry.yarnpkg.com/@esbuild/aix-ppc64/-/aix-ppc64-0.20.2.tgz#a70f4ac11c6a1dfc18b8bbb13284155d933b9537"

View File

@@ -177,50 +177,8 @@ update_website_context = [
jinja = { jinja = {
"methods": [ "methods": [
"lms.page_renderers.get_profile_url",
"lms.overrides.user.get_enrolled_courses",
"lms.overrides.user.get_course_membership",
"lms.overrides.user.get_authored_courses",
"lms.overrides.user.get_palette",
"lms.lms.utils.get_membership",
"lms.lms.utils.get_lessons",
"lms.lms.utils.get_tags",
"lms.lms.utils.get_instructors",
"lms.lms.utils.get_students",
"lms.lms.utils.get_average_rating",
"lms.lms.utils.is_certified",
"lms.lms.utils.get_lesson_index",
"lms.lms.utils.get_lesson_url",
"lms.lms.utils.get_chapters",
"lms.lms.utils.get_slugified_chapter_title",
"lms.lms.utils.get_progress",
"lms.lms.utils.render_html",
"lms.lms.utils.is_mentor",
"lms.lms.utils.is_cohort_staff",
"lms.lms.utils.get_mentors",
"lms.lms.utils.get_reviews",
"lms.lms.utils.is_eligible_to_review",
"lms.lms.utils.get_initial_members",
"lms.lms.utils.get_sorted_reviews",
"lms.lms.utils.is_instructor",
"lms.lms.utils.convert_number_to_character",
"lms.lms.utils.get_signup_optin_checks", "lms.lms.utils.get_signup_optin_checks",
"lms.lms.utils.get_popular_courses", "lms.lms.utils.get_tags",
"lms.lms.utils.format_amount",
"lms.lms.utils.first_lesson_exists",
"lms.lms.utils.get_courses_under_review",
"lms.lms.utils.has_course_instructor_role",
"lms.lms.utils.has_course_moderator_role",
"lms.lms.utils.get_certificates",
"lms.lms.utils.format_number",
"lms.lms.utils.get_lesson_count",
"lms.lms.utils.get_all_memberships",
"lms.lms.utils.get_filtered_membership",
"lms.lms.utils.show_start_learing_cta",
"lms.lms.utils.can_create_courses",
"lms.lms.utils.get_telemetry_boot_info",
"lms.lms.utils.is_onboarding_complete",
"lms.www.utils.is_student",
], ],
"filters": [], "filters": [],
} }

View File

@@ -330,13 +330,12 @@ def get_evaluator_details(evaluator):
@frappe.whitelist(allow_guest=True) @frappe.whitelist(allow_guest=True)
def get_certified_participants(search_query=""): def get_certified_participants():
LMSCertificate = DocType("LMS Certificate") LMSCertificate = DocType("LMS Certificate")
participants = ( participants = (
frappe.qb.from_(LMSCertificate) frappe.qb.from_(LMSCertificate)
.select(LMSCertificate.member) .select(LMSCertificate.member)
.distinct() .distinct()
.where(LMSCertificate.member_name.like(f"%{search_query}%"))
.where(LMSCertificate.published == 1) .where(LMSCertificate.published == 1)
.orderby(LMSCertificate.creation, order=frappe.qb.desc) .orderby(LMSCertificate.creation, order=frappe.qb.desc)
.run(as_dict=1) .run(as_dict=1)
@@ -542,3 +541,21 @@ def update_index(lessons, chapter):
frappe.db.set_value( frappe.db.set_value(
"Lesson Reference", {"lesson": row, "parent": chapter}, "idx", lessons.index(row) + 1 "Lesson Reference", {"lesson": row, "parent": chapter}, "idx", lessons.index(row) + 1
) )
@frappe.whitelist(allow_guest=True)
def get_categories(doctype, filters):
categoryOptions = []
categories = frappe.get_all(
doctype,
filters,
pluck="category",
)
categories = list(set(categories))
for category in categories:
if category:
categoryOptions.append({"label": category, "value": category})
return categoryOptions

View File

@@ -93,7 +93,7 @@ def save_progress(lesson, course):
"LMS Enrollment", {"course": course, "member": frappe.session.user} "LMS Enrollment", {"course": course, "member": frappe.session.user}
) )
if not membership: if not membership:
return 0 return
frappe.db.set_value("LMS Enrollment", membership, "current_lesson", lesson) frappe.db.set_value("LMS Enrollment", membership, "current_lesson", lesson)
@@ -104,7 +104,7 @@ def save_progress(lesson, course):
if frappe.db.exists( if frappe.db.exists(
"LMS Course Progress", {"lesson": lesson, "member": frappe.session.user} "LMS Course Progress", {"lesson": lesson, "member": frappe.session.user}
): ):
return 0 return
frappe.get_doc( frappe.get_doc(
{ {
@@ -116,9 +116,12 @@ def save_progress(lesson, course):
).save(ignore_permissions=True) ).save(ignore_permissions=True)
progress = get_course_progress(course) progress = get_course_progress(course)
frappe.db.set_value("LMS Enrollment", membership, "progress", progress) # Had to get doc, as on_change doesn't trigger when you use set_value. The trigger is necesary for badge to get assigned.
enrollment = frappe.get_doc("LMS Enrollment", membership) enrollment = frappe.get_doc("LMS Enrollment", membership)
enrollment.progress = progress
enrollment.save()
enrollment.run_method("on_change") enrollment.run_method("on_change")
return progress return progress

View File

@@ -86,7 +86,6 @@
"label": "Comments" "label": "Comments"
}, },
{ {
"fetch_from": "course.evaluator",
"fieldname": "evaluator", "fieldname": "evaluator",
"fieldtype": "Link", "fieldtype": "Link",
"label": "Evaluator", "label": "Evaluator",

View File

@@ -103,6 +103,7 @@
"fieldname": "start_date", "fieldname": "start_date",
"fieldtype": "Date", "fieldtype": "Date",
"in_list_view": 1, "in_list_view": 1,
"in_standard_filter": 1,
"label": "Start Date", "label": "Start Date",
"reqd": 1 "reqd": 1
}, },
@@ -127,6 +128,7 @@
{ {
"fieldname": "start_time", "fieldname": "start_time",
"fieldtype": "Time", "fieldtype": "Time",
"in_list_view": 1,
"label": "Start Time", "label": "Start Time",
"reqd": 1 "reqd": 1
}, },
@@ -165,6 +167,7 @@
{ {
"fieldname": "category", "fieldname": "category",
"fieldtype": "Link", "fieldtype": "Link",
"in_standard_filter": 1,
"label": "Category", "label": "Category",
"options": "LMS Category" "options": "LMS Category"
}, },
@@ -325,7 +328,7 @@
], ],
"index_web_pages_for_search": 1, "index_web_pages_for_search": 1,
"links": [], "links": [],
"modified": "2024-06-24 16:24:45.536453", "modified": "2024-07-18 18:06:37.229885",
"modified_by": "Administrator", "modified_by": "Administrator",
"module": "LMS", "module": "LMS",
"name": "LMS Batch", "name": "LMS Batch",

View File

@@ -96,6 +96,7 @@ class LMSBatch(Document):
) )
args = { args = {
"title": self.title,
"student_name": student.student_name, "student_name": student.student_name,
"start_time": self.start_time, "start_time": self.start_time,
"start_date": self.start_date, "start_date": self.start_date,

View File

@@ -10,12 +10,14 @@
"course_title", "course_title",
"member", "member",
"member_name", "member_name",
"column_break_3", "column_break_vwbn",
"template",
"issue_date", "issue_date",
"template",
"published",
"section_break_scyf",
"expiry_date", "expiry_date",
"batch_name", "column_break_slaw",
"published" "batch_name"
], ],
"fields": [ "fields": [
{ {
@@ -25,10 +27,6 @@
"label": "Issue Date", "label": "Issue Date",
"reqd": 1 "reqd": 1
}, },
{
"fieldname": "column_break_3",
"fieldtype": "Column Break"
},
{ {
"fieldname": "course", "fieldname": "course",
"fieldtype": "Link", "fieldtype": "Link",
@@ -85,11 +83,23 @@
"fieldtype": "Data", "fieldtype": "Data",
"label": "Course Title", "label": "Course Title",
"read_only": 1 "read_only": 1
},
{
"fieldname": "column_break_vwbn",
"fieldtype": "Column Break"
},
{
"fieldname": "section_break_scyf",
"fieldtype": "Section Break"
},
{
"fieldname": "column_break_slaw",
"fieldtype": "Column Break"
} }
], ],
"index_web_pages_for_search": 1, "index_web_pages_for_search": 1,
"links": [], "links": [],
"modified": "2024-06-21 18:14:30.491841", "modified": "2024-07-16 15:29:19.708888",
"modified_by": "Administrator", "modified_by": "Administrator",
"module": "LMS", "module": "LMS",
"name": "LMS Certificate", "name": "LMS Certificate",
@@ -120,13 +130,15 @@
"write": 1 "write": 1
}, },
{ {
"create": 1,
"email": 1, "email": 1,
"export": 1, "export": 1,
"print": 1, "print": 1,
"read": 1, "read": 1,
"report": 1, "report": 1,
"role": "LMS Student", "role": "LMS Student",
"share": 1 "share": 1,
"write": 1
} }
], ],
"sort_field": "modified", "sort_field": "modified",

View File

@@ -73,6 +73,8 @@ class LMSCertificate(Document):
def has_website_permission(doc, ptype, user, verbose=False): def has_website_permission(doc, ptype, user, verbose=False):
if ptype in ["read", "print"]: if ptype in ["read", "print"]:
return True return True
if doc.member == user and ptype == "create":
return True
return False return False
@@ -81,7 +83,9 @@ def create_certificate(course):
certificate = is_certified(course) certificate = is_certified(course)
if certificate: if certificate:
return certificate return frappe.db.get_value(
"LMS Certificate", certificate, ["name", "course", "template"], as_dict=True
)
else: else:
expires_after_yrs = int(frappe.db.get_value("LMS Course", course, "expiry")) expires_after_yrs = int(frappe.db.get_value("LMS Course", course, "expiry"))

View File

@@ -47,7 +47,7 @@
"fieldtype": "Rating", "fieldtype": "Rating",
"in_list_view": 1, "in_list_view": 1,
"label": "Rating", "label": "Rating",
"mandatory_depends_on": "eval:doc.status != 'Pending' && doc.status != 'In Progress'" "mandatory_depends_on": "eval:doc.status == 'Pass'"
}, },
{ {
"fieldname": "summary", "fieldname": "summary",
@@ -107,7 +107,7 @@
], ],
"index_web_pages_for_search": 1, "index_web_pages_for_search": 1,
"links": [], "links": [],
"modified": "2024-04-15 11:22:43.189908", "modified": "2024-07-16 14:06:11.977666",
"modified_by": "Administrator", "modified_by": "Administrator",
"module": "LMS", "module": "LMS",
"name": "LMS Certificate Evaluation", "name": "LMS Certificate Evaluation",

View File

@@ -42,7 +42,7 @@ class LMSCertificateRequest(Document):
): ):
frappe.throw( frappe.throw(
_( _(
"Evaluator is unavailable from {0} to {1}. Please select a date after {1}" "The evaluator of this course is unavailable from {0} to {1}. Please select a date after {1}"
).format( ).format(
format_date(unavailable.unavailable_from, "medium"), format_date(unavailable.unavailable_from, "medium"),
format_date(unavailable.unavailable_to, "medium"), format_date(unavailable.unavailable_to, "medium"),
@@ -56,6 +56,7 @@ class LMSCertificateRequest(Document):
"evaluator": self.evaluator, "evaluator": self.evaluator,
"date": self.date, "date": self.date,
"start_time": self.start_time, "start_time": self.start_time,
"member": ["!=", self.member],
}, },
): ):
frappe.throw(_("The slot is already booked by another participant.")) frappe.throw(_("The slot is already booked by another participant."))

View File

@@ -30,23 +30,23 @@
"disable_self_learning", "disable_self_learning",
"section_break_18", "section_break_18",
"short_introduction", "short_introduction",
"column_break_viqw",
"description", "description",
"section_break_gglp",
"chapters", "chapters",
"related_courses", "related_courses",
"pricing_tab",
"pricing_section", "pricing_section",
"paid_course", "paid_course",
"column_break_acoj", "column_break_acoj",
"course_price", "course_price",
"currency", "currency",
"amount_usd", "amount_usd",
"certification_tab",
"certification_section", "certification_section",
"enable_certification", "enable_certification",
"expiry",
"max_attempts",
"column_break_rxww", "column_break_rxww",
"grant_certificate_after", "expiry"
"evaluator",
"duration"
], ],
"fields": [ "fields": [
{ {
@@ -129,8 +129,7 @@
}, },
{ {
"fieldname": "certification_section", "fieldname": "certification_section",
"fieldtype": "Section Break", "fieldtype": "Section Break"
"label": "Certification"
}, },
{ {
"default": "0", "default": "0",
@@ -170,25 +169,9 @@
"fieldname": "column_break_10", "fieldname": "column_break_10",
"fieldtype": "Column Break" "fieldtype": "Column Break"
}, },
{
"depends_on": "enable_certification",
"fieldname": "grant_certificate_after",
"fieldtype": "Select",
"label": "Grant Certificate After",
"options": "Completion\nEvaluation"
},
{
"depends_on": "eval: doc.grant_certificate_after == \"Evaluation\"",
"fieldname": "evaluator",
"fieldtype": "Link",
"label": "Evaluator",
"mandatory_depends_on": "eval: doc.grant_certificate_after == \"Evaluation\"",
"options": "Course Evaluator"
},
{ {
"fieldname": "pricing_section", "fieldname": "pricing_section",
"fieldtype": "Section Break", "fieldtype": "Section Break"
"label": "Pricing"
}, },
{ {
"depends_on": "paid_course", "depends_on": "paid_course",
@@ -198,20 +181,6 @@
"mandatory_depends_on": "paid_course", "mandatory_depends_on": "paid_course",
"options": "Currency" "options": "Currency"
}, },
{
"default": "1",
"depends_on": "eval: doc.grant_certificate_after == \"Evaluation\"",
"fieldname": "max_attempts",
"fieldtype": "Int",
"label": "Max Attempts for Evaluations"
},
{
"depends_on": "eval: doc.grant_certificate_after == \"Evaluation\"",
"fieldname": "duration",
"fieldtype": "Select",
"label": "Duration for Attempts",
"options": "1\n2\n3\n4\n5\n6\n7\n8\n9\n10\n11\n12"
},
{ {
"default": "0", "default": "0",
"fieldname": "paid_course", "fieldname": "paid_course",
@@ -250,6 +219,24 @@
"fieldname": "featured", "fieldname": "featured",
"fieldtype": "Check", "fieldtype": "Check",
"label": "Featured" "label": "Featured"
},
{
"fieldname": "column_break_viqw",
"fieldtype": "Column Break"
},
{
"fieldname": "section_break_gglp",
"fieldtype": "Section Break"
},
{
"fieldname": "pricing_tab",
"fieldtype": "Tab Break",
"label": "Pricing"
},
{
"fieldname": "certification_tab",
"fieldtype": "Tab Break",
"label": "Certification"
} }
], ],
"is_published_field": "published", "is_published_field": "published",
@@ -276,7 +263,7 @@
} }
], ],
"make_attachments_public": 1, "make_attachments_public": 1,
"modified": "2024-06-24 17:44:45.903164", "modified": "2024-07-12 13:54:40.474097",
"modified_by": "Administrator", "modified_by": "Administrator",
"module": "LMS", "module": "LMS",
"name": "LMS Course", "name": "LMS Course",

View File

@@ -195,7 +195,8 @@
], ],
"index_web_pages_for_search": 1, "index_web_pages_for_search": 1,
"links": [], "links": [],
"modified": "2024-05-24 16:12:26.331351", "make_attachments_public": 1,
"modified": "2024-08-01 13:01:55.000072",
"modified_by": "Administrator", "modified_by": "Administrator",
"module": "LMS", "module": "LMS",
"name": "LMS Question", "name": "LMS Question",

View File

@@ -5,5 +5,5 @@
<p> {{ _("Hey {0}").format(doc.member_name) }} </p> <p> {{ _("Hey {0}").format(doc.member_name) }} </p>
<p> {{ _('Your evaluation for the course {0} has been scheduled on {1} at {2} {3}.').format(title, frappe.utils.format_date(doc.date, "medium"), frappe.utils.format_time(doc.start_time, "short"), timezone) }}</p> <p> {{ _('Your evaluation for the course {0} has been scheduled on {1} at {2} {3}.').format(title, frappe.utils.format_date(doc.date, "medium"), frappe.utils.format_time(doc.start_time, "short"), timezone) }}</p>
<p> {{ _("Your evaluator is {0}").format(evaluator_name) }} <p> {{ _("Your evaluator is {0}").format(evaluator_name) }} </p>
<p> {{ _("Please prepare well and be on time for the evaluations.") }} </p> <p> {{ _("Please prepare well and be on time for the evaluations.") }} </p>

View File

@@ -1,11 +1,7 @@
import unittest import unittest
import frappe import frappe
from frappe.utils import getdate from .utils import slugify
from lms.lms.doctype.lms_course.test_lms_course import new_course, new_user
from .utils import get_evaluation_details, slugify
class TestUtils(unittest.TestCase): class TestUtils(unittest.TestCase):
@@ -20,58 +16,3 @@ class TestUtils(unittest.TestCase):
self.assertEqual( self.assertEqual(
slugify("Hello World", ["hello-world", "hello-world-2"]), "hello-world-3" slugify("Hello World", ["hello-world", "hello-world-2"]), "hello-world-3"
) )
def test_evaluation_details(self):
user = new_user("Eval", "eval@test.com")
course = new_course(
"Test Evaluation Details",
{
"enable_certification": 1,
"grant_certificate_after": "Evaluation",
"evaluator": "evaluator@example.com",
"max_attempts": 3,
"duration": 2,
"instructors": [{"instructor": user.name}],
},
)
# Two evaluations failed within max attempts. Check eligibility for a third evaluation
create_evaluation(user.name, course.name, getdate("21-03-2022"), 0.4, "Fail")
create_evaluation(user.name, course.name, getdate("12-04-2022"), 0.4, "Fail")
details = get_evaluation_details(course.name, user.name)
self.assertTrue(details.eligible)
# Three evaluations failed within max attempts. Check eligibility for a forth evaluation
create_evaluation(user.name, course.name, getdate("21-03-2022"), 0.4, "Fail")
create_evaluation(user.name, course.name, getdate("12-04-2022"), 0.4, "Fail")
create_evaluation(user.name, course.name, getdate("16-04-2022"), 0.4, "Fail")
details = get_evaluation_details(course.name, user.name)
self.assertFalse(details.eligible)
# Three evaluations failed within max attempts. Check eligibility for a forth evaluation. Different Dates
create_evaluation(user.name, course.name, getdate("01-03-2022"), 0.4, "Fail")
create_evaluation(user.name, course.name, getdate("12-04-2022"), 0.4, "Fail")
create_evaluation(user.name, course.name, getdate("16-04-2022"), 0.4, "Fail")
details = get_evaluation_details(course.name, user.name)
self.assertFalse(details.eligible)
frappe.db.delete("LMS Certificate Evaluation", {"course": course.name})
frappe.db.delete("LMS Course", course.name)
frappe.db.delete("User", user.name)
def create_evaluation(user, course, date, rating, status):
evaluation = frappe.get_doc(
{
"doctype": "LMS Certificate Evaluation",
"member": user,
"course": course,
"date": date,
"start_time": "12:00:00",
"end_time": "13:00:00",
"rating": rating,
"status": status,
}
)
evaluation.save()

View File

@@ -452,45 +452,6 @@ def get_popular_courses():
return course_membership[:3] return course_membership[:3]
def get_evaluation_details(course, member=None):
info = frappe.db.get_value(
"LMS Course",
course,
["grant_certificate_after", "max_attempts", "duration"],
as_dict=True,
)
request = frappe.db.get_value(
"LMS Certificate Request",
{
"course": course,
"member": member or frappe.session.user,
"date": [">=", getdate()],
},
["date", "start_time", "end_time"],
as_dict=True,
)
no_of_attempts = frappe.db.count(
"LMS Certificate Evaluation",
{
"course": course,
"member": member or frappe.session.user,
"status": ["!=", "Pass"],
"creation": [">=", add_months(getdate(), -abs(cint(info.duration)))],
},
)
return frappe._dict(
{
"eligible": info.grant_certificate_after == "Evaluation"
and not request
and no_of_attempts < info.max_attempts,
"request": request,
"no_of_attempts": no_of_attempts,
}
)
def format_amount(amount, currency): def format_amount(amount, currency):
amount_reduced = amount / 1000 amount_reduced = amount / 1000
if amount_reduced < 1: if amount_reduced < 1:
@@ -612,14 +573,6 @@ def get_courses_under_review():
) )
def get_certificates(member=None):
return frappe.get_all(
"LMS Certificate",
{"member": member or frappe.session.user},
["course", "member", "issue_date", "expiry_date", "name"],
)
def validate_image(path): def validate_image(path):
if path and "/private" in path: if path and "/private" in path:
file = frappe.get_doc("File", {"file_url": path}) file = frappe.get_doc("File", {"file_url": path})
@@ -944,19 +897,13 @@ def has_graded_assessment(submission):
return False if status == "Not Graded" else True return False if status == "Not Graded" else True
def get_evaluator(course, batch=None): def get_evaluator(course, batch):
evaluator = None evaluator = None
if batch:
evaluator = frappe.db.get_value( evaluator = frappe.db.get_value(
"Batch Course", "Batch Course",
{"parent": batch, "course": course}, {"parent": batch, "course": course},
"evaluator", "evaluator",
) )
if not evaluator:
evaluator = frappe.db.get_value("LMS Course", course, "evaluator")
return evaluator return evaluator
@@ -1285,6 +1232,7 @@ def get_course_details(course):
"course_price", "course_price",
"currency", "currency",
"amount_usd", "amount_usd",
"enable_certification",
], ],
as_dict=1, as_dict=1,
) )
@@ -1508,6 +1456,7 @@ def get_batch_details(batch):
"evaluation_end_date", "evaluation_end_date",
"allow_self_enrollment", "allow_self_enrollment",
"timezone", "timezone",
"category",
], ],
as_dict=True, as_dict=True,
) )

View File

@@ -8,10 +8,11 @@
<br> <br>
<p> <p>
<b> <b>
{{ _("Important Details:") }} {{ title }}
</b> </b>
</p> </p>
<p> <p>
<b>{{ _("Batch Start Date:") }}</b> {{ frappe.utils.format_date(start_date, "medium") }} <b>{{ _("Batch Start Date:") }}</b> {{ frappe.utils.format_date(start_date, "medium") }}
</p> </p>

View File

@@ -26,5 +26,8 @@
"devDependencies": { "devDependencies": {
"cypress": "^13.9.0", "cypress": "^13.9.0",
"cypress-file-upload": "^5.0.8" "cypress-file-upload": "^5.0.8"
},
"dependencies": {
"pre-commit": "^1.2.2"
} }
} }

514
yarn.lock

File diff suppressed because it is too large Load Diff