Compare commits

..

1 Commits

Author SHA1 Message Date
frappe-pr-bot
d4c0ddb191 chore: update POT file 2025-03-28 16:04:20 +00:00
134 changed files with 10224 additions and 30306 deletions

View File

@@ -7,27 +7,8 @@ on:
branches: [ main ] branches: [ main ]
jobs: jobs:
commit-lint:
name: 'Semantic Commits'
runs-on: ubuntu-latest
if: github.event_name == 'pull_request'
steps:
- uses: actions/checkout@v4
with:
fetch-depth: 200
- uses: actions/setup-node@v4
with:
node-version: 20
check-latest: true
- name: Check commit titles
run: |
npm install @commitlint/cli @commitlint/config-conventional
npx commitlint --verbose --from ${{ github.event.pull_request.base.sha }} --to ${{ github.event.pull_request.head.sha }}
linters: linters:
name: Semgrep Rules name: Semantic Commits
runs-on: ubuntu-latest runs-on: ubuntu-latest
if: github.event_name == 'pull_request' if: github.event_name == 'pull_request'
@@ -39,17 +20,8 @@ jobs:
with: with:
python-version: '3.10' python-version: '3.10'
- name: Cache pip
uses: actions/cache@v4
with:
path: ~/.cache/pip
key: ${{ runner.os }}-pip-${{ hashFiles('**/*requirements.txt', '**/pyproject.toml', '**/setup.py') }}
restore-keys: |
${{ runner.os }}-pip-
${{ runner.os }}-
- name: Install and Run Pre-commit - name: Install and Run Pre-commit
uses: pre-commit/action@v3.0.1 uses: pre-commit/action@v2.0.3
- name: Download Semgrep rules - name: Download Semgrep rules
run: git clone --depth 1 https://github.com/frappe/semgrep-rules.git frappe-semgrep-rules run: git clone --depth 1 https://github.com/frappe/semgrep-rules.git frappe-semgrep-rules

View File

@@ -1,7 +1,8 @@
name: Create weekly release name: Create weekly release
on: on:
schedule: schedule:
- cron: '30 4 15 * *' # 13:00 UTC -> 7pm IST on every Wednesday
- cron: '30 4 * * 3'
workflow_dispatch: workflow_dispatch:
jobs: jobs:

View File

@@ -70,7 +70,7 @@ jobs:
id: yarn-cache-dir-path id: yarn-cache-dir-path
run: echo "dir=$(yarn cache dir)" >> $GITHUB_OUTPUT run: echo "dir=$(yarn cache dir)" >> $GITHUB_OUTPUT
- uses: actions/cache@v4 - uses: actions/cache@v3
id: yarn-cache id: yarn-cache
with: with:
path: ${{ steps.yarn-cache-dir-path.outputs.dir }} path: ${{ steps.yarn-cache-dir-path.outputs.dir }}
@@ -79,7 +79,7 @@ jobs:
${{ runner.os }}-yarn-ui- ${{ runner.os }}-yarn-ui-
- name: Cache cypress binary - name: Cache cypress binary
uses: actions/cache@v4 uses: actions/cache@v3
with: with:
path: ~/.cache/Cypress path: ~/.cache/Cypress
key: ${{ runner.os }}-cypress key: ${{ runner.os }}-cypress
@@ -100,7 +100,6 @@ jobs:
bench --site lms.test execute frappe.utils.install.complete_setup_wizard bench --site lms.test execute frappe.utils.install.complete_setup_wizard
bench --site lms.test execute frappe.tests.ui_test_helpers.create_test_user bench --site lms.test execute frappe.tests.ui_test_helpers.create_test_user
bench --site lms.test set-password frappe@example.com admin bench --site lms.test set-password frappe@example.com admin
bench --site lms.test execute lms.lms.utils.persona_captured
- name: cypress pre-requisites - name: cypress pre-requisites
run: | run: |

View File

@@ -1,26 +0,0 @@
module.exports = {
parserPreset: "conventional-changelog-conventionalcommits",
rules: {
"subject-empty": [2, "never"],
"type-case": [2, "always", "lower-case"],
"type-empty": [2, "never"],
"type-enum": [
2,
"always",
[
"build",
"chore",
"ci",
"docs",
"feat",
"fix",
"perf",
"refactor",
"revert",
"style",
"test",
"deprecate", // deprecation decision
],
],
},
};

View File

@@ -13,6 +13,6 @@ module.exports = defineConfig({
openMode: 0, openMode: 0,
}, },
e2e: { e2e: {
baseUrl: "http://pertest:8000", baseUrl: "http://testui:8000",
}, },
}); });

View File

@@ -19,11 +19,7 @@ describe("Course Creation", () => {
); );
cy.fixture("profile.png", "base64").then((fileContent) => { cy.fixture("profile.png", "base64").then((fileContent) => {
cy.get("div") cy.get('input[type="file"]').attachFile({
.contains("Course Image")
.siblings("div")
.children('input[type="file"]')
.attachFile({
fileContent, fileContent,
fileName: "profile.png", fileName: "profile.png",
mimeType: "image/png", mimeType: "image/png",

1
frappe-ui Submodule

Submodule frappe-ui added at 704a098eb1

View File

@@ -16,7 +16,6 @@ declare module 'vue' {
AssessmentPlugin: typeof import('./src/components/AssessmentPlugin.vue')['default'] AssessmentPlugin: typeof import('./src/components/AssessmentPlugin.vue')['default']
Assessments: typeof import('./src/components/Assessments.vue')['default'] Assessments: typeof import('./src/components/Assessments.vue')['default']
Assignment: typeof import('./src/components/Assignment.vue')['default'] Assignment: typeof import('./src/components/Assignment.vue')['default']
AssignmentForm: typeof import('./src/components/Modals/AssignmentForm.vue')['default']
AudioBlock: typeof import('./src/components/AudioBlock.vue')['default'] AudioBlock: typeof import('./src/components/AudioBlock.vue')['default']
Autocomplete: typeof import('./src/components/Controls/Autocomplete.vue')['default'] Autocomplete: typeof import('./src/components/Controls/Autocomplete.vue')['default']
BatchCard: typeof import('./src/components/BatchCard.vue')['default'] BatchCard: typeof import('./src/components/BatchCard.vue')['default']
@@ -69,9 +68,9 @@ declare module 'vue' {
NoPermission: typeof import('./src/components/NoPermission.vue')['default'] NoPermission: typeof import('./src/components/NoPermission.vue')['default']
NoSidebarLayout: typeof import('./src/components/NoSidebarLayout.vue')['default'] NoSidebarLayout: typeof import('./src/components/NoSidebarLayout.vue')['default']
NotPermitted: typeof import('./src/components/NotPermitted.vue')['default'] NotPermitted: typeof import('./src/components/NotPermitted.vue')['default']
OnboardingBanner: typeof import('./src/components/OnboardingBanner.vue')['default']
PageModal: typeof import('./src/components/Modals/PageModal.vue')['default'] PageModal: typeof import('./src/components/Modals/PageModal.vue')['default']
PaymentSettings: typeof import('./src/components/PaymentSettings.vue')['default'] PaymentSettings: typeof import('./src/components/PaymentSettings.vue')['default']
Play: typeof import('./src/components/Icons/Play.vue')['default']
ProgressBar: typeof import('./src/components/ProgressBar.vue')['default'] ProgressBar: typeof import('./src/components/ProgressBar.vue')['default']
Question: typeof import('./src/components/Modals/Question.vue')['default'] Question: typeof import('./src/components/Modals/Question.vue')['default']
Quiz: typeof import('./src/components/Quiz.vue')['default'] Quiz: typeof import('./src/components/Quiz.vue')['default']

View File

@@ -2,9 +2,9 @@
<html lang="en"> <html lang="en">
<head> <head>
<meta charset="UTF-8" /> <meta charset="UTF-8" />
<link rel="icon" href="{{ favicon }}" /> <link rel="icon" href="{{ favicon or '/assets/lms/frontend/favicon.png' }}" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" /> <meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>{{ title }}</title> <title>Frappe Learning</title>
<meta name="title" content="{{ meta.title }}" /> <meta name="title" content="{{ meta.title }}" />
<meta name="image" content="{{ meta.image }}" /> <meta name="image" content="{{ meta.image }}" />
<meta name="description" content="{{ meta.description }}" /> <meta name="description" content="{{ meta.description }}" />
@@ -23,10 +23,26 @@
<p> <p>
{{ meta.description }} {{ meta.description }}
</p> </p>
<p>
The content here is just for seo purposes. The actual content will be loaded in a few seconds.
</p>
<p>
Seo checks if a page has more than 300 words. So, here are some more words to make it more than 300 words.
Page descriptions are the HTML meta tags that provide a brief summary of a web page.
Search engines use meta descriptions to help identify the page's topic - they don't use them to rank the page, but they do use them to determine whether or not to display the page in search results.
Meta descriptions are important because they're often the first thing people see when they're deciding which search result to click on.
They're also important because they can help improve your click-through rate (CTR) from search results.
A good meta description can entice people to click on your page instead of someone else's.
</p>
<a href="{{ meta.link }}">Know More</a> <a href="{{ meta.link }}">Know More</a>
</div> </div>
</div> </div>
<div id="modals"></div>
<div id="popovers"></div>
<script> <script>
window.csrf_token = '{{ csrf_token }}'
window.setup_complete = '{{ setup_complete }}'
document.getElementById('seo-content').style.display = 'none'; document.getElementById('seo-content').style.display = 'none';
</script> </script>
<script type="module" src="/src/main.js"></script> <script type="module" src="/src/main.js"></script>

View File

@@ -26,12 +26,11 @@
"codemirror-editor-vue3": "^2.8.0", "codemirror-editor-vue3": "^2.8.0",
"dayjs": "^1.11.6", "dayjs": "^1.11.6",
"feather-icons": "^4.28.0", "feather-icons": "^4.28.0",
"frappe-ui": "^0.1.134", "frappe-ui": "^0.1.122",
"highlight.js": "^11.11.1", "highlight.js": "^11.11.1",
"lucide-vue-next": "^0.383.0", "lucide-vue-next": "^0.383.0",
"markdown-it": "^14.0.0", "markdown-it": "^14.0.0",
"pinia": "^2.0.33", "pinia": "^2.0.33",
"plyr": "^3.7.8",
"socket.io-client": "^4.7.2", "socket.io-client": "^4.7.2",
"tailwindcss": "3.4.15", "tailwindcss": "3.4.15",
"typescript": "^5.7.2", "typescript": "^5.7.2",

View File

@@ -1,4 +0,0 @@
<svg width="80" height="79" viewBox="0 0 80 79" fill="none" xmlns="http://www.w3.org/2000/svg">
<path d="M57.1285 0.580383H22.8514C10.2309 0.580383 0 10.5649 0 22.8815V56.3332C0 68.6497 10.2309 78.6343 22.8514 78.6343H57.1285C69.749 78.6343 79.9799 68.6497 79.9799 56.3332V22.8815C79.9799 10.5649 69.749 0.580383 57.1285 0.580383Z" fill="#0E7159"/>
<path d="M62.8434 23.6906L60.7869 23.1052C53.6744 21.0702 45.9048 22.4641 39.992 26.8128C35.8502 23.7742 30.7943 22.1854 25.7099 22.2133H17.1406V27.8163H25.7099C29.6232 27.8163 33.508 29.015 36.6787 31.3845L39.992 33.8377L43.3056 31.3845C47.2475 28.4575 52.3032 27.2588 57.1306 28.0393V50.647C51.1035 49.9223 44.9051 51.4834 39.992 55.0795C35.8502 52.0688 30.8515 50.4798 25.7671 50.4798C24.7959 50.4798 23.8247 50.5355 22.8535 50.647V35.0642H17.1406V57.0588H62.8434V23.7185V23.6906Z" fill="white"/>
</svg>

Before

Width:  |  Height:  |  Size: 856 B

View File

@@ -24,7 +24,7 @@ const router = useRouter()
const noSidebar = ref(false) const noSidebar = ref(false)
router.beforeEach((to, from, next) => { router.beforeEach((to, from, next) => {
if (to.query.fromLesson || to.path === '/persona') { if (to.query.fromLesson) {
noSidebar.value = true noSidebar.value = true
} else { } else {
noSidebar.value = false noSidebar.value = false

View File

@@ -39,11 +39,7 @@
{{ __('More') }} {{ __('More') }}
</span> </span>
</div> </div>
<Button <Button v-if="isModerator" variant="ghost" @click="openPageModal()">
v-if="isModerator && !readOnlyMode"
variant="ghost"
@click="openPageModal()"
>
<template #icon> <template #icon>
<Plus class="h-4 w-4 text-ink-gray-7 stroke-1.5" /> <Plus class="h-4 w-4 text-ink-gray-7 stroke-1.5" />
</template> </template>
@@ -67,16 +63,6 @@
</div> </div>
</div> </div>
<div class="m-2 flex flex-col gap-1"> <div class="m-2 flex flex-col gap-1">
<div
v-if="readOnlyMode && !sidebarStore.isSidebarCollapsed"
class="z-10 m-2 bg-surface-modal py-2.5 px-3 text-xs text-ink-gray-7 leading-5 rounded-md"
>
{{
__(
'This site is being updated. You will not be able to make any changes. Full access will be restored shortly.'
)
}}
</div>
<TrialBanner <TrialBanner
v-if=" v-if="
userResource.data?.is_system_manager && userResource.data?.is_fc_site userResource.data?.is_system_manager && userResource.data?.is_fc_site
@@ -88,69 +74,43 @@
:isSidebarCollapsed="sidebarStore.isSidebarCollapsed" :isSidebarCollapsed="sidebarStore.isSidebarCollapsed"
appName="learning" appName="learning"
/> />
<SidebarLink
<div v-if="isOnboardingStepsCompleted"
class="flex items-center mt-4" :link="{
:class=" label: __('Help'),
sidebarStore.isSidebarCollapsed ? 'flex-col space-y-3' : 'flex-row' }"
" :isCollapsed="sidebarStore.isSidebarCollapsed"
>
<div
class="flex items-center flex-1"
:class="
sidebarStore.isSidebarCollapsed
? 'flex-col space-y-3'
: 'flex-row space-x-3'
"
>
<Tooltip v-if="readOnlyMode && sidebarStore.isSidebarCollapsed">
<CircleAlert
class="size-4 stroke-1.5 text-ink-gray-7 cursor-pointer"
/>
<template #body>
<div
class="max-w-[30ch] rounded bg-surface-gray-7 px-2 py-1 text-center text-p-xs text-ink-white shadow-xl"
>
{{
__(
'This site is being updated. You will not be able to make any changes. Full access will be restored shortly.'
)
}}
</div>
</template>
</Tooltip>
<Tooltip :text="__('Powered by Learning')">
<Zap
class="size-4 stroke-1.5 text-ink-gray-7 cursor-pointer"
@click="redirectToWebsite()"
/>
</Tooltip>
<Tooltip :text="__('Help')">
<CircleHelp
class="size-4 stroke-1.5 text-ink-gray-7 cursor-pointer"
@click=" @click="
() => { () => {
showHelpModal = minimize ? true : !showHelpModal showHelpModal = minimize ? true : !showHelpModal
minimize = !showHelpModal minimize = !showHelpModal
} }
" "
/>
</Tooltip>
</div>
<Tooltip
:text="
sidebarStore.isSidebarCollapsed ? __('Expand') : __('Collapse')
"
> >
<template #icon>
<span class="grid h-5 w-6 flex-shrink-0 place-items-center">
<CircleHelp class="h-4 w-4 stroke-1.5" />
</span>
</template>
</SidebarLink>
<SidebarLink
:link="{
label: sidebarStore.isSidebarCollapsed ? 'Expand' : 'Collapse',
}"
:isCollapsed="sidebarStore.isSidebarCollapsed"
@click="toggleSidebar()"
>
<template #icon>
<span class="grid h-5 w-6 flex-shrink-0 place-items-center">
<CollapseSidebar <CollapseSidebar
class="size-4 text-ink-gray-7 duration-300 stroke-1.5 ease-in-out cursor-pointer" class="h-4 w-4 text-ink-gray-7 duration-300 ease-in-out"
:class="{ :class="{
'[transform:rotateY(180deg)]': sidebarStore.isSidebarCollapsed, '[transform:rotateY(180deg)]': sidebarStore.isSidebarCollapsed,
}" }"
@click="toggleSidebar()"
/> />
</Tooltip> </span>
</div> </template>
</SidebarLink>
</div> </div>
<HelpModal <HelpModal
v-if="showOnboarding && showHelpModal" v-if="showOnboarding && showHelpModal"
@@ -188,7 +148,7 @@ import { usersStore } from '@/stores/user'
import { sessionStore } from '@/stores/session' import { sessionStore } from '@/stores/session'
import { useSidebar } from '@/stores/sidebar' import { useSidebar } from '@/stores/sidebar'
import { useSettings } from '@/stores/settings' import { useSettings } from '@/stores/settings'
import { Button, createResource, Tooltip } from 'frappe-ui' import { Button, createResource } from 'frappe-ui'
import PageModal from '@/components/Modals/PageModal.vue' import PageModal from '@/components/Modals/PageModal.vue'
import { capture } from '@/telemetry' import { capture } from '@/telemetry'
import LMSLogo from '@/components/Icons/LMSLogo.vue' import LMSLogo from '@/components/Icons/LMSLogo.vue'
@@ -196,7 +156,6 @@ import { useRouter } from 'vue-router'
import InviteIcon from './Icons/InviteIcon.vue' import InviteIcon from './Icons/InviteIcon.vue'
import { import {
BookOpen, BookOpen,
CircleAlert,
ChevronRight, ChevronRight,
Plus, Plus,
CircleHelp, CircleHelp,
@@ -205,7 +164,6 @@ import {
UserPlus, UserPlus,
Users, Users,
BookText, BookText,
Zap,
} from 'lucide-vue-next' } from 'lucide-vue-next'
import { import {
TrialBanner, TrialBanner,
@@ -234,7 +192,6 @@ const currentStep = ref({})
const router = useRouter() const router = useRouter()
let onboardingDetails let onboardingDetails
let isOnboardingStepsCompleted = false let isOnboardingStepsCompleted = false
const readOnlyMode = window.read_only_mode
const iconProps = { const iconProps = {
strokeWidth: 1.5, strokeWidth: 1.5,
width: 16, width: 16,
@@ -621,8 +578,4 @@ watch(userResource, () => {
setUpOnboarding() setUpOnboarding()
} }
}) })
const redirectToWebsite = () => {
window.open('https://frappe.io/learning', '_blank')
}
</script> </script>

View File

@@ -3,7 +3,7 @@
<template #target="{ togglePopover }"> <template #target="{ togglePopover }">
<button <button
:class="[ :class="[
'group w-full flex h-7 items-center justify-between rounded px-2 text-base text-ink-gray-7 hover:bg-surface-gray-2', 'group w-full flex h-7 items-center justify-between rounded px-2 text-base text-ink-gray-8 hover:bg-surface-gray-2',
]" ]"
@click.prevent="togglePopover()" @click.prevent="togglePopover()"
> >

View File

@@ -4,7 +4,7 @@
<div class="text-lg font-semibold text-ink-gray-9"> <div class="text-lg font-semibold text-ink-gray-9">
{{ __('Assessments') }} {{ __('Assessments') }}
</div> </div>
<Button v-if="canAddAssessments()" @click="showModal = true"> <Button v-if="canSeeAddButton()" @click="showModal = true">
<template #prefix> <template #prefix>
<Plus class="h-4 w-4" /> <Plus class="h-4 w-4" />
</template> </template>
@@ -100,7 +100,6 @@ import { Plus, Trash2 } from 'lucide-vue-next'
const user = inject('$user') const user = inject('$user')
const showModal = ref(false) const showModal = ref(false)
const readOnlyMode = window.read_only_mode
const props = defineProps({ const props = defineProps({
batch: { batch: {
@@ -182,8 +181,7 @@ const getRowRoute = (row) => {
} }
} }
const canAddAssessments = () => { const canSeeAddButton = () => {
if (readOnlyMode) return false
return user.data?.is_moderator || user.data?.is_evaluator return user.data?.is_moderator || user.data?.is_evaluator
} }

View File

@@ -1,6 +1,6 @@
<template> <template>
<div <div
class="flex flex-col border hover:border-outline-gray-4 rounded-md p-4 h-full" class="flex flex-col border-2 hover:bg-surface-gray-2 rounded-md p-4 h-full"
style="min-height: 150px" style="min-height: 150px"
> >
<div class="text-lg leading-5 font-semibold mb-2 text-ink-gray-9"> <div class="text-lg leading-5 font-semibold mb-2 text-ink-gray-9">

View File

@@ -89,7 +89,6 @@ import {
} from 'frappe-ui' } from 'frappe-ui'
import { Plus, Trash2 } from 'lucide-vue-next' import { Plus, Trash2 } from 'lucide-vue-next'
import { showToast } from '@/utils' import { showToast } from '@/utils'
const readOnlyMode = window.read_only_mode
const showCourseModal = ref(false) const showCourseModal = ref(false)
const user = inject('$user') const user = inject('$user')
@@ -160,9 +159,6 @@ const removeCourses = (selections, unselectAll) => {
} }
const canSeeAddButton = () => { const canSeeAddButton = () => {
if (readOnlyMode) {
return false
}
return user.data?.is_moderator || user.data?.is_evaluator return user.data?.is_moderator || user.data?.is_evaluator
} }
</script> </script>

View File

@@ -111,6 +111,7 @@ import {
FormControl, FormControl,
ListView, ListView,
ListHeader, ListHeader,
ListHeaderItem,
ListRows, ListRows,
ListRow, ListRow,
ListRowItem, ListRowItem,

View File

@@ -24,10 +24,7 @@
> >
{{ formatNumberIntoCurrency(batch.data.amount, batch.data.currency) }} {{ formatNumberIntoCurrency(batch.data.amount, batch.data.currency) }}
</div> </div>
<div <div class="flex items-center mb-3 text-ink-gray-7">
v-if="batch.data.courses.length"
class="flex items-center mb-3 text-ink-gray-7"
>
<BookOpen class="h-4 w-4 stroke-1.5 mr-2" /> <BookOpen class="h-4 w-4 stroke-1.5 mr-2" />
<span> {{ batch.data.courses.length }} {{ __('Courses') }} </span> <span> {{ batch.data.courses.length }} {{ __('Courses') }} </span>
</div> </div>
@@ -49,7 +46,6 @@
{{ batch.data.timezone }} {{ batch.data.timezone }}
</span> </span>
</div> </div>
<div v-if="!readOnlyMode">
<router-link <router-link
v-if="isModerator || isStudent" v-if="isModerator || isStudent"
:to="{ :to="{
@@ -113,7 +109,6 @@
</Button> </Button>
</router-link> </router-link>
</div> </div>
</div>
</template> </template>
<script setup> <script setup>
import { inject, computed } from 'vue' import { inject, computed } from 'vue'
@@ -125,7 +120,7 @@ import { useRouter } from 'vue-router'
const router = useRouter() const router = useRouter()
const user = inject('$user') const user = inject('$user')
const readOnlyMode = window.read_only_mode const dayjs = inject('$dayjs')
const props = defineProps({ const props = defineProps({
batch: { batch: {

View File

@@ -110,7 +110,7 @@
<div class="text-ink-gray-7 font-medium"> <div class="text-ink-gray-7 font-medium">
{{ __('Students') }} {{ __('Students') }}
</div> </div>
<Button v-if="!readOnlyMode" @click="openStudentModal()"> <Button @click="openStudentModal()">
<template #prefix> <template #prefix>
<Plus class="h-4 w-4" /> <Plus class="h-4 w-4" />
</template> </template>
@@ -247,7 +247,6 @@ const chartData = ref(null)
const chartOptions = ref(null) const chartOptions = ref(null)
const showProgressChart = ref(false) const showProgressChart = ref(false)
const assessmentCount = ref(0) const assessmentCount = ref(0)
const readOnlyMode = window.read_only_mode
const props = defineProps({ const props = defineProps({
batch: { batch: {

View File

@@ -28,7 +28,9 @@
</template> </template>
<template #body="{ isOpen }"> <template #body="{ isOpen }">
<div v-show="isOpen"> <div v-show="isOpen">
<div class="mt-1 rounded-lg bg-surface-white py-1 text-base border-2"> <div
class="mt-1 rounded-lg bg-surface-white py-1 text-base shadow-2xl"
>
<div class="relative px-1.5 pt-0.5"> <div class="relative px-1.5 pt-0.5">
<ComboboxInput <ComboboxInput
ref="search" ref="search"
@@ -47,7 +49,7 @@
class="absolute right-1.5 inline-flex h-7 w-7 items-center justify-center" class="absolute right-1.5 inline-flex h-7 w-7 items-center justify-center"
@click="selectedValue = null" @click="selectedValue = null"
> >
<X class="h-4 w-4 stroke-1.5 text-ink-gray-7" /> <X class="h-4 w-4 stroke-1.5" />
</button> </button>
</div> </div>
<ComboboxOptions <ComboboxOptions
@@ -87,7 +89,7 @@
name="item-label" name="item-label"
v-bind="{ active, selected, option }" v-bind="{ active, selected, option }"
> >
<div class="flex flex-col space-y-1 text-ink-gray-8"> <div class="flex flex-col space-y-1">
<div> <div>
{{ option.label }} {{ option.label }}
</div> </div>

View File

@@ -146,6 +146,7 @@ function resetEditor(value: string, resetHistory = false) {
value = getModelValue() value = getModelValue()
aceEditor?.setValue(value) aceEditor?.setValue(value)
aceEditor?.clearSelection() aceEditor?.clearSelection()
console.log(isDark.value)
aceEditor?.setTheme(isDark.value ? 'ace/theme/twilight' : 'ace/theme/chrome') aceEditor?.setTheme(isDark.value ? 'ace/theme/twilight' : 'ace/theme/chrome')
props.autofocus && aceEditor?.focus() props.autofocus && aceEditor?.focus()
if (resetHistory) { if (resetHistory) {

View File

@@ -4,7 +4,7 @@
{{ label }} {{ label }}
<span class="text-ink-red-3" v-if="required">*</span> <span class="text-ink-red-3" v-if="required">*</span>
</label> </label>
<div class="grid grid-cols-3 gap-2"> <div class="grid grid-cols-3 gap-1">
<Button <Button
ref="emails" ref="emails"
v-for="value in values" v-for="value in values"
@@ -12,7 +12,7 @@
:label="value" :label="value"
theme="gray" theme="gray"
variant="subtle" variant="subtle"
class="rounded-md word-break-all" class="rounded-md"
@keydown.delete.capture.stop="removeLastValue" @keydown.delete.capture.stop="removeLastValue"
> >
<template #suffix> <template #suffix>
@@ -42,7 +42,7 @@
<template #body="{ isOpen }"> <template #body="{ isOpen }">
<div v-show="isOpen"> <div v-show="isOpen">
<div <div
class="mt-1 rounded-lg bg-surface-white py-1 text-base border-2" class="mt-1 rounded-lg bg-surface-white py-1 text-base shadow-2xl"
> >
<ComboboxOptions <ComboboxOptions
class="my-1 max-h-[12rem] overflow-y-auto px-1.5" class="my-1 max-h-[12rem] overflow-y-auto px-1.5"
@@ -61,7 +61,7 @@
]" ]"
> >
<div class="flex flex-col gap-1 p-1"> <div class="flex flex-col gap-1 p-1">
<div class="text-base font-medium text-ink-gray-8"> <div class="text-base font-medium">
{{ option.description }} {{ option.description }}
</div> </div>
<div class="text-sm text-ink-gray-5"> <div class="text-sm text-ink-gray-5">

View File

@@ -9,20 +9,16 @@
: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 items-center flex-wrap relative top-4 px-2 w-fit"> <div
<Badge class="flex items-center flex-wrap space-x-1 relative top-4 px-2 w-fit"
v-if="course.featured"
variant="subtle"
theme="green"
size="md"
class="mb-1 mr-1"
> >
<Badge v-if="course.featured" variant="subtle" theme="green" size="md">
{{ __('Featured') }} {{ __('Featured') }}
</Badge> </Badge>
<div <div
v-if="course.tags" v-if="course.tags"
v-for="tag in course.tags?.split(', ')" v-for="tag in course.tags?.split(', ')"
class="text-xs bg-white text-gray-800 px-2 py-0.5 rounded-md mb-1 mr-1" class="text-xs bg-white text-gray-800 px-2 py-0.5 rounded-md"
> >
{{ tag }} {{ tag }}
</div> </div>

View File

@@ -9,7 +9,6 @@
<div v-if="course.data.paid_course" class="text-2xl font-semibold mb-3"> <div v-if="course.data.paid_course" class="text-2xl font-semibold mb-3">
{{ course.data.price }} {{ course.data.price }}
</div> </div>
<div v-if="!readOnlyMode">
<div v-if="course.data.membership" class="space-y-2"> <div v-if="course.data.membership" class="space-y-2">
<router-link <router-link
:to="{ :to="{
@@ -49,13 +48,12 @@
</span> </span>
</Button> </Button>
</router-link> </router-link>
<Badge <div
v-else-if="course.data.disable_self_learning" v-else-if="course.data.disable_self_learning"
theme="blue" class="bg-surface-blue-2 text-blue-900 text-sm rounded-md py-1 px-3"
size="lg"
> >
{{ __('Contact the Administrator to enroll for this course.') }} {{ __('Contact the Administrator to enroll for this course.') }}
</Badge> </div>
<Button <Button
v-else v-else
@click="enrollStudent()" @click="enrollStudent()"
@@ -91,12 +89,8 @@
</span> </span>
</Button> </Button>
</router-link> </router-link>
</div>
<div class="space-y-4"> <div class="space-y-4">
<div <div class="mt-8 font-medium text-ink-gray-9">
class="font-medium text-ink-gray-9"
:class="{ 'mt-8': !readOnlyMode }"
>
{{ __('This course has:') }} {{ __('This course has:') }}
</div> </div>
<div class="flex items-center text-ink-gray-9"> <div class="flex items-center text-ink-gray-9">
@@ -146,7 +140,7 @@
<script setup> <script setup>
import { BookOpen, Users, Star, GraduationCap } from 'lucide-vue-next' import { BookOpen, Users, Star, GraduationCap } from 'lucide-vue-next'
import { computed, inject } from 'vue' import { computed, inject } from 'vue'
import { Badge, Button, createResource } from 'frappe-ui' import { Button, createResource, Tooltip } from 'frappe-ui'
import { showToast, formatAmount } from '@/utils/' import { showToast, formatAmount } from '@/utils/'
import { capture } from '@/telemetry' import { capture } from '@/telemetry'
import { useRouter } from 'vue-router' import { useRouter } from 'vue-router'
@@ -154,7 +148,6 @@ import CertificationLinks from '@/components/CertificationLinks.vue'
const router = useRouter() const router = useRouter()
const user = inject('$user') const user = inject('$user')
const readOnlyMode = window.read_only_mode
const props = defineProps({ const props = defineProps({
course: { course: {
@@ -179,7 +172,7 @@ function enrollStudent() {
) )
setTimeout(() => { setTimeout(() => {
window.location.href = `/login?redirect-to=${window.location.pathname}` window.location.href = `/login?redirect-to=${window.location.pathname}`
}, 1000) }, 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',

View File

@@ -1,5 +1,5 @@
<template> <template>
<div class=""> <div class="h-full">
<div <div
v-if="title && (outline.data?.length || allowEdit)" v-if="title && (outline.data?.length || allowEdit)"
class="flex items-center justify-between space-x-2 mb-4 px-2" class="flex items-center justify-between space-x-2 mb-4 px-2"
@@ -17,6 +17,9 @@
<Button size="sm" v-if="allowEdit" @click="openChapterModal()"> <Button size="sm" v-if="allowEdit" @click="openChapterModal()">
{{ __('Add Chapter') }} {{ __('Add Chapter') }}
</Button> </Button>
<!-- <span class="font-medium cursor-pointer" @click="expandAllChapters()">
{{ expandAll ? __("Collapse all chapters") : __("Expand all chapters") }}
</span> -->
</div> </div>
<div <div
:class="{ :class="{
@@ -139,7 +142,6 @@
</div> </div>
</div> </div>
<ChapterModal <ChapterModal
v-if="user.data"
v-model="showChapterModal" v-model="showChapterModal"
v-model:outline="outline" v-model:outline="outline"
:course="courseName" :course="courseName"

View File

@@ -27,9 +27,7 @@
</span> </span>
</div> </div>
<Dropdown <Dropdown
v-if=" v-if="user.data.name == reply.owner && !reply.editable"
user.data.name == reply.owner && !reply.editable && !readOnlyMode
"
:options="[ :options="[
{ {
label: 'Edit', label: 'Edit',
@@ -73,7 +71,7 @@
</div> </div>
<TextEditor <TextEditor
v-if="renderEditor && !readOnlyMode" v-if="renderEditor"
class="mt-5" class="mt-5"
:content="newReply" :content="newReply"
:mentions="mentionUsers" :mentions="mentionUsers"
@@ -82,7 +80,7 @@
:fixedMenu="true" :fixedMenu="true"
editorClass="ProseMirror prose prose-table:table-fixed prose-td:p-2 prose-th:p-2 prose-td:border prose-th:border prose-td:border-outline-gray-2 prose-th:border-outline-gray-2 prose-td:relative prose-th:relative prose-th:bg-surface-gray-2 prose-sm max-w-none border border-outline-gray-2 rounded-b-md min-h-[7rem] py-1 px-2" editorClass="ProseMirror prose prose-table:table-fixed prose-td:p-2 prose-th:p-2 prose-td:border prose-th:border prose-td:border-outline-gray-2 prose-th:border-outline-gray-2 prose-td:relative prose-th:relative prose-th:bg-surface-gray-2 prose-sm max-w-none border border-outline-gray-2 rounded-b-md min-h-[7rem] py-1 px-2"
/> />
<div v-if="!readOnlyMode" class="flex justify-between mt-2"> <div class="flex justify-between mt-2">
<span> </span> <span> </span>
<Button @click="postReply()"> <Button @click="postReply()">
<span> <span>
@@ -107,7 +105,6 @@ const user = inject('$user')
const allUsers = inject('$allUsers') const allUsers = inject('$allUsers')
const mentionUsers = ref([]) const mentionUsers = ref([])
const renderEditor = ref(false) const renderEditor = ref(false)
const readOnlyMode = window.read_only_mode
const props = defineProps({ const props = defineProps({
topic: { topic: {

View File

@@ -1,10 +1,6 @@
<template> <template>
<div> <div>
<Button <Button v-if="!singleThread" class="float-right" @click="openTopicModal()">
v-if="!singleThread && !readOnlyMode"
class="float-right"
@click="openTopicModal()"
>
{{ __('New {0}').format(singularize(title)) }} {{ __('New {0}').format(singularize(title)) }}
</Button> </Button>
<div class="text-xl font-semibold text-ink-gray-9"> <div class="text-xl font-semibold text-ink-gray-9">
@@ -81,7 +77,6 @@ const currentTopic = ref(null)
const socket = inject('$socket') const socket = inject('$socket')
const user = inject('$user') const user = inject('$user')
const showTopicModal = ref(false) const showTopicModal = ref(false)
const readOnlyMode = window.read_only_mode
const props = defineProps({ const props = defineProps({
title: { title: {

View File

@@ -97,7 +97,7 @@ const evaluators = createResource({
return { return {
doctype: 'Course Evaluator', doctype: 'Course Evaluator',
fields: ['evaluator', 'full_name', 'user_image', 'username'], fields: ['evaluator', 'full_name', 'user_image', 'username'],
filters: search.value ? { evaluator: ['like', `%${search.value}%`] } : {}, filters: search.value ? [['evaluator', 'like', search.value]] : [],
} }
}, },
auto: true, auto: true,

View File

@@ -1,18 +1,36 @@
<template> <template>
<svg <svg
width="80" width="118"
height="79" height="118"
viewBox="0 0 80 79" viewBox="0 0 118 118"
fill="none" fill="none"
xmlns="http://www.w3.org/2000/svg" xmlns="http://www.w3.org/2000/svg"
> >
<path <path
d="M57.1285 0.580383H22.8514C10.2309 0.580383 0 10.5649 0 22.8815V56.3332C0 68.6497 10.2309 78.6343 22.8514 78.6343H57.1285C69.749 78.6343 79.9799 68.6497 79.9799 56.3332V22.8815C79.9799 10.5649 69.749 0.580383 57.1285 0.580383Z" d="M93.9278 0H23.1013C10.3428 0 0 10.3428 0 23.1013V93.9278C0 106.686 10.3428 117.029 23.1013 117.029H93.9278C106.686 117.029 117.029 106.686 117.029 93.9278V23.1013C117.029 10.3428 106.686 0 93.9278 0Z"
fill="#0E7159" fill="url(#paint0_radial_174_336)"
/> />
<path <path
d="M62.8434 23.6906L60.7869 23.1052C53.6744 21.0702 45.9048 22.4641 39.992 26.8128C35.8502 23.7742 30.7943 22.1854 25.7099 22.2133H17.1406V27.8163H25.7099C29.6232 27.8163 33.508 29.015 36.6787 31.3845L39.992 33.8377L43.3056 31.3845C47.2475 28.4575 52.3032 27.2588 57.1306 28.0393V50.647C51.1035 49.9223 44.9051 51.4834 39.992 55.0795C35.8502 52.0688 30.8515 50.4798 25.7671 50.4798C24.7959 50.4798 23.8247 50.5355 22.8535 50.647V35.0642H17.1406V57.0588H62.8434V23.7185V23.6906Z" d="M93.9278 0H23.1013C10.3428 0 0 10.3428 0 23.1013V93.9278C0 106.686 10.3428 117.029 23.1013 117.029H93.9278C106.686 117.029 117.029 106.686 117.029 93.9278V23.1013C117.029 10.3428 106.686 0 93.9278 0Z"
fill="white" fill="#0B3D3D"
fill-opacity="0.8"
/> />
<path
d="M95.1879 33.1294L91.4077 32.0268C80.1721 28.7716 67.9389 30.9242 58.5409 37.7496C52.083 33.0769 43.9975 30.5042 36.1746 30.5042H21.8938V41.0048H36.2796C42.2649 41.0048 48.1978 42.9999 52.923 46.6226L58.5934 50.9279L64.2637 46.6226C70.144 42.1599 77.5469 40.2698 84.7923 41.2673V76.1818C75.5518 75.2367 66.2063 77.7044 58.6459 83.2172C51.0854 77.7044 41.6349 75.2367 32.4994 76.1818V52.8705H21.9988V86.4724H95.3454V33.1294H95.1879Z"
fill="#58FF9B"
/>
<defs>
<radialGradient
id="paint0_radial_174_336"
cx="0"
cy="0"
r="1"
gradientUnits="userSpaceOnUse"
gradientTransform="translate(117.24 -101.5) rotate(105.042) scale(226.282)"
>
<stop offset="0.445162" stop-color="#1F7676" />
<stop offset="1" stop-color="#0A4B4B" />
</radialGradient>
</defs>
</svg> </svg>
</template> </template>

View File

@@ -1,16 +0,0 @@
<template>
<svg
width="20"
height="20"
viewBox="0 0 68 75"
fill="none"
xmlns="http://www.w3.org/2000/svg"
>
<path
fill-rule="evenodd"
clip-rule="evenodd"
d="M0 6.78182C0 1.60212 5.5742 -1.65958 10.09 0.879521L64.09 31.2545C68.6916 33.8443 68.6916 40.4693 64.09 43.0595L10.09 73.4345C5.5744 75.9736 0 72.7119 0 67.5322V6.78182ZM26.2695 38.5201C26.2695 37.3248 25.2265 37.9342 26.2695 38.5201C27.332 39.1178 27.332 37.9225 26.2695 38.5201Z"
fill="white"
/>
</svg>
</template>

View File

@@ -1,52 +1,41 @@
<template> <template>
<div <div class="flex space-x-4 border rounded-md p-2">
class="flex flex-col border rounded-md p-3 h-full hover:border-outline-gray-4" <img :src="job.company_logo" class="size-10 rounded-full object-contain" />
>
<div class="flex space-x-4 mb-4">
<div class="flex flex-col space-y-2 flex-1"> <div class="flex flex-col space-y-2 flex-1">
<div class="text-lg font-semibold text-ink-gray-9"> <div class="flex items-center justify-between">
{{ job.company_name }} <span class="font-semibold text-ink-gray-9">
</div>
<span class="font-medium text-ink-gray-7 leading-5">
{{ job.job_title }} {{ job.job_title }}
</span> </span>
<div class="flex items-center space-x-1 text-sm text-ink-gray-7"> </div>
<MapPin class="size-3" /> <div class="flex items-center space-x-2 text-ink-gray-5">
<Building2 class="w-4 h-4 stroke-1.5" />
<span> <span>
{{ job.location }}{{ job.country ? `, ${job.country}` : '' }} {{ job.company_name }}
</span> </span>
</div> </div>
<div <div class="flex items-center space-x-2 text-ink-gray-5">
v-if="job.applicants" <MapPin class="w-4 h-4 stroke-1.5" />
class="flex items-center space-x-1 text-sm text-ink-gray-7"
>
<User class="size-3" />
<span> <span>
{{ job.applicants }} {{ job.location }}
{{ job.applicants > 1 ? __('applicants') : __('applicant') }}
</span> </span>
</div> </div>
</div> <div class="flex items-center space-x-2 text-ink-gray-5">
<!-- <img :src="job.company_logo" alt="Company Logo" class="size-8 rounded-full object-contain bg-white" /> --> <Shapes class="w-4 h-4 stroke-1.5" />
</div> <span>
<div class="space-x-2 mt-auto">
<Badge>
{{ job.type }} {{ job.type }}
</Badge> </span>
<Badge> </div>
{{ dayjs(job.creation).fromNow() }} <div class="flex items-center space-x-2 text-ink-gray-5">
</Badge> <Calendar class="w-4 h-4 stroke-1.5" />
<span> {{ __('posted') }} {{ dayjs(job.creation).fromNow() }} </span>
</div>
</div> </div>
<!-- <div
class="description text-ink-gray-9 text-sm"
v-html="job.description"
></div> -->
</div> </div>
</template> </template>
<script setup> <script setup>
import { Building2, Calendar, MapPin, Shapes } from 'lucide-vue-next'
import { inject } from 'vue' import { inject } from 'vue'
import { Badge } from 'frappe-ui' import { Avatar } from 'frappe-ui'
import { MapPin, User } from 'lucide-vue-next'
const dayjs = inject('$dayjs') const dayjs = inject('$dayjs')
const props = defineProps({ const props = defineProps({
@@ -56,15 +45,3 @@ const props = defineProps({
}, },
}) })
</script> </script>
<style>
.description {
display: -webkit-box;
-webkit-line-clamp: 2;
-webkit-box-orient: vertical;
text-overflow: ellipsis;
width: 100%;
overflow: hidden;
margin-top: auto;
line-height: 1.5;
}
</style>

View File

@@ -3,7 +3,7 @@
<div class="text-lg font-semibold text-ink-gray-9"> <div class="text-lg font-semibold text-ink-gray-9">
{{ __('Live Class') }} {{ __('Live Class') }}
</div> </div>
<Button v-if="canCreateClass()" @click="openLiveClassModal"> <Button v-if="user.data.is_moderator" @click="openLiveClassModal">
<template #prefix> <template #prefix>
<Plus class="h-4 w-4" /> <Plus class="h-4 w-4" />
</template> </template>
@@ -87,7 +87,6 @@ import { formatTime } from '@/utils/'
const user = inject('$user') const user = inject('$user')
const showLiveClassModal = ref(false) const showLiveClassModal = ref(false)
const dayjs = inject('$dayjs') const dayjs = inject('$dayjs')
const readOnlyMode = window.read_only_mode
const props = defineProps({ const props = defineProps({
batch: { batch: {
@@ -117,11 +116,6 @@ const liveClasses = createListResource({
const openLiveClassModal = () => { const openLiveClassModal = () => {
showLiveClassModal.value = true showLiveClassModal.value = true
} }
const canCreateClass = () => {
if (readOnlyMode) return false
return user.data?.is_moderator || user.data?.is_evaluator
}
</script> </script>
<style> <style>
.short-introduction { .short-introduction {

View File

@@ -118,23 +118,6 @@ import { ref, watch, reactive, inject } from 'vue'
import { RefreshCw, Plus, X } from 'lucide-vue-next' import { RefreshCw, Plus, X } from 'lucide-vue-next'
import { useOnboarding } from 'frappe-ui/frappe' import { useOnboarding } from 'frappe-ui/frappe'
interface User {
data: {
email: string
name: string
enabled: boolean
user_image: string
full_name: string
user_type: ['System User', 'Website User']
username: string
is_moderator: boolean
is_system_manager: boolean
is_evaluator: boolean
is_instructor: boolean
is_fc_site: boolean
}
}
const router = useRouter() const router = useRouter()
const show = defineModel('show') const show = defineModel('show')
const search = ref('') const search = ref('')
@@ -143,7 +126,6 @@ const memberList = ref([])
const hasNextPage = ref(false) const hasNextPage = ref(false)
const showForm = ref(false) const showForm = ref(false)
const dayjs = inject('$dayjs') const dayjs = inject('$dayjs')
const user = inject<User | null>('$user')
const { updateOnboardingStep } = useOnboarding('learning') const { updateOnboardingStep } = useOnboarding('learning')
const member = reactive({ const member = reactive({
@@ -205,9 +187,7 @@ const newMember = createResource({
auto: false, auto: false,
onSuccess(data) { onSuccess(data) {
show.value = false show.value = false
updateOnboardingStep('invite_students')
if (user?.data?.is_system_manager) updateOnboardingStep('invite_students')
router.push({ router.push({
name: 'Profile', name: 'Profile',
params: { params: {

View File

@@ -1,163 +0,0 @@
<template>
<Dialog
v-model="show"
:options="{
size: 'lg',
}"
>
<template #body>
<div class="p-5 text-base max-h-[75vh] overflow-y-auto">
<div class="text-lg text-ink-gray-9 font-semibold mb-5">
{{
assignmentID === 'new'
? __('Create an Assignment')
: __('Edit Assignment')
}}
</div>
<div class="space-y-4">
<FormControl
v-model="assignment.title"
:label="__('Title')"
:required="true"
/>
<FormControl
v-model="assignment.type"
type="select"
:options="assignmentOptions"
:label="__('Submission Type')"
:required="true"
/>
<div>
<div class="text-xs text-ink-gray-5 mb-2">
{{ __('Question') }}
<span class="text-ink-red-3">*</span>
</div>
<TextEditor
:content="assignment.question"
@change="(val) => (assignment.question = 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]"
/>
</div>
</div>
<div class="flex justify-end space-x-2 mt-5">
<router-link
:to="{
name: 'AssignmentSubmissionList',
query: {
assignmentID: assignmentID,
},
}"
>
<Button v-if="assignmentID !== 'new'" variant="subtle">
{{ __('Check Submissions') }}
</Button>
</router-link>
<Button variant="solid" @click="saveAssignment">
{{ __('Save') }}
</Button>
</div>
</div>
</template>
</Dialog>
</template>
<script setup lang="ts">
import { Button, Dialog, FormControl, TextEditor } from 'frappe-ui'
import { computed, reactive, watch } from 'vue'
import { showToast } from '@/utils'
const show = defineModel()
const assignments = defineModel<Assignments>('assignments')
interface Assignment {
title: string
type: string
question: string
}
interface Assignments {
data: Assignment[]
get: (params: { doctype: string; name: string }) => Promise<Assignment>
insert: {
submit: (params: Assignment, options: { onSuccess: () => void }) => void
}
}
const assignment = reactive({
title: '',
type: '',
question: '',
})
const props = defineProps({
assignmentID: {
type: String,
default: 'new',
},
})
watch(
() => props.assignmentID,
(val) => {
if (val !== 'new') {
assignments.value?.data.forEach((row) => {
if (row.name === val) {
assignment.title = row.title
assignment.type = row.type
assignment.question = row.question
}
})
}
},
{ flush: 'post' }
)
const saveAssignment = () => {
if (props.assignmentID == 'new') {
assignments.value.insert.submit(
{
...assignment,
},
{
onSuccess() {
show.value = false
showToast(
__('Success'),
__('Assignment created successfully'),
'check'
)
},
}
)
} else {
assignments.value.setValue.submit(
{
...assignment,
name: props.assignmentID,
},
{
onSuccess() {
show.value = false
showToast(
__('Success'),
__('Assignment updated successfully'),
'check'
)
},
}
)
}
}
const assignmentOptions = computed(() => {
return [
{ label: 'PDF', value: 'PDF' },
{ label: 'Image', value: 'Image' },
{ label: 'Document', value: 'Document' },
{ label: 'Text', value: 'Text' },
{ label: 'URL', value: 'URL' },
]
})
</script>

View File

@@ -32,7 +32,7 @@
</template> </template>
<script setup> <script setup>
import { Dialog, createResource } from 'frappe-ui' import { Dialog, createResource } from 'frappe-ui'
import { ref, inject } from 'vue' import { ref } from 'vue'
import Link from '@/components/Controls/Link.vue' import Link from '@/components/Controls/Link.vue'
import { showToast } from '@/utils' import { showToast } from '@/utils'
import { useOnboarding } from 'frappe-ui/frappe' import { useOnboarding } from 'frappe-ui/frappe'
@@ -41,7 +41,6 @@ import { useSettings } from '@/stores/settings'
const show = defineModel() const show = defineModel()
const course = ref(null) const course = ref(null)
const evaluator = ref(null) const evaluator = ref(null)
const user = inject('$user')
const courses = defineModel('courses') const courses = defineModel('courses')
const { updateOnboardingStep } = useOnboarding('learning') const { updateOnboardingStep } = useOnboarding('learning')
const settingsStore = useSettings() const settingsStore = useSettings()
@@ -74,11 +73,9 @@ const addCourse = (close) => {
{}, {},
{ {
onSuccess() { onSuccess() {
if (user.data?.is_system_manager)
updateOnboardingStep('add_batch_course')
close()
courses.value.reload() courses.value.reload()
updateOnboardingStep('add_batch_course')
close()
course.value = null course.value = null
evaluator.value = null evaluator.value = null
}, },

View File

@@ -38,7 +38,7 @@
<div class="mb-4"> <div class="mb-4">
<Button @click="openFileSelector" :loading="uploading"> <Button @click="openFileSelector" :loading="uploading">
{{ {{
uploading ? `Uploading ${progress}%` : 'Upload an ZIP file' uploading ? `Uploading ${progress}%` : 'Upload an zip file'
}} }}
</Button> </Button>
</div> </div>
@@ -77,7 +77,7 @@ import {
FormControl, FormControl,
Switch, Switch,
} from 'frappe-ui' } from 'frappe-ui'
import { reactive, watch, inject } from 'vue' import { reactive, watch } from 'vue'
import { showToast, getFileSize } from '@/utils/' import { showToast, getFileSize } from '@/utils/'
import { capture } from '@/telemetry' import { capture } from '@/telemetry'
import { FileText, X } from 'lucide-vue-next' import { FileText, X } from 'lucide-vue-next'
@@ -85,7 +85,6 @@ import { useOnboarding } from 'frappe-ui/frappe'
const show = defineModel() const show = defineModel()
const outline = defineModel('outline') const outline = defineModel('outline')
const user = inject('$user')
const { updateOnboardingStep } = useOnboarding('learning') const { updateOnboardingStep } = useOnboarding('learning')
const props = defineProps({ const props = defineProps({
@@ -140,10 +139,8 @@ const addChapter = async (close) => {
return validateChapter() return validateChapter()
}, },
onSuccess: (data) => { onSuccess: (data) => {
if (user.data?.is_system_manager)
updateOnboardingStep('create_first_chapter')
capture('chapter_created') capture('chapter_created')
updateOnboardingStep('create_first_chapter')
chapterReference.submit( chapterReference.submit(
{ name: data.name }, { name: data.name },
{ {

View File

@@ -1,27 +1,38 @@
<template> <template>
<Dialog <Dialog v-model="show" :options="dialogOptions">
v-model="show" <template #body-content>
:options="{ <div class="space-y-4">
size: '3xl',
}"
>
<template #body>
<div class="p-5 space-y-5">
<div class="text-lg font-semibold text-ink-gray-9 mb-5">
{{ __(props.title) }}
</div>
<div <div
v-if="!editMode" v-if="!editMode"
class="flex items-center text-xs text-ink-gray-7 space-x-5" class="flex items-center text-xs text-ink-gray-7 space-x-5"
> >
<Switch <div class="flex items-center space-x-2">
size="sm" <input
:label="__('Choose an existing question')" type="radio"
v-model="chooseFromExisting" id="existing"
class="!p-0" value="existing"
v-model="questionType"
class="w-3 h-3 cursor-pointer"
/> />
<label for="existing" class="cursor-pointer">
{{ __('Add an existing question') }}
</label>
</div> </div>
<div v-if="!chooseFromExisting || editMode" class="space-y-2">
<div class="flex items-center space-x-2">
<input
type="radio"
id="new"
value="new"
v-model="questionType"
class="w-3 h-3 cursor-pointer"
/>
<label for="new" class="cursor-pointer">
{{ __('Create a new question') }}
</label>
</div>
</div>
<div v-if="questionType == 'new' || editMode" class="space-y-2">
<div> <div>
<label class="block text-xs text-ink-gray-5 mb-1"> <label class="block text-xs text-ink-gray-5 mb-1">
{{ __('Question') }} {{ __('Question') }}
@@ -34,7 +45,6 @@
editorClass="prose-sm max-w-none border-b border-x bg-surface-gray-2 rounded-b-md py-1 px-2 min-h-[7rem]" editorClass="prose-sm max-w-none border-b border-x bg-surface-gray-2 rounded-b-md py-1 px-2 min-h-[7rem]"
/> />
</div> </div>
<div class="grid grid-cols-2 gap-4">
<FormControl <FormControl
v-model="question.marks" v-model="question.marks"
:label="__('Marks')" :label="__('Marks')"
@@ -48,20 +58,7 @@
class="pb-2" class="pb-2"
:required="true" :required="true"
/> />
</div> <div v-if="question.type == 'Choices'" class="divide-y border-t">
<div
v-if="question.type == 'Choices'"
class="text-base font-semibold text-ink-gray-9 mb-5 mt-5"
>
{{ __('Options') }}
</div>
<div
v-else-if="question.type == 'User Input'"
class="text-base font-semibold text-ink-gray-9 mb-5 mt-5"
>
{{ __('Possibilities') }}
</div>
<div v-if="question.type == 'Choices'" class="grid grid-cols-2 gap-4">
<div v-for="n in 4" class="space-y-4 py-2"> <div v-for="n in 4" class="space-y-4 py-2">
<FormControl <FormControl
:label="__('Option') + ' ' + n" :label="__('Option') + ' ' + n"
@@ -81,9 +78,9 @@
</div> </div>
<div <div
v-else-if="question.type == 'User Input'" v-else-if="question.type == 'User Input'"
class="grid grid-cols-2 gap-4 py-2" v-for="n in 4"
class="space-y-2"
> >
<div v-for="n in 4">
<FormControl <FormControl
:label="__('Possibility') + ' ' + n" :label="__('Possibility') + ' ' + n"
v-model="question[`possibility_${n}`]" v-model="question[`possibility_${n}`]"
@@ -91,8 +88,7 @@
/> />
</div> </div>
</div> </div>
</div> <div v-else-if="questionType == 'existing'" class="space-y-2">
<div v-else-if="chooseFromExisting" class="space-y-2">
<Link <Link
v-model="existingQuestion.question" v-model="existingQuestion.question"
:label="__('Select a question')" :label="__('Select a question')"
@@ -104,39 +100,26 @@
type="number" type="number"
/> />
</div> </div>
<div class="flex items-center justify-end space-x-2 mt-5">
<Button variant="solid" @click="submitQuestion()">
{{ __('Submit') }}
</Button>
</div>
</div> </div>
</template> </template>
</Dialog> </Dialog>
</template> </template>
<script setup> <script setup>
import { import { Dialog, FormControl, TextEditor, createResource } from 'frappe-ui'
Dialog, import { computed, watch, reactive, ref } from 'vue'
FormControl,
TextEditor,
createResource,
Switch,
Button,
} from 'frappe-ui'
import { computed, watch, reactive, ref, inject } from 'vue'
import Link from '@/components/Controls/Link.vue' import Link from '@/components/Controls/Link.vue'
import { showToast } from '@/utils' import { showToast } from '@/utils'
import { useOnboarding } from 'frappe-ui/frappe' import { useOnboarding } from 'frappe-ui/frappe'
const show = defineModel() const show = defineModel()
const quiz = defineModel('quiz') const quiz = defineModel('quiz')
const chooseFromExisting = ref(false) const questionType = ref(null)
const editMode = ref(false) const editMode = ref(false)
const user = inject('$user')
const { updateOnboardingStep } = useOnboarding('learning') const { updateOnboardingStep } = useOnboarding('learning')
const existingQuestion = reactive({ const existingQuestion = reactive({
question: '', question: '',
marks: 1, marks: 0,
}) })
const question = reactive({ const question = reactive({
question: '', question: '',
@@ -198,12 +181,11 @@ watch(show, () => {
editMode.value = false editMode.value = false
if (props.questionDetail.question) questionData.fetch() if (props.questionDetail.question) questionData.fetch()
else { else {
question.question = '' ;(question.question = ''), (question.marks = 0)
question.marks = 1
question.type = 'Choices' question.type = 'Choices'
existingQuestion.question = '' existingQuestion.question = ''
existingQuestion.marks = 1 existingQuestion.marks = 0
chooseFromExisting.value = false questionType.value = null
populateFields() populateFields()
} }
@@ -238,26 +220,32 @@ const questionCreation = createResource({
}, },
}) })
const submitQuestion = () => { const submitQuestion = (close) => {
if (props.questionDetail?.question) updateQuestion() if (props.questionDetail?.question) updateQuestion(close)
else addQuestion() else addQuestion(close)
} }
const addQuestion = () => { const addQuestion = (close) => {
if (chooseFromExisting.value) { if (questionType.value == 'existing') {
addQuestionRow({ addQuestionRow(
{
question: existingQuestion.question, question: existingQuestion.question,
marks: existingQuestion.marks, marks: existingQuestion.marks,
}) },
close
)
} else { } else {
questionCreation.submit( questionCreation.submit(
{}, {},
{ {
onSuccess(data) { onSuccess(data) {
addQuestionRow({ addQuestionRow(
{
question: data.name, question: data.name,
marks: question.marks, marks: question.marks,
}) },
close
)
}, },
onError(err) { onError(err) {
showToast(__('Error'), __(err.messages?.[0] || err), 'x') showToast(__('Error'), __(err.messages?.[0] || err), 'x')
@@ -267,24 +255,22 @@ const addQuestion = () => {
} }
} }
const addQuestionRow = (question) => { const addQuestionRow = (question, close) => {
questionRow.submit( questionRow.submit(
{ {
...question, ...question,
}, },
{ {
onSuccess() { onSuccess() {
if (user.data?.is_system_manager)
updateOnboardingStep('create_first_quiz')
show.value = false show.value = false
updateOnboardingStep('create_first_quiz')
showToast(__('Success'), __('Question added successfully'), 'check') showToast(__('Success'), __('Question added successfully'), 'check')
quiz.value.reload() quiz.value.reload()
show.value = false close()
}, },
onError(err) { onError(err) {
showToast(__('Error'), __(err.messages?.[0] || err), 'x') showToast(__('Error'), __(err.messages?.[0] || err), 'x')
show.value = false close()
}, },
} }
) )
@@ -318,7 +304,7 @@ const marksUpdate = createResource({
}, },
}) })
const updateQuestion = () => { const updateQuestion = (close) => {
questionUpdate.submit( questionUpdate.submit(
{}, {},
{ {
@@ -334,6 +320,7 @@ const updateQuestion = () => {
'check' 'check'
) )
quiz.value.reload() quiz.value.reload()
close()
}, },
} }
) )
@@ -344,6 +331,22 @@ const updateQuestion = () => {
} }
) )
} }
const dialogOptions = computed(() => {
return {
title: __(props.title),
size: 'xl',
actions: [
{
label: __('Submit'),
variant: 'solid',
onClick: (close) => {
submitQuestion(close)
},
},
],
}
})
</script> </script>
<style> <style>
input[type='radio']:checked { input[type='radio']:checked {

View File

@@ -315,6 +315,12 @@ const tabsStructure = computed(() => {
doctype: 'Email Template', doctype: 'Email Template',
type: 'Link', type: 'Link',
}, },
{
label: 'Assignment Submission Template',
name: 'assignment_submission_template',
doctype: 'Email Template',
type: 'Link',
},
], ],
}, },
{ {
@@ -322,52 +328,18 @@ const tabsStructure = computed(() => {
icon: 'LogIn', icon: 'LogIn',
fields: [ fields: [
{ {
label: 'Identify User Category', label: 'Custom Content',
name: 'user_category',
type: 'checkbox',
description:
'Enable this option to identify the user category during signup.',
},
{
label: 'Disable signup',
name: 'disable_signup',
type: 'checkbox',
description:
'New users will have to be manually registered by Admins.',
},
{
label: 'Signup Consent HTML',
name: 'custom_signup_content', name: 'custom_signup_content',
type: 'Code', type: 'Code',
mode: 'htmlmixed', mode: 'htmlmixed',
rows: 10, rows: 10,
}, },
],
},
{ {
label: 'SEO', label: 'Ask for Occupation',
icon: 'Search', name: 'user_category',
fields: [ type: 'checkbox',
{
label: 'Meta Description',
name: 'meta_description',
type: 'textarea',
rows: 4,
description: description:
"This description will be shown on lists and pages that don't have meta description", 'Enable this option to ask users to select their occupation during the signup process.',
},
{
label: 'Meta Keywords',
name: 'meta_keywords',
type: 'textarea',
rows: 4,
description:
'Keywords for search engines to find your website. Separated by commas.',
},
{
label: 'Meta Image',
name: 'meta_image',
type: 'Upload',
}, },
], ],
}, },

View File

@@ -26,14 +26,13 @@
</template> </template>
<script setup> <script setup>
import { Dialog, createResource } from 'frappe-ui' import { Dialog, createResource } from 'frappe-ui'
import { ref, inject } from 'vue' import { ref } from 'vue'
import Link from '@/components/Controls/Link.vue' import Link from '@/components/Controls/Link.vue'
import { showToast } from '@/utils' import { showToast } from '@/utils'
import { useOnboarding } from 'frappe-ui/frappe' import { useOnboarding } from 'frappe-ui/frappe'
const students = defineModel('reloadStudents') const students = defineModel('reloadStudents')
const student = ref() const student = ref()
const user = inject('$user')
const { updateOnboardingStep } = useOnboarding('learning') const { updateOnboardingStep } = useOnboarding('learning')
const show = defineModel() const show = defineModel()
@@ -62,11 +61,9 @@ const addStudent = (close) => {
{}, {},
{ {
onSuccess() { onSuccess() {
if (user.data?.is_system_manager)
updateOnboardingStep('add_batch_student')
students.value.reload() students.value.reload()
student.value = null student.value = null
updateOnboardingStep('add_batch_student')
close() close()
}, },
onError(err) { onError(err) {

View File

@@ -0,0 +1,159 @@
<template>
<div v-if="showOnboardingBanner && onboardingDetails.data">
<Tooltip :text="__('Skip Onboarding')" placement="left">
<X
class="w-4 h-4 stroke-1 absolute top-2 right-2 cursor-pointer mr-1"
@click="skipOnboarding.reload()"
/>
</Tooltip>
<div class="flex items-center justify-evenly bg-surface-gray-2 p-10">
<div
@click="redirectToCourseForm()"
class="flex items-center space-x-2"
:class="{
'cursor-pointer': !onboardingDetails.data.course_created?.length,
}"
>
<span
v-if="onboardingDetails.data.course_created?.length"
class="py-1 px-1 bg-surface-white rounded-full"
>
<Check class="h-4 w-4 stroke-2 text-ink-green-3" />
</span>
<span
v-else
class="font-semibold bg-surface-white px-2 py-1 rounded-full"
>
1
</span>
<span class="text-lg font-semibold">
{{ __('Create a course') }}
</span>
</div>
<div
@click="redirectToChapterForm()"
class="flex items-center space-x-2"
:class="{
'cursor-pointer':
onboardingDetails.data.course_created?.length &&
!onboardingDetails.data.chapter_created?.length,
'text-ink-gray-3': !onboardingDetails.data.course_created?.length,
}"
>
<span
v-if="onboardingDetails.data.chapter_created?.length"
class="py-1 px-1 bg-surface-white rounded-full"
>
<Check class="h-4 w-4 stroke-2 text-ink-green-3" />
</span>
<span
v-else
class="font-semibold bg-surface-white px-2 py-1 rounded-full"
>
2
</span>
<span class="text-lg font-semibold">
{{ __('Add a chapter') }}
</span>
</div>
<div
@click="redirectToLessonForm()"
class="flex items-center space-x-2"
:class="{
'cursor-pointer':
onboardingDetails.data.course_created?.length &&
onboardingDetails.data.chapter_created?.length,
'text-ink-gray-3':
!onboardingDetails.data.course_created?.length ||
!onboardingDetails.data.chapter_created?.length,
}"
>
<span
v-if="onboardingDetails.data.lesson_created?.length"
class="py-1 px-1 bg-surface-white rounded-full"
>
<Check class="h-4 w-4 stroke-2 text-ink-green-3" />
</span>
<span class="font-semibold bg-surface-white px-2 py-1 rounded-full">
3
</span>
<span class="text-lg font-semibold">
{{ __('Add a lesson') }}
</span>
</div>
</div>
</div>
</template>
<script setup lang="ts">
import { ref, watch } from 'vue'
import { Check, X } from 'lucide-vue-next'
import { useRouter } from 'vue-router'
import { useSettings } from '@/stores/settings'
import { createResource, Tooltip } from 'frappe-ui'
const showOnboardingBanner = ref(false)
const settings = useSettings()
const onboardingDetails = settings.onboardingDetails
const router = useRouter()
watch(onboardingDetails, () => {
if (!onboardingDetails.data?.is_onboarded) {
showOnboardingBanner.value = true
} else {
showOnboardingBanner.value = false
}
})
const redirectToCourseForm = () => {
if (onboardingDetails.data?.course_created.length) {
return
} else {
router.push({ name: 'CourseForm', params: { courseName: 'new' } })
}
}
const redirectToChapterForm = () => {
if (!onboardingDetails.data?.course_created.length) {
return
} else {
router.push({
name: 'CourseForm',
params: {
courseName: onboardingDetails.data?.first_course,
},
})
}
}
const redirectToLessonForm = () => {
if (!onboardingDetails.data?.course_created.length) {
return
} else if (!onboardingDetails.data?.chapter_created.length) {
return
} else {
router.push({
name: 'LessonForm',
params: {
courseName: onboardingDetails.data?.first_course,
chapterNumber: 1,
lessonNumber: 1,
},
})
}
}
const skipOnboarding = createResource({
url: 'frappe.client.set_value',
makeParams() {
return {
doctype: 'LMS Settings',
name: 'LMS Settings',
fieldname: 'is_onboarding_complete',
value: 1,
}
},
onSuccess(data) {
onboardingDetails.reload()
},
})
</script>

View File

@@ -1,7 +1,7 @@
<template> <template>
<div v-if="quiz.data"> <div v-if="quiz.data">
<div <div
class="bg-surface-blue-2 space-y-1 py-2 px-2 mb-4 rounded-md text-sm text-ink-blue-3" class="bg-surface-blue-2 space-y-1 py-2 px-2 mb-4 rounded-md text-sm text-ink-blue-2"
> >
<div class="leading-5"> <div class="leading-5">
{{ {{
@@ -653,8 +653,3 @@ const getSubmissionColumns = () => {
] ]
} }
</script> </script>
<style>
p {
line-height: 1.5rem;
}
</style>

View File

@@ -51,9 +51,7 @@ const props = defineProps({
const update = () => { const update = () => {
props.fields.forEach((f) => { props.fields.forEach((f) => {
if (f.type == 'Upload') { if (f.type != 'Column Break') {
props.data.doc[f.name] = f.value ? f.value.file_url : null
} else if (f.type != 'Column Break') {
props.data.doc[f.name] = f.value props.data.doc[f.name] = f.value
} }
}) })

View File

@@ -54,30 +54,21 @@
<div v-else> <div v-else>
<div class="flex items-center text-sm space-x-2"> <div class="flex items-center text-sm space-x-2">
<div <div
class="flex items-center justify-center rounded border border-outline-gray-modals bg-white w-[10rem] py-2" class="flex items-center justify-center rounded border border-outline-gray-modals w-[10rem] py-5"
> >
<img <img :src="data[field.name]?.file_url" class="h-6 rounded" />
:src="data[field.name]?.file_url || data[field.name]"
class="w-[80%] rounded"
/>
</div> </div>
<div class="flex flex-col flex-wrap"> <div class="flex flex-col flex-wrap">
<span class="break-all text-ink-gray-9"> <span class="break-all text-ink-gray-9">
{{ {{ data[field.name]?.file_name }}
data[field.name]?.file_name ||
data[field.name].split('/').pop()
}}
</span> </span>
<span <span class="text-sm text-ink-gray-5 mt-1">
v-if="data[field.name]?.file_size"
class="text-sm text-ink-gray-5 mt-1"
>
{{ getFileSize(data[field.name]?.file_size) }} {{ getFileSize(data[field.name]?.file_size) }}
</span> </span>
</div> </div>
<X <X
@click="data[field.name] = null" @click="data[field.name] = null"
class="border text-ink-gray-7 border-outline-gray-3 rounded-md cursor-pointer stroke-1.5 w-5 h-5 p-1 ml-4" class="bg-surface-gray-5 rounded-md cursor-pointer stroke-1.5 w-5 h-5 p-1 ml-4"
/> />
</div> </div>
</div> </div>

View File

@@ -36,7 +36,7 @@
<span v-else> Learning </span> <span v-else> Learning </span>
</div> </div>
<div <div
v-if="userResource.data" v-if="userResource"
class="mt-1 text-sm text-ink-gray-7 leading-none" class="mt-1 text-sm text-ink-gray-7 leading-none"
> >
{{ convertToTitleCase(userResource.data?.full_name) }} {{ convertToTitleCase(userResource.data?.full_name) }}
@@ -194,6 +194,18 @@ const userDropdownOptions = computed(() => {
) )
}, },
}, },
],
},
{
group: '',
items: [
{
icon: Zap,
label: 'Powered by Learning',
onClick: () => {
window.open('https://frappe.io/learning', '_blank')
},
},
{ {
icon: LogOut, icon: LogOut,
label: 'Log out', label: 'Log out',

View File

@@ -1,53 +1,32 @@
<template> <template>
<div ref="videoContainer" class="video-block relative group"> <div ref="videoContainer" class="video-block group relative">
<video <video
@timeupdate="updateTime" @timeupdate="updateTime"
@ended="videoEnded" @ended="videoEnded"
@click="togglePlay" @click="togglePlay"
oncontextmenu="return false" oncontextmenu="return false"
class="rounded-md border border-gray-100 cursor-pointer" class="rounded-lg border border-gray-100 group cursor-pointer"
ref="videoRef" ref="videoRef"
> >
<source :src="fileURL" :type="type" /> <source :src="fileURL" :type="type" />
</video> </video>
<div <div
v-if="!playing" class="flex items-center space-x-2 bg-surface-gray-3 rounded-md p-0.5 absolute bottom-3 w-[98%] left-0 right-0 mx-auto invisible group-hover:visible"
class="absolute inset-0 flex items-center justify-center cursor-pointer"
@click="playVideo"
>
<div
class="rounded-full p-4 pl-4.5"
style="
background: radial-gradient(
circle,
rgba(0, 0, 0, 0.3) 0%,
rgba(0, 0, 0, 0.4) 50%
);
"
>
<Play />
</div>
</div>
<div
class="flex items-center space-x-2 py-2 px-1 text-ink-white bg-gradient-to-b from-transparent to-black/75 absolute bottom-0 left-0 right-0 mx-auto rounded-md"
:class="{
'invisible group-hover:visible': playing,
}"
> >
<Button variant="ghost"> <Button variant="ghost">
<template #icon> <template #icon>
<Play <Play
v-if="!playing" v-if="!playing"
@click="playVideo" @click="playVideo"
class="size-4 text-ink-gray-9" class="w-4 h-4 text-ink-gray-9"
/> />
<Pause v-else @click="pauseVideo" class="size-5 text-ink-white" /> <Pause v-else @click="pauseVideo" class="w-4 h-4 text-ink-gray-9" />
</template> </template>
</Button> </Button>
<Button variant="ghost" @click="toggleMute"> <Button variant="ghost" @click="toggleMute">
<template #icon> <template #icon>
<Volume2 v-if="!muted" class="size-5 text-ink-white" /> <Volume2 v-if="!muted" class="w-4 h-4 text-ink-gray-9" />
<VolumeX v-else class="size-5 text-ink-white" /> <VolumeX v-else class="w-4 h-4 text-ink-gray-9" />
</template> </template>
</Button> </Button>
<input <input
@@ -59,12 +38,12 @@
@input="changeCurrentTime" @input="changeCurrentTime"
class="duration-slider w-full h-1" class="duration-slider w-full h-1"
/> />
<span class="text-sm font-semibold"> <span class="text-xs font-medium">
{{ formatTime(currentTime) }} / {{ formatTime(duration) }} {{ formatTime(currentTime) }} / {{ formatTime(duration) }}
</span> </span>
<Button variant="ghost" @click="toggleFullscreen"> <Button variant="ghost" @click="toggleFullscreen">
<template #icon> <template #icon>
<Maximize class="size-5 text-ink-white" /> <Maximize class="w-4 h-4 text-ink-gray-9" />
</template> </template>
</Button> </Button>
</div> </div>
@@ -72,9 +51,8 @@
</template> </template>
<script setup> <script setup>
import { ref, onMounted, computed } from 'vue' import { ref, onMounted, computed } from 'vue'
import { Pause, Maximize, Volume2, VolumeX } from 'lucide-vue-next' import { Play, Pause, Maximize, Volume2, VolumeX } from 'lucide-vue-next'
import { Button } from 'frappe-ui' import { Button } from 'frappe-ui'
import Play from '@/components/Icons/Play.vue'
const videoRef = ref(null) const videoRef = ref(null)
const videoContainer = ref(null) const videoContainer = ref(null)
@@ -169,6 +147,7 @@ const toggleFullscreen = () => {
<style scoped> <style scoped>
.video-block { .video-block {
width: 100%; width: 100%;
max-width: 900px;
margin: 0 auto; margin: 0 auto;
} }
@@ -186,16 +165,15 @@ iframe {
flex: 1; flex: 1;
-webkit-appearance: none; -webkit-appearance: none;
appearance: none; appearance: none;
border-radius: 10px; background-color: theme('colors.gray.400');
background-color: theme('colors.gray.100');
cursor: pointer; cursor: pointer;
} }
.duration-slider::-webkit-slider-thumb { .duration-slider::-webkit-slider-thumb {
width: 2px; height: 10px;
border-radius: 50%; width: 10px;
-webkit-appearance: none; -webkit-appearance: none;
background-color: theme('colors.gray.500'); background-color: theme('colors.gray.900');
} }
@media screen and (-webkit-min-device-pixel-ratio: 0) { @media screen and (-webkit-min-device-pixel-ratio: 0) {
@@ -208,7 +186,7 @@ iframe {
input[type='range']::-webkit-slider-thumb { input[type='range']::-webkit-slider-thumb {
-webkit-appearance: none; -webkit-appearance: none;
cursor: pointer; cursor: pointer;
box-shadow: -500px 0 0 500px theme('colors.gray.600'); box-shadow: -500px 0 0 500px theme('colors.gray.900');
} }
} }
</style> </style>

View File

@@ -26,6 +26,5 @@ app.mount('#app')
const { userResource, allUsers } = usersStore() const { userResource, allUsers } = usersStore()
app.provide('$user', userResource) app.provide('$user', userResource)
app.provide('$allUsers', allUsers) app.provide('$allUsers', allUsers)
app.config.globalProperties.$user = userResource app.config.globalProperties.$user = userResource
app.config.globalProperties.$dialog = createDialog app.config.globalProperties.$dialog = createDialog

View File

@@ -0,0 +1,191 @@
<template>
<header
class="sticky top-0 z-10 flex items-center justify-between border-b bg-surface-white px-3 py-2.5 sm:px-5"
>
<Breadcrumbs :items="breadcrumbs" />
<div class="space-x-2">
<router-link
v-if="assignment.doc?.name"
:to="{
name: 'AssignmentSubmissionList',
query: {
assignmentID: assignment.doc.name,
},
}"
>
<Button>
{{ __('Submission List') }}
</Button>
</router-link>
<Button variant="solid" @click="saveAssignment()">
{{ __('Save') }}
</Button>
</div>
</header>
<div class="w-3/4 mx-auto py-5">
<div class="font-semibold mb-4">
{{ __('Details') }}
</div>
<div class="grid grid-cols-2 gap-5 mt-4 mb-8">
<FormControl
v-model="model.title"
:label="__('Title')"
:required="true"
/>
<FormControl
v-model="model.type"
type="select"
:options="assignmentOptions"
:label="__('Type')"
:required="true"
/>
</div>
<div>
<div class="text-xs text-ink-gray-5 mb-2">
{{ __('Question') }}
<span class="text-ink-red-3">*</span>
</div>
<TextEditor
:content="model.question"
@change="(val) => (model.question = 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]"
/>
</div>
</div>
</template>
<script setup>
import {
Breadcrumbs,
Button,
createDocumentResource,
createResource,
FormControl,
TextEditor,
} from 'frappe-ui'
import {
computed,
inject,
onMounted,
onBeforeUnmount,
reactive,
watch,
} from 'vue'
import { showToast } from '@/utils'
import { useRouter } from 'vue-router'
const user = inject('$user')
const router = useRouter()
const props = defineProps({
assignmentID: {
type: String,
required: true,
},
})
const model = reactive({
title: '',
type: 'PDF',
question: '',
})
onMounted(() => {
if (
props.assignmentID == 'new' &&
!user.data?.is_moderator &&
!user.data?.is_instructor
) {
router.push({ name: 'Courses' })
}
if (props.assignmentID !== 'new') {
assignment.reload()
}
window.addEventListener('keydown', keyboardShortcut)
})
const keyboardShortcut = (e) => {
if (e.key === 's' && (e.ctrlKey || e.metaKey)) {
saveAssignment()
e.preventDefault()
}
}
onBeforeUnmount(() => {
window.removeEventListener('keydown', keyboardShortcut)
})
const assignment = createDocumentResource({
doctype: 'LMS Assignment',
name: props.assignmentID,
auto: false,
})
const newAssignment = createResource({
url: 'frappe.client.insert',
makeParams(values) {
return {
doc: {
doctype: 'LMS Assignment',
...values,
},
}
},
onSuccess(data) {
router.push({ name: 'AssignmentForm', params: { assignmentID: data.name } })
},
onError(err) {
showToast(__('Error'), __(err.messages?.[0] || err), 'x')
},
})
const saveAssignment = () => {
if (props.assignmentID == 'new') {
newAssignment.submit({
...model,
})
} else {
assignment.setValue.submit(
{
...model,
},
{
onSuccess(data) {
showToast(__('Success'), __('Assignment saved successfully'), 'check')
assignment.reload()
},
onError(err) {
showToast(__('Error'), __(err.messages?.[0] || err), 'x')
},
}
)
}
}
watch(assignment, () => {
Object.keys(assignment.doc).forEach((key) => {
model[key] = assignment.doc[key]
})
})
const breadcrumbs = computed(() => [
{
label: __('Assignments'),
route: { name: 'Assignments' },
},
{
label: assignment.doc ? assignment.doc.title : __('New Assignment'),
},
])
const assignmentOptions = computed(() => {
return [
{ label: 'PDF', value: 'PDF' },
{ label: 'Image', value: 'Image' },
{ label: 'Document', value: 'Document' },
{ label: 'Text', value: 'Text' },
{ label: 'URL', value: 'URL' },
]
})
</script>

View File

@@ -14,14 +14,12 @@
</div> </div>
</template> </template>
<script setup> <script setup>
import { Breadcrumbs, createResource, usePageMeta } from 'frappe-ui' import { Breadcrumbs, createResource } from 'frappe-ui'
import { computed, inject, onMounted, ref } from 'vue' import { computed, inject, onMounted, onBeforeUnmount, ref } from 'vue'
import { sessionStore } from '../stores/session'
import Assignment from '@/components/Assignment.vue' import Assignment from '@/components/Assignment.vue'
const user = inject('$user') const user = inject('$user')
const fromLesson = ref(false) const fromLesson = ref(false)
const { brand } = sessionStore()
const props = defineProps({ const props = defineProps({
assignmentID: { assignmentID: {
@@ -74,11 +72,4 @@ const breadcrumbs = computed(() => {
] ]
return crumbs return crumbs
}) })
usePageMeta(() => {
return {
title: title.data?.title,
icon: brand.favicon,
}
})
</script> </script>

View File

@@ -84,17 +84,14 @@ import {
ListRows, ListRows,
ListRow, ListRow,
ListRowItem, ListRowItem,
usePageMeta,
} from 'frappe-ui' } from 'frappe-ui'
import { computed, inject, onMounted, ref, watch } from 'vue' import { computed, inject, onMounted, ref, watch } from 'vue'
import { useRouter } from 'vue-router' import { useRouter } from 'vue-router'
import { Pencil } from 'lucide-vue-next' import { Pencil } from 'lucide-vue-next'
import { sessionStore } from '../stores/session'
import Link from '@/components/Controls/Link.vue' import Link from '@/components/Controls/Link.vue'
const user = inject('$user') const user = inject('$user')
const dayjs = inject('$dayjs') const dayjs = inject('$dayjs')
const { brand } = sessionStore()
const router = useRouter() const router = useRouter()
const assignmentID = ref('') const assignmentID = ref('')
const member = ref('') const member = ref('')
@@ -217,11 +214,4 @@ const breadcrumbs = computed(() => {
}, },
] ]
}) })
usePageMeta(() => {
return {
title: __('Assignment Submissions'),
icon: brand.favicon,
}
})
</script> </script>

View File

@@ -3,21 +3,21 @@
class="sticky top-0 z-10 flex items-center justify-between border-b bg-surface-white px-3 py-2.5 sm:px-5" class="sticky top-0 z-10 flex items-center justify-between border-b bg-surface-white px-3 py-2.5 sm:px-5"
> >
<Breadcrumbs :items="breadcrumbs" /> <Breadcrumbs :items="breadcrumbs" />
<Button <router-link
v-if="!readOnlyMode" :to="{
variant="solid" name: 'AssignmentForm',
@click=" params: {
() => { assignmentID: 'new',
assignmentID = 'new' },
showAssignmentForm = true }"
}
"
> >
<Button variant="solid">
<template #prefix> <template #prefix>
<Plus class="w-4 h-4" /> <Plus class="w-4 h-4" />
</template> </template>
{{ __('New') }} {{ __('New') }}
</Button> </Button>
</router-link>
</header> </header>
<div class="md:w-3/4 md:mx-auto py-5 mx-5"> <div class="md:w-3/4 md:mx-auto py-5 mx-5">
@@ -38,11 +38,12 @@
:options="{ :options="{
showTooltip: false, showTooltip: false,
selectable: false, selectable: false,
onRowClick: (row) => { getRowRoute: (row) => ({
if (readOnlyMode) return name: 'AssignmentForm',
assignmentID = row.name params: {
showAssignmentForm = true assignmentID: row.name,
}, },
}),
}" }"
> >
</ListView> </ListView>
@@ -71,11 +72,6 @@
</Button> </Button>
</div> </div>
</div> </div>
<AssignmentForm
v-model="showAssignmentForm"
v-model:assignments="assignments"
:assignmentID="assignmentID"
/>
</template> </template>
<script setup> <script setup>
import { import {
@@ -84,23 +80,16 @@ import {
createListResource, createListResource,
FormControl, FormControl,
ListView, ListView,
usePageMeta,
} from 'frappe-ui' } from 'frappe-ui'
import { computed, inject, onMounted, ref, watch } from 'vue' import { computed, inject, onMounted, ref, watch } from 'vue'
import { Plus, Pencil } from 'lucide-vue-next' import { Plus, Pencil } from 'lucide-vue-next'
import { useRouter } from 'vue-router' import { useRouter } from 'vue-router'
import { sessionStore } from '../stores/session'
import AssignmentForm from '@/components/Modals/AssignmentForm.vue'
const user = inject('$user') const user = inject('$user')
const dayjs = inject('$dayjs') const dayjs = inject('$dayjs')
const titleFilter = ref('') const titleFilter = ref('')
const typeFilter = ref('') const typeFilter = ref('')
const showAssignmentForm = ref(false)
const assignmentID = ref('new')
const { brand } = sessionStore()
const router = useRouter() const router = useRouter()
const readOnlyMode = window.read_only_mode
onMounted(() => { onMounted(() => {
if (!user.data?.is_moderator && !user.data?.is_instructor) { if (!user.data?.is_moderator && !user.data?.is_instructor) {
@@ -144,7 +133,7 @@ const assignmentFilter = computed(() => {
const assignments = createListResource({ const assignments = createListResource({
doctype: 'LMS Assignment', doctype: 'LMS Assignment',
fields: ['name', 'title', 'type', 'creation', 'question'], fields: ['name', 'title', 'type', 'creation'],
orderBy: 'modified desc', orderBy: 'modified desc',
cache: ['assignments'], cache: ['assignments'],
transform(data) { transform(data) {
@@ -174,7 +163,7 @@ const assignmentColumns = computed(() => {
label: __('Created'), label: __('Created'),
key: 'creation', key: 'creation',
width: 1, width: 1,
align: 'right', align: 'center',
}, },
] ]
}) })
@@ -195,11 +184,4 @@ const breadcrumbs = computed(() => [
route: { name: 'Assignments' }, route: { name: 'Assignments' },
}, },
]) ])
usePageMeta(() => {
return {
title: __('Assignments'),
icon: brand.favicon,
}
})
</script> </script>

View File

@@ -24,12 +24,10 @@
</div> </div>
</template> </template>
<script setup> <script setup>
import { createResource, usePageMeta } from 'frappe-ui' import { createDocumentResource, createResource } from 'frappe-ui'
import { computed, inject } from 'vue' import { computed, inject } from 'vue'
import { sessionStore } from '../stores/session'
const dayjs = inject('$dayjs') const dayjs = inject('$dayjs')
const { brand } = sessionStore()
const props = defineProps({ const props = defineProps({
badgeName: { badgeName: {
@@ -72,11 +70,4 @@ const breadcrumbs = computed(() => {
}, },
] ]
}) })
usePageMeta(() => {
return {
title: badge.data.badge,
icon: brand.favicon,
}
})
</script> </script>

View File

@@ -11,7 +11,7 @@
> >
{{ __('Generate Certificates') }} {{ __('Generate Certificates') }}
</Button> </Button>
<Button v-if="canMakeAnnouncement()" @click="openAnnouncementModal()"> <Button v-if="user.data?.is_moderator" @click="openAnnouncementModal()">
<span> <span>
{{ __('Make an Announcement') }} {{ __('Make an Announcement') }}
</span> </span>
@@ -199,14 +199,9 @@
<script setup> <script setup>
import { computed, inject, ref, onMounted, watch } from 'vue' import { computed, inject, ref, onMounted, watch } from 'vue'
import { useRoute, useRouter } from 'vue-router' import { useRoute, useRouter } from 'vue-router'
import { import { Breadcrumbs, Button, createResource, Tabs, Badge } from 'frappe-ui'
Breadcrumbs, import CourseInstructors from '@/components/CourseInstructors.vue'
Button, import UserAvatar from '@/components/UserAvatar.vue'
createResource,
Tabs,
Badge,
usePageMeta,
} from 'frappe-ui'
import { import {
Clock, Clock,
LayoutDashboard, LayoutDashboard,
@@ -219,10 +214,7 @@ import {
Globe, Globe,
ClipboardPen, ClipboardPen,
} from 'lucide-vue-next' } from 'lucide-vue-next'
import { formatTime } from '@/utils' import { formatTime, updateDocumentTitle } from '@/utils'
import { sessionStore } from '@/stores/session'
import CourseInstructors from '@/components/CourseInstructors.vue'
import UserAvatar from '@/components/UserAvatar.vue'
import BatchDashboard from '@/components/BatchDashboard.vue' import BatchDashboard from '@/components/BatchDashboard.vue'
import BatchCourses from '@/components/BatchCourses.vue' import BatchCourses from '@/components/BatchCourses.vue'
import LiveClass from '@/components/LiveClass.vue' import LiveClass from '@/components/LiveClass.vue'
@@ -240,9 +232,7 @@ const showAnnouncementModal = ref(false)
const openCertificateDialog = ref(false) const openCertificateDialog = ref(false)
const route = useRoute() const route = useRoute()
const router = useRouter() const router = useRouter()
const { brand } = sessionStore()
const tabIndex = ref(0) const tabIndex = ref(0)
const readOnlyMode = window.read_only_mode
const tabs = computed(() => { const tabs = computed(() => {
let batchTabs = [] let batchTabs = []
@@ -355,15 +345,12 @@ watch(tabIndex, () => {
} }
}) })
const canMakeAnnouncement = () => { const pageMeta = computed(() => {
if (readOnlyMode) return false
return user.data?.is_moderator || user.data?.is_evaluator
}
usePageMeta(() => {
return { return {
title: batch?.data?.title, title: batch.data?.title,
icon: brand.favicon, description: batch.data?.description,
} }
}) })
updateDocumentTitle(pageMeta)
</script> </script>

View File

@@ -14,16 +14,13 @@
{{ batch.data.description }} {{ batch.data.description }}
</div> </div>
<div <div
class="flex flex-col gap-2 lg:gap-0 lg:flex-row lg:items-center space-x-0 md:space-x-5 lg:w-1/2" class="flex flex-col gap-2 lg:gap-0 lg:flex-row lg:items-center justify-between lg:w-1/2"
> >
<div <div class="flex items-center text-ink-gray-7">
v-if="batch.data?.courses?.length" <BookOpen class="h-4 w-4 mr-2" />
class="flex items-center text-ink-gray-7"
>
<BookOpen class="h-4 w-4 mr-2 stroke-1.5" />
<span> {{ batch.data?.courses?.length }} {{ __('Courses') }} </span> <span> {{ batch.data?.courses?.length }} {{ __('Courses') }} </span>
</div> </div>
<span v-if="batch.data?.courses?.length" class="hidden lg:block" <span class="hidden lg:block" v-if="batch.data.courses"
>&middot;</span >&middot;</span
> >
<DateRange <DateRange
@@ -34,7 +31,7 @@
>&middot;</span >&middot;</span
> >
<div class="flex items-center text-ink-gray-7"> <div class="flex items-center text-ink-gray-7">
<Clock class="h-4 w-4 mr-2 stroke-1.5" /> <Clock class="h-4 w-4 mr-2" />
<span> <span>
{{ formatTime(batch.data.start_time) }} - {{ formatTime(batch.data.start_time) }} -
{{ formatTime(batch.data.end_time) }} {{ formatTime(batch.data.end_time) }}
@@ -105,9 +102,8 @@
import { computed, inject } from 'vue' import { computed, inject } from 'vue'
import { useRouter } from 'vue-router' import { useRouter } from 'vue-router'
import { BookOpen, Clock } from 'lucide-vue-next' import { BookOpen, Clock } from 'lucide-vue-next'
import { formatTime } from '@/utils' import { formatTime, updateDocumentTitle } from '@/utils'
import { Breadcrumbs, createResource, usePageMeta } from 'frappe-ui' import { Breadcrumbs, createResource } from 'frappe-ui'
import { sessionStore } from '@/stores/session'
import CourseCard from '@/components/CourseCard.vue' import CourseCard from '@/components/CourseCard.vue'
import BatchOverlay from '@/components/BatchOverlay.vue' import BatchOverlay from '@/components/BatchOverlay.vue'
import DateRange from '../components/Common/DateRange.vue' import DateRange from '../components/Common/DateRange.vue'
@@ -116,7 +112,6 @@ import UserAvatar from '@/components/UserAvatar.vue'
const user = inject('$user') const user = inject('$user')
const router = useRouter() const router = useRouter()
const { brand } = sessionStore()
const props = defineProps({ const props = defineProps({
batchName: { batchName: {
@@ -157,12 +152,14 @@ const breadcrumbs = computed(() => {
return items return items
}) })
usePageMeta(() => { const pageMeta = computed(() => {
return { return {
title: batch?.data?.title, title: batch.data?.title,
icon: brand.favicon, description: batch.data?.description,
} }
}) })
updateDocumentTitle(pageMeta)
</script> </script>
<style> <style>
.batch-description p { .batch-description p {

View File

@@ -8,30 +8,19 @@
{{ __('Save') }} {{ __('Save') }}
</Button> </Button>
</header> </header>
<div class="w-3/4 mx-auto py-5"> <div class="w-1/2 mx-auto py-5">
<div class=""> <div class="">
<div class="text-lg text-ink-gray-9 font-semibold mb-4"> <div class="text-lg font-semibold mb-4">
{{ __('Details') }} {{ __('Details') }}
</div> </div>
<div class="space-y-10 mb-4"> <div class="space-y-4 mb-4">
<div class="space-y-4">
<FormControl <FormControl
v-model="batch.title" v-model="batch.title"
:label="__('Title')" :label="__('Title')"
:required="true" :required="true"
class="w-full" class="w-full"
/> />
<MultiSelect <div class="flex items-center space-x-5">
v-model="instructors"
doctype="User"
:label="__('Instructors')"
:required="true"
:filters="{ ignore_user_type: 1 }"
/>
</div>
<div class="grid grid-cols-2 gap-10">
<div class="flex flex-col space-y-5">
<FormControl <FormControl
v-model="batch.published" v-model="batch.published"
type="checkbox" type="checkbox"
@@ -48,8 +37,9 @@
:label="__('Certification')" :label="__('Certification')"
/> />
</div> </div>
</div>
<div> </div>
<div class="mb-4">
<div class="text-xs text-ink-gray-5 mb-2"> <div class="text-xs text-ink-gray-5 mb-2">
{{ __('Meta Image') }} {{ __('Meta Image') }}
</div> </div>
@@ -59,9 +49,7 @@
:validateFile="validateFile" :validateFile="validateFile"
@success="(file) => saveImage(file)" @success="(file) => saveImage(file)"
> >
<template <template v-slot="{ file, progress, uploading, openFileSelector }">
v-slot="{ file, progress, uploading, openFileSelector }"
>
<div class="flex items-center"> <div class="flex items-center">
<div class="border rounded-md w-fit py-5 px-20"> <div class="border rounded-md w-fit py-5 px-20">
<Image class="size-5 stroke-1 text-ink-gray-7" /> <Image class="size-5 stroke-1 text-ink-gray-7" />
@@ -83,10 +71,7 @@
</FileUploader> </FileUploader>
<div v-else class="mb-4"> <div v-else class="mb-4">
<div class="flex items-center"> <div class="flex items-center">
<img <img :src="batch.image.file_url" class="border rounded-md w-40" />
:src="batch.image.file_url"
class="border rounded-md w-40"
/>
<div class="ml-4"> <div class="ml-4">
<Button @click="removeImage()"> <Button @click="removeImage()">
{{ __('Remove') }} {{ __('Remove') }}
@@ -102,15 +87,19 @@
</div> </div>
</div> </div>
</div> </div>
</div> <MultiSelect
</div> v-model="instructors"
</div> doctype="User"
:label="__('Instructors')"
:required="true"
:filters="{ ignore_user_type: 1 }"
/>
<div class="my-10"> <div class="my-10">
<div class="text-lg text-ink-gray-9 font-semibold mb-4"> <div class="text-lg font-semibold mb-4">
{{ __('Date and Time') }} {{ __('Date and Time') }}
</div> </div>
<div class="grid grid-cols-3 gap-10"> <div class="grid grid-cols-2 gap-10">
<div> <div>
<FormControl <FormControl
v-model="batch.start_date" v-model="batch.start_date"
@@ -126,6 +115,14 @@
class="mb-4" class="mb-4"
:required="true" :required="true"
/> />
<FormControl
v-model="batch.timezone"
:label="__('Timezone')"
type="text"
:placeholder="__('Example: IST (+5:30)')"
class="mb-4"
:required="true"
/>
</div> </div>
<div> <div>
<FormControl <FormControl
@@ -143,24 +140,14 @@
:required="true" :required="true"
/> />
</div> </div>
<div>
<FormControl
v-model="batch.timezone"
:label="__('Timezone')"
type="text"
:placeholder="__('Example: IST (+5:30)')"
class="mb-4"
:required="true"
/>
</div>
</div> </div>
</div> </div>
<div class="mb-10"> <div class="mb-10">
<div class="text-lg text-ink-gray-9 font-semibold mb-4"> <div class="text-lg font-semibold mb-4">
{{ __('Settings') }} {{ __('Settings') }}
</div> </div>
<div class="grid grid-cols-3 gap-10"> <div class="grid grid-cols-2 gap-10">
<div> <div>
<FormControl <FormControl
v-model="batch.seat_count" v-model="batch.seat_count"
@@ -175,6 +162,11 @@
type="date" type="date"
class="mb-4" class="mb-4"
/> />
<Link
doctype="Email Template"
:label="__('Email Template')"
v-model="batch.confirmation_email_template"
/>
</div> </div>
<div> <div>
<FormControl <FormControl
@@ -199,30 +191,24 @@
v-model="batch.category" v-model="batch.category"
/> />
</div> </div>
<div>
<Link
doctype="Email Template"
:label="__('Email Template')"
v-model="batch.confirmation_email_template"
/>
</div>
</div> </div>
</div> </div>
<div class=""> <div class="">
<div class="text-lg text-ink-gray-9 font-semibold mb-4"> <div class="text-lg font-semibold mb-4">
{{ __('Payment') }} {{ __('Payment') }}
</div> </div>
<div>
<FormControl <FormControl
v-model="batch.paid_batch" v-model="batch.paid_batch"
type="checkbox" type="checkbox"
:label="__('Paid Batch')" :label="__('Paid Batch')"
/> />
<div class="grid grid-cols-3 gap-10 mt-4">
<FormControl <FormControl
v-model="batch.amount" v-model="batch.amount"
:label="__('Amount')" :label="__('Amount')"
type="number" type="number"
class="my-4"
/> />
<Link <Link
doctype="Currency" doctype="Currency"
@@ -234,7 +220,7 @@
</div> </div>
<div class="my-10"> <div class="my-10">
<div class="text-lg text-ink-gray-9 font-semibold mb-4"> <div class="text-lg font-semibold mb-4">
{{ __('Description') }} {{ __('Description') }}
</div> </div>
<FormControl <FormControl
@@ -278,20 +264,17 @@ import {
Button, Button,
TextEditor, TextEditor,
createResource, createResource,
usePageMeta,
} from 'frappe-ui' } from 'frappe-ui'
import Link from '@/components/Controls/Link.vue'
import { useRouter } from 'vue-router' import { useRouter } from 'vue-router'
import { showToast } from '@/utils' import { showToast } from '@/utils'
import { Image } from 'lucide-vue-next' import { Image } from 'lucide-vue-next'
import { capture } from '@/telemetry' import { capture } from '@/telemetry'
import { useOnboarding } from 'frappe-ui/frappe'
import { sessionStore } from '../stores/session'
import MultiSelect from '@/components/Controls/MultiSelect.vue' import MultiSelect from '@/components/Controls/MultiSelect.vue'
import Link from '@/components/Controls/Link.vue' import { useOnboarding } from 'frappe-ui/frappe'
const router = useRouter() const router = useRouter()
const user = inject('$user') const user = inject('$user')
const { brand } = sessionStore()
const { updateOnboardingStep } = useOnboarding('learning') const { updateOnboardingStep } = useOnboarding('learning')
const props = defineProps({ const props = defineProps({
@@ -444,13 +427,10 @@ const createNewBatch = () => {
{}, {},
{ {
onSuccess(data) { onSuccess(data) {
if (user.data?.is_system_manager) { capture('batch_created')
updateOnboardingStep('create_first_batch', true, false, () => { updateOnboardingStep('create_first_batch', true, false, () => {
localStorage.setItem('firstBatch', data.name) localStorage.setItem('firstBatch', data.name)
}) })
}
capture('batch_created')
router.push({ router.push({
name: 'BatchDetail', name: 'BatchDetail',
params: { params: {
@@ -459,7 +439,7 @@ const createNewBatch = () => {
}) })
}, },
onError(err) { onError(err) {
showToast('Message', err.messages?.[0] || err, 'alert-circle') showToast('Error', err.messages?.[0] || err, 'x')
}, },
} }
) )
@@ -478,7 +458,7 @@ const editBatchDetails = () => {
}) })
}, },
onError(err) { onError(err) {
showToast('Message', err.messages?.[0] || err, 'alert-circle') showToast('Error', err.messages?.[0] || err, 'x')
}, },
} }
) )
@@ -525,11 +505,4 @@ const breadcrumbs = computed(() => {
}) })
return crumbs return crumbs
}) })
usePageMeta(() => {
return {
title: props.batchName == 'new' ? 'New Batch' : batchDetail.data?.title,
icon: brand.favicon,
}
})
</script> </script>

View File

@@ -4,7 +4,7 @@
> >
<Breadcrumbs :items="breadcrumbs" /> <Breadcrumbs :items="breadcrumbs" />
<router-link <router-link
v-if="canCreateBatch()" v-if="user.data?.is_moderator"
:to="{ :to="{
name: 'BatchForm', name: 'BatchForm',
params: { batchName: 'new' }, params: { batchName: 'new' },
@@ -104,16 +104,14 @@ import {
FormControl, FormControl,
Select, Select,
TabButtons, TabButtons,
usePageMeta,
} from 'frappe-ui' } from 'frappe-ui'
import { computed, inject, onMounted, ref, watch } from 'vue' import { computed, inject, onMounted, ref, watch } from 'vue'
import { BookOpen, Plus } from 'lucide-vue-next' import { BookOpen, Plus } from 'lucide-vue-next'
import { sessionStore } from '@/stores/session' import { updateDocumentTitle } from '@/utils'
import BatchCard from '@/components/BatchCard.vue' import BatchCard from '@/components/BatchCard.vue'
const user = inject('$user') const user = inject('$user')
const dayjs = inject('$dayjs') const dayjs = inject('$dayjs')
const { brand } = sessionStore()
const start = ref(0) const start = ref(0)
const pageLength = ref(20) const pageLength = ref(20)
const categories = ref([]) const categories = ref([])
@@ -124,7 +122,6 @@ const filters = ref({})
const is_student = computed(() => user.data?.is_student) const is_student = computed(() => user.data?.is_student)
const currentTab = ref(is_student.value ? 'All' : 'Upcoming') const currentTab = ref(is_student.value ? 'All' : 'Upcoming')
const orderBy = ref('start_date') const orderBy = ref('start_date')
const readOnlyMode = window.read_only_mode
onMounted(() => { onMounted(() => {
setFiltersFromQuery() setFiltersFromQuery()
@@ -300,12 +297,6 @@ const batchTabs = computed(() => {
return tabs return tabs
}) })
const canCreateBatch = () => {
if (readOnlyMode) return false
if (user.data?.is_moderator || user.data?.is_instructor) return true
return false
}
const breadcrumbs = computed(() => [ const breadcrumbs = computed(() => [
{ {
label: __('Batches'), label: __('Batches'),
@@ -313,10 +304,12 @@ const breadcrumbs = computed(() => [
}, },
]) ])
usePageMeta(() => { const pageMeta = computed(() => {
return { return {
title: __('Batches'), title: 'Batches',
icon: brand.favicon, description: 'All upcoming batches.',
} }
}) })
updateDocumentTitle(pageMeta)
</script> </script>

View File

@@ -151,20 +151,19 @@
</template> </template>
<script setup> <script setup>
import { import {
Input,
Button, Button,
createResource, createResource,
FormControl, FormControl,
Breadcrumbs, Breadcrumbs,
usePageMeta, Tooltip,
} from 'frappe-ui' } from 'frappe-ui'
import { reactive, inject, onMounted, computed } from 'vue' import { reactive, inject, onMounted, computed } from 'vue'
import { showToast } from '@/utils/'
import { sessionStore } from '../stores/session'
import Link from '@/components/Controls/Link.vue' import Link from '@/components/Controls/Link.vue'
import NotPermitted from '@/components/NotPermitted.vue' import NotPermitted from '@/components/NotPermitted.vue'
import { showToast } from '@/utils/'
const user = inject('$user') const user = inject('$user')
const { brand } = sessionStore()
onMounted(() => { onMounted(() => {
const script = document.createElement('script') const script = document.createElement('script')
@@ -357,11 +356,4 @@ const redirectTo = computed(() => {
return `/lms/courses/${props.name}/certification` return `/lms/courses/${props.name}/certification`
} }
}) })
usePageMeta(() => {
return {
title: __('Billing Details'),
icon: brand.favicon,
}
})
</script> </script>

View File

@@ -12,13 +12,12 @@
</Button> </Button>
</router-link> </router-link>
</header> </header>
<div class="p-5 lg:w-3/4 mx-auto">
<div <div
v-if="participants.data?.length" class="flex flex-col lg:flex-row lg:items-center space-y-4 lg:space-y-0 justify-between mb-5"
class="mx-auto w-full max-w-4xl pt-6 pb-10"
> >
<div class="flex flex-col md:flex-row justify-between mb-4 px-3"> <div class="text-lg text-ink-gray-9 font-semibold">
<div class="text-xl font-semibold text-ink-gray-7 mb-4 md:mb-0"> {{ __('All Certified Participants') }}
{{ memberCount }} {{ __('certified members') }}
</div> </div>
<div class="grid grid-cols-2 gap-2"> <div class="grid grid-cols-2 gap-2">
<FormControl <FormControl
@@ -41,56 +40,36 @@
</div> </div>
</div> </div>
</div> </div>
<div class="divide-y"> <div v-if="participants.data?.length">
<template v-for="participant in participants.data"> <div class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-5">
<router-link <router-link
v-for="participant in participants.data"
:to="{ :to="{
name: 'ProfileCertificates', name: 'ProfileCertificates',
params: { params: { username: participant.username },
username: participant.username,
},
}" }"
class="flex sm:rounded px-3 py-2 sm:h-15 hover:bg-surface-gray-2"
> >
<div class="flex items-center w-full space-x-3"> <div
class="flex items-center space-x-2 border rounded-md hover:bg-surface-menu-bar p-2 text-ink-gray-7"
>
<Avatar <Avatar
:image="participant.user_image" :image="participant.user_image"
class="size-8 rounded-full object-contain"
:label="participant.full_name" :label="participant.full_name"
size="2xl" size="2xl"
/> />
<div class="flex flex-col md:flex-row w-full"> <div class="flex flex-col space-y-2">
<div class="flex-1"> <div class="font-medium">
<div class="text-base font-medium text-ink-gray-8">
{{ participant.full_name }} {{ participant.full_name }}
</div> </div>
<div <div
v-if="participant.headline" v-if="participant.headline"
class="mt-1.5 text-base text-ink-gray-5" class="headline text-sm text-ink-gray-7"
> >
{{ participant.headline }} {{ participant.headline }}
</div> </div>
</div> </div>
<div
class="flex items-center space-x-3 md:space-x-24 text-sm md:text-base mt-1.5"
>
<div class="text-ink-gray-5">
{{ participant.certificate_count }}
{{
participant.certificate_count > 1
? __('certificates')
: __('certificate')
}}
</div>
<span class="text-ink-gray-4 md:hidden">·</span>
<div class="text-ink-gray-5">
{{ dayjs(participant.issue_date).format('DD MMM YYYY') }}
</div>
</div>
</div>
</div> </div>
</router-link> </router-link>
</template>
</div> </div>
<div <div
v-if="!participants.list.loading && participants.hasNextPage" v-if="!participants.list.loading && participants.hasNextPage"
@@ -102,19 +81,16 @@
</div> </div>
</div> </div>
<div <div
v-else v-else-if="!participants.list.loading"
class="flex flex-col items-center justify-center text-sm text-ink-gray-5 mt-48" class="flex flex-col items-center justify-center text-sm text-ink-gray-5 italic mt-48"
> >
<BookOpen class="size-10 mx-auto stroke-1 text-ink-gray-4" /> <BookOpen class="size-10 mx-auto stroke-1 text-ink-gray-4" />
<div class="text-lg font-medium mb-1"> <div class="text-lg font-medium mb-1">
{{ __('No certified members') }} {{ __('No participants found') }}
</div> </div>
<div class="leading-5 w-2/5 text-center"> <div class="leading-5 w-2/5 text-center">
{{ {{ __('There are no participants matching this criteria.') }}
__( </div>
'No certified members found. Please check again later or get certified yourself.'
)
}}
</div> </div>
</div> </div>
</template> </template>
@@ -123,22 +99,17 @@ import {
Avatar, Avatar,
Breadcrumbs, Breadcrumbs,
Button, Button,
call,
createListResource, createListResource,
FormControl, FormControl,
Select, Select,
usePageMeta,
} from 'frappe-ui' } from 'frappe-ui'
import { computed, inject, onMounted, ref } from 'vue' import { computed, onMounted, ref } from 'vue'
import { updateDocumentTitle } from '@/utils'
import { BookOpen, GraduationCap } from 'lucide-vue-next' import { BookOpen, GraduationCap } from 'lucide-vue-next'
import { sessionStore } from '../stores/session'
const currentCategory = ref('') const currentCategory = ref('')
const filters = ref({}) const filters = ref({})
const nameFilter = ref('') const nameFilter = ref('')
const { brand } = sessionStore()
const memberCount = ref(0)
const dayjs = inject('$dayjs')
onMounted(() => { onMounted(() => {
updateParticipants() updateParticipants()
@@ -152,12 +123,6 @@ const participants = createListResource({
pageLength: 30, pageLength: 30,
}) })
const count = call('lms.lms.api.get_count_of_certified_members').then(
(data) => {
memberCount.value = data
}
)
const categories = createListResource({ const categories = createListResource({
doctype: 'LMS Certificate', doctype: 'LMS Certificate',
url: 'lms.lms.api.get_certification_categories', url: 'lms.lms.api.get_certification_categories',
@@ -193,17 +158,18 @@ const updateFilters = () => {
const breadcrumbs = computed(() => [ const breadcrumbs = computed(() => [
{ {
label: __('Certified Members'), label: __('Certified Participants'),
route: { name: 'CertifiedParticipants' }, route: { name: 'CertifiedParticipants' },
}, },
]) ])
usePageMeta(() => { const pageMeta = computed(() => {
return { return {
title: __('Certified Members'), title: 'Certified Participants',
icon: brand.favicon, description: 'All participants that have been certified.',
} }
}) })
updateDocumentTitle(pageMeta)
</script> </script>
<style> <style>
.headline { .headline {

View File

@@ -36,14 +36,12 @@
</template> </template>
<script setup> <script setup>
import { computed, inject, onMounted, ref } from 'vue' import { computed, inject, onMounted, ref } from 'vue'
import { Breadcrumbs, call, createResource, usePageMeta } from 'frappe-ui' import { Breadcrumbs, call, createResource } from 'frappe-ui'
import { useRouter } from 'vue-router' import { useRouter } from 'vue-router'
import { sessionStore } from '../stores/session'
import UpcomingEvaluations from '@/components/UpcomingEvaluations.vue' import UpcomingEvaluations from '@/components/UpcomingEvaluations.vue'
const courseTitle = ref(null) const courseTitle = ref(null)
const evaluator = ref(null) const evaluator = ref(null)
const { brand } = sessionStore()
const courses = ref([]) const courses = ref([])
const user = inject('$user') const user = inject('$user')
const dayjs = inject('$dayjs') const dayjs = inject('$dayjs')
@@ -135,11 +133,4 @@ const breadcrumbs = computed(() => [
label: __('Certification'), label: __('Certification'),
}, },
]) ])
usePageMeta(() => {
return {
title: courseTitle.value,
icon: brand.favicon,
}
})
</script> </script>

View File

@@ -56,7 +56,7 @@
<CourseInstructors :instructors="course.data.instructors" /> <CourseInstructors :instructors="course.data.instructors" />
</div> </div>
</div> </div>
<div v-if="course.data.tags" class="flex my-4 w-fit"> <div v-if="course.data.tags" class="flex mt-4 w-fit">
<Badge <Badge
theme="gray" theme="gray"
size="lg" size="lg"
@@ -69,7 +69,7 @@
<CourseCardOverlay :course="course" class="md:hidden mb-4" /> <CourseCardOverlay :course="course" class="md:hidden mb-4" />
<div <div
v-html="course.data.description" v-html="course.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-outline-gray-2 prose-th:border-outline-gray-2 prose-td:relative prose-th:relative prose-th:bg-surface-gray-2 prose-sm max-w-none !whitespace-normal mt-10" class="ProseMirror prose prose-table:table-fixed prose-td:p-2 prose-th:p-2 prose-td:border prose-th:border prose-td:border-outline-gray-2 prose-th:border-outline-gray-2 prose-td:relative prose-th:relative prose-th:bg-surface-gray-2 prose-sm max-w-none !whitespace-normal mt-4"
></div> ></div>
<div class="mt-10"> <div class="mt-10">
<CourseOutline <CourseOutline
@@ -92,24 +92,16 @@
</div> </div>
</template> </template>
<script setup> <script setup>
import { import { createResource, Breadcrumbs, Badge, Tooltip } from 'frappe-ui'
createResource,
Breadcrumbs,
Badge,
Tooltip,
usePageMeta,
} from 'frappe-ui'
import { computed } from 'vue' import { computed } from 'vue'
import { Users, Star } from 'lucide-vue-next' import { Users, Star } from 'lucide-vue-next'
import { sessionStore } from '@/stores/session'
import CourseCardOverlay from '@/components/CourseCardOverlay.vue' import CourseCardOverlay from '@/components/CourseCardOverlay.vue'
import CourseOutline from '@/components/CourseOutline.vue' import CourseOutline from '@/components/CourseOutline.vue'
import CourseReviews from '@/components/CourseReviews.vue' import CourseReviews from '@/components/CourseReviews.vue'
import UserAvatar from '@/components/UserAvatar.vue' import UserAvatar from '@/components/UserAvatar.vue'
import { updateDocumentTitle } from '@/utils'
import CourseInstructors from '@/components/CourseInstructors.vue' import CourseInstructors from '@/components/CourseInstructors.vue'
const { brand } = sessionStore()
const props = defineProps({ const props = defineProps({
courseName: { courseName: {
type: String, type: String,
@@ -135,12 +127,14 @@ const breadcrumbs = computed(() => {
return items return items
}) })
usePageMeta(() => { const pageMeta = computed(() => {
return { return {
title: course?.data?.title, title: course?.data?.title,
icon: brand.favicon, description: course?.data?.short_introduction,
} }
}) })
updateDocumentTitle(pageMeta)
</script> </script>
<style> <style>
.avatar-group { .avatar-group {

View File

@@ -3,11 +3,15 @@
<div class="grid md:grid-cols-[70%,30%] h-full"> <div class="grid md:grid-cols-[70%,30%] h-full">
<div> <div>
<header <header
class="sticky top-0 z-10 flex flex-col md:flex-row md:items-center justify-between border-b bg-surface-white px-3 py-2.5 sm:px-5" class="sticky top-0 z-10 group flex flex-col md:flex-row md:items-center justify-between border-b bg-surface-white px-3 py-2.5 sm:px-5"
> >
<Breadcrumbs class="h-7" :items="breadcrumbs" /> <Breadcrumbs class="h-7" :items="breadcrumbs" />
<div class="flex items-center mt-3 md:mt-0"> <div class="flex items-center mt-3 md:mt-0">
<Button v-if="courseResource.data?.name" @click="trashCourse()"> <Button
v-if="courseResource.data?.name"
@click="trashCourse()"
class="invisible group-hover:visible"
>
<template #icon> <template #icon>
<Trash2 class="w-4 h-4 stroke-1.5" /> <Trash2 class="w-4 h-4 stroke-1.5" />
</template> </template>
@@ -249,7 +253,6 @@ import {
createResource, createResource,
FormControl, FormControl,
FileUploader, FileUploader,
usePageMeta,
} from 'frappe-ui' } from 'frappe-ui'
import { import {
inject, inject,
@@ -261,20 +264,18 @@ import {
watch, watch,
getCurrentInstance, getCurrentInstance,
} from 'vue' } from 'vue'
import { showToast } from '@/utils' import { showToast, updateDocumentTitle } from '@/utils'
import Link from '@/components/Controls/Link.vue'
import { Image, Trash2, X } from 'lucide-vue-next' import { Image, Trash2, X } from 'lucide-vue-next'
import { useRouter } from 'vue-router' import { useRouter } from 'vue-router'
import { capture } from '@/telemetry'
import { useOnboarding } from 'frappe-ui/frappe'
import { sessionStore } from '../stores/session'
import { useSettings } from '@/stores/settings'
import Link from '@/components/Controls/Link.vue'
import CourseOutline from '@/components/CourseOutline.vue' import CourseOutline from '@/components/CourseOutline.vue'
import MultiSelect from '@/components/Controls/MultiSelect.vue' import MultiSelect from '@/components/Controls/MultiSelect.vue'
import { capture } from '@/telemetry'
import { useOnboarding } from 'frappe-ui/frappe'
import { useSettings } from '@/stores/settings'
const user = inject('$user') const user = inject('$user')
const newTag = ref('') const newTag = ref('')
const { brand } = sessionStore()
const router = useRouter() const router = useRouter()
const instructors = ref([]) const instructors = ref([])
const settingsStore = useSettings() const settingsStore = useSettings()
@@ -310,7 +311,11 @@ const course = reactive({
}) })
onMounted(() => { onMounted(() => {
if (!user.data?.is_moderator && !user.data?.is_instructor) { if (
props.courseName == 'new' &&
!user.data?.is_moderator &&
!user.data?.is_instructor
) {
router.push({ name: 'Courses' }) router.push({ name: 'Courses' })
} }
@@ -396,7 +401,7 @@ const courseResource = createResource({
'paid_course', 'paid_course',
'featured', 'featured',
'enable_certification', 'enable_certification',
'paid_certificate', 'paid_certifiate',
] ]
for (let idx in checkboxes) { for (let idx in checkboxes) {
let key = checkboxes[idx] let key = checkboxes[idx]
@@ -439,14 +444,11 @@ const submitCourse = () => {
} else { } else {
courseCreationResource.submit(course, { courseCreationResource.submit(course, {
onSuccess(data) { onSuccess(data) {
if (user.data?.is_system_manager) { capture('course_created')
showToast('Success', 'Course created successfully', 'check')
updateOnboardingStep('create_first_course', true, false, () => { updateOnboardingStep('create_first_course', true, false, () => {
localStorage.setItem('firstCourse', data.name) localStorage.setItem('firstCourse', data.name)
}) })
}
capture('course_created')
showToast('Success', 'Course created successfully', 'check')
router.push({ router.push({
name: 'CourseForm', name: 'CourseForm',
params: { courseName: data.name }, params: { courseName: data.name },
@@ -572,10 +574,12 @@ const breadcrumbs = computed(() => {
return crumbs return crumbs
}) })
usePageMeta(() => { const pageMeta = computed(() => {
return { return {
title: courseResource.data?.title || __('New Course'), title: 'Create a Course',
icon: brand.favicon, description: 'Create or edit a course for your learning system.',
} }
}) })
updateDocumentTitle(pageMeta)
</script> </script>

View File

@@ -4,7 +4,7 @@
> >
<Breadcrumbs :items="breadcrumbs" /> <Breadcrumbs :items="breadcrumbs" />
<router-link <router-link
v-if="canCreateCourse()" v-if="user.data?.is_moderator"
:to="{ :to="{
name: 'CourseForm', name: 'CourseForm',
params: { courseName: 'new' }, params: { courseName: 'new' },
@@ -57,7 +57,7 @@
</div> </div>
<div <div
v-if="courses.data?.length" v-if="courses.data?.length"
class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 2xl:grid-cols-4 gap-4" class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 2xl:grid-cols-4 gap-5"
> >
<router-link <router-link
v-for="course in courses.data" v-for="course in courses.data"
@@ -96,19 +96,15 @@
import { import {
Breadcrumbs, Breadcrumbs,
Button, Button,
call,
createListResource, createListResource,
FormControl, FormControl,
Select, Select,
TabButtons, TabButtons,
usePageMeta,
} from 'frappe-ui' } from 'frappe-ui'
import { computed, inject, onMounted, ref, watch } from 'vue' import { computed, inject, onMounted, ref, watch } from 'vue'
import { BookOpen, Plus } from 'lucide-vue-next' import { BookOpen, Plus } from 'lucide-vue-next'
import { sessionStore } from '@/stores/session' import { updateDocumentTitle } from '@/utils'
import { canCreateCourse } from '@/utils'
import CourseCard from '@/components/CourseCard.vue' import CourseCard from '@/components/CourseCard.vue'
import router from '../router'
const user = inject('$user') const user = inject('$user')
const dayjs = inject('$dayjs') const dayjs = inject('$dayjs')
@@ -120,11 +116,8 @@ const title = ref('')
const certification = ref(false) const certification = ref(false)
const filters = ref({}) const filters = ref({})
const currentTab = ref('Live') const currentTab = ref('Live')
const { brand } = sessionStore()
const readOnlyMode = window.read_only_mode
onMounted(() => { onMounted(() => {
identifyUserPersona()
setFiltersFromQuery() setFiltersFromQuery()
updateCourses() updateCourses()
categories.value = [ categories.value = [
@@ -149,11 +142,6 @@ const courses = createListResource({
pageLength: pageLength.value, pageLength: pageLength.value,
start: start.value, start: start.value,
onSuccess(data) { onSuccess(data) {
setCategories(data)
},
})
const setCategories = (data) => {
let allCategories = data.map((course) => course.category) let allCategories = data.map((course) => course.category)
allCategories = allCategories.filter( allCategories = allCategories.filter(
(category, index) => allCategories.indexOf(category) === index && category (category, index) => allCategories.indexOf(category) === index && category
@@ -161,32 +149,8 @@ const setCategories = (data) => {
if (categories.value.length <= allCategories.length) { if (categories.value.length <= allCategories.length) {
updateCategories(data) updateCategories(data)
} }
} },
})
const isPersonaCaptured = async () => {
let persona = await call('frappe.client.get_single_value', {
doctype: 'LMS Settings',
field: 'persona_captured',
})
return persona
}
const identifyUserPersona = async () => {
if (user.data?.is_system_manager && !user.data?.developer_mode) {
let personaCaptured = await isPersonaCaptured()
if (personaCaptured) return
call('frappe.client.get_count', {
doctype: 'LMS Course',
}).then((data) => {
if (!data) {
router.push({
name: 'PersonaForm',
})
}
})
}
}
const updateCourses = () => { const updateCourses = () => {
updateFilters() updateFilters()
@@ -339,10 +303,12 @@ const breadcrumbs = computed(() => [
}, },
]) ])
usePageMeta(() => { const pageMeta = computed(() => {
return { return {
title: __('Courses'), title: 'Courses',
icon: brand.favicon, description: 'All published courses.',
} }
}) })
updateDocumentTitle(pageMeta)
</script> </script>

View File

@@ -0,0 +1,39 @@
<template>
<div class="max-w-3xl py-12 mx-auto">
<Button
icon-left="code"
@click="$resources.ping.fetch"
:loading="$resources.ping.loading"
>
Click to send 'ping' request
</Button>
<div>
{{ $resources.ping.data }}
</div>
<pre>{{ $resources.ping }}</pre>
<Button @click="showDialog = true">Open Dialog</Button>
<Dialog title="Title" v-model="showDialog"> Dialog content </Dialog>
</div>
</template>
<script>
import { Dialog } from 'frappe-ui'
export default {
name: 'Home',
data() {
return {
showDialog: false,
}
},
resources: {
ping: {
url: 'ping',
},
},
components: {
Dialog,
},
}
</script>

View File

@@ -13,22 +13,17 @@
<div class="text-lg font-semibold mb-4"> <div class="text-lg font-semibold mb-4">
{{ __('Job Details') }} {{ __('Job Details') }}
</div> </div>
<div class="grid grid-cols-2 gap-5"> <div class="grid grid-cols-2 gap-4">
<div class="space-y-4"> <div>
<FormControl <FormControl
v-model="job.job_title" v-model="job.job_title"
:label="__('Title')" :label="__('Title')"
class="mb-4"
:required="true" :required="true"
/> />
<FormControl <FormControl
v-model="job.location" v-model="job.location"
:label="__('City')" :label="__('Location')"
:required="true"
/>
<Link
v-model="job.country"
doctype="Country"
:label="__('Country')"
:required="true" :required="true"
/> />
</div> </div>
@@ -50,12 +45,25 @@
/> />
</div> </div>
</div> </div>
<div class="mt-4">
<label class="block text-ink-gray-5 text-xs mb-1">
{{ __('Description') }}
<span class="text-ink-red-3">*</span>
</label>
<TextEditor
:content="job.description"
@change="(val) => (job.description = 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] mb-4"
/>
</div> </div>
<div class="container border-b mb-4 pb-4"> </div>
<div class="container mb-4 pb-4">
<div class="text-lg font-semibold mb-4"> <div class="text-lg font-semibold mb-4">
{{ __('Company Details') }} {{ __('Company Details') }}
</div> </div>
<div class="grid grid-cols-2 gap-5"> <div class="grid grid-cols-2 gap-4">
<div> <div>
<FormControl <FormControl
v-model="job.company_name" v-model="job.company_name"
@@ -120,19 +128,6 @@
</div> </div>
</div> </div>
</div> </div>
<div class="container mt-4">
<label class="block text-ink-gray-5 text-xs mb-1">
{{ __('Description') }}
<span class="text-ink-red-3">*</span>
</label>
<TextEditor
:content="job.description"
@change="(val) => (job.description = 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] mb-4"
/>
</div>
</div> </div>
</div> </div>
</template> </template>
@@ -144,17 +139,14 @@ import {
Button, Button,
TextEditor, TextEditor,
FileUploader, FileUploader,
usePageMeta,
} from 'frappe-ui' } from 'frappe-ui'
import { computed, onMounted, reactive, inject } from 'vue' import { computed, onMounted, reactive, inject } from 'vue'
import { FileText, X } from 'lucide-vue-next' import { FileText, X } from 'lucide-vue-next'
import { sessionStore } from '@/stores/session'
import { useRouter } from 'vue-router' import { useRouter } from 'vue-router'
import { getFileSize, showToast } from '../utils' import { getFileSize, showToast } from '../utils'
const user = inject('$user') const user = inject('$user')
const router = useRouter() const router = useRouter()
const { brand } = sessionStore()
const props = defineProps({ const props = defineProps({
jobName: { jobName: {
@@ -222,7 +214,6 @@ const imageResource = createResource({
const job = reactive({ const job = reactive({
job_title: '', job_title: '',
location: '', location: '',
country: '',
type: 'Full Time', type: 'Full Time',
status: 'Open', status: 'Open',
company_name: '', company_name: '',
@@ -323,16 +314,9 @@ const breadcrumbs = computed(() => {
}, },
{ {
label: props.jobName == 'new' ? 'New Job' : 'Edit Job', label: props.jobName == 'new' ? 'New Job' : 'Edit Job',
route: { name: 'JobForm' }, route: { name: 'JobCreation' },
}, },
] ]
return crumbs return crumbs
}) })
usePageMeta(() => {
return {
title: props.jobName == 'new' ? 'New Job' : jobDetail.data?.title,
icon: brand.favicon,
}
})
</script> </script>

View File

@@ -16,30 +16,21 @@
}, },
]" ]"
/> />
<div <div v-if="user.data?.name" class="flex">
v-if="user.data?.name && !readOnlyMode"
class="flex items-center space-x-2"
>
<router-link <router-link
v-if="user.data.name == job.data?.owner" v-if="user.data.name == job.data?.owner"
:to="{ :to="{
name: 'JobForm', name: 'JobCreation',
params: { jobName: job.data?.name }, params: { jobName: job.data?.name },
}" }"
> >
<Button> <Button class="mr-2">
<template #prefix> <template #prefix>
<Pencil class="h-4 w-4 stroke-1.5" /> <Pencil class="h-4 w-4 stroke-1.5" />
</template> </template>
{{ __('Edit') }} {{ __('Edit') }}
</Button> </Button>
</router-link> </router-link>
<Button @click="redirectToWebsite(job.data?.company_website)">
<template #prefix>
<SquareArrowOutUpRight class="h-4 w-4 stroke-1.5" />
</template>
{{ __('Visit Website') }}
</Button>
<Button <Button
v-if="!jobApplication.data?.length" v-if="!jobApplication.data?.length"
variant="solid" variant="solid"
@@ -50,14 +41,8 @@
</template> </template>
{{ __('Apply') }} {{ __('Apply') }}
</Button> </Button>
<Badge v-else variant="subtle" theme="green" size="lg">
<template #prefix>
<Check class="h-4 w-4" />
</template>
{{ __('You have applied') }}
</Badge>
</div> </div>
<div v-else-if="!readOnlyMode"> <div v-else>
<Button @click="redirectToLogin(job.data?.name)"> <Button @click="redirectToLogin(job.data?.name)">
<span> <span>
{{ __('Login to apply') }} {{ __('Login to apply') }}
@@ -65,17 +50,16 @@
</Button> </Button>
</div> </div>
</header> </header>
<div v-if="job.data" class="max-w-3xl mx-auto pt-5"> <div v-if="job.data" class="max-w-3xl mx-auto">
<div class="p-4"> <div class="p-4">
<div class="space-y-5 mb-10"> <div class="space-y-5 mb-10">
<div class="flex items-center"> <div class="flex items-center">
<img <img
:src="job.data.company_logo" :src="job.data.company_logo"
class="size-10 rounded-lg object-contain cursor-pointer mr-4" class="w-16 h-16 rounded-lg object-contain mr-4"
:alt="job.data.company_name" :alt="job.data.company_name"
@click="redirectToWebsite(job.data.company_website)"
/> />
<div class="text-2xl text-ink-gray-9 font-semibold"> <div class="text-2xl text-ink-gray-9 font-semibold mb-4">
{{ job.data.job_title }} {{ job.data.job_title }}
</div> </div>
</div> </div>
@@ -84,8 +68,8 @@
class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-x-10 gap-y-5 md:gap-y-5" class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-x-10 gap-y-5 md:gap-y-5"
> >
<div class="flex items-center space-x-4"> <div class="flex items-center space-x-4">
<Building2 class="size-4 stroke-1.5 text-ink-gray-7" /> <Building2 class="h-4 w-4 text-ink-green-2" />
<div class="flex flex-col space-y-1 text-ink-gray-7"> <div class="flex flex-col space-y-2 text-ink-gray-7">
<span class="text-xs text-ink-gray-5 font-medium uppercase"> <span class="text-xs text-ink-gray-5 font-medium uppercase">
{{ __('Organisation') }} {{ __('Organisation') }}
</span> </span>
@@ -95,20 +79,20 @@
</div> </div>
</div> </div>
<div class="flex items-center space-x-4"> <div class="flex items-center space-x-4">
<MapPin class="size-4 stroke-1.5 text-ink-gray-7" /> <MapPin class="size-4 text-ink-red-3" />
<div class="flex flex-col space-y-1 text-ink-gray-7"> <div class="flex flex-col space-y-2 text-ink-gray-7">
<span class="text-xs text-ink-gray-5 font-medium uppercase"> <span class="text-xs font-medium uppercase">
{{ __('Location') }} {{ __('Location') }}
</span> </span>
<span class="text-sm font-semibold"> <span class="text-sm font-semibold">
{{ job.data.location }}, {{ job.data.country }} {{ job.data.location }}
</span> </span>
</div> </div>
</div> </div>
<div class="flex items-center space-x-4"> <div class="flex items-center space-x-4">
<ClipboardType class="size-4 stroke-1.5 text-ink-gray-7" /> <ClipboardType class="h-4 w-4 text-yellow-500" />
<div class="flex flex-col space-y-1 text-ink-gray-7"> <div class="flex flex-col space-y-2 text-ink-gray-7">
<span class="text-xs text-ink-gray-5 font-medium uppercase"> <span class="text-xs font-medium uppercase">
{{ __('Category') }} {{ __('Category') }}
</span> </span>
<span class="text-sm font-semibold"> <span class="text-sm font-semibold">
@@ -117,9 +101,9 @@
</div> </div>
</div> </div>
<div class="flex items-center space-x-4"> <div class="flex items-center space-x-4">
<CalendarDays class="size-4 stroke-1.5 text-ink-gray-7" /> <CalendarDays class="h-4 w-4 text-ink-blue-2" />
<div class="flex flex-col space-y-1 text-ink-gray-7"> <div class="flex flex-col space-y-2 text-ink-gray-7">
<span class="text-xs text-ink-gray-5 font-medium uppercase"> <span class="text-xs font-medium uppercase">
{{ __('Posted on') }} {{ __('Posted on') }}
</span> </span>
<span class="text-sm font-semibold"> <span class="text-sm font-semibold">
@@ -131,9 +115,9 @@
v-if="applicationCount.data" v-if="applicationCount.data"
class="flex items-center space-x-4" class="flex items-center space-x-4"
> >
<SquareUserRound class="size-4 stroke-1.5 text-ink-gray-7" /> <SquareUserRound class="h-4 w-4 text-purple-500" />
<div class="flex flex-col space-y-1 text-ink-gray-7"> <div class="flex flex-col space-y-2 text-ink-gray-7">
<span class="text-xs text-ink-gray-5 font-medium uppercase"> <span class="text-xs font-medium uppercase">
{{ __('Applications Received') }} {{ __('Applications Received') }}
</span> </span>
<span class="text-sm font-semibold"> <span class="text-sm font-semibold">
@@ -158,33 +142,23 @@
</div> </div>
</template> </template>
<script setup> <script setup>
import { import { Button, Breadcrumbs, createResource } from 'frappe-ui'
Badge, import { inject, ref, computed } from 'vue'
Button, import { updateDocumentTitle } from '@/utils'
Breadcrumbs,
createResource,
usePageMeta,
} from 'frappe-ui'
import { inject, ref } from 'vue'
import { sessionStore } from '../stores/session'
import JobApplicationModal from '@/components/Modals/JobApplicationModal.vue' import JobApplicationModal from '@/components/Modals/JobApplicationModal.vue'
import { import {
MapPin, MapPin,
Check,
SendHorizonal, SendHorizonal,
Pencil, Pencil,
Building2, Building2,
CalendarDays, CalendarDays,
ClipboardType, ClipboardType,
SquareUserRound, SquareUserRound,
SquareArrowOutUpRight,
} from 'lucide-vue-next' } from 'lucide-vue-next'
const user = inject('$user') const user = inject('$user')
const dayjs = inject('$dayjs') const dayjs = inject('$dayjs')
const { brand } = sessionStore()
const showApplicationModal = ref(false) const showApplicationModal = ref(false)
const readOnlyMode = window.read_only_mode
const props = defineProps({ const props = defineProps({
job: { job: {
@@ -241,14 +215,12 @@ const redirectToLogin = (job) => {
window.location.href = `/login?redirect-to=/job-openings/${job}` window.location.href = `/login?redirect-to=/job-openings/${job}`
} }
const redirectToWebsite = (url) => { const pageMeta = computed(() => {
window.open(url, '_blank')
}
usePageMeta(() => {
return { return {
title: job.data?.job_title, title: job.data?.job_title,
icon: brand.favicon, description: job.data?.description,
} }
}) })
updateDocumentTitle(pageMeta)
</script> </script>

View File

@@ -10,13 +10,13 @@
<router-link <router-link
v-if="user.data?.name" v-if="user.data?.name"
:to="{ :to="{
name: 'JobForm', name: 'JobCreation',
params: { params: {
jobName: 'new', jobName: 'new',
}, },
}" }"
> >
<Button v-if="!readOnlyMode" variant="solid"> <Button variant="solid">
<template #prefix> <template #prefix>
<Plus class="h-4 w-4" /> <Plus class="h-4 w-4" />
</template> </template>
@@ -25,16 +25,14 @@
</router-link> </router-link>
</header> </header>
<div> <div>
<div class="lg:w-3/4 mx-auto p-5">
<div <div
class="flex flex-col lg:flex-row space-y-4 lg:space-y-0 lg:items-center justify-between w-full md:w-4/5 mx-auto p-5" class="flex flex-col lg:flex-row space-y-4 lg:space-y-0 lg:items-center justify-between mb-5"
> >
<div <div class="text-xl text-ink-gray-9 font-semibold">
v-if="jobCount" {{ __('Find the perfect job for you') }}
class="text-xl font-semibold text-ink-gray-7 mb-4 md:mb-0"
>
{{ __('{0} Open Jobs').format(jobCount) }}
</div> </div>
<div class="grid grid-cols-1 md:grid-cols-3 gap-2"> <div class="grid grid-cols-2 gap-2">
<FormControl <FormControl
type="text" type="text"
:placeholder="__('Search')" :placeholder="__('Search')"
@@ -49,12 +47,6 @@
/> />
</template> </template>
</FormControl> </FormControl>
<Link
doctype="Country"
v-model="country"
:placeholder="__('Country')"
class="min-w-40 lg:min-w-0 lg:w-32 xl:w-40"
/>
<FormControl <FormControl
v-model="jobType" v-model="jobType"
type="select" type="select"
@@ -65,8 +57,11 @@
/> />
</div> </div>
</div> </div>
<div v-if="jobs.data?.length" class="w-full md:w-4/5 mx-auto p-5 pt-0">
<div class="grid grid-cols-1 lg:grid-cols-3 gap-4"> <div
v-if="jobs.data?.length"
class="grid grid-cols-1 lg:grid-cols-2 gap-5"
>
<router-link <router-link
v-for="job in jobs.data" v-for="job in jobs.data"
:to="{ :to="{
@@ -78,49 +73,25 @@
<JobCard :job="job" /> <JobCard :job="job" />
</router-link> </router-link>
</div> </div>
</div> <div v-else class="text-ink-gray-7 italic p-5 w-fit mx-auto">
<div {{ __('No jobs posted') }}
v-else
class="flex flex-col items-center justify-center text-sm text-ink-gray-5 mt-56"
>
<Laptop class="size-10 mx-auto stroke-1 text-ink-gray-4" />
<div class="text-lg font-medium mb-1">
{{ __('No jobs found') }}
</div>
<div class="leading-5 w-2/5 text-center">
{{ __('There are no jobs available at the moment.') }}
</div>
<div class="leading-5 w-1/5 text-center">
{{ __('Post a new job or check again later.') }}
</div> </div>
</div> </div>
</div> </div>
</div> </div>
</template> </template>
<script setup> <script setup>
import { import { Button, Breadcrumbs, createResource, FormControl } from 'frappe-ui'
Button, import { Plus, Search } from 'lucide-vue-next'
Breadcrumbs, import { inject, computed, ref, onMounted } from 'vue'
call,
createResource,
FormControl,
usePageMeta,
} from 'frappe-ui'
import { Laptop, Plus, Search } from 'lucide-vue-next'
import { sessionStore } from '../stores/session'
import { inject, computed, ref, onMounted, watch } from 'vue'
import JobCard from '@/components/JobCard.vue' import JobCard from '@/components/JobCard.vue'
import Link from '@/components/Controls/Link.vue' import { updateDocumentTitle } from '@/utils'
const user = inject('$user') const user = inject('$user')
const jobType = ref(null) const jobType = ref(null)
const { brand } = sessionStore()
const searchQuery = ref('') const searchQuery = ref('')
const country = ref(null)
const filters = ref({}) const filters = ref({})
const orFilters = ref({}) const orFilters = ref({})
const jobCount = ref(0)
const readOnlyMode = window.read_only_mode
onMounted(() => { onMounted(() => {
let queries = new URLSearchParams(location.search) let queries = new URLSearchParams(location.search)
@@ -128,7 +99,6 @@ onMounted(() => {
jobType.value = queries.get('type') jobType.value = queries.get('type')
} }
updateJobs() updateJobs()
getJobCount()
}) })
const jobs = createResource({ const jobs = createResource({
@@ -166,30 +136,8 @@ const updateFilters = () => {
} else { } else {
orFilters.value = {} orFilters.value = {}
} }
if (country.value) {
filters.value.country = country.value
} else {
delete filters.value.country
}
} }
const getJobCount = () => {
call('frappe.client.get_count', {
doctype: 'Job Opportunity',
filters: {
status: 'Open',
disabled: 0,
},
}).then((data) => {
jobCount.value = data
})
}
watch(country, (val) => {
updateJobs()
})
const jobTypes = computed(() => { const jobTypes = computed(() => {
return [ return [
'', '',
@@ -199,11 +147,12 @@ const jobTypes = computed(() => {
{ label: __('Freelance'), value: 'Freelance' }, { label: __('Freelance'), value: 'Freelance' },
] ]
}) })
const pageMeta = computed(() => {
usePageMeta(() => {
return { return {
title: __('Jobs'), title: 'Jobs',
icon: brand.favicon, description: 'An open job board for the community',
} }
}) })
updateDocumentTitle(pageMeta)
</script> </script>

View File

@@ -4,102 +4,33 @@
class="sticky top-0 z-10 flex items-center justify-between border-b bg-surface-white px-3 py-2.5 sm:px-5" class="sticky top-0 z-10 flex items-center justify-between border-b bg-surface-white px-3 py-2.5 sm:px-5"
> >
<Breadcrumbs class="h-7" :items="breadcrumbs" /> <Breadcrumbs class="h-7" :items="breadcrumbs" />
<div class="flex items-center space-x-2">
<Tooltip v-if="canGoZen()" :text="__('Zen Mode')">
<Button @click="goFullScreen()">
<template #icon>
<Focus class="w-4 h-4 stroke-2" />
</template>
</Button>
</Tooltip>
<CertificationLinks :courseName="courseName" /> <CertificationLinks :courseName="courseName" />
</div>
</header> </header>
<div class="grid md:grid-cols-[70%,30%] h-screen"> <div class="grid md:grid-cols-[70%,30%] h-screen">
<div v-if="lesson.data.no_preview" class="border-r"> <div
<div class="shadow rounded-md w-3/4 mt-10 mx-auto text-center p-4"> v-if="lesson.data.no_preview"
<div class="flex items-center justify-center mt-4 space-x-2"> class="border-r text-center pt-10 px-5 md:px-0 pb-10"
<LockKeyholeIcon class="size-4 stroke-2 text-ink-gray-5" /> >
<div class="text-lg font-semibold text-ink-gray-7"> <p class="mb-4">
{{ __('This lesson is locked') }}
</div>
</div>
<div class="mt-1 mb-4 text-ink-gray-7">
{{ {{
__( __(
'This lesson is not available for preview. Please enroll in the course to access it.' 'This lesson is not available for preview. Please enroll in the course to access it.'
) )
}} }}
</div> </p>
<Button <Button v-if="user.data" @click="enrollStudent()" variant="solid">
v-if="user.data && !lesson.data.disable_self_learning"
@click="enrollStudent()"
variant="solid"
>
{{ __('Start Learning') }} {{ __('Start Learning') }}
</Button> </Button>
<Badge
theme="blue"
size="lg"
v-else-if="lesson.data.disable_self_learning"
class="mt-2"
>
{{ __('Contact the Administrator to enroll for this course.') }}
</Badge>
<Button v-else @click="redirectToLogin()"> <Button v-else @click="redirectToLogin()">
<template #prefix>
<LogIn class="w-4 h-4 stroke-1" />
</template>
{{ __('Login') }} {{ __('Login') }}
</Button> </Button>
</div> </div>
</div> <div v-else class="border-r container pt-5 pb-10 px-5">
<div <div class="flex flex-col md:flex-row md:items-center justify-between">
v-else
ref="lessonContainer"
class="bg-surface-white"
:class="{
'overflow-y-auto': zenModeEnabled,
}"
>
<div
class="border-r container pt-5 pb-10 px-5 h-full"
:class="{
'w-full md:w-3/4 mx-auto border-none !pt-10': zenModeEnabled,
}"
>
<div
class="flex flex-col md:flex-row md:items-center justify-between"
>
<div class="flex flex-col">
<div class="text-3xl font-semibold text-ink-gray-9"> <div class="text-3xl font-semibold text-ink-gray-9">
{{ lesson.data.title }} {{ lesson.data.title }}
</div> </div>
<div class="flex items-center mt-2 md:mt-0">
<div
v-if="zenModeEnabled"
class="relative flex items-center space-x-2 text-sm mt-1 text-ink-gray-7 group w-fit mt-2"
>
<span>
{{ lesson.data.chapter_title }} -
{{ lesson.data.course_title }}
</span>
<Info class="size-3" />
<div
class="hidden group-hover:block rounded bg-gray-900 px-2 py-1 text-xs text-white shadow-xl absolute left-0 top-full mt-2"
>
{{ Math.ceil(lesson.data.membership.progress) }}%
{{ __('completed') }}
</div>
</div>
</div>
<div class="flex items-center space-x-2 mt-2 md:mt-0">
<Button v-if="zenModeEnabled" @click="showDiscussionsInZenMode()">
<template #icon>
<MessageCircleQuestion class="w-4 h-4 stroke-1.5" />
</template>
</Button>
<router-link <router-link
v-if="lesson.data.prev" v-if="lesson.data.prev"
:to="{ :to="{
@@ -111,7 +42,7 @@
}, },
}" }"
> >
<Button> <Button class="mr-2">
<template #prefix> <template #prefix>
<ChevronLeft class="w-4 h-4 stroke-1" /> <ChevronLeft class="w-4 h-4 stroke-1" />
</template> </template>
@@ -131,7 +62,7 @@
}, },
}" }"
> >
<Button> <Button class="mr-2">
{{ __('Edit') }} {{ __('Edit') }}
</Button> </Button>
</router-link> </router-link>
@@ -169,7 +100,7 @@
</div> </div>
</div> </div>
<div v-if="!zenModeEnabled" class="flex items-center mt-2"> <div class="flex items-center mt-2">
<span <span
class="h-6 mr-1" class="h-6 mr-1"
:class="{ :class="{
@@ -186,7 +117,6 @@
:instructors="lesson.data.instructors" :instructors="lesson.data.instructors"
/> />
</div> </div>
<div <div
v-if=" v-if="
lesson.data.instructor_content && lesson.data.instructor_content &&
@@ -205,19 +135,19 @@
</div> </div>
<div <div
v-else-if="lesson.data.instructor_notes" v-else-if="lesson.data.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-outline-gray-2 prose-th:border-outline-gray-2 prose-td:relative prose-th:relative prose-th:bg-surface-gray-2 prose-sm max-w-none !whitespace-normal mt-8" class="ProseMirror prose prose-table:table-fixed prose-td:p-2 prose-th:p-2 prose-td:border prose-th:border prose-td:border-outline-gray-2 prose-th:border-outline-gray-2 prose-td:relative prose-th:relative prose-th:bg-surface-gray-2 prose-sm max-w-none !whitespace-normal mt-6"
> >
<LessonContent :content="lesson.data.instructor_notes" /> <LessonContent :content="lesson.data.instructor_notes" />
</div> </div>
<div <div
v-if="lesson.data.content" v-if="lesson.data.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-outline-gray-2 prose-th:border-outline-gray-2 prose-td:relative prose-th:relative prose-th:bg-surface-gray-2 prose-sm max-w-none !whitespace-normal mt-8" class="ProseMirror prose prose-table:table-fixed prose-td:p-2 prose-th:p-2 prose-td:border prose-th:border prose-td:border-outline-gray-2 prose-th:border-outline-gray-2 prose-td:relative prose-th:relative prose-th:bg-surface-gray-2 prose-sm max-w-none !whitespace-normal mt-5"
> >
<div id="editor"></div> <div id="editor"></div>
</div> </div>
<div <div
v-else v-else
class="ProseMirror prose prose-table:table-fixed prose-td:p-2 prose-th:p-2 prose-td:border prose-th:border prose-td:border-outline-gray-2 prose-th:border-outline-gray-2 prose-td:relative prose-th:relative prose-th:bg-surface-gray-2 prose-sm max-w-none !whitespace-normal mt-8" class="ProseMirror prose prose-table:table-fixed prose-td:p-2 prose-th:p-2 prose-td:border prose-th:border prose-td:border-outline-gray-2 prose-th:border-outline-gray-2 prose-td:relative prose-th:relative prose-th:bg-surface-gray-2 prose-sm max-w-none !whitespace-normal mt-5"
> >
<LessonContent <LessonContent
v-if="lesson.data?.body" v-if="lesson.data?.body"
@@ -226,7 +156,7 @@
:quizId="lesson.data.quiz_id" :quizId="lesson.data.quiz_id"
/> />
</div> </div>
<div class="mt-20" ref="discussionsContainer"> <div class="mt-20">
<Discussions <Discussions
v-if="allowDiscussions" v-if="allowDiscussions"
:title="'Questions'" :title="'Questions'"
@@ -236,7 +166,6 @@
/> />
</div> </div>
</div> </div>
</div>
<div class="sticky top-10"> <div class="sticky top-10">
<div class="bg-surface-menu-bar py-5 px-2 border-b"> <div class="bg-surface-menu-bar py-5 px-2 border-b">
<div class="text-lg font-semibold text-ink-gray-9"> <div class="text-lg font-semibold text-ink-gray-9">
@@ -264,38 +193,14 @@
</div> </div>
</template> </template>
<script setup> <script setup>
import { import { createResource, Breadcrumbs, Button } from 'frappe-ui'
createResource, import { computed, watch, inject, ref, onMounted, onBeforeUnmount } from 'vue'
Badge,
Breadcrumbs,
Button,
Tooltip,
usePageMeta,
} from 'frappe-ui'
import {
computed,
watch,
inject,
ref,
onMounted,
onBeforeUnmount,
nextTick,
} 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 { useRouter, useRoute } from 'vue-router' import { useRouter, useRoute } from 'vue-router'
import { import { ChevronLeft, ChevronRight, GraduationCap } from 'lucide-vue-next'
ChevronLeft,
ChevronRight,
LockKeyholeIcon,
LogIn,
Focus,
Info,
MessageCircleQuestion,
} from 'lucide-vue-next'
import Discussions from '@/components/Discussions.vue' import Discussions from '@/components/Discussions.vue'
import { getEditorTools, enablePlyr } from '@/utils' import { getEditorTools, updateDocumentTitle } from '../utils'
import { sessionStore } from '@/stores/session'
import EditorJS from '@editorjs/editorjs' import EditorJS from '@editorjs/editorjs'
import LessonContent from '@/components/LessonContent.vue' import LessonContent from '@/components/LessonContent.vue'
import CourseInstructors from '@/components/CourseInstructors.vue' import CourseInstructors from '@/components/CourseInstructors.vue'
@@ -309,12 +214,7 @@ 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 lessonProgress = ref(0)
const lessonContainer = ref(null)
const zenModeEnabled = ref(false)
const hasQuiz = ref(false)
const discussionsContainer = ref(null)
const timer = ref(0) const timer = ref(0)
const { brand } = sessionStore()
let timerInterval let timerInterval
const props = defineProps({ const props = defineProps({
@@ -334,28 +234,11 @@ const props = defineProps({
onMounted(() => { onMounted(() => {
startTimer() startTimer()
enablePlyr()
document.addEventListener('fullscreenchange', attachFullscreenEvent)
})
const attachFullscreenEvent = () => {
if (document.fullscreenElement) {
zenModeEnabled.value = true
allowDiscussions.value = false
} else {
zenModeEnabled.value = false
if (!hasQuiz.value) {
allowDiscussions.value = true
}
}
}
onBeforeUnmount(() => {
document.removeEventListener('fullscreenchange', attachFullscreenEvent)
}) })
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],
makeParams(values) { makeParams(values) {
return { return {
course: props.courseName, course: props.courseName,
@@ -364,9 +247,7 @@ const lesson = createResource({
} }
}, },
auto: true, auto: true,
}) onSuccess(data) {
const setupLesson = (data) => {
if (Object.keys(data).length === 0) { if (Object.keys(data).length === 0) {
router.push({ router.push({
name: 'CourseDetail', name: 'CourseDetail',
@@ -390,10 +271,11 @@ const setupLesson = (data) => {
if (!editor.value && data.body) { if (!editor.value && data.body) {
const quizRegex = /\{\{ Quiz\(".*"\) \}\}/ const quizRegex = /\{\{ Quiz\(".*"\) \}\}/
hasQuiz.value = quizRegex.test(data.body) const hasQuiz = quizRegex.test(data.body)
if (!hasQuiz.value && !zenModeEnabled) allowDiscussions.value = true if (!hasQuiz) allowDiscussions.value = true
} }
} },
})
const renderEditor = (holder, content) => { const renderEditor = (holder, content) => {
// empty the holder // empty the holder
@@ -464,18 +346,10 @@ watch(
clearInterval(timerInterval) clearInterval(timerInterval)
timer.value = 0 timer.value = 0
startTimer() startTimer()
enablePlyr()
} }
} }
) )
watch(
() => lesson.data,
(data) => {
setupLesson(data)
}
)
const startTimer = () => { const startTimer = () => {
timerInterval = setInterval(() => { timerInterval = setInterval(() => {
timer.value++ timer.value++
@@ -491,13 +365,13 @@ onBeforeUnmount(() => {
}) })
const checkIfDiscussionsAllowed = () => { const checkIfDiscussionsAllowed = () => {
let quizPresent = false
JSON.parse(lesson.data?.content)?.blocks?.forEach((block) => { JSON.parse(lesson.data?.content)?.blocks?.forEach((block) => {
if (block.type === 'quiz') hasQuiz.value = true if (block.type === 'quiz') quizPresent = true
}) })
if ( if (
!hasQuiz.value && !quizPresent &&
!zenModeEnabled.value &&
(lesson.data?.membership || (lesson.data?.membership ||
user.data?.is_moderator || user.data?.is_moderator ||
user.data?.is_instructor) user.data?.is_instructor)
@@ -506,7 +380,6 @@ const checkIfDiscussionsAllowed = () => {
} }
const allowEdit = () => { const allowEdit = () => {
if (window.read_only_mode) return false
if (user.data?.is_moderator) return true if (user.data?.is_moderator) return true
if (lesson.data?.instructors?.includes(user.data?.name)) return true if (lesson.data?.instructors?.includes(user.data?.name)) return true
return false return false
@@ -542,58 +415,18 @@ const enrollStudent = () => {
) )
} }
const canGoZen = () => {
if (
user.data?.is_moderator ||
user.data?.is_instructor ||
user.data?.is_evaluator
)
return false
if (lesson.data?.membership) return true
return false
}
const goFullScreen = () => {
if (lessonContainer.value.requestFullscreen) {
lessonContainer.value.requestFullscreen()
} else if (lessonContainer.value.mozRequestFullScreen) {
lessonContainer.value.mozRequestFullScreen()
} else if (lessonContainer.value.webkitRequestFullscreen) {
lessonContainer.value.webkitRequestFullscreen()
} else if (lessonContainer.value.msRequestFullscreen) {
lessonContainer.value.msRequestFullscreen()
}
}
const showDiscussionsInZenMode = () => {
if (allowDiscussions.value) {
allowDiscussions.value = false
} else {
allowDiscussions.value = true
scrollDiscussionsIntoView()
}
}
const scrollDiscussionsIntoView = () => {
nextTick(() => {
discussionsContainer.value?.scrollIntoView({
behavior: 'smooth',
block: 'center',
inline: 'nearest',
})
})
}
const redirectToLogin = () => { const redirectToLogin = () => {
window.location.href = `/login?redirect-to=/lms/courses/${props.courseName}` window.location.href = `/login?redirect-to=/lms/courses/${props.courseName}`
} }
usePageMeta(() => { const pageMeta = computed(() => {
return { return {
title: lesson?.data?.title, title: lesson.data?.title,
icon: brand.favicon, description: lesson.data?.course,
} }
}) })
updateDocumentTitle(pageMeta)
</script> </script>
<style> <style>
.avatar-group { .avatar-group {
@@ -757,30 +590,4 @@ usePageMeta(() => {
.tc-table { .tc-table {
border-left: 1px solid #e8e8eb; border-left: 1px solid #e8e8eb;
} }
.plyr__volume input[type='range'] {
display: none;
}
.plyr__control--overlaid {
background: radial-gradient(
circle,
rgba(0, 0, 0, 0.4) 0%,
rgba(0, 0, 0, 0.5) 50%
);
}
.plyr__control:hover {
background: none;
}
.plyr--video {
border: 1px solid theme('colors.gray.200');
border-radius: 8px;
}
:root {
--plyr-range-fill-background: white;
--plyr-video-control-background-hover: transparent;
}
</style> </style>

View File

@@ -78,13 +78,7 @@
</div> </div>
</template> </template>
<script setup> <script setup>
import { import { Breadcrumbs, Button, createResource, FormControl } from 'frappe-ui'
Breadcrumbs,
Button,
createResource,
FormControl,
usePageMeta,
} from 'frappe-ui'
import { import {
computed, computed,
reactive, reactive,
@@ -93,15 +87,13 @@ import {
ref, ref,
onBeforeUnmount, onBeforeUnmount,
} from 'vue' } from 'vue'
import { sessionStore } from '../stores/session'
import EditorJS from '@editorjs/editorjs' import EditorJS from '@editorjs/editorjs'
import LessonHelp from '@/components/LessonHelp.vue' import LessonHelp from '@/components/LessonHelp.vue'
import { ChevronRight } from 'lucide-vue-next' import { ChevronRight } from 'lucide-vue-next'
import { createToast, getEditorTools, enablePlyr } from '@/utils' import { updateDocumentTitle, createToast, getEditorTools } from '@/utils'
import { capture } from '@/telemetry' import { capture } from '@/telemetry'
import { useOnboarding } from 'frappe-ui/frappe' import { useOnboarding } from 'frappe-ui/frappe'
const { brand } = sessionStore()
const editor = ref(null) const editor = ref(null)
const instructorEditor = ref(null) const instructorEditor = ref(null)
const user = inject('$user') const user = inject('$user')
@@ -133,7 +125,6 @@ onMounted(() => {
editor.value = renderEditor('content') editor.value = renderEditor('content')
instructorEditor.value = renderEditor('instructor-notes') instructorEditor.value = renderEditor('instructor-notes')
window.addEventListener('keydown', keyboardShortcut) window.addEventListener('keydown', keyboardShortcut)
enablePlyr()
}) })
const renderEditor = (holder) => { const renderEditor = (holder) => {
@@ -142,9 +133,6 @@ const renderEditor = (holder) => {
tools: getEditorTools(true), tools: getEditorTools(true),
autofocus: true, autofocus: true,
defaultBlock: 'markdown', defaultBlock: 'markdown',
onChange: async (api, event) => {
enablePlyr()
},
}) })
} }
@@ -406,10 +394,8 @@ const createNewLesson = () => {
{ lesson: data.name }, { lesson: data.name },
{ {
onSuccess() { onSuccess() {
if (user.data?.is_system_manager)
updateOnboardingStep('create_first_lesson')
capture('lesson_created') capture('lesson_created')
updateOnboardingStep('create_first_lesson')
showToast('Success', 'Lesson created successfully', 'check') showToast('Success', 'Lesson created successfully', 'check')
lessonDetails.reload() lessonDetails.reload()
}, },
@@ -506,14 +492,14 @@ const breadcrumbs = computed(() => {
return crumbs return crumbs
}) })
usePageMeta(() => { const pageMeta = computed(() => {
return { return {
title: lessonDetails?.data?.lesson title: 'Lesson Editor',
? lessonDetails.data.lesson.title description: 'Create and edit lessons for your course',
: 'New Lesson',
icon: brand.favicon,
} }
}) })
updateDocumentTitle(pageMeta)
</script> </script>
<style> <style>
.embed-tool__caption, .embed-tool__caption,
@@ -628,7 +614,8 @@ usePageMeta(() => {
} }
iframe { iframe {
border: none !important; border-top: 3px solid theme('colors.gray.700');
border-bottom: 3px solid theme('colors.gray.700');
} }
.tc-table { .tc-table {
@@ -642,30 +629,4 @@ iframe {
.ce-popover-item[data-item-name='markdown'] { .ce-popover-item[data-item-name='markdown'] {
display: none !important; display: none !important;
} }
.plyr__volume input[type='range'] {
display: none;
}
.plyr__control--overlaid {
background: radial-gradient(
circle,
rgba(0, 0, 0, 0.4) 0%,
rgba(0, 0, 0, 0.5) 50%
);
}
.plyr__control:hover {
background: none;
}
.plyr--video {
border: 1px solid theme('colors.gray.200');
border-radius: 8px;
}
:root {
--plyr-range-fill-background: white;
--plyr-video-control-background-hover: transparent;
}
</style> </style>

View File

@@ -65,14 +65,12 @@ import {
TabButtons, TabButtons,
Button, Button,
Tooltip, Tooltip,
usePageMeta,
} from 'frappe-ui' } from 'frappe-ui'
import { sessionStore } from '../stores/session'
import { computed, inject, ref, onMounted } from 'vue' import { computed, inject, ref, onMounted } from 'vue'
import { useRouter } from 'vue-router' import { useRouter } from 'vue-router'
import { X } from 'lucide-vue-next' import { X } from 'lucide-vue-next'
import { updateDocumentTitle } from '@/utils'
const { brand } = sessionStore()
const user = inject('$user') const user = inject('$user')
const socket = inject('$socket') const socket = inject('$socket')
const activeTab = ref('Unread') const activeTab = ref('Unread')
@@ -147,12 +145,14 @@ const breadcrumbs = computed(() => {
return crumbs return crumbs
}) })
usePageMeta(() => { const pageMeta = computed(() => {
return { return {
title: 'Notifications', title: 'Notifications',
icon: brand.favicon, description: 'All your notifications in one place.',
} }
}) })
updateDocumentTitle(pageMeta)
</script> </script>
<style> <style>
.notification strong { .notification strong {

View File

@@ -1,136 +0,0 @@
<template>
<div class="flex h-screen overflow-hidden sm:bg-gray-50">
<div class="relative h-full z-10 mx-auto pt-8 sm:w-max sm:pt-32">
<div class="mx-auto flex items-center justify-center space-x-2">
<LMSLogo class="size-7" />
<span
class="select-none text-xl font-semibold tracking-tight text-gray-900"
>
Learning
</span>
</div>
<div
class="mx-auto w-full h-fit bg-white py-8 sm:mt-6 sm:w-96 sm:rounded-lg sm:px-8 sm:shadow-xl"
>
<div class="font-medium text-center mb-8">
{{ __('Help us understand your needs') }}
</div>
<div class="mb-5">
<div class="text-sm text-gray-700 mb-2">
{{ __('What is your main use case for Frappe Learning?') }}
</div>
<FormControl
v-model="persona.useCase"
type="select"
:options="useCaseOptions"
/>
</div>
<div class="mb-5">
<div class="text-sm text-gray-700 mb-2">
{{ __('How many students are you planning to teach?') }}
</div>
<FormControl
v-model="persona.noOfStudents"
type="select"
:options="noOfStudentsOptions"
/>
</div>
<div class="flex w-full">
<Button variant="solid" class="mx-auto" @click="submitPersona()">
{{ __('Submit and Continue') }}
</Button>
</div>
</div>
<div
class="text-center absolute bottom-0 right-0 left-0 mx-auto cursor-pointer text-sm pb-4"
@click="skipPersonaForm()"
>
{{ __('Skip') }}
</div>
</div>
</div>
</template>
<script setup>
import LMSLogo from '@/components/Icons/LMSLogo.vue'
import { Button, call, FormControl, usePageMeta } from 'frappe-ui'
import { computed, inject, reactive } from 'vue'
import { useRouter } from 'vue-router'
import { sessionStore } from '@/stores/session'
const user = inject('$user')
const router = useRouter()
const { brand } = sessionStore()
const persona = reactive({
noOfStudents: null,
useCase: null,
})
const submitPersona = () => {
let responses = {
site: user.data?.sitename,
no_of_students: persona.noOfStudents,
use_case: persona.useCase,
}
call('lms.lms.api.capture_user_persona', {
responses: JSON.stringify(responses),
}).then(() => {
router.push({
name: 'Courses',
})
})
}
const skipPersonaForm = () => {
call('frappe.client.set_value', {
doctype: 'LMS Settings',
name: null,
fieldname: 'persona_captured',
value: 1,
}).then(() => {
router.push({
name: 'Courses',
})
})
}
const noOfStudentsOptions = computed(() => {
const options = [
'Less than 50',
'50-200',
'200-1000',
'1000+',
'Not sure yet',
]
return options.map((option) => ({
label: option,
value: option,
}))
})
const useCaseOptions = computed(() => {
const options = [
'Teaching students in a school/university',
'Training employees in my company',
'Onboarding and educating my users/community',
'Selling courses and earning income',
'Other',
]
return options.map((option) => ({
label: option,
value: option,
}))
})
usePageMeta(() => {
return {
title: 'Persona',
icon: brand.favicon,
}
})
</script>

View File

@@ -25,11 +25,7 @@
@select="(imageUrl) => coverImage.submit({ url: imageUrl })" @select="(imageUrl) => coverImage.submit({ url: imageUrl })"
> >
<template v-slot="{ togglePopover }"> <template v-slot="{ togglePopover }">
<Button <Button variant="outline" @click="togglePopover()">
v-if="!readOnlyMode"
variant="outline"
@click="togglePopover()"
>
<template #prefix> <template #prefix>
<Edit class="w-4 h-4 stroke-1.5 text-ink-gray-7" /> <Edit class="w-4 h-4 stroke-1.5 text-ink-gray-7" />
</template> </template>
@@ -62,7 +58,7 @@
</div> </div>
</div> </div>
<Button <Button
v-if="isSessionUser() && !readOnlyMode" v-if="isSessionUser()"
class="mt-3 sm:mt-0 md:ml-auto" class="mt-3 sm:mt-0 md:ml-auto"
@click="editProfile()" @click="editProfile()"
> >
@@ -90,30 +86,23 @@
/> />
</template> </template>
<script setup> <script setup>
import { import { Breadcrumbs, createResource, Button, TabButtons } from 'frappe-ui'
Breadcrumbs,
createResource,
Button,
TabButtons,
usePageMeta,
} from 'frappe-ui'
import { computed, inject, watch, ref, onMounted, watchEffect } from 'vue' import { computed, inject, watch, ref, onMounted, watchEffect } from 'vue'
import { sessionStore } from '@/stores/session' import { sessionStore } from '@/stores/session'
import { Edit } from 'lucide-vue-next' import { Edit } from 'lucide-vue-next'
import UserAvatar from '@/components/UserAvatar.vue' import UserAvatar from '@/components/UserAvatar.vue'
import { useRoute, useRouter } from 'vue-router' import { useRoute, useRouter } from 'vue-router'
import NoPermission from '@/components/NoPermission.vue' import NoPermission from '@/components/NoPermission.vue'
import { convertToTitleCase } from '@/utils' import { convertToTitleCase, updateDocumentTitle } from '@/utils'
import EditProfile from '@/components/Modals/EditProfile.vue' import EditProfile from '@/components/Modals/EditProfile.vue'
import EditCoverImage from '@/components/Modals/EditCoverImage.vue' import EditCoverImage from '@/components/Modals/EditCoverImage.vue'
const { user, brand } = sessionStore() const { user } = sessionStore()
const $user = inject('$user') const $user = inject('$user')
const route = useRoute() const route = useRoute()
const router = useRouter() const router = useRouter()
const activeTab = ref('') const activeTab = ref('')
const showProfileModal = ref(false) const showProfileModal = ref(false)
const readOnlyMode = window.read_only_mode
const props = defineProps({ const props = defineProps({
username: { username: {
@@ -226,10 +215,12 @@ const breadcrumbs = computed(() => {
return crumbs return crumbs
}) })
usePageMeta(() => { const pageMeta = computed(() => {
return { return {
title: profile.data?.full_name, title: profile.data?.full_name,
icon: brand.favicon, description: profile.data?.headline,
} }
}) })
updateDocumentTitle(pageMeta)
</script> </script>

View File

@@ -4,21 +4,7 @@
{{ __('My availability') }} {{ __('My availability') }}
</h2> </h2>
<div <div class="">
v-if="readOnlyMode"
class="flex items-center space-x-2 text-sm text-ink-gray-7 bg-surface-gray-1 px-3 py-2 rounded-md w-full text-center"
>
<CircleAlert class="size-4 stroke-1.5" />
<span>
{{
__(
'You cannot change the availability when the site is being updated.'
)
}}
</span>
</div>
<div v-else>
<div>
<div <div
class="grid grid-cols-3 md:grid-cols-4 gap-4 text-sm text-ink-gray-7 mb-4" class="grid grid-cols-3 md:grid-cols-4 gap-4 text-sm text-ink-gray-7 mb-4"
> >
@@ -138,16 +124,14 @@
</Button> </Button>
</div> </div>
</div> </div>
</div>
</template> </template>
<script setup> <script setup>
import { createResource, FormControl, Button, Badge } from 'frappe-ui' import { createResource, FormControl, Button } from 'frappe-ui'
import { computed, reactive, ref, onMounted, inject } from 'vue' import { computed, reactive, ref, onMounted, inject } from 'vue'
import { showToast, convertToTitleCase } from '@/utils' import { showToast, convertToTitleCase } from '@/utils'
import { Plus, X, Check, CircleAlert } from 'lucide-vue-next' import { Plus, X, Check } from 'lucide-vue-next'
const user = inject('$user') const user = inject('$user')
const readOnlyMode = window.read_only_mode
const props = defineProps({ const props = defineProps({
profile: { profile: {

View File

@@ -4,16 +4,6 @@
{{ __('Settings') }} {{ __('Settings') }}
</h2> </h2>
<div <div
v-if="readOnlyMode"
class="flex items-center space-x-2 text-sm text-ink-gray-7 bg-surface-gray-1 px-3 py-2 rounded-md w-full text-center"
>
<CircleAlert class="size-4 stroke-1.5" />
<span>
{{ __('You cannot change the roles in read-only mode.') }}
</span>
</div>
<div
v-else
class="flex flex-col md:flex-row gap-4 md:gap-0 justify-between w-3/4 mt-5" class="flex flex-col md:flex-row gap-4 md:gap-0 justify-between w-3/4 mt-5"
> >
<FormControl <FormControl
@@ -47,13 +37,11 @@
import { FormControl, createResource } from 'frappe-ui' import { FormControl, createResource } from 'frappe-ui'
import { ref } from 'vue' import { ref } from 'vue'
import { showToast, convertToTitleCase } from '@/utils' import { showToast, convertToTitleCase } from '@/utils'
import { CircleAlert } from 'lucide-vue-next'
const moderator = ref(false) const moderator = ref(false)
const course_creator = ref(false) const course_creator = ref(false)
const batch_evaluator = ref(false) const batch_evaluator = ref(false)
const lms_student = ref(false) const lms_student = ref(false)
const readOnlyMode = window.read_only_mode
const props = defineProps({ const props = defineProps({
profile: { profile: {

View File

@@ -13,7 +13,7 @@
<!-- Courses --> <!-- Courses -->
<div> <div>
<div class="flex items-center justify-between mb-2"> <div class="flex items-center justify-between mb-2">
<div class="text-lg text-ink-gray-9 font-semibold"> <div class="text-lg font-semibold">
{{ __('Program Courses') }} {{ __('Program Courses') }}
</div> </div>
<Button <Button
@@ -75,7 +75,7 @@
<!-- Members --> <!-- Members -->
<div> <div>
<div class="flex items-center justify-between mb-2"> <div class="flex items-center justify-between mb-2">
<div class="text-lg text-ink-gray-9 font-semibold"> <div class="text-lg font-semibold">
{{ __('Program Members') }} {{ __('Program Members') }}
</div> </div>
<Button <Button
@@ -186,17 +186,14 @@ import {
ListHeader, ListHeader,
ListHeaderItem, ListHeaderItem,
ListSelectBanner, ListSelectBanner,
usePageMeta,
} from 'frappe-ui' } from 'frappe-ui'
import { computed, ref } from 'vue' import { computed, ref } from 'vue'
import { Plus, Trash2 } from 'lucide-vue-next' import { Plus, Trash2 } from 'lucide-vue-next'
import { showToast } from '@/utils/'
import { useRouter } from 'vue-router'
import { sessionStore } from '../stores/session'
import Draggable from 'vuedraggable'
import Link from '@/components/Controls/Link.vue' import Link from '@/components/Controls/Link.vue'
import { showToast } from '@/utils/'
import Draggable from 'vuedraggable'
import { useRouter } from 'vue-router'
const { brand } = sessionStore()
const showDialog = ref(false) const showDialog = ref(false)
const currentForm = ref(null) const currentForm = ref(null)
const course = ref(null) const course = ref(null)
@@ -367,11 +364,4 @@ const breadbrumbs = computed(() => {
}, },
] ]
}) })
usePageMeta(() => {
return {
title: program.doc?.title,
icon: brand.favicon,
}
})
</script> </script>

View File

@@ -4,7 +4,7 @@
> >
<Breadcrumbs :items="breadbrumbs" /> <Breadcrumbs :items="breadbrumbs" />
<Button <Button
v-if="canCreateProgram()" v-if="user.data?.is_moderator || user.data?.is_instructor"
@click="showDialog = true" @click="showDialog = true"
variant="solid" variant="solid"
> >
@@ -46,7 +46,7 @@
params: { programName: program.name }, params: { programName: program.name },
}" }"
> >
<Button v-if="!readOnlyMode"> <Button>
<template #prefix> <template #prefix>
<Edit class="h-4 w-4 stroke-1.5" /> <Edit class="h-4 w-4 stroke-1.5" />
</template> </template>
@@ -126,23 +126,19 @@ import {
createResource, createResource,
Dialog, Dialog,
FormControl, FormControl,
usePageMeta,
} from 'frappe-ui' } from 'frappe-ui'
import { computed, inject, onMounted, ref } from 'vue' import { computed, inject, onMounted, ref } from 'vue'
import { BookOpen, Edit, Plus, LockKeyhole } from 'lucide-vue-next' import { BookOpen, Edit, Plus, LockKeyhole } from 'lucide-vue-next'
import CourseCard from '@/components/CourseCard.vue' import CourseCard from '@/components/CourseCard.vue'
import { useRouter } from 'vue-router' import { useRouter } from 'vue-router'
import { sessionStore } from '../stores/session'
import { showToast } from '@/utils' import { showToast } from '@/utils'
import { useSettings } from '@/stores/settings' import { useSettings } from '@/stores/settings'
const { brand } = sessionStore()
const user = inject('$user') const user = inject('$user')
const showDialog = ref(false) const showDialog = ref(false)
const router = useRouter() const router = useRouter()
const title = ref('') const title = ref('')
const settings = useSettings() const settings = useSettings()
const readOnlyMode = window.read_only_mode
onMounted(() => { onMounted(() => {
if ( if (
@@ -209,22 +205,9 @@ const lockCourse = (course) => {
return true return true
} }
const canCreateProgram = () => {
if (readOnlyMode) return false
if (user.data?.is_moderator || user.data?.is_instructor) return true
return false
}
const breadbrumbs = computed(() => [ const breadbrumbs = computed(() => [
{ {
label: __('Programs'), label: 'Programs',
}, },
]) ])
usePageMeta(() => {
return {
title: __('Programs'),
icon: brand.favicon,
}
})
</script> </script>

View File

@@ -3,7 +3,7 @@
class="sticky top-0 z-10 flex items-center justify-between border-b bg-surface-white px-3 py-2.5 sm:px-5" class="sticky top-0 z-10 flex items-center justify-between border-b bg-surface-white px-3 py-2.5 sm:px-5"
> >
<Breadcrumbs :items="breadcrumbs" /> <Breadcrumbs :items="breadcrumbs" />
<div v-if="!readOnlyMode" class="space-x-2"> <div class="space-x-2">
<router-link <router-link
v-if="quizDetails.data?.name" v-if="quizDetails.data?.name"
:to="{ :to="{
@@ -38,7 +38,7 @@
<div class="w-3/4 mx-auto py-5"> <div class="w-3/4 mx-auto py-5">
<!-- Details --> <!-- Details -->
<div class="mb-8"> <div class="mb-8">
<div class="font-semibold text-ink-gray-9 mb-4"> <div class="font-semibold mb-4">
{{ __('Details') }} {{ __('Details') }}
</div> </div>
<FormControl <FormControl
@@ -75,7 +75,7 @@
<!-- Settings --> <!-- Settings -->
<div class="mb-8"> <div class="mb-8">
<div class="font-semibold text-ink-gray-9 mb-4"> <div class="font-semibold mb-4">
{{ __('Settings') }} {{ __('Settings') }}
</div> </div>
<div class="grid grid-cols-3 gap-5 my-4"> <div class="grid grid-cols-3 gap-5 my-4">
@@ -93,7 +93,7 @@
</div> </div>
<div class="mb-8"> <div class="mb-8">
<div class="font-semibold text-ink-gray-9 mb-4"> <div class="font-semibold mb-4">
{{ __('Shuffle Settings') }} {{ __('Shuffle Settings') }}
</div> </div>
<div class="grid grid-cols-3"> <div class="grid grid-cols-3">
@@ -113,10 +113,10 @@
<!-- Questions --> <!-- Questions -->
<div> <div>
<div class="flex items-center justify-between mb-4"> <div class="flex items-center justify-between mb-4">
<div class="font-semibold text-ink-gray-9"> <div class="font-semibold">
{{ __('Questions') }} {{ __('Questions') }}
</div> </div>
<Button v-if="!readOnlyMode" @click="openQuestionModal()"> <Button @click="openQuestionModal()">
<template #prefix> <template #prefix>
<Plus class="w-4 h-4" /> <Plus class="w-4 h-4" />
</template> </template>
@@ -197,7 +197,6 @@ import {
ListRowItem, ListRowItem,
ListSelectBanner, ListSelectBanner,
Button, Button,
usePageMeta,
} from 'frappe-ui' } from 'frappe-ui'
import { import {
computed, computed,
@@ -208,13 +207,11 @@ import {
onBeforeUnmount, onBeforeUnmount,
watch, watch,
} from 'vue' } from 'vue'
import { sessionStore } from '../stores/session'
import { Plus, Trash2 } from 'lucide-vue-next' import { Plus, Trash2 } from 'lucide-vue-next'
import Question from '@/components/Modals/Question.vue'
import { showToast, updateDocumentTitle } from '@/utils' import { showToast, updateDocumentTitle } from '@/utils'
import { useRouter } from 'vue-router' import { useRouter } from 'vue-router'
import Question from '@/components/Modals/Question.vue'
const { brand } = sessionStore()
const showQuestionModal = ref(false) const showQuestionModal = ref(false)
const currentQuestion = reactive({ const currentQuestion = reactive({
question: '', question: '',
@@ -223,7 +220,6 @@ const currentQuestion = reactive({
}) })
const user = inject('$user') const user = inject('$user')
const router = useRouter() const router = useRouter()
const readOnlyMode = window.read_only_mode
const props = defineProps({ const props = defineProps({
quizID: { quizID: {
@@ -445,7 +441,11 @@ const breadcrumbs = computed(() => {
}, },
}, },
] ]
/* if (quizDetails.data) {
crumbs.push({
label: quiz.title,
})
} */
crumbs.push({ crumbs.push({
label: props.quizID == 'new' ? __('New Quiz') : quizDetails.data?.title, label: props.quizID == 'new' ? __('New Quiz') : quizDetails.data?.title,
route: { name: 'QuizForm', params: { quizID: props.quizID } }, route: { name: 'QuizForm', params: { quizID: props.quizID } },
@@ -453,10 +453,12 @@ const breadcrumbs = computed(() => {
return crumbs return crumbs
}) })
usePageMeta(() => { const pageMeta = computed(() => {
return { return {
title: props.quizID == 'new' ? __('New Quiz') : quizDetails.data?.title, title: props.quizID == 'new' ? __('New Quiz') : quizDetails.data?.title,
icon: brand.favicon, description: __('Form to create and edit quizzes'),
} }
}) })
updateDocumentTitle(pageMeta)
</script> </script>

View File

@@ -14,12 +14,11 @@
</template> </template>
<script setup> <script setup>
import Quiz from '@/components/Quiz.vue' import Quiz from '@/components/Quiz.vue'
import { createResource, Breadcrumbs, usePageMeta } from 'frappe-ui' import { createResource, Breadcrumbs } from 'frappe-ui'
import { computed, inject, onMounted, ref } from 'vue' import { computed, inject, onMounted, ref } from 'vue'
import { useRouter } from 'vue-router' import { useRouter } from 'vue-router'
import { sessionStore } from '../stores/session' import { updateDocumentTitle } from '@/utils'
const { brand } = sessionStore()
const user = inject('$user') const user = inject('$user')
const router = useRouter() const router = useRouter()
const fromLesson = ref(false) const fromLesson = ref(false)
@@ -57,10 +56,12 @@ const breadcrumbs = computed(() => {
return [{ label: __('Quiz Submission') }, { label: title.data?.title }] return [{ label: __('Quiz Submission') }, { label: title.data?.title }]
}) })
usePageMeta(() => { const pageMeta = computed(() => {
return { return {
title: `${title.data?.title}`, title: title.data?.title,
icon: brand.favicon, description: __('Quiz Submission'),
} }
}) })
updateDocumentTitle(pageMeta)
</script> </script>

View File

@@ -79,14 +79,11 @@ import {
FormControl, FormControl,
Button, Button,
Badge, Badge,
usePageMeta,
} from 'frappe-ui' } from 'frappe-ui'
import { computed, onBeforeUnmount, onMounted, inject } from 'vue' import { computed, onBeforeUnmount, onMounted, inject } from 'vue'
import { useRouter } from 'vue-router' import { useRouter } from 'vue-router'
import { showToast } from '@/utils' import { showToast } from '@/utils'
import { sessionStore } from '@/stores/session'
const { brand } = sessionStore()
const router = useRouter() const router = useRouter()
const user = inject('$user') const user = inject('$user')
@@ -152,11 +149,4 @@ const saveSubmission = () => {
} }
) )
} }
usePageMeta(() => {
return {
title: `${submisisonDetails.doc.quiz_title}`,
icon: brand.favicon,
}
})
</script> </script>

View File

@@ -40,18 +40,6 @@
</Button> </Button>
</div> </div>
</div> </div>
<div
v-else
class="text-center p-5 text-ink-gray-5 mt-52 w-3/4 md:w-1/2 mx-auto space-y-2"
>
<BookOpen class="size-10 mx-auto stroke-1 text-ink-gray-4" />
<div class="text-xl font-medium">
{{ __('No submissions') }}
</div>
<div class="leading-5">
{{ __('No quiz submissions found. Please check again later.') }}
</div>
</div>
</template> </template>
<script setup> <script setup>
import { import {
@@ -63,14 +51,10 @@ import {
ListRows, ListRows,
ListHeader, ListHeader,
ListHeaderItem, ListHeaderItem,
usePageMeta,
} from 'frappe-ui' } from 'frappe-ui'
import { BookOpen } from 'lucide-vue-next'
import { computed, onMounted, inject } from 'vue' import { computed, onMounted, inject } from 'vue'
import { sessionStore } from '../stores/session'
import { useRouter } from 'vue-router' import { useRouter } from 'vue-router'
const { brand } = sessionStore()
const router = useRouter() const router = useRouter()
const user = inject('$user') const user = inject('$user')
@@ -121,11 +105,4 @@ const quizColumns = computed(() => {
const breadcrumbs = computed(() => { const breadcrumbs = computed(() => {
return [{ label: __('Quiz Submissions') }] return [{ label: __('Quiz Submissions') }]
}) })
usePageMeta(() => {
return {
title: __('Quiz Submissions'),
icon: brand.favicon,
}
})
</script> </script>

View File

@@ -4,7 +4,6 @@
> >
<Breadcrumbs :items="breadcrumbs" /> <Breadcrumbs :items="breadcrumbs" />
<router-link <router-link
v-if="!readOnlyMode"
:to="{ :to="{
name: 'QuizForm', name: 'QuizForm',
params: { params: {
@@ -80,17 +79,14 @@ import {
ListRow, ListRow,
ListHeader, ListHeader,
ListHeaderItem, ListHeaderItem,
usePageMeta,
} from 'frappe-ui' } from 'frappe-ui'
import { useRouter } from 'vue-router' import { useRouter } from 'vue-router'
import { computed, inject, onMounted } from 'vue' import { computed, inject, onMounted } from 'vue'
import { BookOpen, Plus } from 'lucide-vue-next' import { BookOpen, Plus } from 'lucide-vue-next'
import { sessionStore } from '@/stores/session' import { updateDocumentTitle } from '@/utils'
const { brand } = sessionStore()
const user = inject('$user') const user = inject('$user')
const router = useRouter() const router = useRouter()
const readOnlyMode = window.read_only_mode
onMounted(() => { onMounted(() => {
if (!user.data?.is_moderator && !user.data?.is_instructor) { if (!user.data?.is_moderator && !user.data?.is_instructor) {
@@ -147,10 +143,12 @@ const breadcrumbs = computed(() => {
] ]
}) })
usePageMeta(() => { const pageMeta = computed(() => {
return { return {
title: __('Quizzes'), title: __('Quizzes'),
icon: brand.favicon, description: __('List of quizzes'),
} }
}) })
updateDocumentTitle(pageMeta)
</script> </script>

View File

@@ -39,13 +39,11 @@ import {
createDocumentResource, createDocumentResource,
createListResource, createListResource,
createResource, createResource,
usePageMeta,
} from 'frappe-ui' } from 'frappe-ui'
import { computed, inject, onBeforeMount, ref } from 'vue' import { computed, inject, onBeforeMount, ref } from 'vue'
import { useSidebar } from '@/stores/sidebar' import { useSidebar } from '@/stores/sidebar'
import { sessionStore } from '../stores/session' import { updateDocumentTitle } from '@/utils'
const { brand } = sessionStore()
const sidebarStore = useSidebar() const sidebarStore = useSidebar()
const user = inject('$user') const user = inject('$user')
const readyToRender = ref(false) const readyToRender = ref(false)
@@ -197,10 +195,14 @@ const breadcrumbs = computed(() => {
] ]
}) })
usePageMeta(() => { const pageMeta = computed(() => {
return { return {
title: chapter.doc?.title, title: chapter?.doc?.title,
icon: brand.favicon, description: __('This is a chapter in the course {0}').format(
chapter?.doc?.course_title
),
} }
}) })
updateDocumentTitle(pageMeta)
</script> </script>

View File

@@ -7,115 +7,109 @@
</header> </header>
<div v-if="chartDetails.data" class="p-5"> <div v-if="chartDetails.data" class="p-5">
<div class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-5 gap-4"> <div class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-5 gap-4">
<NumberChart <div
class="border rounded-md" class="flex items-center border py-2 px-3 rounded-md text-ink-gray-7"
:config="{ title: 'Courses', value: chartDetails.data.courses }" >
/> <div class="p-2 rounded-md bg-surface-gray-2 mr-3">
<NumberChart <BookOpen class="w-18 h-18 stroke-1.5" />
class="border rounded-md" </div>
:config="{ title: 'Signups', value: chartDetails.data.users }" <div>
/> <div class="text-xl font-semibold mb-1">
<NumberChart {{ formatNumber(chartDetails.data.courses) }}
class="border rounded-md" </div>
:config="{ <div>
title: 'Enrollments', {{ __('Courses') }}
value: chartDetails.data.enrollments, </div>
}" </div>
/> </div>
<NumberChart <div
class="border rounded-md" class="flex items-center border py-2 px-3 rounded-md text-ink-gray-7"
:config="{ >
title: 'Completions', <div class="p-2 rounded-md bg-surface-gray-2 mr-3">
value: chartDetails.data.completions, <LogIn class="w-18 h-18 stroke-1.5" />
}" </div>
/> <div>
<NumberChart <div class="text-xl font-semibold mb-1">
class="border rounded-md" {{ formatNumber(chartDetails.data.users) }}
:config="{ </div>
title: 'Certifications', <div>
value: chartDetails.data.certifications, {{ __('Signups') }}
}" </div>
/> </div>
</div>
<div
class="flex items-center border py-2 px-3 rounded-md text-ink-gray-7"
>
<div class="p-2 rounded-md bg-surface-gray-2 mr-3">
<BookOpenCheck class="w-18 h-18 stroke-1.5" />
</div>
<div>
<div class="text-xl font-semibold mb-1">
{{ formatNumber(chartDetails.data.enrollments) }}
</div>
<div>
{{ __('Enrollments') }}
</div>
</div>
</div>
<div
class="flex items-center border py-2 px-3 rounded-md text-ink-gray-7"
>
<div class="p-2 rounded-md bg-surface-gray-2 mr-3">
<FileCheck class="w-18 h-18 stroke-1.5" />
</div>
<div>
<div class="text-xl font-semibold mb-1">
{{ formatNumber(chartDetails.data.completions) }}
</div>
<div>
{{ __('Completions') }}
</div>
</div>
</div>
<div
class="flex items-center border py-2 px-3 rounded-md text-ink-gray-7"
>
<div class="p-2 rounded-md bg-surface-gray-2 mr-3">
<FileCheck2 class="w-18 h-18 stroke-1.5" />
</div>
<div>
<div class="text-xl font-semibold mb-1">
{{ formatNumber(chartDetails.data.lesson_completions) }}
</div>
<div class="text-ink-gray-7">
{{ __('Milestones') }}
</div>
</div>
</div>
</div> </div>
<div class="grid grid-cols-1 lg:grid-cols-2 gap-4 mt-4"> <div class="grid grid-cols-1 lg:grid-cols-2 gap-4 mt-4">
<div class="border rounded-md min-h-72"> <div class="border rounded-md p-5 min-h-72">
<AxisChart <Line
v-if="signupsChart.data" v-if="signupsChart.data"
:config="{ :data="signupsChart.data"
data: signupsChart.data, :options="signupChartOptions()"
title: 'Signups',
subtitle: 'Signups per month',
xAxis: {
key: 'date',
type: 'time',
title: 'Date',
timeGrain: 'day',
},
yAxis: {
title: 'Signups',
},
series: [{ name: 'signups', type: 'line', showDataPoints: true }],
}"
/> />
</div> </div>
<div class="border rounded-md min-h-72"> <div class="border rounded-md p-5 min-h-72">
<AxisChart <Line
v-if="enrollmentChart.data" v-if="enrollmentChart.data"
:config="{ :data="enrollmentChart.data"
data: enrollmentChart.data, :options="enrollmentChartOptions()"
title: 'Enrollments',
subtitle: 'Enrollments per month',
xAxis: {
key: 'date',
type: 'time',
title: 'Date',
timeGrain: 'day',
},
yAxis: {
title: 'Enrollments',
},
series: [
{ name: 'enrollments', type: 'line', showDataPoints: true },
],
}"
/> />
</div> </div>
<div class="border rounded-md"> <div class="border rounded-md p-5">
<AxisChart <Line
v-if="certification.data" v-if="lessonCompletion.data"
:config="{ :data="lessonCompletion.data"
data: certification.data, :options="lessonChartOptions()"
title: 'Certifications',
subtitle: 'Certifications per month',
xAxis: {
key: 'date',
type: 'time',
title: 'Date',
timeGrain: 'day',
},
yAxis: {
title: 'Certifications',
},
series: [
{
name: 'certifications',
type: 'line',
showDataPoints: true,
},
],
}"
/> />
</div> </div>
<div class="border rounded-md"> <div class="border rounded-md p-5">
<DonutChart <Pie
v-if="courseCompletion.data" v-if="courseCompletion.data"
:config="{ :data="courseCompletion.data"
data: courseCompletion.data, :options="courseChartOptions()"
title: 'Completions',
subtitle: 'Course Completion',
categoryColumn: 'label',
valueColumn: 'value',
}"
/> />
</div> </div>
</div> </div>
@@ -123,18 +117,44 @@
</div> </div>
</template> </template>
<script setup> <script setup>
import { createResource, Breadcrumbs } from 'frappe-ui'
import { computed, inject } from 'vue'
import { updateDocumentTitle } from '@/utils'
import { formatNumber } from '@/utils'
import { Line, Pie } from 'vue-chartjs'
import { import {
AxisChart, Chart as ChartJS,
Breadcrumbs, Title,
createResource, Tooltip,
DonutChart, Legend,
NumberChart, LineElement,
usePageMeta, CategoryScale,
} from 'frappe-ui' LinearScale,
import { computed } from 'vue' PointElement,
import { sessionStore } from '../stores/session' ArcElement,
Filler,
} from 'chart.js'
const { brand } = sessionStore() ChartJS.register(
Title,
Tooltip,
Legend,
LineElement,
CategoryScale,
LinearScale,
PointElement,
ArcElement,
Filler
)
import {
BookOpen,
LogIn,
FileCheck,
FileCheck2,
BookOpenCheck,
} from 'lucide-vue-next'
const dayjs = inject('$dayjs')
const breadcrumbs = computed(() => { const breadcrumbs = computed(() => {
return [ return [
@@ -155,18 +175,11 @@ const chartDetails = createResource({
const signupsChart = createResource({ const signupsChart = createResource({
url: 'lms.lms.utils.get_chart_data', url: 'lms.lms.utils.get_chart_data',
cache: ['signups'],
params: { params: {
chart_name: 'New Signups', chart_name: 'New Signups',
}, },
auto: true, auto: true,
transform(data) {
return data.map((item) => {
return {
date: new Date(item.date),
signups: item.count,
}
})
},
}) })
const enrollmentChart = createResource({ const enrollmentChart = createResource({
@@ -176,31 +189,15 @@ const enrollmentChart = createResource({
chart_name: 'Course Enrollments', chart_name: 'Course Enrollments',
}, },
auto: true, auto: true,
transform(data) {
return data.map((item) => {
return {
date: new Date(item.date),
enrollments: item.count,
}
})
},
}) })
const certification = createResource({ const lessonCompletion = createResource({
url: 'lms.lms.utils.get_chart_data', url: 'lms.lms.utils.get_chart_data',
cache: ['certifications'], cache: ['lessonCompletion'],
params: { params: {
chart_name: 'Certification', chart_name: 'Lesson Completion',
}, },
auto: true, auto: true,
transform(data) {
return data.map((item) => {
return {
date: new Date(item.date),
certifications: item.count,
}
})
},
}) })
const courseCompletion = createResource({ const courseCompletion = createResource({
@@ -209,10 +206,123 @@ const courseCompletion = createResource({
cache: ['courseCompletion'], cache: ['courseCompletion'],
}) })
usePageMeta(() => { const signupChartOptions = () => {
let options = chartOptions(false)
options.plugins.title.text = 'Signups'
options.borderColor = '#4563f0'
options.backgroundColor = (ctx) => {
const canvas = ctx.chart.ctx
const gradient = canvas.createLinearGradient(0, 0, 0, 160)
gradient.addColorStop(0, '#4563f0')
gradient.addColorStop(0.5, '#e8ecfe')
gradient.addColorStop(1, '#f6f7ff')
return gradient
}
return options
}
const enrollmentChartOptions = () => {
let options = chartOptions(false)
options.plugins.title.text = 'Enrollments'
options.borderColor = '#4563f0'
options.backgroundColor = (ctx) => {
const canvas = ctx.chart.ctx
const gradient = canvas.createLinearGradient(0, 0, 0, 160)
gradient.addColorStop(0, '#4563f0')
gradient.addColorStop(0.5, '#e8ecfe')
gradient.addColorStop(1, '#f6f7ff')
return gradient
}
return options
}
const lessonChartOptions = () => {
let options = chartOptions(false)
options.plugins.title.text = 'Milestones'
options.borderColor = '#4563f0'
options.backgroundColor = (ctx) => {
const canvas = ctx.chart.ctx
const gradient = canvas.createLinearGradient(0, 0, 0, 160)
gradient.addColorStop(0, '#B6DEC5')
gradient.addColorStop(0.5, '#e8ecfe')
gradient.addColorStop(1, '#f6f7ff')
return gradient
}
return options
}
const courseChartOptions = () => {
let options = chartOptions(true)
options.plugins.title.text = 'Completions'
options.backgroundColor = ['#4563f0', '#f683ae']
return options
}
const chartOptions = (isPie) => {
return { return {
title: __('Statistics'), responsive: true,
icon: brand.favicon, maintainAspectRatio: false,
fill: true,
borderWidth: 2,
pointRadius: 2,
pointStyle: 'cross',
ticks: {
autoSkip: true,
maxTicksLimit: 5,
},
plugins: {
legend: {
display: isPie ? true : false,
},
title: {
display: true,
align: 'start',
font: {
size: 14,
weight: '500',
},
color: '#171717',
padding: {
bottom: 20,
},
},
tooltip: {
backgroundColor: '#000',
},
},
scales: {
x: {
display: isPie ? false : true,
grid: {
display: false,
},
border: {
display: isPie ? false : true,
},
},
y: {
beginAtZero: true,
display: isPie ? false : true,
grid: {
display: false,
},
border: {
display: isPie ? false : true,
},
},
},
}
}
const pageMeta = computed(() => {
return {
title: 'Statistics',
description: 'Statistics of the platform',
} }
}) })
updateDocumentTitle(pageMeta)
</script> </script>

View File

@@ -134,8 +134,8 @@ const routes = [
}, },
{ {
path: '/job-opening/:jobName/edit', path: '/job-opening/:jobName/edit',
name: 'JobForm', name: 'JobCreation',
component: () => import('@/pages/JobForm.vue'), component: () => import('@/pages/JobCreation.vue'),
props: true, props: true,
}, },
{ {
@@ -199,6 +199,12 @@ const routes = [
name: 'Assignments', name: 'Assignments',
component: () => import('@/pages/Assignments.vue'), component: () => import('@/pages/Assignments.vue'),
}, },
{
path: '/assignments/:assignmentID',
name: 'AssignmentForm',
component: () => import('@/pages/AssignmentForm.vue'),
props: true,
},
{ {
path: '/assignment-submission/:assignmentID/:submissionName', path: '/assignment-submission/:assignmentID/:submissionName',
name: 'AssignmentSubmission', name: 'AssignmentSubmission',
@@ -210,11 +216,6 @@ const routes = [
name: 'AssignmentSubmissionList', name: 'AssignmentSubmissionList',
component: () => import('@/pages/AssignmentSubmissionList.vue'), component: () => import('@/pages/AssignmentSubmissionList.vue'),
}, },
{
path: '/persona',
name: 'PersonaForm',
component: () => import('@/pages/PersonaForm.vue'),
},
] ]
let router = createRouter({ let router = createRouter({

View File

@@ -2,11 +2,10 @@ import { defineStore } from 'pinia'
import { createResource } from 'frappe-ui' import { createResource } from 'frappe-ui'
import { usersStore } from './user' import { usersStore } from './user'
import router from '@/router' import router from '@/router'
import { computed, reactive, ref } from 'vue' import { ref, computed } from 'vue'
export const sessionStore = defineStore('lms-session', () => { export const sessionStore = defineStore('lms-session', () => {
let { userResource } = usersStore() let { userResource } = usersStore()
const brand = reactive({})
function sessionUser() { function sessionUser() {
let cookies = new URLSearchParams(document.cookie.split('; ').join('&')) let cookies = new URLSearchParams(document.cookie.split('; ').join('&'))
@@ -47,10 +46,7 @@ export const sessionStore = defineStore('lms-session', () => {
cache: 'brand', cache: 'brand',
auto: true, auto: true,
onSuccess(data) { onSuccess(data) {
brand.name = data.app_name document.querySelector("link[rel='icon']").href = data.favicon
brand.logo = data.app_logo
brand.favicon =
data.favicon?.file_url || '/assets/lms/frontend/learning.svg'
}, },
}) })
@@ -65,7 +61,6 @@ export const sessionStore = defineStore('lms-session', () => {
isLoggedIn, isLoggedIn,
login, login,
logout, logout,
brand,
branding, branding,
sidebarSettings, sidebarSettings,
} }

View File

@@ -9,9 +9,15 @@ export const useSettings = defineStore('settings', () => {
const activeTab = ref(null) const activeTab = ref(null)
const learningPaths = createResource({ const learningPaths = createResource({
url: 'lms.lms.api.is_learning_path_enabled', url: 'frappe.client.get_single_value',
auto: true, makeParams(values) {
cache: ['learningPath'], return {
doctype: 'LMS Settings',
field: 'enable_learning_paths',
}
},
auto: isLoggedIn ? true : false,
cache: ['learningPaths'],
}) })
const allowGuestAccess = createResource({ const allowGuestAccess = createResource({
@@ -20,6 +26,12 @@ export const useSettings = defineStore('settings', () => {
cache: ['allowGuestAccess'], cache: ['allowGuestAccess'],
}) })
/* const onboardingDetails = createResource({
url: 'lms.lms.utils.is_onboarding_complete',
auto: isLoggedIn ? true : false,
cache: ['onboardingDetails'],
}) */
return { return {
isSettingsOpen, isSettingsOpen,
activeTab, activeTab,

View File

@@ -14,11 +14,6 @@ import dayjs from '@/utils/dayjs'
import Embed from '@editorjs/embed' import Embed from '@editorjs/embed'
import SimpleImage from '@editorjs/simple-image' import SimpleImage from '@editorjs/simple-image'
import Table from '@editorjs/table' import Table from '@editorjs/table'
import { usersStore } from '../stores/user'
import Plyr from 'plyr'
import 'plyr/dist/plyr.css'
const readOnlyMode = window.read_only_mode
export function createToast(options) { export function createToast(options) {
toast({ toast({
@@ -113,7 +108,7 @@ export function showToast(title, text, icon, iconClasses = null) {
icon: icon, icon: icon,
iconClasses: iconClasses, iconClasses: iconClasses,
position: icon == 'check' ? 'bottom-right' : 'top-center', position: icon == 'check' ? 'bottom-right' : 'top-center',
timeout: icon != 'check' ? 10 : 5, timeout: 5,
}) })
} }
@@ -203,50 +198,78 @@ export function getEditorTools() {
services: { services: {
youtube: { youtube: {
regex: /(?:https?:\/\/)?(?:www\.)?(?:(?:youtu\.be\/)|(?:youtube\.com)\/(?:v\/|u\/\w\/|embed\/|watch))(?:(?:\?v=)?([^#&?=]*))?((?:[?&]\w*=\w*)*)/, regex: /(?:https?:\/\/)?(?:www\.)?(?:(?:youtu\.be\/)|(?:youtube\.com)\/(?:v\/|u\/\w\/|embed\/|watch))(?:(?:\?v=)?([^#&?=]*))?((?:[?&]\w*=\w*)*)/,
embedUrl: '<%= remote_id %>',
/* 'https://www.youtube.com/embed/<%= remote_id %>?origin=https://plyr.io&amp;iv_load_policy=3&amp;modestbranding=1&amp;playsinline=1&amp;showinfo=0&amp;rel=0&amp;enablejsapi=1' */
html: `<div class="video-player" data-plyr-provider="youtube"></div>`,
id: ([id]) => id,
},
vimeo: {
regex: /(?:http[s]?:\/\/)?(?:www\.)?vimeo\.com\/(\d+)/,
embedUrl: '<%= remote_id %>',
html: `<div class="video-player" data-plyr-provider="vimeo"></div>`,
id: ([id]) => id,
},
cloudflareStream: {
regex: /https:\/\/customer-[a-z0-9]+\.cloudflarestream\.com\/([a-f0-9]{32})\/watch/,
embedUrl: embedUrl:
'https://iframe.videodelivery.net/<%= remote_id %>', 'https://www.youtube.com/embed/<%= remote_id %>',
html: `<iframe style="width:100%; height: ${ html: '<iframe style="width:100%; height: 30rem;" frameborder="0" allowfullscreen></iframe>',
window.innerWidth < 640 ? '15rem' : '30rem' height: 320,
};" frameborder="0" allowfullscreen></iframe>`, 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,
codepen: true, codepen: true,
aparat: { aparat: {
regex: /(?:http[s]?:\/\/)?(?:www.)?aparat\.com\/v\/([^\/\?\&]+)\/?/, regex: /(?:http[s]?:\/\/)?(?:www.)?aparat\.com\/v\/([^\/\?\&]+)\/?/,
embedUrl: embedUrl:
'https://www.aparat.com/video/video/embed/videohash/<%= remote_id %>/vt/frame', 'https://www.aparat.com/video/video/embed/videohash/<%= remote_id %>/vt/frame',
html: `<iframe style="margin: 0 auto; width: 100%; height: ${ html: '<iframe style="margin: 0 auto; width: 100%; height: 25rem;" frameborder="0" scrolling="no" allowtransparency="true"></iframe>',
window.innerWidth < 640 ? '15rem' : '30rem' height: 300,
};" frameborder="0" scrolling="no" allowtransparency="true"></iframe>`, width: 600,
}, },
github: true, github: true,
slides: { slides: {
regex: /https:\/\/docs\.google\.com\/presentation\/d\/([A-Za-z0-9_-]+)\/pub/, regex: /https:\/\/docs\.google\.com\/presentation\/d\/([A-Za-z0-9_-]+)\/pub/,
embedUrl: embedUrl:
'https://docs.google.com/presentation/d/<%= remote_id %>/embed', 'https://docs.google.com/presentation/d/<%= remote_id %>/embed',
html: `<iframe style='width: 100%; height: ${ html: "<iframe style='width: 100%; height: 30rem; border: 1px solid #D3D3D3; border-radius: 12px; margin: 1rem 0' frameborder='0' allowfullscreen='true'></iframe>",
window.innerWidth < 640 ? '15rem' : '30rem'
}; border: 1px solid #D3D3D3; border-radius: 12px; margin: 1rem 0' frameborder='0' allowfullscreen='true'></iframe>`,
}, },
drive: { drive: {
regex: /https:\/\/drive\.google\.com\/file\/d\/([A-Za-z0-9_-]+)\/view(\?.+)?/, regex: /https:\/\/drive\.google\.com\/file\/d\/([A-Za-z0-9_-]+)\/view(\?.+)?/,
embedUrl: embedUrl:
'https://drive.google.com/file/d/<%= remote_id %>/preview', 'https://drive.google.com/file/d/<%= remote_id %>/preview',
html: `<iframe style='width: 100%; height: ${ html: "<iframe style='width: 100%; height: 25rem; border: 1px solid #D3D3D3; border-radius: 12px;' frameborder='0' allowfullscreen='true'></iframe>",
window.innerWidth < 640 ? '15rem' : '30rem'
}; border: 1px solid #D3D3D3; border-radius: 12px;' frameborder='0' allowfullscreen='true'></iframe>`,
}, },
docsPublic: { docsPublic: {
regex: /https:\/\/docs\.google\.com\/document\/d\/([A-Za-z0-9_-]+)\/edit(\?.+)?/, regex: /https:\/\/docs\.google\.com\/document\/d\/([A-Za-z0-9_-]+)\/edit(\?.+)?/,
@@ -455,7 +478,7 @@ export function getSidebarLinks() {
activeFor: ['Batches', 'BatchDetail', 'Batch', 'BatchForm'], activeFor: ['Batches', 'BatchDetail', 'Batch', 'BatchForm'],
}, },
{ {
label: 'Certified Members', label: 'Certified Participants',
icon: 'GraduationCap', icon: 'GraduationCap',
to: 'CertifiedParticipants', to: 'CertifiedParticipants',
activeFor: ['CertifiedParticipants'], activeFor: ['CertifiedParticipants'],
@@ -544,38 +567,3 @@ export const escapeHTML = (text) => {
(char) => escape_html_mapping[char] || char (char) => escape_html_mapping[char] || char
) )
} }
export const canCreateCourse = () => {
const { userResource } = usersStore()
return (
!readOnlyMode &&
(userResource.data?.is_instructor || userResource.data?.is_moderator)
)
}
export const enablePlyr = () => {
setTimeout(() => {
const videoElement = document.getElementsByClassName('video-player')
if (videoElement.length === 0) return
const src = videoElement[0].getAttribute('src')
if (src) {
let videoID = src.split('/').pop()
videoElement[0].setAttribute('data-plyr-embed-id', videoID)
}
new Plyr('.video-player', {
youtube: {
noCookie: true,
},
controls: [
'play-large',
'play',
'progress',
'current-time',
'mute',
'volume',
'fullscreen',
],
})
}, 500)
}

View File

@@ -25,7 +25,7 @@ export default defineConfig({
}), }),
], ],
server: { server: {
allowedHosts: ['fs', 'persona'], allowedHosts: ['fs', 'onb1'],
}, },
resolve: { resolve: {
alias: { alias: {
@@ -40,7 +40,6 @@ export default defineConfig({
'engine.io-client', 'engine.io-client',
'tailwind.config.js', 'tailwind.config.js',
'highlight.js', 'highlight.js',
'plyr',
], ],
}, },
}) })

File diff suppressed because it is too large Load Diff

View File

@@ -1 +1 @@
__version__ = "2.28.0" __version__ = "2.26.0"

View File

@@ -246,7 +246,7 @@ on_login = "lms.lms.user.on_login"
add_to_apps_screen = [ add_to_apps_screen = [
{ {
"name": "lms", "name": "lms",
"logo": "/assets/lms/frontend/learning.svg", "logo": "/assets/lms/images/lms-logo.png",
"title": "Learning", "title": "Learning",
"route": "/lms", "route": "/lms",
"has_permission": "lms.lms.api.check_app_permission", "has_permission": "lms.lms.api.check_app_permission",

View File

@@ -9,19 +9,18 @@
"field_order": [ "field_order": [
"job_title", "job_title",
"location", "location",
"country", "disabled",
"column_break_5", "column_break_5",
"type", "type",
"status", "status",
"disabled",
"section_break_6", "section_break_6",
"description",
"company_details_section",
"company_name", "company_name",
"company_website", "company_website",
"column_break_phkm", "column_break_11",
"company_logo", "company_logo",
"company_email_address", "company_email_address"
"company_details_section",
"description"
], ],
"fields": [ "fields": [
{ {
@@ -37,7 +36,7 @@
"fieldtype": "Data", "fieldtype": "Data",
"in_list_view": 1, "in_list_view": 1,
"in_standard_filter": 1, "in_standard_filter": 1,
"label": "City", "label": "Location",
"reqd": 1 "reqd": 1
}, },
{ {
@@ -63,8 +62,7 @@
}, },
{ {
"fieldname": "section_break_6", "fieldname": "section_break_6",
"fieldtype": "Section Break", "fieldtype": "Section Break"
"label": "Company Details"
}, },
{ {
"fieldname": "description", "fieldname": "description",
@@ -74,7 +72,8 @@
}, },
{ {
"fieldname": "company_details_section", "fieldname": "company_details_section",
"fieldtype": "Section Break" "fieldtype": "Section Break",
"label": "Company Details"
}, },
{ {
"fieldname": "company_name", "fieldname": "company_name",
@@ -90,6 +89,10 @@
"label": "Company Website", "label": "Company Website",
"reqd": 1 "reqd": 1
}, },
{
"fieldname": "column_break_11",
"fieldtype": "Column Break"
},
{ {
"fieldname": "company_logo", "fieldname": "company_logo",
"fieldtype": "Attach Image", "fieldtype": "Attach Image",
@@ -108,30 +111,13 @@
"label": "Company Email Address", "label": "Company Email Address",
"options": "Email", "options": "Email",
"reqd": 1 "reqd": 1
},
{
"fieldname": "column_break_phkm",
"fieldtype": "Column Break"
},
{
"fieldname": "country",
"fieldtype": "Link",
"label": "Country",
"options": "Country",
"reqd": 1
} }
], ],
"grid_page_length": 50,
"index_web_pages_for_search": 1, "index_web_pages_for_search": 1,
"links": [ "links": [],
{
"link_doctype": "LMS Job Application",
"link_fieldname": "job"
}
],
"make_attachments_public": 1, "make_attachments_public": 1,
"modified": "2025-04-24 14:34:35.920242", "modified": "2025-01-17 12:38:57.134919",
"modified_by": "sayali@frappe.io", "modified_by": "Administrator",
"module": "Job", "module": "Job",
"name": "Job Opportunity", "name": "Job Opportunity",
"owner": "Administrator", "owner": "Administrator",
@@ -171,7 +157,6 @@
"write": 1 "write": 1
} }
], ],
"row_format": "Dynamic",
"sort_field": "modified", "sort_field": "modified",
"sort_order": "DESC", "sort_order": "DESC",
"states": [], "states": [],

View File

@@ -19,8 +19,6 @@ from frappe.utils import (
format_date, format_date,
date_diff, date_diff,
) )
from frappe.query_builder import DocType
from pypika.functions import DistinctOptionFunction
from lms.lms.utils import get_average_rating, get_lesson_count from lms.lms.utils import get_average_rating, get_lesson_count
from xml.dom.minidom import parseString from xml.dom.minidom import parseString
from lms.lms.doctype.course_lesson.course_lesson import save_progress from lms.lms.doctype.course_lesson.course_lesson import save_progress
@@ -184,10 +182,9 @@ def get_user_info():
) )
user.is_fc_site = is_fc_site() user.is_fc_site = is_fc_site()
user.is_system_manager = "System Manager" in user.roles user.is_system_manager = "System Manager" in user.roles
user.sitename = frappe.local.site
user.developer_mode = frappe.conf.developer_mode
if user.is_fc_site and user.is_system_manager: if user.is_fc_site and user.is_system_manager:
user.site_info = current_site_info() user.site_info = current_site_info()
user.sitename = frappe.local.site
return user return user
@@ -240,11 +237,6 @@ def validate_billing_access(billing_type, name):
access = False access = False
message = _("Batch is sold out.") message = _("Batch is sold out.")
start_date = frappe.get_cached_value("LMS Batch", name, "start_date")
if start_date and date_diff(start_date, now()) < 0:
access = False
message = _("Batch has already started.")
elif access and billing_type == "certificate": elif access and billing_type == "certificate":
purchased_certificate = frappe.db.exists( purchased_certificate = frappe.db.exists(
"LMS Enrollment", "LMS Enrollment",
@@ -286,11 +278,9 @@ def get_job_details(job):
[ [
"job_title", "job_title",
"location", "location",
"country",
"type", "type",
"company_name", "company_name",
"company_logo", "company_logo",
"company_website",
"name", "name",
"creation", "creation",
"description", "description",
@@ -312,20 +302,14 @@ def get_job_opportunities(filters=None, orFilters=None):
fields=[ fields=[
"job_title", "job_title",
"location", "location",
"country",
"type", "type",
"company_name", "company_name",
"company_logo", "company_logo",
"name", "name",
"creation", "creation",
"description",
], ],
order_by="creation desc", order_by="creation desc",
) )
for job in jobs:
job.description = frappe.utils.strip_html_tags(job.description)
job.applicants = frappe.db.count("LMS Job Application", {"job": job.name})
return jobs return jobs
@@ -346,7 +330,7 @@ def get_chart_details():
details.completions = frappe.db.count( details.completions = frappe.db.count(
"LMS Enrollment", {"progress": ["like", "%100%"]} "LMS Enrollment", {"progress": ["like", "%100%"]}
) )
details.certifications = frappe.db.count("LMS Certificate", {"published": 1}) details.lesson_completions = frappe.db.count("LMS Course Progress")
return details return details
@@ -426,50 +410,29 @@ def get_certified_participants(filters=None, start=0, page_length=30):
or_filters["course_title"] = ["like", f"%{category}%"] or_filters["course_title"] = ["like", f"%{category}%"]
or_filters["batch_title"] = ["like", f"%{category}%"] or_filters["batch_title"] = ["like", f"%{category}%"]
participants = frappe.db.get_all( participants = frappe.get_all(
"LMS Certificate", "LMS Certificate",
filters=filters, filters=filters,
or_filters=or_filters, or_filters=or_filters,
fields=["member", "issue_date"], fields=["member"],
group_by="member", group_by="member",
order_by="issue_date desc", order_by="creation desc",
start=start, start=start,
page_length=page_length, page_length=page_length,
) )
for participant in participants: for participant in participants:
count = frappe.db.count("LMS Certificate", {"member": participant.member})
details = frappe.db.get_value( details = frappe.db.get_value(
"User", "User",
participant.member, participant.member,
["full_name", "user_image", "username", "country", "headline"], ["full_name", "user_image", "username", "country", "headline"],
as_dict=1, as_dict=1,
) )
details["certificate_count"] = count
participant.update(details) participant.update(details)
return participants return participants
class CountDistinct(DistinctOptionFunction):
def __init__(self, field):
super().__init__("COUNT", field, distinct=True)
@frappe.whitelist(allow_guest=True)
def get_count_of_certified_members():
Certificate = DocType("LMS Certificate")
query = (
frappe.qb.from_(Certificate)
.select(CountDistinct(Certificate.member).as_("total"))
.where(Certificate.published == 1)
)
result = query.run(as_dict=True)
return result[0]["total"] if result else 0
@frappe.whitelist(allow_guest=True) @frappe.whitelist(allow_guest=True)
def get_certification_categories(): def get_certification_categories():
categories = [] categories = []
@@ -1295,11 +1258,6 @@ def is_guest_allowed():
return frappe.get_cached_value("LMS Settings", None, "allow_guest_access") return frappe.get_cached_value("LMS Settings", None, "allow_guest_access")
@frappe.whitelist(allow_guest=True)
def is_learning_path_enabled():
return frappe.get_cached_value("LMS Settings", None, "enable_learning_paths")
@frappe.whitelist() @frappe.whitelist()
def cancel_evaluation(evaluation): def cancel_evaluation(evaluation):
evaluation = frappe._dict(evaluation) evaluation = frappe._dict(evaluation)
@@ -1402,17 +1360,3 @@ def add_an_evaluator(email):
evaluator.insert() evaluator.insert()
return evaluator return evaluator
@frappe.whitelist()
def capture_user_persona(responses):
frappe.only_for("System Manager")
data = frappe.parse_json(responses)
data = json.dumps(data)
response = frappe.integrations.utils.make_post_request(
"https://school.frappe.io/api/method/capture-persona",
data={"response": data},
)
if response.get("message").get("name"):
frappe.db.set_single_value("LMS Settings", "persona_captured", True)
return response

View File

@@ -1,31 +0,0 @@
{
"based_on": "issue_date",
"chart_name": "Certification",
"chart_type": "Count",
"creation": "2025-04-28 17:47:28.517149",
"docstatus": 0,
"doctype": "Dashboard Chart",
"document_type": "LMS Certificate",
"dynamic_filters_json": "[]",
"filters_json": "[[\"LMS Certificate\",\"published\",\"=\",1,false]]",
"group_by_type": "Count",
"idx": 0,
"is_public": 1,
"is_standard": 1,
"modified": "2025-04-28 17:47:28.517149",
"modified_by": "sayali@frappe.io",
"module": "LMS",
"name": "Certification",
"number_of_groups": 0,
"owner": "sayali@frappe.io",
"parent_document_type": "",
"roles": [],
"source": "",
"time_interval": "Daily",
"timeseries": 1,
"timespan": "Last Month",
"type": "Line",
"use_report_chart": 0,
"value_based_on": "",
"y_axis": []
}

View File

@@ -9,14 +9,14 @@
"doctype": "Dashboard Chart", "doctype": "Dashboard Chart",
"document_type": "User", "document_type": "User",
"dynamic_filters_json": "[]", "dynamic_filters_json": "[]",
"filters_json": "[[\"User\",\"enabled\",\"=\",1,false]]", "filters_json": "[]",
"group_by_type": "Count", "group_by_type": "Count",
"idx": 5, "idx": 1,
"is_public": 1, "is_public": 1,
"is_standard": 1, "is_standard": 1,
"last_synced_on": "2025-04-28 15:09:52.161688", "last_synced_on": "2022-10-20 10:46:56.849265",
"modified": "2025-04-28 17:47:58.168293", "modified": "2022-10-20 11:31:17.184897",
"modified_by": "sayali@frappe.io", "modified_by": "Administrator",
"module": "LMS", "module": "LMS",
"name": "New Signups", "name": "New Signups",
"number_of_groups": 0, "number_of_groups": 0,

View File

@@ -161,7 +161,7 @@
], ],
"index_web_pages_for_search": 1, "index_web_pages_for_search": 1,
"links": [], "links": [],
"modified": "2025-04-10 15:19:22.400932", "modified": "2024-11-14 13:46:56.838659",
"modified_by": "Administrator", "modified_by": "Administrator",
"module": "LMS", "module": "LMS",
"name": "Course Lesson", "name": "Course Lesson",
@@ -189,26 +189,12 @@
"print": 1, "print": 1,
"read": 1, "read": 1,
"report": 1, "report": 1,
"role": "Course Creator", "role": "LMS Student",
"select": 1,
"share": 1,
"write": 1
},
{
"create": 1,
"delete": 1,
"email": 1,
"export": 1,
"print": 1,
"read": 1,
"report": 1,
"role": "Moderator",
"select": 1, "select": 1,
"share": 1, "share": 1,
"write": 1 "write": 1
} }
], ],
"row_format": "Dynamic",
"sort_field": "modified", "sort_field": "modified",
"sort_order": "DESC", "sort_order": "DESC",
"states": [], "states": [],

View File

@@ -11,34 +11,66 @@ import json
class CourseLesson(Document): class CourseLesson(Document):
def on_update(self): def validate(self):
# self.check_and_create_folder()
self.validate_quiz_id() self.validate_quiz_id()
def validate_quiz_id(self): def validate_quiz_id(self):
if self.quiz_id and not frappe.db.exists("LMS Quiz", self.quiz_id): if self.quiz_id and not frappe.db.exists("LMS Quiz", self.quiz_id):
frappe.throw(_("Invalid Quiz ID")) frappe.throw(_("Invalid Quiz ID"))
if self.content: def on_update(self):
self.save_lesson_details_in_quiz(self.content) dynamic_documents = ["Exercise", "Quiz"]
for section in dynamic_documents:
self.update_lesson_name_in_document(section)
if self.instructor_content: def update_lesson_name_in_document(self, section):
self.save_lesson_details_in_quiz(self.instructor_content) doctype_map = {"Exercise": "LMS Exercise", "Quiz": "LMS Quiz"}
macros = find_macros(self.body)
documents = [value for name, value in macros if name == section]
index = 1
for name in documents:
e = frappe.get_doc(doctype_map[section], name)
e.lesson = self.name
e.index_ = index
e.course = self.course
e.save(ignore_permissions=True)
index += 1
self.update_orphan_documents(doctype_map[section], documents)
def save_lesson_details_in_quiz(self, content): def update_orphan_documents(self, doctype, documents):
content = json.loads(self.content) """Updates the documents that were previously part of this lesson,
for block in content.get("blocks"): but not any more.
if block.get("type") == "quiz": """
quiz = block.get("data").get("quiz") linked_documents = {
if not frappe.db.exists("LMS Quiz", quiz): row["name"] for row in frappe.get_all(doctype, {"lesson": self.name})
frappe.throw(_("Invalid Quiz ID in content")) }
frappe.db.set_value( active_documents = set(documents)
"LMS Quiz", orphan_documents = linked_documents - active_documents
quiz, for name in orphan_documents:
{ ex = frappe.get_doc(doctype, name)
"course": self.course, ex.lesson = None
"lesson": self.name, ex.course = None
}, ex.index_ = 0
) ex.save(ignore_permissions=True)
def check_and_create_folder(self):
args = {
"doctype": "File",
"is_folder": True,
"file_name": f"{self.name} {self.course}",
}
if not frappe.db.exists(args):
folder = frappe.get_doc(args)
folder.save(ignore_permissions=True)
def get_exercises(self):
if not self.body:
return []
macros = find_macros(self.body)
exercises = [value for name, value in macros if name == "Exercise"]
return [frappe.get_doc("LMS Exercise", name) for name in exercises]
@frappe.whitelist() @frappe.whitelist()
@@ -70,7 +102,7 @@ def save_progress(lesson, course):
progress = get_course_progress(course) progress = get_course_progress(course)
capture_progress_for_analytics(progress, course) capture_progress_for_analytics(progress, course)
# Had to get doc, as on_change doesn't trigger when you use set_value. The trigger is necessary for badge to get assigned. # 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.progress = progress
enrollment.save() enrollment.save()

View File

@@ -4,7 +4,8 @@
import frappe import frappe
from frappe import _ from frappe import _
from frappe.model.document import Document from frappe.model.document import Document
from frappe.utils import validate_url from frappe.utils import validate_url, validate_email_address
from frappe.email.doctype.email_template.email_template import get_email_template
from frappe.desk.doctype.notification_log.notification_log import make_notification_logs from frappe.desk.doctype.notification_log.notification_log import make_notification_logs
@@ -14,6 +15,14 @@ class LMSAssignmentSubmission(Document):
self.validate_url() self.validate_url()
self.validate_status() self.validate_status()
def after_insert(self):
if not frappe.flags.in_test:
outgoing_email_account = frappe.get_cached_value(
"Email Account", {"default_outgoing": 1, "enable_outgoing": 1}, "name"
)
if outgoing_email_account or frappe.conf.get("mail_login"):
self.send_mail()
def validate_duplicates(self): def validate_duplicates(self):
if frappe.db.exists( if frappe.db.exists(
"LMS Assignment Submission", "LMS Assignment Submission",
@@ -30,6 +39,38 @@ class LMSAssignmentSubmission(Document):
if self.type == "URL" and not validate_url(self.answer): if self.type == "URL" and not validate_url(self.answer):
frappe.throw(_("Please enter a valid URL.")) frappe.throw(_("Please enter a valid URL."))
def send_mail(self):
subject = _("New Assignment Submission")
template = "assignment_submission"
custom_template = frappe.db.get_single_value(
"LMS Settings", "assignment_submission_template"
)
args = {
"member_name": self.member_name,
"assignment_name": self.assignment,
"assignment_title": self.assignment_title,
"submission_name": self.name,
}
moderators = frappe.get_all("Has Role", {"role": "Moderator"}, pluck="parent")
for moderator in moderators:
if not validate_email_address(moderator):
moderators.remove(moderator)
if custom_template:
email_template = get_email_template(custom_template, args)
subject = email_template.get("subject")
content = email_template.get("message")
frappe.sendmail(
recipients=moderators,
subject=subject,
template=template if not custom_template else None,
content=content if custom_template else None,
args=args,
header=[subject, "green"],
)
def validate_status(self): def validate_status(self):
if not self.is_new(): if not self.is_new():
doc_before_save = self.get_doc_before_save() doc_before_save = self.get_doc_before_save()

View File

@@ -53,12 +53,7 @@ class LMSBatch(Document):
if self.paid_batch: if self.paid_batch:
installed_apps = frappe.get_installed_apps() installed_apps = frappe.get_installed_apps()
if "payments" not in installed_apps: if "payments" not in installed_apps:
documentation_link = "https://docs.frappe.io/learning/setting-up-payment-gateway" frappe.throw(_("Please install the Payments app to create a paid batches."))
frappe.throw(
_(
"Please install the Payments App to create a paid batch. Refer to the documentation for more details. {0}"
).format(documentation_link)
)
def validate_amount_and_currency(self): def validate_amount_and_currency(self):
if self.paid_batch and (not self.amount or not self.currency): if self.paid_batch and (not self.amount or not self.currency):

View File

@@ -50,12 +50,7 @@ class LMSCourse(Document):
if self.paid_course: if self.paid_course:
installed_apps = frappe.get_installed_apps() installed_apps = frappe.get_installed_apps()
if "payments" not in installed_apps: if "payments" not in installed_apps:
documentation_link = "https://docs.frappe.io/learning/setting-up-payment-gateway" frappe.throw(_("Please install the Payments app to create a paid courses."))
frappe.throw(
_(
"Please install the Payments App to create a paid course. Refer to the documentation for more details. {0}"
).format(documentation_link)
)
def validate_certification(self): def validate_certification(self):
if self.enable_certification and self.paid_certificate: if self.enable_certification and self.paid_certificate:

View File

@@ -91,7 +91,7 @@
"fetch_from": "member.username", "fetch_from": "member.username",
"fieldname": "member_username", "fieldname": "member_username",
"fieldtype": "Data", "fieldtype": "Data",
"label": "Member Username", "label": "Memeber Username",
"read_only": 1 "read_only": 1
}, },
{ {
@@ -145,11 +145,10 @@
"options": "LMS Certificate" "options": "LMS Certificate"
} }
], ],
"grid_page_length": 50,
"index_web_pages_for_search": 1, "index_web_pages_for_search": 1,
"links": [], "links": [],
"modified": "2025-04-25 10:06:25.824119", "modified": "2025-02-21 17:11:37.986157",
"modified_by": "sayali@frappe.io", "modified_by": "Administrator",
"module": "LMS", "module": "LMS",
"name": "LMS Enrollment", "name": "LMS Enrollment",
"owner": "Administrator", "owner": "Administrator",
@@ -193,7 +192,6 @@
} }
], ],
"quick_entry": 1, "quick_entry": 1,
"row_format": "Dynamic",
"show_title_field_in_link": 1, "show_title_field_in_link": 1,
"sort_field": "modified", "sort_field": "modified",
"sort_order": "DESC", "sort_order": "DESC",

Some files were not shown because too many files have changed in this diff Show More