Compare commits
106 Commits
learn-page
...
certificat
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
d7e1745c09 | ||
|
|
ef238c1b25 | ||
|
|
cb60d97bb7 | ||
|
|
f0ee8d7b88 | ||
|
|
7e5203f058 | ||
|
|
a3672e9d91 | ||
|
|
7017382451 | ||
|
|
6c9d49bf8c | ||
|
|
2de058246b | ||
|
|
798ea30382 | ||
|
|
3e2c6b3343 | ||
|
|
5ea744de5c | ||
|
|
aedb3d3d45 | ||
|
|
83a2f42df9 | ||
|
|
66aace247c | ||
|
|
bc3db06960 | ||
|
|
ddaa063587 | ||
|
|
f9b4fe468e | ||
|
|
6cbca8d1bb | ||
|
|
d5067a4bcd | ||
|
|
04d44510de | ||
|
|
844fcc9bca | ||
|
|
145b5efab0 | ||
|
|
4079ed97b9 | ||
|
|
63d70fc037 | ||
|
|
ce86b5deda | ||
|
|
037e946bbe | ||
|
|
a51c8de1eb | ||
|
|
53dc517180 | ||
|
|
44ca940c6b | ||
|
|
c0b688c720 | ||
|
|
861d5f231d | ||
|
|
d14b4f55a6 | ||
|
|
db9a6c3eda | ||
|
|
a667643681 | ||
|
|
f278e4b6a5 | ||
|
|
33a12c2dec | ||
|
|
508f90f459 | ||
|
|
709f0c2274 | ||
|
|
be47700e7c | ||
|
|
40842830a4 | ||
|
|
11d070fa0d | ||
|
|
dd2f830a33 | ||
|
|
5431fcb450 | ||
|
|
324033e9ee | ||
|
|
86596d0cfe | ||
|
|
9323cfd748 | ||
|
|
d125b02cec | ||
|
|
282c4c5351 | ||
|
|
276d64a66a | ||
|
|
79eb381a41 | ||
|
|
44f9c0dfd3 | ||
|
|
0ca4cd724e | ||
|
|
8a3e31f021 | ||
|
|
9be8a1af0b | ||
|
|
b9cac20613 | ||
|
|
e6d5e6d37b | ||
|
|
0abfcac7da | ||
|
|
b70e8b9acc | ||
|
|
3b1e1aa3c3 | ||
|
|
8f74c74d50 | ||
|
|
d2f435016c | ||
|
|
389b35802b | ||
|
|
a9192a74f9 | ||
|
|
1366c7cf75 | ||
|
|
eaa9e8e3ea | ||
|
|
5ecae0df61 | ||
|
|
4891be1d8c | ||
|
|
ec852fc255 | ||
|
|
47f2d3cb7b | ||
|
|
37820c1e19 | ||
|
|
230fab3bb2 | ||
|
|
d292d2d093 | ||
|
|
51d5db01e9 | ||
|
|
0fd760df81 | ||
|
|
71d0a89968 | ||
|
|
d939a63412 | ||
|
|
daaa2d2fe2 | ||
|
|
6dd7cb19df | ||
|
|
b1de2481a8 | ||
|
|
27c01b3b0c | ||
|
|
b7aa9aff51 | ||
|
|
524a041fb9 | ||
|
|
9de0203914 | ||
|
|
0ed5309b97 | ||
|
|
68fd32d536 | ||
|
|
5ea3b25d21 | ||
|
|
2c24412633 | ||
|
|
1b8a45ba4a | ||
|
|
3dd4adbc1f | ||
|
|
0c52c9c4bc | ||
|
|
9caf44cdbd | ||
|
|
45d88bdc08 | ||
|
|
94b3ccd3d9 | ||
|
|
ee8273fd30 | ||
|
|
60c1449f40 | ||
|
|
67708325ae | ||
|
|
3e99577401 | ||
|
|
5e916dc2c8 | ||
|
|
0c64d46e99 | ||
|
|
3aa974f8bd | ||
|
|
621d01d502 | ||
|
|
aa20136223 | ||
|
|
9bc5408a44 | ||
|
|
5a7afb3092 | ||
|
|
f8948ac2ef |
3
.gitignore
vendored
@@ -5,3 +5,6 @@
|
|||||||
tags
|
tags
|
||||||
community/docs/current
|
community/docs/current
|
||||||
community/public/dist
|
community/public/dist
|
||||||
|
__pycache__/
|
||||||
|
*.py[cod]
|
||||||
|
*$py.class
|
||||||
|
|||||||
@@ -0,0 +1,8 @@
|
|||||||
|
// Copyright (c) 2021, FOSS United and contributors
|
||||||
|
// For license information, please see license.txt
|
||||||
|
|
||||||
|
frappe.ui.form.on('Discussion Message', {
|
||||||
|
// refresh: function(frm) {
|
||||||
|
|
||||||
|
// }
|
||||||
|
});
|
||||||
@@ -1,67 +1,53 @@
|
|||||||
{
|
{
|
||||||
"actions": [],
|
"actions": [],
|
||||||
"creation": "2021-03-19 12:19:32.355307",
|
"creation": "2021-08-11 10:59:38.597046",
|
||||||
"doctype": "DocType",
|
"doctype": "DocType",
|
||||||
"editable_grid": 1,
|
"editable_grid": 1,
|
||||||
"engine": "InnoDB",
|
"engine": "InnoDB",
|
||||||
"field_order": [
|
"field_order": [
|
||||||
"author",
|
"thread",
|
||||||
"batch",
|
"column_break_2",
|
||||||
"column_break_3",
|
"parent_message",
|
||||||
"author_name",
|
"section_break_4",
|
||||||
"pin",
|
|
||||||
"section_break_6",
|
|
||||||
"message"
|
"message"
|
||||||
],
|
],
|
||||||
"fields": [
|
"fields": [
|
||||||
{
|
|
||||||
"fieldname": "batch",
|
|
||||||
"fieldtype": "Link",
|
|
||||||
"in_list_view": 1,
|
|
||||||
"label": "Batch",
|
|
||||||
"options": "LMS Batch"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"fieldname": "author",
|
|
||||||
"fieldtype": "Link",
|
|
||||||
"in_list_view": 1,
|
|
||||||
"label": "Author",
|
|
||||||
"options": "User"
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
"fieldname": "message",
|
"fieldname": "message",
|
||||||
"fieldtype": "Markdown Editor",
|
"fieldtype": "Long Text",
|
||||||
"in_list_view": 1,
|
"in_list_view": 1,
|
||||||
"label": "Message"
|
"label": "Message"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"default": "0",
|
"fieldname": "thread",
|
||||||
"fieldname": "pin",
|
"fieldtype": "Link",
|
||||||
"fieldtype": "Check",
|
"in_list_view": 1,
|
||||||
"label": "Pin"
|
"in_standard_filter": 1,
|
||||||
|
"label": "Thread",
|
||||||
|
"options": "Discussion Thread"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"fetch_from": "author.full_name",
|
"fieldname": "column_break_2",
|
||||||
"fieldname": "author_name",
|
|
||||||
"fieldtype": "Data",
|
|
||||||
"label": "Author Name",
|
|
||||||
"read_only": 1
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"fieldname": "column_break_3",
|
|
||||||
"fieldtype": "Column Break"
|
"fieldtype": "Column Break"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"fieldname": "section_break_6",
|
"fieldname": "parent_message",
|
||||||
|
"fieldtype": "Link",
|
||||||
|
"in_list_view": 1,
|
||||||
|
"label": "Parent Message",
|
||||||
|
"options": "Discussion Message"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "section_break_4",
|
||||||
"fieldtype": "Section Break"
|
"fieldtype": "Section Break"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"index_web_pages_for_search": 1,
|
"index_web_pages_for_search": 1,
|
||||||
"links": [],
|
"links": [],
|
||||||
"modified": "2021-05-21 11:49:34.911479",
|
"modified": "2021-08-12 15:59:04.811286",
|
||||||
"modified_by": "Administrator",
|
"modified_by": "Administrator",
|
||||||
"module": "LMS",
|
"module": "Community",
|
||||||
"name": "LMS Message",
|
"name": "Discussion Message",
|
||||||
"owner": "Administrator",
|
"owner": "Administrator",
|
||||||
"permissions": [
|
"permissions": [
|
||||||
{
|
{
|
||||||
@@ -77,9 +63,7 @@
|
|||||||
"write": 1
|
"write": 1
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"quick_entry": 1,
|
|
||||||
"sort_field": "modified",
|
"sort_field": "modified",
|
||||||
"sort_order": "DESC",
|
"sort_order": "DESC",
|
||||||
"title_field": "author",
|
|
||||||
"track_changes": 1
|
"track_changes": 1
|
||||||
}
|
}
|
||||||
@@ -0,0 +1,22 @@
|
|||||||
|
# Copyright (c) 2021, FOSS United and contributors
|
||||||
|
# For license information, please see license.txt
|
||||||
|
|
||||||
|
import frappe
|
||||||
|
from frappe.model.document import Document
|
||||||
|
from community.widgets import Widget, Widgets
|
||||||
|
|
||||||
|
class DiscussionMessage(Document):
|
||||||
|
def after_insert(self):
|
||||||
|
data = {
|
||||||
|
"message": self,
|
||||||
|
"widgets": Widgets()
|
||||||
|
}
|
||||||
|
template = frappe.render_template("community/templates/message_card.html", data)
|
||||||
|
thread_info = frappe.db.get_value("Discussion Thread", self.thread, ["reference_doctype", "reference_docname"], as_dict=True)
|
||||||
|
frappe.publish_realtime(event="publish_message",
|
||||||
|
message = {
|
||||||
|
"thread": self.thread,
|
||||||
|
"template": template,
|
||||||
|
"thread_info": thread_info
|
||||||
|
},
|
||||||
|
after_commit=True)
|
||||||
@@ -1,10 +1,8 @@
|
|||||||
# -*- coding: utf-8 -*-
|
|
||||||
# Copyright (c) 2021, FOSS United and Contributors
|
# Copyright (c) 2021, FOSS United and Contributors
|
||||||
# See license.txt
|
# See license.txt
|
||||||
from __future__ import unicode_literals
|
|
||||||
|
|
||||||
# import frappe
|
# import frappe
|
||||||
import unittest
|
import unittest
|
||||||
|
|
||||||
class TestLMSSketch(unittest.TestCase):
|
class TestDiscussionMessage(unittest.TestCase):
|
||||||
pass
|
pass
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
// Copyright (c) 2021, FOSS United and contributors
|
// Copyright (c) 2021, FOSS United and contributors
|
||||||
// For license information, please see license.txt
|
// For license information, please see license.txt
|
||||||
|
|
||||||
frappe.ui.form.on('LMS Message', {
|
frappe.ui.form.on('Discussion Thread', {
|
||||||
// refresh: function(frm) {
|
// refresh: function(frm) {
|
||||||
|
|
||||||
// }
|
// }
|
||||||
@@ -1,15 +1,13 @@
|
|||||||
{
|
{
|
||||||
"actions": [],
|
"actions": [],
|
||||||
"autoname": "format:SKETCH-{#}",
|
"creation": "2021-08-11 10:55:29.341674",
|
||||||
"creation": "2021-03-09 16:31:50.523524",
|
|
||||||
"doctype": "DocType",
|
"doctype": "DocType",
|
||||||
"editable_grid": 1,
|
"editable_grid": 1,
|
||||||
"engine": "InnoDB",
|
"engine": "InnoDB",
|
||||||
"field_order": [
|
"field_order": [
|
||||||
"title",
|
"title",
|
||||||
"runtime",
|
"reference_doctype",
|
||||||
"code",
|
"reference_docname"
|
||||||
"svg"
|
|
||||||
],
|
],
|
||||||
"fields": [
|
"fields": [
|
||||||
{
|
{
|
||||||
@@ -18,28 +16,24 @@
|
|||||||
"label": "Title"
|
"label": "Title"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"fieldname": "runtime",
|
"fieldname": "reference_doctype",
|
||||||
"fieldtype": "Data",
|
"fieldtype": "Link",
|
||||||
"label": "Runtime"
|
"label": "Reference Doctype",
|
||||||
|
"options": "DocType"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"fieldname": "code",
|
"fieldname": "reference_docname",
|
||||||
"fieldtype": "Code",
|
"fieldtype": "Dynamic Link",
|
||||||
"label": "Code"
|
"label": "Reference Docname",
|
||||||
},
|
"options": "reference_doctype"
|
||||||
{
|
|
||||||
"fieldname": "svg",
|
|
||||||
"fieldtype": "Code",
|
|
||||||
"label": "SVG",
|
|
||||||
"read_only": 1
|
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"index_web_pages_for_search": 1,
|
"index_web_pages_for_search": 1,
|
||||||
"links": [],
|
"links": [],
|
||||||
"modified": "2021-03-12 08:42:56.671658",
|
"modified": "2021-08-11 12:29:43.564123",
|
||||||
"modified_by": "Administrator",
|
"modified_by": "Administrator",
|
||||||
"module": "LMS",
|
"module": "Community",
|
||||||
"name": "LMS Sketch",
|
"name": "Discussion Thread",
|
||||||
"owner": "Administrator",
|
"owner": "Administrator",
|
||||||
"permissions": [
|
"permissions": [
|
||||||
{
|
{
|
||||||
@@ -55,8 +49,9 @@
|
|||||||
"write": 1
|
"write": 1
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
"search_fields": "title",
|
||||||
"sort_field": "modified",
|
"sort_field": "modified",
|
||||||
"sort_order": "DESC",
|
"sort_order": "DESC",
|
||||||
"track_changes": 1,
|
"title_field": "title",
|
||||||
"track_views": 1
|
"track_changes": 1
|
||||||
}
|
}
|
||||||
@@ -0,0 +1,48 @@
|
|||||||
|
# Copyright (c) 2021, FOSS United and contributors
|
||||||
|
# For license information, please see license.txt
|
||||||
|
|
||||||
|
import frappe
|
||||||
|
from frappe.model.document import Document
|
||||||
|
|
||||||
|
class DiscussionThread(Document):
|
||||||
|
pass
|
||||||
|
|
||||||
|
@frappe.whitelist()
|
||||||
|
def submit_discussion(doctype, docname, message, title=None, thread_name=None):
|
||||||
|
thread = []
|
||||||
|
filters = {}
|
||||||
|
if doctype and docname:
|
||||||
|
filters = {
|
||||||
|
"reference_doctype": doctype,
|
||||||
|
"reference_docname": docname
|
||||||
|
}
|
||||||
|
|
||||||
|
elif thread_name:
|
||||||
|
filters = {
|
||||||
|
"name": thread_name
|
||||||
|
}
|
||||||
|
|
||||||
|
if filters:
|
||||||
|
thread = frappe.get_all("Discussion Thread",filters)
|
||||||
|
if len(thread):
|
||||||
|
thread = thread[0]
|
||||||
|
save_message(message, thread)
|
||||||
|
|
||||||
|
else:
|
||||||
|
thread = frappe.get_doc({
|
||||||
|
"doctype": "Discussion Thread",
|
||||||
|
"title": title,
|
||||||
|
"reference_doctype": doctype,
|
||||||
|
"reference_docname": docname
|
||||||
|
})
|
||||||
|
thread.save(ignore_permissions=True)
|
||||||
|
save_message(message, thread)
|
||||||
|
|
||||||
|
return thread.name
|
||||||
|
|
||||||
|
def save_message(message, thread):
|
||||||
|
frappe.get_doc({
|
||||||
|
"doctype": "Discussion Message",
|
||||||
|
"message": message,
|
||||||
|
"thread": thread.name
|
||||||
|
}).save(ignore_permissions=True)
|
||||||
@@ -1,10 +1,8 @@
|
|||||||
# -*- coding: utf-8 -*-
|
|
||||||
# Copyright (c) 2021, FOSS United and Contributors
|
# Copyright (c) 2021, FOSS United and Contributors
|
||||||
# See license.txt
|
# See license.txt
|
||||||
from __future__ import unicode_literals
|
|
||||||
|
|
||||||
# import frappe
|
# import frappe
|
||||||
import unittest
|
import unittest
|
||||||
|
|
||||||
class TestLMSMessage(unittest.TestCase):
|
class TestDiscussionThread(unittest.TestCase):
|
||||||
pass
|
pass
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
{% set color = member.get_palette() %}
|
{% set color = member.get_palette() %}
|
||||||
<a href="/{{member.username}}">
|
<a class="button-links" href="/{{member.username}}">
|
||||||
<span class="avatar {{ avatar_class }}" title="{{ member.full_name }}">
|
<span class="avatar {{ avatar_class }}" title="{{ member.full_name }}">
|
||||||
{% if member.user_image %}
|
{% if member.user_image %}
|
||||||
<img class="avatar-frame standard-image" style="object-fit: cover;" src="{{ member.user_image }}" title="{{ member.full_name }}">
|
<img class="avatar-frame standard-image" style="object-fit: cover;" src="{{ member.user_image }}" title="{{ member.full_name }}">
|
||||||
|
|||||||
70
community/community/widgets/DiscussionComment.html
Normal file
@@ -0,0 +1,70 @@
|
|||||||
|
<div class="discussions">
|
||||||
|
|
||||||
|
<form class="discussion-form {% if doctype or thread %} discussion-on-page {% endif %}" id="discussion-form">
|
||||||
|
|
||||||
|
|
||||||
|
<div class="form-group" {% if title or thread %} style="display: none;" {% endif %}>
|
||||||
|
<div class="control-input-wrapper">
|
||||||
|
<div class="control-input">
|
||||||
|
<input type="text" autocomplete="off" class="input-with-feedback form-control thread-title"
|
||||||
|
data-fieldtype="Data" data-fieldname="feedback_comments" placeholder="Title" spellcheck="false" {% if title
|
||||||
|
%} value="{{ title }}" {% endif %}></input>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="form-group">
|
||||||
|
<div class="control-input-wrapper">
|
||||||
|
<div class="control-input">
|
||||||
|
<textarea type="text" autocomplete="off" class="input-with-feedback form-control comment-field"
|
||||||
|
data-fieldtype="Text" data-fieldname="feedback_comments" placeholder="Enter a comment..."
|
||||||
|
spellcheck="false"></textarea>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="comment-footer">
|
||||||
|
<div class="button is-secondary pull-right" id="submit-discussion"
|
||||||
|
{% if doctype %} data-doctype="{{ doctype | urlencode}}" {% endif %}
|
||||||
|
{% if docname %} data-docname="{{ docname | urlencode}}" {% endif %}
|
||||||
|
{% if thread %} data-thread="{{ thread }}" {% endif %}>
|
||||||
|
Post</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
<script>
|
||||||
|
frappe.ready(() => {
|
||||||
|
$("#submit-discussion").click((e) => {
|
||||||
|
submit_discussion(e);
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
var submit_discussion = (e) => {
|
||||||
|
var message = $(".comment-field").val().trim();
|
||||||
|
|
||||||
|
if (message) {
|
||||||
|
var doctype = $(e.currentTarget).attr("data-doctype");
|
||||||
|
doctype = doctype ? decodeURIComponent(doctype) : doctype;
|
||||||
|
|
||||||
|
var docname = $(e.currentTarget).attr("data-docname");
|
||||||
|
docname = docname ? decodeURIComponent(docname) : docname;
|
||||||
|
|
||||||
|
frappe.call({
|
||||||
|
method: "community.community.doctype.discussion_thread.discussion_thread.submit_discussion",
|
||||||
|
args: {
|
||||||
|
"doctype": doctype ? doctype : "",
|
||||||
|
"docname": docname ? docname : "",
|
||||||
|
"message": $(".comment-field").val(),
|
||||||
|
"title": $(".thread-title").val(),
|
||||||
|
"thread_name": $(e.currentTarget).attr("data-thread")
|
||||||
|
},
|
||||||
|
callback: (data) => {
|
||||||
|
if (! $(".discussion-on-page").length) {
|
||||||
|
$("#discussion-modal").modal("hide");
|
||||||
|
window.location.href = `/discussions/${data.message}`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
80
community/community/widgets/DiscussionMessage.html
Normal file
@@ -0,0 +1,80 @@
|
|||||||
|
{% if doctype and docname and not thread %}
|
||||||
|
|
||||||
|
{% set thread_info = frappe.get_all("Discussion Thread", {"reference_doctype": doctype, "reference_docname": docname},
|
||||||
|
["name"]) %}
|
||||||
|
|
||||||
|
{% if thread_info | length %}
|
||||||
|
{% set thread = thread_info[0].name %}
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
{% if thread %}
|
||||||
|
{% set messages = frappe.get_all("Discussion Message", {"thread": thread}, ["name", "message", "owner", "creation"],
|
||||||
|
order_by="creation") %}
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
{% if doctype %}
|
||||||
|
<div class="course-home-headings mt-5"> Discussions </div>
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
<div class="messages mt-5">
|
||||||
|
{% for message in messages %}
|
||||||
|
{% include "community/templates/message_card.html" %}
|
||||||
|
{% endfor %}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{% if frappe.session.user == "Guest" or (condition is defined and not condition) %}
|
||||||
|
<div class="d-flex flex-column align-items-center font-weight-bold">
|
||||||
|
Want to join the discussion?
|
||||||
|
{% if frappe.session.user == "Guest" %}
|
||||||
|
<div class="button is-primary" id="login-from-discussion">Log In</div>
|
||||||
|
{% elif not condition %}
|
||||||
|
<div class="button is-primary" id="login-from-discussion" data-redirect="{{ redirect_to }}">{{ button_name }}</div>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
{% else %}
|
||||||
|
{{ widgets.DiscussionComment(doctype=doctype, docname=docname, title=title, thread=thread ) }}
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
<script>
|
||||||
|
frappe.ready(() => {
|
||||||
|
setup_socket_io();
|
||||||
|
|
||||||
|
$("#login-from-discussion").click((e) => {
|
||||||
|
login_from_discussion(e);
|
||||||
|
})
|
||||||
|
|
||||||
|
})
|
||||||
|
|
||||||
|
var setup_socket_io = () => {
|
||||||
|
const assets = [
|
||||||
|
"/assets/frappe/js/lib/socket.io.min.js",
|
||||||
|
"/assets/frappe/js/frappe/socketio_client.js",
|
||||||
|
]
|
||||||
|
frappe.require(assets, () => {
|
||||||
|
|
||||||
|
if (window.dev_server) {
|
||||||
|
frappe.boot.socketio_port = "9000";
|
||||||
|
}
|
||||||
|
|
||||||
|
frappe.socketio.init(9000);
|
||||||
|
var target = $("#submit-discussion");
|
||||||
|
|
||||||
|
frappe.socketio.socket.on("publish_message", (data) => {
|
||||||
|
if (target.attr("data-thread") == data.thread
|
||||||
|
|| (decodeURIComponent(target.attr("data-doctype")) == data.thread_info.reference_doctype
|
||||||
|
&& decodeURIComponent(target.attr("data-docname")) == data.thread_info.reference_docname)) {
|
||||||
|
$(".comment-field").val("");
|
||||||
|
$(".messages").append(data.template);
|
||||||
|
}
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
var login_from_discussion = (e) => {
|
||||||
|
var redirect = $(e.currentTarget).attr("data-redirect") || window.location.href;
|
||||||
|
window.location.href = `/login?redirect-to=${redirect}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
</script>
|
||||||
373
community/fixtures/custom_field.json
Normal file
@@ -0,0 +1,373 @@
|
|||||||
|
[
|
||||||
|
{
|
||||||
|
"allow_in_quick_entry": 0,
|
||||||
|
"allow_on_submit": 0,
|
||||||
|
"bold": 0,
|
||||||
|
"collapsible": 0,
|
||||||
|
"collapsible_depends_on": null,
|
||||||
|
"columns": 0,
|
||||||
|
"default": null,
|
||||||
|
"depends_on": null,
|
||||||
|
"description": null,
|
||||||
|
"docstatus": 0,
|
||||||
|
"doctype": "Custom Field",
|
||||||
|
"dt": "User",
|
||||||
|
"fetch_from": null,
|
||||||
|
"fetch_if_empty": 0,
|
||||||
|
"fieldname": "linkedin",
|
||||||
|
"fieldtype": "Data",
|
||||||
|
"hidden": 0,
|
||||||
|
"hide_border": 0,
|
||||||
|
"hide_days": 0,
|
||||||
|
"hide_seconds": 0,
|
||||||
|
"ignore_user_permissions": 0,
|
||||||
|
"ignore_xss_filter": 0,
|
||||||
|
"in_global_search": 0,
|
||||||
|
"in_list_view": 0,
|
||||||
|
"in_preview": 0,
|
||||||
|
"in_standard_filter": 0,
|
||||||
|
"insert_after": "mobile_no",
|
||||||
|
"label": "LinkedIn ID",
|
||||||
|
"length": 0,
|
||||||
|
"mandatory_depends_on": null,
|
||||||
|
"modified": "2021-06-30 14:46:55.834145",
|
||||||
|
"name": "User-linkedin",
|
||||||
|
"no_copy": 0,
|
||||||
|
"non_negative": 0,
|
||||||
|
"options": null,
|
||||||
|
"parent": null,
|
||||||
|
"parentfield": null,
|
||||||
|
"parenttype": null,
|
||||||
|
"permlevel": 0,
|
||||||
|
"precision": "",
|
||||||
|
"print_hide": 0,
|
||||||
|
"print_hide_if_no_value": 0,
|
||||||
|
"print_width": null,
|
||||||
|
"read_only": 0,
|
||||||
|
"read_only_depends_on": null,
|
||||||
|
"report_hide": 0,
|
||||||
|
"reqd": 0,
|
||||||
|
"search_index": 0,
|
||||||
|
"translatable": 1,
|
||||||
|
"unique": 0,
|
||||||
|
"width": null
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"allow_in_quick_entry": 0,
|
||||||
|
"allow_on_submit": 0,
|
||||||
|
"bold": 0,
|
||||||
|
"collapsible": 0,
|
||||||
|
"collapsible_depends_on": null,
|
||||||
|
"columns": 0,
|
||||||
|
"default": null,
|
||||||
|
"depends_on": null,
|
||||||
|
"description": null,
|
||||||
|
"docstatus": 0,
|
||||||
|
"doctype": "Custom Field",
|
||||||
|
"dt": "User",
|
||||||
|
"fetch_from": null,
|
||||||
|
"fetch_if_empty": 0,
|
||||||
|
"fieldname": "github",
|
||||||
|
"fieldtype": "Data",
|
||||||
|
"hidden": 0,
|
||||||
|
"hide_border": 0,
|
||||||
|
"hide_days": 0,
|
||||||
|
"hide_seconds": 0,
|
||||||
|
"ignore_user_permissions": 0,
|
||||||
|
"ignore_xss_filter": 0,
|
||||||
|
"in_global_search": 0,
|
||||||
|
"in_list_view": 0,
|
||||||
|
"in_preview": 0,
|
||||||
|
"in_standard_filter": 0,
|
||||||
|
"insert_after": "linkedin",
|
||||||
|
"label": "Github ID",
|
||||||
|
"length": 0,
|
||||||
|
"mandatory_depends_on": null,
|
||||||
|
"modified": "2021-06-30 14:46:55.834145",
|
||||||
|
"name": "User-github",
|
||||||
|
"no_copy": 0,
|
||||||
|
"non_negative": 0,
|
||||||
|
"options": null,
|
||||||
|
"parent": null,
|
||||||
|
"parentfield": null,
|
||||||
|
"parenttype": null,
|
||||||
|
"permlevel": 0,
|
||||||
|
"precision": "",
|
||||||
|
"print_hide": 0,
|
||||||
|
"print_hide_if_no_value": 0,
|
||||||
|
"print_width": null,
|
||||||
|
"read_only": 0,
|
||||||
|
"read_only_depends_on": null,
|
||||||
|
"report_hide": 0,
|
||||||
|
"reqd": 0,
|
||||||
|
"search_index": 0,
|
||||||
|
"translatable": 1,
|
||||||
|
"unique": 0,
|
||||||
|
"width": null
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"allow_in_quick_entry": 0,
|
||||||
|
"allow_on_submit": 0,
|
||||||
|
"bold": 0,
|
||||||
|
"collapsible": 0,
|
||||||
|
"collapsible_depends_on": null,
|
||||||
|
"columns": 0,
|
||||||
|
"default": null,
|
||||||
|
"depends_on": null,
|
||||||
|
"description": null,
|
||||||
|
"docstatus": 0,
|
||||||
|
"doctype": "Custom Field",
|
||||||
|
"dt": "User",
|
||||||
|
"fetch_from": null,
|
||||||
|
"fetch_if_empty": 0,
|
||||||
|
"fieldname": "medium",
|
||||||
|
"fieldtype": "Data",
|
||||||
|
"hidden": 0,
|
||||||
|
"hide_border": 0,
|
||||||
|
"hide_days": 0,
|
||||||
|
"hide_seconds": 0,
|
||||||
|
"ignore_user_permissions": 0,
|
||||||
|
"ignore_xss_filter": 0,
|
||||||
|
"in_global_search": 0,
|
||||||
|
"in_list_view": 0,
|
||||||
|
"in_preview": 0,
|
||||||
|
"in_standard_filter": 0,
|
||||||
|
"insert_after": "github",
|
||||||
|
"label": "Medium ID",
|
||||||
|
"length": 0,
|
||||||
|
"mandatory_depends_on": null,
|
||||||
|
"modified": "2021-06-30 14:46:55.834145",
|
||||||
|
"name": "User-medium",
|
||||||
|
"no_copy": 0,
|
||||||
|
"non_negative": 0,
|
||||||
|
"options": null,
|
||||||
|
"parent": null,
|
||||||
|
"parentfield": null,
|
||||||
|
"parenttype": null,
|
||||||
|
"permlevel": 0,
|
||||||
|
"precision": "",
|
||||||
|
"print_hide": 0,
|
||||||
|
"print_hide_if_no_value": 0,
|
||||||
|
"print_width": null,
|
||||||
|
"read_only": 0,
|
||||||
|
"read_only_depends_on": null,
|
||||||
|
"report_hide": 0,
|
||||||
|
"reqd": 0,
|
||||||
|
"search_index": 0,
|
||||||
|
"translatable": 1,
|
||||||
|
"unique": 0,
|
||||||
|
"width": null
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"allow_in_quick_entry": 0,
|
||||||
|
"allow_on_submit": 0,
|
||||||
|
"bold": 0,
|
||||||
|
"collapsible": 0,
|
||||||
|
"collapsible_depends_on": null,
|
||||||
|
"columns": 0,
|
||||||
|
"default": null,
|
||||||
|
"depends_on": null,
|
||||||
|
"description": null,
|
||||||
|
"docstatus": 0,
|
||||||
|
"doctype": "Custom Field",
|
||||||
|
"dt": "User",
|
||||||
|
"fetch_from": null,
|
||||||
|
"fetch_if_empty": 0,
|
||||||
|
"fieldname": "city",
|
||||||
|
"fieldtype": "Data",
|
||||||
|
"hidden": 0,
|
||||||
|
"hide_border": 0,
|
||||||
|
"hide_days": 0,
|
||||||
|
"hide_seconds": 0,
|
||||||
|
"ignore_user_permissions": 0,
|
||||||
|
"ignore_xss_filter": 0,
|
||||||
|
"in_global_search": 0,
|
||||||
|
"in_list_view": 0,
|
||||||
|
"in_preview": 0,
|
||||||
|
"in_standard_filter": 0,
|
||||||
|
"insert_after": "mute_sounds",
|
||||||
|
"label": "City",
|
||||||
|
"length": 0,
|
||||||
|
"mandatory_depends_on": null,
|
||||||
|
"modified": "2021-06-30 14:46:55.834145",
|
||||||
|
"name": "User-city",
|
||||||
|
"no_copy": 0,
|
||||||
|
"non_negative": 0,
|
||||||
|
"options": null,
|
||||||
|
"parent": null,
|
||||||
|
"parentfield": null,
|
||||||
|
"parenttype": null,
|
||||||
|
"permlevel": 0,
|
||||||
|
"precision": "",
|
||||||
|
"print_hide": 0,
|
||||||
|
"print_hide_if_no_value": 0,
|
||||||
|
"print_width": null,
|
||||||
|
"read_only": 0,
|
||||||
|
"read_only_depends_on": null,
|
||||||
|
"report_hide": 0,
|
||||||
|
"reqd": 0,
|
||||||
|
"search_index": 0,
|
||||||
|
"translatable": 1,
|
||||||
|
"unique": 0,
|
||||||
|
"width": null
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"allow_in_quick_entry": 0,
|
||||||
|
"allow_on_submit": 0,
|
||||||
|
"bold": 0,
|
||||||
|
"collapsible": 0,
|
||||||
|
"collapsible_depends_on": null,
|
||||||
|
"columns": 0,
|
||||||
|
"default": null,
|
||||||
|
"depends_on": null,
|
||||||
|
"description": null,
|
||||||
|
"docstatus": 0,
|
||||||
|
"doctype": "Custom Field",
|
||||||
|
"dt": "User",
|
||||||
|
"fetch_from": null,
|
||||||
|
"fetch_if_empty": 0,
|
||||||
|
"fieldname": "college",
|
||||||
|
"fieldtype": "Data",
|
||||||
|
"hidden": 0,
|
||||||
|
"hide_border": 0,
|
||||||
|
"hide_days": 0,
|
||||||
|
"hide_seconds": 0,
|
||||||
|
"ignore_user_permissions": 0,
|
||||||
|
"ignore_xss_filter": 0,
|
||||||
|
"in_global_search": 0,
|
||||||
|
"in_list_view": 0,
|
||||||
|
"in_preview": 0,
|
||||||
|
"in_standard_filter": 0,
|
||||||
|
"insert_after": "city",
|
||||||
|
"label": "College Name",
|
||||||
|
"length": 0,
|
||||||
|
"mandatory_depends_on": null,
|
||||||
|
"modified": "2021-06-30 14:46:55.834145",
|
||||||
|
"name": "User-college",
|
||||||
|
"no_copy": 0,
|
||||||
|
"non_negative": 0,
|
||||||
|
"options": null,
|
||||||
|
"parent": null,
|
||||||
|
"parentfield": null,
|
||||||
|
"parenttype": null,
|
||||||
|
"permlevel": 0,
|
||||||
|
"precision": "",
|
||||||
|
"print_hide": 0,
|
||||||
|
"print_hide_if_no_value": 0,
|
||||||
|
"print_width": null,
|
||||||
|
"read_only": 0,
|
||||||
|
"read_only_depends_on": null,
|
||||||
|
"report_hide": 0,
|
||||||
|
"reqd": 0,
|
||||||
|
"search_index": 0,
|
||||||
|
"translatable": 1,
|
||||||
|
"unique": 0,
|
||||||
|
"width": null
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"allow_in_quick_entry": 0,
|
||||||
|
"allow_on_submit": 0,
|
||||||
|
"bold": 0,
|
||||||
|
"collapsible": 0,
|
||||||
|
"collapsible_depends_on": null,
|
||||||
|
"columns": 0,
|
||||||
|
"default": null,
|
||||||
|
"depends_on": null,
|
||||||
|
"description": null,
|
||||||
|
"docstatus": 0,
|
||||||
|
"doctype": "Custom Field",
|
||||||
|
"dt": "User",
|
||||||
|
"fetch_from": null,
|
||||||
|
"fetch_if_empty": 0,
|
||||||
|
"fieldname": "branch",
|
||||||
|
"fieldtype": "Data",
|
||||||
|
"hidden": 0,
|
||||||
|
"hide_border": 0,
|
||||||
|
"hide_days": 0,
|
||||||
|
"hide_seconds": 0,
|
||||||
|
"ignore_user_permissions": 0,
|
||||||
|
"ignore_xss_filter": 0,
|
||||||
|
"in_global_search": 0,
|
||||||
|
"in_list_view": 0,
|
||||||
|
"in_preview": 0,
|
||||||
|
"in_standard_filter": 0,
|
||||||
|
"insert_after": "college",
|
||||||
|
"label": "Branch",
|
||||||
|
"length": 0,
|
||||||
|
"mandatory_depends_on": null,
|
||||||
|
"modified": "2021-06-30 14:46:55.834145",
|
||||||
|
"name": "User-branch",
|
||||||
|
"no_copy": 0,
|
||||||
|
"non_negative": 0,
|
||||||
|
"options": null,
|
||||||
|
"parent": null,
|
||||||
|
"parentfield": null,
|
||||||
|
"parenttype": null,
|
||||||
|
"permlevel": 0,
|
||||||
|
"precision": "",
|
||||||
|
"print_hide": 0,
|
||||||
|
"print_hide_if_no_value": 0,
|
||||||
|
"print_width": null,
|
||||||
|
"read_only": 0,
|
||||||
|
"read_only_depends_on": null,
|
||||||
|
"report_hide": 0,
|
||||||
|
"reqd": 0,
|
||||||
|
"search_index": 0,
|
||||||
|
"translatable": 1,
|
||||||
|
"unique": 0,
|
||||||
|
"width": null
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"allow_in_quick_entry": 0,
|
||||||
|
"allow_on_submit": 0,
|
||||||
|
"bold": 0,
|
||||||
|
"collapsible": 0,
|
||||||
|
"collapsible_depends_on": null,
|
||||||
|
"columns": 0,
|
||||||
|
"default": null,
|
||||||
|
"depends_on": null,
|
||||||
|
"description": null,
|
||||||
|
"docstatus": 0,
|
||||||
|
"doctype": "Custom Field",
|
||||||
|
"dt": "User",
|
||||||
|
"fetch_from": null,
|
||||||
|
"fetch_if_empty": 0,
|
||||||
|
"fieldname": "profession",
|
||||||
|
"fieldtype": "Data",
|
||||||
|
"hidden": 0,
|
||||||
|
"hide_border": 0,
|
||||||
|
"hide_days": 0,
|
||||||
|
"hide_seconds": 0,
|
||||||
|
"ignore_user_permissions": 0,
|
||||||
|
"ignore_xss_filter": 0,
|
||||||
|
"in_global_search": 0,
|
||||||
|
"in_list_view": 0,
|
||||||
|
"in_preview": 0,
|
||||||
|
"in_standard_filter": 0,
|
||||||
|
"insert_after": "medium",
|
||||||
|
"label": "Profession",
|
||||||
|
"length": 0,
|
||||||
|
"mandatory_depends_on": null,
|
||||||
|
"modified": "2021-06-30 14:46:55.834145",
|
||||||
|
"name": "User-profession",
|
||||||
|
"no_copy": 0,
|
||||||
|
"non_negative": 0,
|
||||||
|
"options": null,
|
||||||
|
"parent": null,
|
||||||
|
"parentfield": null,
|
||||||
|
"parenttype": null,
|
||||||
|
"permlevel": 0,
|
||||||
|
"precision": "",
|
||||||
|
"print_hide": 0,
|
||||||
|
"print_hide_if_no_value": 0,
|
||||||
|
"print_width": null,
|
||||||
|
"read_only": 0,
|
||||||
|
"read_only_depends_on": null,
|
||||||
|
"report_hide": 0,
|
||||||
|
"reqd": 0,
|
||||||
|
"search_index": 0,
|
||||||
|
"translatable": 1,
|
||||||
|
"unique": 0,
|
||||||
|
"width": null
|
||||||
|
}
|
||||||
|
]
|
||||||
@@ -104,6 +104,8 @@ doc_events = {
|
|||||||
# ]
|
# ]
|
||||||
#}
|
#}
|
||||||
|
|
||||||
|
fixtures = ["Custom Field"]
|
||||||
|
|
||||||
# Testing
|
# Testing
|
||||||
# -------
|
# -------
|
||||||
|
|
||||||
@@ -131,10 +133,9 @@ doc_events = {
|
|||||||
primary_rules = [
|
primary_rules = [
|
||||||
{"from_route": "/sketches/<sketch>", "to_route": "sketches/sketch"},
|
{"from_route": "/sketches/<sketch>", "to_route": "sketches/sketch"},
|
||||||
{"from_route": "/courses/<course>", "to_route": "courses/course"},
|
{"from_route": "/courses/<course>", "to_route": "courses/course"},
|
||||||
{"from_route": "/courses/<course>/<topic>", "to_route": "courses/topic"},
|
{"from_route": "/courses/<course>/<certificate>", "to_route": "courses/certificate"},
|
||||||
{"from_route": "/hackathons/<hackathon>", "to_route": "hackathons/hackathon"},
|
{"from_route": "/hackathons/<hackathon>", "to_route": "hackathons/hackathon"},
|
||||||
{"from_route": "/hackathons/<hackathon>/<project>", "to_route": "hackathons/project"},
|
{"from_route": "/hackathons/<hackathon>/<project>", "to_route": "hackathons/project"},
|
||||||
{"from_route": "/dashboard", "to_route": ""},
|
|
||||||
{"from_route": "/add-a-new-batch", "to_route": "add-a-new-batch"},
|
{"from_route": "/add-a-new-batch", "to_route": "add-a-new-batch"},
|
||||||
{"from_route": "/courses/<course>/home", "to_route": "batch/home"},
|
{"from_route": "/courses/<course>/home", "to_route": "batch/home"},
|
||||||
{"from_route": "/courses/<course>/learn", "to_route": "batch/learn"},
|
{"from_route": "/courses/<course>/learn", "to_route": "batch/learn"},
|
||||||
@@ -144,7 +145,8 @@ primary_rules = [
|
|||||||
{"from_route": "/courses/<course>/discuss", "to_route": "batch/discuss"},
|
{"from_route": "/courses/<course>/discuss", "to_route": "batch/discuss"},
|
||||||
{"from_route": "/courses/<course>/about", "to_route": "batch/about"},
|
{"from_route": "/courses/<course>/about", "to_route": "batch/about"},
|
||||||
{"from_route": "/courses/<course>/progress", "to_route": "batch/progress"},
|
{"from_route": "/courses/<course>/progress", "to_route": "batch/progress"},
|
||||||
{"from_route": "/courses/<course>/join", "to_route": "batch/join"}
|
{"from_route": "/courses/<course>/join", "to_route": "batch/join"},
|
||||||
|
{"from_route": "/discussions/<discussion>", "to_route": "discussions/discussion"}
|
||||||
]
|
]
|
||||||
|
|
||||||
# Any frappe default URL is blocked by profile-rules, add it here to unblock it
|
# Any frappe default URL is blocked by profile-rules, add it here to unblock it
|
||||||
@@ -165,7 +167,9 @@ whitelist = [
|
|||||||
"/add-a-new-batch",
|
"/add-a-new-batch",
|
||||||
"/new-sign-up",
|
"/new-sign-up",
|
||||||
"/message",
|
"/message",
|
||||||
"/about"
|
"/about",
|
||||||
|
"/edit-profile",
|
||||||
|
"/discussions"
|
||||||
]
|
]
|
||||||
whitelist_rules = [{"from_route": p, "to_route": p[1:]} for p in whitelist]
|
whitelist_rules = [{"from_route": p, "to_route": p[1:]} for p in whitelist]
|
||||||
|
|
||||||
@@ -176,6 +180,10 @@ profile_rules = [
|
|||||||
|
|
||||||
website_route_rules = primary_rules + whitelist_rules + profile_rules
|
website_route_rules = primary_rules + whitelist_rules + profile_rules
|
||||||
|
|
||||||
|
website_redirects = [
|
||||||
|
{"source": "/update-profile", "target": "/edit-profile"},
|
||||||
|
]
|
||||||
|
|
||||||
update_website_context = 'community.widgets.update_website_context'
|
update_website_context = 'community.widgets.update_website_context'
|
||||||
|
|
||||||
## Specify the additional tabs to be included in the user profile page.
|
## Specify the additional tabs to be included in the user profile page.
|
||||||
@@ -187,6 +195,10 @@ update_website_context = 'community.widgets.update_website_context'
|
|||||||
## subclass of community.community.plugins.PageExtension
|
## subclass of community.community.plugins.PageExtension
|
||||||
# community_lesson_page_extension = None
|
# community_lesson_page_extension = None
|
||||||
|
|
||||||
|
community_lesson_page_extensions = [
|
||||||
|
"community.plugins.LiveCodeExtension"
|
||||||
|
]
|
||||||
|
|
||||||
## Markdown Macros for Lessons
|
## Markdown Macros for Lessons
|
||||||
community_markdown_macro_renderers = {
|
community_markdown_macro_renderers = {
|
||||||
"Exercise": "community.plugins.exercise_renderer",
|
"Exercise": "community.plugins.exercise_renderer",
|
||||||
|
|||||||
@@ -2,7 +2,15 @@
|
|||||||
// For license information, please see license.txt
|
// For license information, please see license.txt
|
||||||
|
|
||||||
frappe.ui.form.on('Chapter', {
|
frappe.ui.form.on('Chapter', {
|
||||||
// refresh: function(frm) {
|
|
||||||
|
|
||||||
// }
|
onload: function (frm) {
|
||||||
|
frm.set_query("lesson", "lessons", function () {
|
||||||
|
return {
|
||||||
|
filters: {
|
||||||
|
"chapter": frm.doc.name,
|
||||||
|
}
|
||||||
|
};
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -9,8 +9,7 @@
|
|||||||
"course",
|
"course",
|
||||||
"title",
|
"title",
|
||||||
"description",
|
"description",
|
||||||
"locked",
|
"lessons"
|
||||||
"index_"
|
|
||||||
],
|
],
|
||||||
"fields": [
|
"fields": [
|
||||||
{
|
{
|
||||||
@@ -24,12 +23,6 @@
|
|||||||
"fieldtype": "Markdown Editor",
|
"fieldtype": "Markdown Editor",
|
||||||
"label": "Description"
|
"label": "Description"
|
||||||
},
|
},
|
||||||
{
|
|
||||||
"default": "0",
|
|
||||||
"fieldname": "locked",
|
|
||||||
"fieldtype": "Check",
|
|
||||||
"label": "Locked"
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
"fieldname": "course",
|
"fieldname": "course",
|
||||||
"fieldtype": "Link",
|
"fieldtype": "Link",
|
||||||
@@ -38,10 +31,10 @@
|
|||||||
"options": "LMS Course"
|
"options": "LMS Course"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"default": "1",
|
"fieldname": "lessons",
|
||||||
"fieldname": "index_",
|
"fieldtype": "Table",
|
||||||
"fieldtype": "Int",
|
"label": "Lessons",
|
||||||
"label": "Index"
|
"options": "Lessons"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"index_web_pages_for_search": 1,
|
"index_web_pages_for_search": 1,
|
||||||
@@ -52,7 +45,7 @@
|
|||||||
"link_fieldname": "chapter"
|
"link_fieldname": "chapter"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"modified": "2021-05-13 21:05:20.531890",
|
"modified": "2021-07-27 16:28:08.667964",
|
||||||
"modified_by": "Administrator",
|
"modified_by": "Administrator",
|
||||||
"module": "LMS",
|
"module": "LMS",
|
||||||
"name": "Chapter",
|
"name": "Chapter",
|
||||||
|
|||||||
@@ -7,9 +7,4 @@ import frappe
|
|||||||
from frappe.model.document import Document
|
from frappe.model.document import Document
|
||||||
|
|
||||||
class Chapter(Document):
|
class Chapter(Document):
|
||||||
def get_lessons(self):
|
pass
|
||||||
rows = frappe.db.get_all("Lesson",
|
|
||||||
filters={"chapter": self.name},
|
|
||||||
fields='name',
|
|
||||||
order_by="index_")
|
|
||||||
return [frappe.get_doc('Lesson', row['name']) for row in rows]
|
|
||||||
|
|||||||
0
community/lms/doctype/chapters/__init__.py
Normal file
32
community/lms/doctype/chapters/chapters.json
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
{
|
||||||
|
"actions": [],
|
||||||
|
"creation": "2021-07-27 16:25:02.903245",
|
||||||
|
"doctype": "DocType",
|
||||||
|
"editable_grid": 1,
|
||||||
|
"engine": "InnoDB",
|
||||||
|
"field_order": [
|
||||||
|
"chapter"
|
||||||
|
],
|
||||||
|
"fields": [
|
||||||
|
{
|
||||||
|
"fieldname": "chapter",
|
||||||
|
"fieldtype": "Link",
|
||||||
|
"in_list_view": 1,
|
||||||
|
"label": "Chapter",
|
||||||
|
"options": "Chapter",
|
||||||
|
"reqd": 1
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"index_web_pages_for_search": 1,
|
||||||
|
"istable": 1,
|
||||||
|
"links": [],
|
||||||
|
"modified": "2021-07-27 16:25:02.903245",
|
||||||
|
"modified_by": "Administrator",
|
||||||
|
"module": "LMS",
|
||||||
|
"name": "Chapters",
|
||||||
|
"owner": "Administrator",
|
||||||
|
"permissions": [],
|
||||||
|
"sort_field": "modified",
|
||||||
|
"sort_order": "DESC",
|
||||||
|
"track_changes": 1
|
||||||
|
}
|
||||||
8
community/lms/doctype/chapters/chapters.py
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
# Copyright (c) 2021, FOSS United and contributors
|
||||||
|
# For license information, please see license.txt
|
||||||
|
|
||||||
|
# import frappe
|
||||||
|
from frappe.model.document import Document
|
||||||
|
|
||||||
|
class Chapters(Document):
|
||||||
|
pass
|
||||||
@@ -3,7 +3,6 @@
|
|||||||
|
|
||||||
import frappe
|
import frappe
|
||||||
from frappe.model.document import Document
|
from frappe.model.document import Document
|
||||||
# from ..lms_sketch.livecode import livecode_to_svg
|
|
||||||
|
|
||||||
class Exercise(Document):
|
class Exercise(Document):
|
||||||
def get_user_submission(self):
|
def get_user_submission(self):
|
||||||
|
|||||||
@@ -6,12 +6,17 @@
|
|||||||
"engine": "InnoDB",
|
"engine": "InnoDB",
|
||||||
"field_order": [
|
"field_order": [
|
||||||
"exercise",
|
"exercise",
|
||||||
"solution",
|
"status",
|
||||||
|
"batch",
|
||||||
|
"column_break_4",
|
||||||
"exercise_title",
|
"exercise_title",
|
||||||
"course",
|
"course",
|
||||||
"batch",
|
|
||||||
"lesson",
|
"lesson",
|
||||||
"image"
|
"section_break_8",
|
||||||
|
"solution",
|
||||||
|
"image",
|
||||||
|
"test_results",
|
||||||
|
"comments"
|
||||||
],
|
],
|
||||||
"fields": [
|
"fields": [
|
||||||
{
|
{
|
||||||
@@ -21,12 +26,6 @@
|
|||||||
"label": "Exercise",
|
"label": "Exercise",
|
||||||
"options": "Exercise"
|
"options": "Exercise"
|
||||||
},
|
},
|
||||||
{
|
|
||||||
"fieldname": "solution",
|
|
||||||
"fieldtype": "Code",
|
|
||||||
"in_list_view": 1,
|
|
||||||
"label": "Solution"
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
"fetch_from": "exercise.title",
|
"fetch_from": "exercise.title",
|
||||||
"fieldname": "exercise_title",
|
"fieldname": "exercise_title",
|
||||||
@@ -61,11 +60,41 @@
|
|||||||
"fieldtype": "Code",
|
"fieldtype": "Code",
|
||||||
"label": "Image",
|
"label": "Image",
|
||||||
"read_only": 1
|
"read_only": 1
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "status",
|
||||||
|
"fieldtype": "Select",
|
||||||
|
"label": "Status",
|
||||||
|
"options": "Correct\nIncorrect"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "test_results",
|
||||||
|
"fieldtype": "Small Text",
|
||||||
|
"label": "Test Results"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "comments",
|
||||||
|
"fieldtype": "Small Text",
|
||||||
|
"label": "Comments"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "solution",
|
||||||
|
"fieldtype": "Code",
|
||||||
|
"in_list_view": 1,
|
||||||
|
"label": "Solution"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "column_break_4",
|
||||||
|
"fieldtype": "Column Break"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "section_break_8",
|
||||||
|
"fieldtype": "Section Break"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"index_web_pages_for_search": 1,
|
"index_web_pages_for_search": 1,
|
||||||
"links": [],
|
"links": [],
|
||||||
"modified": "2021-05-21 11:28:45.833018",
|
"modified": "2021-06-24 16:22:50.570845",
|
||||||
"modified_by": "Administrator",
|
"modified_by": "Administrator",
|
||||||
"module": "LMS",
|
"module": "LMS",
|
||||||
"name": "Exercise Submission",
|
"name": "Exercise Submission",
|
||||||
|
|||||||
@@ -30,10 +30,13 @@ class InviteRequest(Document):
|
|||||||
return user
|
return user
|
||||||
|
|
||||||
def send_email(self):
|
def send_email(self):
|
||||||
subject = _("Your request has been approved.")
|
site_name = "Mon.School"
|
||||||
|
subject = _("Welcome to {0}!").format(site_name)
|
||||||
|
|
||||||
args = {
|
args = {
|
||||||
"full_name": self.full_name,
|
"full_name": self.full_name,
|
||||||
"signup_form_link": "/new-sign-up?invite_code={0}".format(self.name),
|
"signup_form_link": "/new-sign-up?invite_code={0}".format(self.name),
|
||||||
|
"site_name": site_name,
|
||||||
"site_url": frappe.utils.get_url()
|
"site_url": frappe.utils.get_url()
|
||||||
}
|
}
|
||||||
frappe.sendmail(
|
frappe.sendmail(
|
||||||
@@ -42,7 +45,8 @@ class InviteRequest(Document):
|
|||||||
subject=subject,
|
subject=subject,
|
||||||
header=[subject, "green"],
|
header=[subject, "green"],
|
||||||
template = "lms_invite_request_approved",
|
template = "lms_invite_request_approved",
|
||||||
args=args)
|
args=args,
|
||||||
|
now=True)
|
||||||
|
|
||||||
@frappe.whitelist(allow_guest=True)
|
@frappe.whitelist(allow_guest=True)
|
||||||
def create_invite_request(invite_email):
|
def create_invite_request(invite_email):
|
||||||
|
|||||||
@@ -2,7 +2,47 @@
|
|||||||
// For license information, please see license.txt
|
// For license information, please see license.txt
|
||||||
|
|
||||||
frappe.ui.form.on('Lesson', {
|
frappe.ui.form.on('Lesson', {
|
||||||
// refresh: function(frm) {
|
setup: function (frm) {
|
||||||
|
frm.trigger('setup_help');
|
||||||
|
},
|
||||||
|
setup_help(frm) {
|
||||||
|
frm.get_field('help').html(`
|
||||||
|
<p>You can add some more additional content to the lesson using a special syntax. The table below mentions all types of dynamic content that you can add to the lessons and the syntax for the same.</p>
|
||||||
|
<div class="row font-weight-bold mb-3">
|
||||||
|
<div class="col-sm-4">
|
||||||
|
Content Type
|
||||||
|
</div>
|
||||||
|
<div class="col-sm-4">
|
||||||
|
Syntax
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
// }
|
<div class="row mb-3">
|
||||||
|
<div class="col-sm-4">
|
||||||
|
YouTube Video
|
||||||
|
</div>
|
||||||
|
<div class="col-sm-4">
|
||||||
|
{{ YouTubeVideo("unique_embed_id") }}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="row mb-3">
|
||||||
|
<div class="col-sm-4">
|
||||||
|
Exercise
|
||||||
|
</div>
|
||||||
|
<div class="col-sm-4">
|
||||||
|
{{ Exercise("exercise_name") }}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="row mb-3">
|
||||||
|
<div class="col-sm-4">
|
||||||
|
Quiz
|
||||||
|
</div>
|
||||||
|
<div class="col-sm-4">
|
||||||
|
{{ Quiz("lms_quiz_name") }}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
`);
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -10,10 +10,11 @@
|
|||||||
"include_in_preview",
|
"include_in_preview",
|
||||||
"column_break_4",
|
"column_break_4",
|
||||||
"title",
|
"title",
|
||||||
"index_",
|
|
||||||
"index_label",
|
"index_label",
|
||||||
"section_break_6",
|
"section_break_6",
|
||||||
"body"
|
"body",
|
||||||
|
"help_section",
|
||||||
|
"help"
|
||||||
],
|
],
|
||||||
"fields": [
|
"fields": [
|
||||||
{
|
{
|
||||||
@@ -29,13 +30,6 @@
|
|||||||
"in_list_view": 1,
|
"in_list_view": 1,
|
||||||
"label": "Title"
|
"label": "Title"
|
||||||
},
|
},
|
||||||
{
|
|
||||||
"default": "1",
|
|
||||||
"fieldname": "index_",
|
|
||||||
"fieldtype": "Int",
|
|
||||||
"in_list_view": 1,
|
|
||||||
"label": "Index"
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
"fieldname": "body",
|
"fieldname": "body",
|
||||||
"fieldtype": "Markdown Editor",
|
"fieldtype": "Markdown Editor",
|
||||||
@@ -60,11 +54,20 @@
|
|||||||
{
|
{
|
||||||
"fieldname": "column_break_4",
|
"fieldname": "column_break_4",
|
||||||
"fieldtype": "Column Break"
|
"fieldtype": "Column Break"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "help_section",
|
||||||
|
"fieldtype": "Section Break",
|
||||||
|
"label": "Help"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "help",
|
||||||
|
"fieldtype": "HTML"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"index_web_pages_for_search": 1,
|
"index_web_pages_for_search": 1,
|
||||||
"links": [],
|
"links": [],
|
||||||
"modified": "2021-06-23 17:59:52.946515",
|
"modified": "2021-07-27 16:28:29.203624",
|
||||||
"modified_by": "Administrator",
|
"modified_by": "Administrator",
|
||||||
"module": "LMS",
|
"module": "LMS",
|
||||||
"name": "Lesson",
|
"name": "Lesson",
|
||||||
|
|||||||
@@ -63,32 +63,36 @@ class Lesson(Document):
|
|||||||
return
|
return
|
||||||
|
|
||||||
@frappe.whitelist()
|
@frappe.whitelist()
|
||||||
def save_progress(lesson, course):
|
def save_progress(lesson, course, status):
|
||||||
if not frappe.db.exists("LMS Batch Membership",
|
if not frappe.db.exists("LMS Batch Membership",
|
||||||
{
|
{
|
||||||
"member": frappe.session.user,
|
"member": frappe.session.user,
|
||||||
"course": course
|
"course": course
|
||||||
}):
|
}):
|
||||||
return
|
return
|
||||||
|
|
||||||
if frappe.db.exists("LMS Course Progress",
|
if frappe.db.exists("LMS Course Progress",
|
||||||
{
|
{
|
||||||
"lesson": lesson,
|
"lesson": lesson,
|
||||||
"owner": frappe.session.user
|
"owner": frappe.session.user,
|
||||||
|
"course": course
|
||||||
}):
|
}):
|
||||||
return
|
doc = frappe.get_doc("LMS Course Progress",
|
||||||
|
{
|
||||||
lesson_details = frappe.get_doc("Lesson", lesson)
|
"lesson": lesson,
|
||||||
dynamic_content = find_macros(lesson_details.body)
|
"owner": frappe.session.user,
|
||||||
|
"course": course
|
||||||
status = "Complete"
|
})
|
||||||
if dynamic_content:
|
doc.status = status
|
||||||
status = "Partially Complete"
|
doc.save(ignore_permissions=True)
|
||||||
|
else:
|
||||||
frappe.get_doc({
|
frappe.get_doc({
|
||||||
"doctype": "LMS Course Progress",
|
"doctype": "LMS Course Progress",
|
||||||
"lesson": lesson_details.name,
|
"lesson": lesson,
|
||||||
"status": status
|
"status": status,
|
||||||
}).save(ignore_permissions=True)
|
}).save(ignore_permissions=True)
|
||||||
|
course_details = frappe.get_doc("LMS Course", course)
|
||||||
|
return course_details.get_course_progress()
|
||||||
|
|
||||||
def update_progress(lesson):
|
def update_progress(lesson):
|
||||||
user = frappe.session.user
|
user = frappe.session.user
|
||||||
|
|||||||
0
community/lms/doctype/lessons/__init__.py
Normal file
31
community/lms/doctype/lessons/lessons.json
Normal file
@@ -0,0 +1,31 @@
|
|||||||
|
{
|
||||||
|
"actions": [],
|
||||||
|
"creation": "2021-07-27 16:25:48.269536",
|
||||||
|
"doctype": "DocType",
|
||||||
|
"editable_grid": 1,
|
||||||
|
"engine": "InnoDB",
|
||||||
|
"field_order": [
|
||||||
|
"lesson"
|
||||||
|
],
|
||||||
|
"fields": [
|
||||||
|
{
|
||||||
|
"fieldname": "lesson",
|
||||||
|
"fieldtype": "Link",
|
||||||
|
"in_list_view": 1,
|
||||||
|
"label": "Lesson",
|
||||||
|
"options": "Lesson"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"index_web_pages_for_search": 1,
|
||||||
|
"istable": 1,
|
||||||
|
"links": [],
|
||||||
|
"modified": "2021-07-27 16:53:52.732191",
|
||||||
|
"modified_by": "Administrator",
|
||||||
|
"module": "LMS",
|
||||||
|
"name": "Lessons",
|
||||||
|
"owner": "Administrator",
|
||||||
|
"permissions": [],
|
||||||
|
"sort_field": "modified",
|
||||||
|
"sort_order": "DESC",
|
||||||
|
"track_changes": 1
|
||||||
|
}
|
||||||
8
community/lms/doctype/lessons/lessons.py
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
# Copyright (c) 2021, FOSS United and contributors
|
||||||
|
# For license information, please see license.txt
|
||||||
|
|
||||||
|
# import frappe
|
||||||
|
from frappe.model.document import Document
|
||||||
|
|
||||||
|
class Lessons(Document):
|
||||||
|
pass
|
||||||
@@ -71,3 +71,38 @@ def save_message(message, batch):
|
|||||||
"message": message
|
"message": message
|
||||||
})
|
})
|
||||||
doc.save(ignore_permissions=True)
|
doc.save(ignore_permissions=True)
|
||||||
|
|
||||||
|
def switch_batch(course_name, email, batch_name):
|
||||||
|
"""Switches the user from the current batch of the course to a new batch.
|
||||||
|
"""
|
||||||
|
membership = frappe.get_last_doc(
|
||||||
|
"LMS Batch Membership",
|
||||||
|
filters={"course": course_name, "member": email})
|
||||||
|
|
||||||
|
batch = frappe.get_doc("LMS Batch", batch_name)
|
||||||
|
if not batch:
|
||||||
|
raise ValueError(f"Invalid Batch: {batch_name}")
|
||||||
|
|
||||||
|
if batch.course != course_name:
|
||||||
|
raise ValueError("Can not switch batches across courses")
|
||||||
|
|
||||||
|
if batch.is_member(email):
|
||||||
|
print(f"{email} is already a member of {batch.title}")
|
||||||
|
return
|
||||||
|
|
||||||
|
old_batch = frappe.get_doc("LMS Batch", membership.batch)
|
||||||
|
|
||||||
|
print("updating membership", membership.name)
|
||||||
|
membership.batch = batch_name
|
||||||
|
membership.save()
|
||||||
|
|
||||||
|
# update exercise submissions
|
||||||
|
filters = {
|
||||||
|
"owner": email,
|
||||||
|
"batch": old_batch.name
|
||||||
|
}
|
||||||
|
for name in frappe.db.get_all("Exercise Submission", filters=filters, pluck='name'):
|
||||||
|
doc = frappe.get_doc("Exercise Submission", name)
|
||||||
|
print("updating exercise submission", name)
|
||||||
|
doc.batch = batch_name
|
||||||
|
doc.save()
|
||||||
|
|||||||
@@ -10,5 +10,5 @@ frappe.ui.form.on('LMS Batch Membership', {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -37,6 +37,7 @@
|
|||||||
"fieldname": "member_type",
|
"fieldname": "member_type",
|
||||||
"fieldtype": "Select",
|
"fieldtype": "Select",
|
||||||
"in_list_view": 1,
|
"in_list_view": 1,
|
||||||
|
"in_standard_filter": 1,
|
||||||
"label": "Member Type",
|
"label": "Member Type",
|
||||||
"options": "\nStudent\nMentor\nStaff"
|
"options": "\nStudent\nMentor\nStaff"
|
||||||
},
|
},
|
||||||
@@ -44,7 +45,6 @@
|
|||||||
"default": "Member",
|
"default": "Member",
|
||||||
"fieldname": "role",
|
"fieldname": "role",
|
||||||
"fieldtype": "Select",
|
"fieldtype": "Select",
|
||||||
"in_standard_filter": 1,
|
|
||||||
"label": "Role",
|
"label": "Role",
|
||||||
"options": "\nMember\nAdmin"
|
"options": "\nMember\nAdmin"
|
||||||
},
|
},
|
||||||
@@ -63,10 +63,10 @@
|
|||||||
{
|
{
|
||||||
"fetch_from": "batch.course",
|
"fetch_from": "batch.course",
|
||||||
"fieldname": "course",
|
"fieldname": "course",
|
||||||
"fieldtype": "Data",
|
"fieldtype": "Link",
|
||||||
"in_list_view": 1,
|
"in_list_view": 1,
|
||||||
"label": "Course",
|
"label": "Course",
|
||||||
"read_only": 1
|
"options": "LMS Course"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"fieldname": "current_lesson",
|
"fieldname": "current_lesson",
|
||||||
@@ -84,7 +84,7 @@
|
|||||||
],
|
],
|
||||||
"index_web_pages_for_search": 1,
|
"index_web_pages_for_search": 1,
|
||||||
"links": [],
|
"links": [],
|
||||||
"modified": "2021-06-21 12:10:28.808803",
|
"modified": "2021-08-04 17:10:42.708479",
|
||||||
"modified_by": "Administrator",
|
"modified_by": "Administrator",
|
||||||
"module": "LMS",
|
"module": "LMS",
|
||||||
"name": "LMS Batch Membership",
|
"name": "LMS Batch Membership",
|
||||||
|
|||||||
0
community/lms/doctype/lms_certification/__init__.py
Normal file
14
community/lms/doctype/lms_certification/lms_certification.js
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
// Copyright (c) 2021, FOSS United and contributors
|
||||||
|
// For license information, please see license.txt
|
||||||
|
|
||||||
|
frappe.ui.form.on('LMS Certification', {
|
||||||
|
onload: function (frm) {
|
||||||
|
frm.set_query("student", function (doc) {
|
||||||
|
return {
|
||||||
|
filters: {
|
||||||
|
"ignore_user_type": 1,
|
||||||
|
}
|
||||||
|
};
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
@@ -0,0 +1,70 @@
|
|||||||
|
{
|
||||||
|
"actions": [],
|
||||||
|
"creation": "2021-08-16 15:47:19.494055",
|
||||||
|
"doctype": "DocType",
|
||||||
|
"editable_grid": 1,
|
||||||
|
"engine": "InnoDB",
|
||||||
|
"field_order": [
|
||||||
|
"student",
|
||||||
|
"issue_date",
|
||||||
|
"column_break_3",
|
||||||
|
"course",
|
||||||
|
"expiry_date"
|
||||||
|
],
|
||||||
|
"fields": [
|
||||||
|
{
|
||||||
|
"fieldname": "student",
|
||||||
|
"fieldtype": "Link",
|
||||||
|
"in_list_view": 1,
|
||||||
|
"in_standard_filter": 1,
|
||||||
|
"label": "Student",
|
||||||
|
"options": "User"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "issue_date",
|
||||||
|
"fieldtype": "Date",
|
||||||
|
"label": "Issue Date"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "column_break_3",
|
||||||
|
"fieldtype": "Column Break"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "course",
|
||||||
|
"fieldtype": "Link",
|
||||||
|
"in_list_view": 1,
|
||||||
|
"in_standard_filter": 1,
|
||||||
|
"label": "Course",
|
||||||
|
"options": "LMS Course"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "expiry_date",
|
||||||
|
"fieldtype": "Date",
|
||||||
|
"label": "Expiry Date"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"index_web_pages_for_search": 1,
|
||||||
|
"links": [],
|
||||||
|
"modified": "2021-08-16 15:47:19.494055",
|
||||||
|
"modified_by": "Administrator",
|
||||||
|
"module": "LMS",
|
||||||
|
"name": "LMS Certification",
|
||||||
|
"owner": "Administrator",
|
||||||
|
"permissions": [
|
||||||
|
{
|
||||||
|
"create": 1,
|
||||||
|
"delete": 1,
|
||||||
|
"email": 1,
|
||||||
|
"export": 1,
|
||||||
|
"print": 1,
|
||||||
|
"read": 1,
|
||||||
|
"report": 1,
|
||||||
|
"role": "System Manager",
|
||||||
|
"share": 1,
|
||||||
|
"write": 1
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"sort_field": "modified",
|
||||||
|
"sort_order": "DESC",
|
||||||
|
"track_changes": 1
|
||||||
|
}
|
||||||
41
community/lms/doctype/lms_certification/lms_certification.py
Normal file
@@ -0,0 +1,41 @@
|
|||||||
|
# Copyright (c) 2021, FOSS United and contributors
|
||||||
|
# For license information, please see license.txt
|
||||||
|
|
||||||
|
import frappe
|
||||||
|
from frappe.model.document import Document
|
||||||
|
from frappe.utils import nowdate, add_years
|
||||||
|
from frappe import _
|
||||||
|
from frappe.utils.pdf import get_pdf
|
||||||
|
|
||||||
|
class LMSCertification(Document):
|
||||||
|
|
||||||
|
def validate(self):
|
||||||
|
certificates = frappe.get_all("LMS Certification", {
|
||||||
|
"student": self.student,
|
||||||
|
"course": self.course,
|
||||||
|
"expiry_date": [">", nowdate()]
|
||||||
|
})
|
||||||
|
if len(certificates):
|
||||||
|
full_name = frappe.db.get_value("User", self.student, "full_name")
|
||||||
|
course_name = frappe.db.get_value("LMS Course", self.course, "title")
|
||||||
|
frappe.throw(_("There is already a valid certificate for user {0} for the course {1}").format(full_name, course_name))
|
||||||
|
|
||||||
|
@frappe.whitelist()
|
||||||
|
def create_certificate(course):
|
||||||
|
course_details = frappe.get_doc("LMS Course", course)
|
||||||
|
certificate = course_details.is_certified()
|
||||||
|
|
||||||
|
if certificate:
|
||||||
|
return certificate
|
||||||
|
|
||||||
|
else:
|
||||||
|
expires_after_yrs = course_details.expiry
|
||||||
|
certificate = frappe.get_doc({
|
||||||
|
"doctype": "LMS Certification",
|
||||||
|
"student": frappe.session.user,
|
||||||
|
"course": course,
|
||||||
|
"issue_date": nowdate(),
|
||||||
|
"expiry_date": add_years(nowdate(), int(expires_after_yrs))
|
||||||
|
})
|
||||||
|
certificate.save(ignore_permissions=True)
|
||||||
|
return certificate.name
|
||||||
@@ -0,0 +1,8 @@
|
|||||||
|
# Copyright (c) 2021, FOSS United and Contributors
|
||||||
|
# See license.txt
|
||||||
|
|
||||||
|
# import frappe
|
||||||
|
import unittest
|
||||||
|
|
||||||
|
class TestLMSCertification(unittest.TestCase):
|
||||||
|
pass
|
||||||
@@ -2,7 +2,15 @@
|
|||||||
// For license information, please see license.txt
|
// For license information, please see license.txt
|
||||||
|
|
||||||
frappe.ui.form.on('LMS Course', {
|
frappe.ui.form.on('LMS Course', {
|
||||||
// refresh: function(frm) {
|
|
||||||
|
|
||||||
// }
|
onload: function (frm) {
|
||||||
|
frm.set_query("chapter", "chapters", function () {
|
||||||
|
return {
|
||||||
|
filters: {
|
||||||
|
"course": frm.doc.name,
|
||||||
|
}
|
||||||
|
};
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,11 +1,5 @@
|
|||||||
{
|
{
|
||||||
"actions": [
|
"actions": [
|
||||||
{
|
|
||||||
"action": "community.lms.doctype.lms_course.lms_course.reindex_lessons",
|
|
||||||
"action_type": "Server Action",
|
|
||||||
"group": "Reindex",
|
|
||||||
"label": "Reindex Lessons"
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
"action": "community.lms.doctype.lms_course.lms_course.reindex_exercises",
|
"action": "community.lms.doctype.lms_course.lms_course.reindex_exercises",
|
||||||
"action_type": "Server Action",
|
"action_type": "Server Action",
|
||||||
@@ -21,14 +15,20 @@
|
|||||||
"engine": "InnoDB",
|
"engine": "InnoDB",
|
||||||
"field_order": [
|
"field_order": [
|
||||||
"title",
|
"title",
|
||||||
"is_published",
|
|
||||||
"disable_self_learning",
|
|
||||||
"column_break_3",
|
|
||||||
"short_code",
|
|
||||||
"video_link",
|
"video_link",
|
||||||
|
"image",
|
||||||
|
"column_break_3",
|
||||||
|
"tags",
|
||||||
|
"is_published",
|
||||||
|
"upcoming",
|
||||||
|
"disable_self_learning",
|
||||||
"section_break_5",
|
"section_break_5",
|
||||||
"short_introduction",
|
"short_introduction",
|
||||||
"description"
|
"description",
|
||||||
|
"chapters",
|
||||||
|
"certification_section",
|
||||||
|
"enable_certification",
|
||||||
|
"expiry"
|
||||||
],
|
],
|
||||||
"fields": [
|
"fields": [
|
||||||
{
|
{
|
||||||
@@ -51,11 +51,6 @@
|
|||||||
"fieldtype": "Check",
|
"fieldtype": "Check",
|
||||||
"label": "Published"
|
"label": "Published"
|
||||||
},
|
},
|
||||||
{
|
|
||||||
"fieldname": "short_code",
|
|
||||||
"fieldtype": "Data",
|
|
||||||
"label": "Short Code"
|
|
||||||
},
|
|
||||||
{
|
{
|
||||||
"fieldname": "column_break_3",
|
"fieldname": "column_break_3",
|
||||||
"fieldtype": "Column Break"
|
"fieldtype": "Column Break"
|
||||||
@@ -80,6 +75,47 @@
|
|||||||
"fieldname": "disable_self_learning",
|
"fieldname": "disable_self_learning",
|
||||||
"fieldtype": "Check",
|
"fieldtype": "Check",
|
||||||
"label": "Disable Self Learning"
|
"label": "Disable Self Learning"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "image",
|
||||||
|
"fieldtype": "Attach Image",
|
||||||
|
"label": "Preview Image"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "tags",
|
||||||
|
"fieldtype": "Data",
|
||||||
|
"label": "Tags"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"default": "0",
|
||||||
|
"fieldname": "upcoming",
|
||||||
|
"fieldtype": "Check",
|
||||||
|
"label": "Is an Upcoming Course"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "chapters",
|
||||||
|
"fieldtype": "Table",
|
||||||
|
"label": "Chapters",
|
||||||
|
"options": "Chapters"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "certification_section",
|
||||||
|
"fieldtype": "Section Break",
|
||||||
|
"label": "Certification"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"default": "0",
|
||||||
|
"fieldname": "enable_certification",
|
||||||
|
"fieldtype": "Check",
|
||||||
|
"label": "Enable Certification"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"default": "0",
|
||||||
|
"depends_on": "enable_certification",
|
||||||
|
"fieldname": "expiry",
|
||||||
|
"fieldtype": "Select",
|
||||||
|
"label": "Certification Expires After Years",
|
||||||
|
"options": "0\n1\n2\n3\n4\n5\n6\n7\n8\n9\n10"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"index_web_pages_for_search": 1,
|
"index_web_pages_for_search": 1,
|
||||||
@@ -99,14 +135,9 @@
|
|||||||
"group": "Mentors",
|
"group": "Mentors",
|
||||||
"link_doctype": "LMS Course Mentor Mapping",
|
"link_doctype": "LMS Course Mentor Mapping",
|
||||||
"link_fieldname": "course"
|
"link_fieldname": "course"
|
||||||
},
|
|
||||||
{
|
|
||||||
"group": "Mentors",
|
|
||||||
"link_doctype": "LMS Mentor Request",
|
|
||||||
"link_fieldname": "course"
|
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"modified": "2021-06-21 11:34:04.552376",
|
"modified": "2021-08-18 18:02:12.623807",
|
||||||
"modified_by": "Administrator",
|
"modified_by": "Administrator",
|
||||||
"module": "LMS",
|
"module": "LMS",
|
||||||
"name": "LMS Course",
|
"name": "LMS Course",
|
||||||
@@ -129,6 +160,5 @@
|
|||||||
"sort_field": "creation",
|
"sort_field": "creation",
|
||||||
"sort_order": "DESC",
|
"sort_order": "DESC",
|
||||||
"title_field": "title",
|
"title_field": "title",
|
||||||
"track_changes": 1,
|
"track_changes": 1
|
||||||
"track_views": 1
|
|
||||||
}
|
}
|
||||||
@@ -8,9 +8,43 @@ from frappe.model.document import Document
|
|||||||
import json
|
import json
|
||||||
from ...utils import slugify
|
from ...utils import slugify
|
||||||
from community.query import find, find_all
|
from community.query import find, find_all
|
||||||
from frappe.utils import flt
|
from frappe.utils import flt, cint
|
||||||
|
from ...utils import slugify
|
||||||
|
|
||||||
class LMSCourse(Document):
|
class LMSCourse(Document):
|
||||||
|
|
||||||
|
def on_update(self):
|
||||||
|
if not self.upcoming and self.has_value_changed("upcoming"):
|
||||||
|
self.send_email_to_interested_users()
|
||||||
|
|
||||||
|
def send_email_to_interested_users(self):
|
||||||
|
interested_users = frappe.get_all("LMS Course Interest",
|
||||||
|
{
|
||||||
|
"course": self.name
|
||||||
|
},
|
||||||
|
["name", "user"])
|
||||||
|
subject = self.title + " is available!"
|
||||||
|
args = {
|
||||||
|
"title": self.title,
|
||||||
|
"course_link": "/courses/{0}".format(self.name),
|
||||||
|
"app_name": frappe.db.get_single_value("System Settings", "app_name"),
|
||||||
|
"site_url": frappe.utils.get_url()
|
||||||
|
}
|
||||||
|
|
||||||
|
for user in interested_users:
|
||||||
|
args["first_name"] = frappe.db.get_value("User", user.user, "first_name")
|
||||||
|
email_args = frappe._dict(
|
||||||
|
recipients = user.user,
|
||||||
|
sender = frappe.db.get_single_value("LMS Settings", "email_sender"),
|
||||||
|
subject = subject,
|
||||||
|
header = [subject, "green"],
|
||||||
|
template = "lms_course_interest",
|
||||||
|
args = args,
|
||||||
|
now = True
|
||||||
|
)
|
||||||
|
frappe.enqueue(method=frappe.sendmail, queue='short', timeout=300, is_async=True, **email_args)
|
||||||
|
frappe.db.set_value("LMS Course Interest", user.name, "email_sent", True)
|
||||||
|
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def find(name):
|
def find(name):
|
||||||
"""Returns the course with specified name.
|
"""Returns the course with specified name.
|
||||||
@@ -72,8 +106,11 @@ class LMSCourse(Document):
|
|||||||
mentors = frappe.get_all("LMS Course Mentor Mapping", {"course": self.name}, ["mentor"])
|
mentors = frappe.get_all("LMS Course Mentor Mapping", {"course": self.name}, ["mentor"])
|
||||||
for mentor in mentors:
|
for mentor in mentors:
|
||||||
member = frappe.get_doc("User", mentor.mentor)
|
member = frappe.get_doc("User", mentor.mentor)
|
||||||
# TODO: change this to count query
|
member.batch_count = frappe.db.count("LMS Batch Membership",
|
||||||
member.batch_count = len(frappe.get_all("LMS Batch Membership", {"member": member.name, "member_type": "Mentor"}))
|
{
|
||||||
|
"member": member.name,
|
||||||
|
"member_type": "Mentor"
|
||||||
|
})
|
||||||
course_mentors.append(member)
|
course_mentors.append(member)
|
||||||
return course_mentors
|
return course_mentors
|
||||||
|
|
||||||
@@ -112,8 +149,55 @@ class LMSCourse(Document):
|
|||||||
def get_chapters(self):
|
def get_chapters(self):
|
||||||
"""Returns all chapters of this course.
|
"""Returns all chapters of this course.
|
||||||
"""
|
"""
|
||||||
# TODO: chapters should have a way to specify the order
|
chapters = []
|
||||||
return find_all("Chapter", course=self.name, order_by="index_")
|
for row in self.chapters:
|
||||||
|
chapter_details = frappe.db.get_value("Chapter", row.chapter,
|
||||||
|
["name", "title", "description"],
|
||||||
|
as_dict=True)
|
||||||
|
chapter_details.idx = row.idx
|
||||||
|
chapters.append(chapter_details)
|
||||||
|
return chapters
|
||||||
|
|
||||||
|
def get_lessons(self, chapter=None):
|
||||||
|
""" If chapter is passed, returns lessons of only that chapter.
|
||||||
|
Else returns lessons of all chapters of the course """
|
||||||
|
lessons = []
|
||||||
|
|
||||||
|
if chapter:
|
||||||
|
return self.get_lesson_details(chapter)
|
||||||
|
|
||||||
|
for chapter in self.get_chapters():
|
||||||
|
lesson = self.get_lesson_details(chapter)
|
||||||
|
lessons += lesson
|
||||||
|
|
||||||
|
return lessons
|
||||||
|
|
||||||
|
def get_lesson_details(self, chapter):
|
||||||
|
lessons = []
|
||||||
|
lesson_list = frappe.get_all("Lessons", {"parent": chapter.name},
|
||||||
|
["lesson", "idx"], order_by="idx")
|
||||||
|
for row in lesson_list:
|
||||||
|
lesson_details = frappe.get_doc("Lesson", row.lesson)
|
||||||
|
lesson_details.number = flt("{}.{}".format(chapter.idx, row.idx))
|
||||||
|
lessons.append(lesson_details)
|
||||||
|
return lessons
|
||||||
|
|
||||||
|
def get_slugified_chapter_title(self, chapter):
|
||||||
|
return slugify(chapter)
|
||||||
|
|
||||||
|
def get_course_progress(self):
|
||||||
|
""" Returns the course progress of the session user """
|
||||||
|
lesson_count = len(self.get_lessons())
|
||||||
|
completed_lessons = frappe.db.count("LMS Course Progress",
|
||||||
|
{
|
||||||
|
"course": self.name,
|
||||||
|
"owner": frappe.session.user,
|
||||||
|
"status": "Complete"
|
||||||
|
})
|
||||||
|
precision = cint(frappe.db.get_default("float_precision")) or 3
|
||||||
|
if not lesson_count:
|
||||||
|
return 0
|
||||||
|
return flt(((completed_lessons/lesson_count) * 100), precision)
|
||||||
|
|
||||||
def get_batch(self, batch_name):
|
def get_batch(self, batch_name):
|
||||||
return find("LMS Batch", name=batch_name, course=self.name)
|
return find("LMS Batch", name=batch_name, course=self.name)
|
||||||
@@ -138,38 +222,18 @@ class LMSCourse(Document):
|
|||||||
visibility="Public")
|
visibility="Public")
|
||||||
return batches
|
return batches
|
||||||
|
|
||||||
def get_chapter(self, index):
|
|
||||||
return find("Chapter", course=self.name, index_=index)
|
|
||||||
|
|
||||||
def get_lesson(self, chapter_index, lesson_index):
|
|
||||||
chapter_name = frappe.get_value(
|
|
||||||
"Chapter",
|
|
||||||
{"course": self.name, "index_": chapter_index},
|
|
||||||
"name")
|
|
||||||
lesson_name = chapter_name and frappe.get_value(
|
|
||||||
"Lesson",
|
|
||||||
{"chapter": chapter_name, "index_": lesson_index},
|
|
||||||
"name")
|
|
||||||
return lesson_name and frappe.get_doc("Lesson", lesson_name)
|
|
||||||
|
|
||||||
def get_lesson_index(self, lesson_name):
|
def get_lesson_index(self, lesson_name):
|
||||||
"""Returns the {chapter_index}.{lesson_index} for the lesson.
|
"""Returns the {chapter_index}.{lesson_index} for the lesson.
|
||||||
"""
|
"""
|
||||||
lesson = frappe.get_doc("Lesson", lesson_name)
|
lesson = frappe.db.get_value("Lessons", {"lesson": lesson_name}, ["idx", "parent"], as_dict=True)
|
||||||
chapter = frappe.get_doc("Chapter", lesson.chapter)
|
if not lesson:
|
||||||
return f"{chapter.index_}.{lesson.index_}"
|
return None
|
||||||
|
|
||||||
def reindex_lessons(self):
|
chapter = frappe.db.get_value("Chapters", {"chapter": lesson.parent}, ["idx"], as_dict=True)
|
||||||
for i, c in enumerate(self.get_chapters(), start=1):
|
if not chapter:
|
||||||
c.index_ = i
|
return None
|
||||||
c.save()
|
|
||||||
self._reindex_lessons_in_chapter(c)
|
|
||||||
|
|
||||||
def _reindex_lessons_in_chapter(self, c):
|
return f"{chapter.idx}.{lesson.idx}"
|
||||||
for i, lesson in enumerate(c.get_lessons(), start=1):
|
|
||||||
lesson.index = i
|
|
||||||
lesson.index_label = f"{c.index_}.{i}"
|
|
||||||
lesson.save()
|
|
||||||
|
|
||||||
def reindex_exercises(self):
|
def reindex_exercises(self):
|
||||||
for i, c in enumerate(self.get_chapters(), start=1):
|
for i, c in enumerate(self.get_chapters(), start=1):
|
||||||
@@ -180,7 +244,7 @@ class LMSCourse(Document):
|
|||||||
|
|
||||||
def _reindex_exercises_in_chapter(self, c):
|
def _reindex_exercises_in_chapter(self, c):
|
||||||
i = 1
|
i = 1
|
||||||
for lesson in c.get_lessons():
|
for lesson in self.get_lessons(c):
|
||||||
for exercise in lesson.get_exercises():
|
for exercise in lesson.get_exercises():
|
||||||
exercise.index_ = i
|
exercise.index_ = i
|
||||||
exercise.index_label = f"{c.index_}.{i}"
|
exercise.index_label = f"{c.index_}.{i}"
|
||||||
@@ -199,7 +263,12 @@ class LMSCourse(Document):
|
|||||||
}
|
}
|
||||||
if batch:
|
if batch:
|
||||||
filters["batch"] = batch
|
filters["batch"] = batch
|
||||||
membership = frappe.db.get_value("LMS Batch Membership", filters, ["name","batch", "current_lesson"], as_dict=True)
|
|
||||||
|
membership = frappe.db.get_value("LMS Batch Membership",
|
||||||
|
filters,
|
||||||
|
["name", "batch", "current_lesson", "member_type"],
|
||||||
|
as_dict=True)
|
||||||
|
|
||||||
if membership and membership.batch:
|
if membership and membership.batch:
|
||||||
membership.batch_title = frappe.db.get_value("LMS Batch", membership.batch, "title")
|
membership.batch_title = frappe.db.get_value("LMS Batch", membership.batch, "title")
|
||||||
return membership
|
return membership
|
||||||
@@ -210,21 +279,6 @@ class LMSCourse(Document):
|
|||||||
membership.batch_title = frappe.db.get_value("LMS Batch", membership.batch, "title")
|
membership.batch_title = frappe.db.get_value("LMS Batch", membership.batch, "title")
|
||||||
return all_memberships
|
return all_memberships
|
||||||
|
|
||||||
def get_mentors(self, batch=None):
|
|
||||||
filters = {
|
|
||||||
"course": self.name,
|
|
||||||
"member_type": "Mentor"
|
|
||||||
}
|
|
||||||
if batch:
|
|
||||||
filters["batch"] = batch
|
|
||||||
|
|
||||||
memberships = frappe.get_all(
|
|
||||||
"LMS Batch Membership",
|
|
||||||
filters,
|
|
||||||
["member"])
|
|
||||||
member_names = [m['member'] for m in memberships]
|
|
||||||
return find_all("User", name=["IN", member_names])
|
|
||||||
|
|
||||||
def get_students(self, batch=None):
|
def get_students(self, batch=None):
|
||||||
"""Returns (email, full_name, username) of all the students of this batch as a list of dict.
|
"""Returns (email, full_name, username) of all the students of this batch as a list of dict.
|
||||||
"""
|
"""
|
||||||
@@ -241,58 +295,67 @@ class LMSCourse(Document):
|
|||||||
member_names = [m['member'] for m in memberships]
|
member_names = [m['member'] for m in memberships]
|
||||||
return find_all("User", name=["IN", member_names])
|
return find_all("User", name=["IN", member_names])
|
||||||
|
|
||||||
def get_outline(self):
|
def get_tags(self):
|
||||||
return CourseOutline(self)
|
return self.tags.split(",") if self.tags else []
|
||||||
|
|
||||||
class CourseOutline:
|
def get_reviews(self):
|
||||||
def __init__(self, course):
|
reviews = frappe.get_all("LMS Course Review",
|
||||||
self.course = course
|
{
|
||||||
self.chapters = self.get_chapters()
|
"course": self.name
|
||||||
self.lessons = self.get_lessons()
|
},
|
||||||
|
["review", "rating", "owner"],
|
||||||
|
order_by= "creation desc")
|
||||||
|
|
||||||
def get_next(self, current):
|
for review in reviews:
|
||||||
|
review.owner_details = frappe.get_doc("User", review.owner)
|
||||||
|
|
||||||
|
return reviews
|
||||||
|
|
||||||
|
def is_eligible_to_review(self, membership):
|
||||||
|
""" Checks if user is eligible to review the course """
|
||||||
|
if not membership:
|
||||||
|
return False
|
||||||
|
if frappe.db.count("LMS Course Review",
|
||||||
|
{
|
||||||
|
"course": self.name,
|
||||||
|
"owner": frappe.session.user
|
||||||
|
}):
|
||||||
|
return False
|
||||||
|
return True
|
||||||
|
|
||||||
|
def get_average_rating(self):
|
||||||
|
ratings = [review.rating for review in self.get_reviews()]
|
||||||
|
if not len(ratings):
|
||||||
|
return None
|
||||||
|
return sum(ratings)/len(ratings)
|
||||||
|
|
||||||
|
def get_progress(self, lesson):
|
||||||
|
return frappe.db.get_value("LMS Course Progress",
|
||||||
|
{
|
||||||
|
"course": self.name,
|
||||||
|
"owner": frappe.session.user,
|
||||||
|
"lesson": lesson
|
||||||
|
},
|
||||||
|
["status"])
|
||||||
|
|
||||||
|
def get_neighbours(self, current, lessons):
|
||||||
current = flt(current)
|
current = flt(current)
|
||||||
numbers = sorted(lesson['number'] for lesson in self.lessons)
|
numbers = sorted(lesson.number for lesson in lessons)
|
||||||
try:
|
|
||||||
index = numbers.index(current)
|
index = numbers.index(current)
|
||||||
return numbers[index+1]
|
return {
|
||||||
except IndexError:
|
"prev": numbers[index-1] if index-1 >= 0 else None,
|
||||||
return None
|
"next": numbers[index+1] if index+1 < len(numbers) else None
|
||||||
|
}
|
||||||
|
|
||||||
def get_prev(self, current):
|
def is_certified(self):
|
||||||
current = flt(current)
|
certificate = frappe.get_all("LMS Certification",
|
||||||
numbers = sorted(lesson['number'] for lesson in self.lessons)
|
{
|
||||||
try:
|
"student": frappe.session.user,
|
||||||
index = numbers.index(current)
|
"course": self.name
|
||||||
if index == 0:
|
})
|
||||||
return None
|
if len(certificate):
|
||||||
return numbers[index-1]
|
return certificate[0].name
|
||||||
except IndexError:
|
return
|
||||||
return None
|
|
||||||
|
|
||||||
def get_chapters(self):
|
|
||||||
return frappe.db.get_all("Chapter",
|
|
||||||
filters={"course": self.course.name},
|
|
||||||
fields=["name", "title", "index_"],
|
|
||||||
order_by="index_")
|
|
||||||
|
|
||||||
def get_lessons(self):
|
|
||||||
chapters = [c['name'] for c in self.chapters]
|
|
||||||
lessons = frappe.db.get_all("Lesson",
|
|
||||||
filters={"chapter": ["IN", chapters]},
|
|
||||||
fields=["name", "title", "chapter", "index_"])
|
|
||||||
|
|
||||||
chapter_numbers = {c['name']: c['index_'] for c in self.chapters}
|
|
||||||
for lesson in lessons:
|
|
||||||
lesson['number'] = flt("{}.{}".format(chapter_numbers[lesson['chapter']], lesson['index_']))
|
|
||||||
return lessons
|
|
||||||
|
|
||||||
@frappe.whitelist()
|
|
||||||
def reindex_lessons(doc):
|
|
||||||
course_data = json.loads(doc)
|
|
||||||
course = frappe.get_doc("LMS Course", course_data['name'])
|
|
||||||
course.reindex_lessons()
|
|
||||||
frappe.msgprint("All lessons in this course have been re-indexed.")
|
|
||||||
|
|
||||||
@frappe.whitelist()
|
@frappe.whitelist()
|
||||||
def reindex_exercises(doc):
|
def reindex_exercises(doc):
|
||||||
|
|||||||
@@ -0,0 +1,8 @@
|
|||||||
|
// Copyright (c) 2021, FOSS United and contributors
|
||||||
|
// For license information, please see license.txt
|
||||||
|
|
||||||
|
frappe.ui.form.on('LMS Course Interest', {
|
||||||
|
// refresh: function(frm) {
|
||||||
|
|
||||||
|
// }
|
||||||
|
});
|
||||||
@@ -0,0 +1,61 @@
|
|||||||
|
{
|
||||||
|
"actions": [],
|
||||||
|
"creation": "2021-08-06 17:37:20.184849",
|
||||||
|
"doctype": "DocType",
|
||||||
|
"editable_grid": 1,
|
||||||
|
"engine": "InnoDB",
|
||||||
|
"field_order": [
|
||||||
|
"course",
|
||||||
|
"user",
|
||||||
|
"email_sent"
|
||||||
|
],
|
||||||
|
"fields": [
|
||||||
|
{
|
||||||
|
"fieldname": "course",
|
||||||
|
"fieldtype": "Link",
|
||||||
|
"in_list_view": 1,
|
||||||
|
"in_standard_filter": 1,
|
||||||
|
"label": "Course",
|
||||||
|
"options": "LMS Course"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "user",
|
||||||
|
"fieldtype": "Link",
|
||||||
|
"in_list_view": 1,
|
||||||
|
"in_standard_filter": 1,
|
||||||
|
"label": "User",
|
||||||
|
"options": "User"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"default": "0",
|
||||||
|
"fieldname": "email_sent",
|
||||||
|
"fieldtype": "Check",
|
||||||
|
"label": "Email Sent",
|
||||||
|
"options": "email_sent"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"index_web_pages_for_search": 1,
|
||||||
|
"links": [],
|
||||||
|
"modified": "2021-08-06 18:06:21.370741",
|
||||||
|
"modified_by": "Administrator",
|
||||||
|
"module": "LMS",
|
||||||
|
"name": "LMS Course Interest",
|
||||||
|
"owner": "Administrator",
|
||||||
|
"permissions": [
|
||||||
|
{
|
||||||
|
"create": 1,
|
||||||
|
"delete": 1,
|
||||||
|
"email": 1,
|
||||||
|
"export": 1,
|
||||||
|
"print": 1,
|
||||||
|
"read": 1,
|
||||||
|
"report": 1,
|
||||||
|
"role": "System Manager",
|
||||||
|
"share": 1,
|
||||||
|
"write": 1
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"sort_field": "modified",
|
||||||
|
"sort_order": "DESC",
|
||||||
|
"track_changes": 1
|
||||||
|
}
|
||||||
@@ -0,0 +1,17 @@
|
|||||||
|
# Copyright (c) 2021, FOSS United and contributors
|
||||||
|
# For license information, please see license.txt
|
||||||
|
|
||||||
|
import frappe
|
||||||
|
from frappe.model.document import Document
|
||||||
|
|
||||||
|
class LMSCourseInterest(Document):
|
||||||
|
pass
|
||||||
|
|
||||||
|
@frappe.whitelist()
|
||||||
|
def capture_interest(course):
|
||||||
|
frappe.get_doc({
|
||||||
|
"doctype": "LMS Course Interest",
|
||||||
|
"course": course,
|
||||||
|
"user": frappe.session.user
|
||||||
|
}).save(ignore_permissions=True)
|
||||||
|
return "OK"
|
||||||
@@ -0,0 +1,8 @@
|
|||||||
|
# Copyright (c) 2021, FOSS United and Contributors
|
||||||
|
# See license.txt
|
||||||
|
|
||||||
|
# import frappe
|
||||||
|
import unittest
|
||||||
|
|
||||||
|
class TestLMSCourseInterest(unittest.TestCase):
|
||||||
|
pass
|
||||||
@@ -2,7 +2,13 @@
|
|||||||
// For license information, please see license.txt
|
// For license information, please see license.txt
|
||||||
|
|
||||||
frappe.ui.form.on('LMS Course Mentor Mapping', {
|
frappe.ui.form.on('LMS Course Mentor Mapping', {
|
||||||
// refresh: function(frm) {
|
onload: function(frm) {
|
||||||
|
frm.set_query('mentor', function(doc) {
|
||||||
// }
|
return {
|
||||||
|
filters: {
|
||||||
|
"ignore_user_type": 1,
|
||||||
|
}
|
||||||
|
};
|
||||||
|
});
|
||||||
|
},
|
||||||
});
|
});
|
||||||
|
|||||||
0
community/lms/doctype/lms_course_review/__init__.py
Normal file
@@ -1,7 +1,7 @@
|
|||||||
// Copyright (c) 2021, FOSS United and contributors
|
// Copyright (c) 2021, FOSS United and contributors
|
||||||
// For license information, please see license.txt
|
// For license information, please see license.txt
|
||||||
|
|
||||||
frappe.ui.form.on('LMS Sketch', {
|
frappe.ui.form.on('LMS Course Review', {
|
||||||
// refresh: function(frm) {
|
// refresh: function(frm) {
|
||||||
|
|
||||||
// }
|
// }
|
||||||
@@ -0,0 +1,57 @@
|
|||||||
|
{
|
||||||
|
"actions": [],
|
||||||
|
"creation": "2021-06-28 13:36:36.146718",
|
||||||
|
"doctype": "DocType",
|
||||||
|
"editable_grid": 1,
|
||||||
|
"engine": "InnoDB",
|
||||||
|
"field_order": [
|
||||||
|
"review",
|
||||||
|
"rating",
|
||||||
|
"course"
|
||||||
|
],
|
||||||
|
"fields": [
|
||||||
|
{
|
||||||
|
"fieldname": "review",
|
||||||
|
"fieldtype": "Small Text",
|
||||||
|
"in_list_view": 1,
|
||||||
|
"label": "Review"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "rating",
|
||||||
|
"fieldtype": "Rating",
|
||||||
|
"in_list_view": 1,
|
||||||
|
"label": "Rating"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "course",
|
||||||
|
"fieldtype": "Link",
|
||||||
|
"in_list_view": 1,
|
||||||
|
"label": "Course",
|
||||||
|
"options": "LMS Course"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"index_web_pages_for_search": 1,
|
||||||
|
"links": [],
|
||||||
|
"modified": "2021-07-05 14:57:03.841430",
|
||||||
|
"modified_by": "Administrator",
|
||||||
|
"module": "LMS",
|
||||||
|
"name": "LMS Course Review",
|
||||||
|
"owner": "Administrator",
|
||||||
|
"permissions": [
|
||||||
|
{
|
||||||
|
"create": 1,
|
||||||
|
"delete": 1,
|
||||||
|
"email": 1,
|
||||||
|
"export": 1,
|
||||||
|
"print": 1,
|
||||||
|
"read": 1,
|
||||||
|
"report": 1,
|
||||||
|
"role": "System Manager",
|
||||||
|
"share": 1,
|
||||||
|
"write": 1
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"sort_field": "modified",
|
||||||
|
"sort_order": "DESC",
|
||||||
|
"track_changes": 1
|
||||||
|
}
|
||||||
18
community/lms/doctype/lms_course_review/lms_course_review.py
Normal file
@@ -0,0 +1,18 @@
|
|||||||
|
# Copyright (c) 2021, FOSS United and contributors
|
||||||
|
# For license information, please see license.txt
|
||||||
|
|
||||||
|
import frappe
|
||||||
|
from frappe.model.document import Document
|
||||||
|
|
||||||
|
class LMSCourseReview(Document):
|
||||||
|
pass
|
||||||
|
|
||||||
|
@frappe.whitelist()
|
||||||
|
def submit_review(rating, review, course):
|
||||||
|
frappe.get_doc({
|
||||||
|
"doctype": "LMS Course Review",
|
||||||
|
"rating": rating,
|
||||||
|
"review": review,
|
||||||
|
"course": course
|
||||||
|
}).save(ignore_permissions=True)
|
||||||
|
return "OK"
|
||||||
@@ -0,0 +1,8 @@
|
|||||||
|
# Copyright (c) 2021, FOSS United and Contributors
|
||||||
|
# See license.txt
|
||||||
|
|
||||||
|
# import frappe
|
||||||
|
import unittest
|
||||||
|
|
||||||
|
class TestLMSCourseReview(unittest.TestCase):
|
||||||
|
pass
|
||||||
@@ -1,110 +0,0 @@
|
|||||||
# -*- coding: utf-8 -*-
|
|
||||||
# Copyright (c) 2021, FOSS United and contributors
|
|
||||||
# For license information, please see license.txt
|
|
||||||
|
|
||||||
from __future__ import unicode_literals
|
|
||||||
import frappe
|
|
||||||
from frappe.model.document import Document
|
|
||||||
from frappe import _
|
|
||||||
from frappe.utils import add_days, nowdate
|
|
||||||
|
|
||||||
class LMSMessage(Document):
|
|
||||||
def after_insert(self):
|
|
||||||
self.publish_message()
|
|
||||||
#Todo: Adding email preference field for users
|
|
||||||
#self.send_email()
|
|
||||||
|
|
||||||
def publish_message(self):
|
|
||||||
template = self.get_message_template()
|
|
||||||
message = frappe._dict({
|
|
||||||
"author_name": self.author_name,
|
|
||||||
"message_time": frappe.utils.format_datetime(self.creation, "dd-mm-yyyy HH:mm"),
|
|
||||||
"message": frappe.utils.md_to_html(self.message)
|
|
||||||
})
|
|
||||||
|
|
||||||
js = """
|
|
||||||
$(".msger-input").val("");
|
|
||||||
var template = `{0}`;
|
|
||||||
var message = {1};
|
|
||||||
var session_user = ("{2}" == frappe.session.user) ? true : false;
|
|
||||||
message.author_name = session_user ? "You" : message.author_name
|
|
||||||
message.is_author = session_user;
|
|
||||||
template = frappe.render_template(template, {{
|
|
||||||
"message": message
|
|
||||||
}})
|
|
||||||
$(".messages").append(template);
|
|
||||||
var message_element = document.getElementsByClassName("messages")[0]
|
|
||||||
message_element.scrollTo(0, message_element.scrollHeight);
|
|
||||||
""".format(template, message, self.owner)
|
|
||||||
|
|
||||||
frappe.publish_realtime(event="eval_js", message=js, after_commit=True)
|
|
||||||
|
|
||||||
def get_message_template(self):
|
|
||||||
return """
|
|
||||||
<li class="{% if message.is_author %} ours {% endif %}">
|
|
||||||
<div class="d-flex justify-content-between">
|
|
||||||
<div class="font-weight-bold">
|
|
||||||
{{ message.author_name }}
|
|
||||||
</div>
|
|
||||||
<small class="">
|
|
||||||
{{ message.message_time }}
|
|
||||||
</small>
|
|
||||||
</div>
|
|
||||||
<div class="message-para">
|
|
||||||
{{ message.message }}
|
|
||||||
</div>
|
|
||||||
</li>
|
|
||||||
"""
|
|
||||||
|
|
||||||
def send_email(self):
|
|
||||||
membership = frappe.get_all("LMS Batch Membership", {"batch": self.batch}, ["member"])
|
|
||||||
for entry in membership:
|
|
||||||
member = frappe.get_doc("User", entry.member)
|
|
||||||
if member.name != self.author:
|
|
||||||
#Todo: wrap sendmail in frappe.enqueue, else messages takes long to display.
|
|
||||||
frappe.sendmail(
|
|
||||||
recipients = member.email,
|
|
||||||
subject = _("New Message on ") + self.batch,
|
|
||||||
header = _("New Message on ") + self.batch,
|
|
||||||
template = "lms_message",
|
|
||||||
args = {
|
|
||||||
"author": self.author,
|
|
||||||
"message": frappe.utils.md_to_html(self.message),
|
|
||||||
"creation": frappe.utils.format_datetime(self.creation, "medium"),
|
|
||||||
"course": frappe.db.get_value("LMS Batch", self.batch, ["course"])
|
|
||||||
}
|
|
||||||
)
|
|
||||||
|
|
||||||
def send_daily_digest():
|
|
||||||
#Todo: Optimize this
|
|
||||||
emails = frappe._dict()
|
|
||||||
messages = frappe.get_all("LMS Message", {"creation": [">=", add_days(nowdate(), -1)]}, ["message", "batch", "author", "creation"])
|
|
||||||
for message in messages:
|
|
||||||
membership = frappe.get_all("LMS Batch Membership", {"batch": message.batch}, ["member"])
|
|
||||||
for entry in membership:
|
|
||||||
member = frappe.db.get_value("User", entry.member, ["name", "email"], as_dict=1)
|
|
||||||
if member.name != message.author:
|
|
||||||
if member.name in emails.keys():
|
|
||||||
emails[member.name]["messages"].append(message)
|
|
||||||
else:
|
|
||||||
emails[member.name] = frappe._dict({
|
|
||||||
"email": member.email,
|
|
||||||
"messages": [message]
|
|
||||||
})
|
|
||||||
for email in emails:
|
|
||||||
group_by_batch = frappe._dict()
|
|
||||||
for message in emails[email]["messages"]:
|
|
||||||
if message.batch in group_by_batch.keys():
|
|
||||||
group_by_batch[message.batch].append(message)
|
|
||||||
else:
|
|
||||||
group_by_batch[message.batch] = [message]
|
|
||||||
frappe.sendmail(
|
|
||||||
recipients = frappe.db.get_value("User", email, "email"),
|
|
||||||
subject = _("Message Digest"),
|
|
||||||
header = _("Message Digest"),
|
|
||||||
template = "lms_daily_digest",
|
|
||||||
args = {
|
|
||||||
"batches": group_by_batch
|
|
||||||
},
|
|
||||||
delayed = False
|
|
||||||
)
|
|
||||||
@@ -27,12 +27,13 @@
|
|||||||
"fieldname": "lesson",
|
"fieldname": "lesson",
|
||||||
"fieldtype": "Link",
|
"fieldtype": "Link",
|
||||||
"label": "Lesson",
|
"label": "Lesson",
|
||||||
"options": "Lesson"
|
"options": "Lesson",
|
||||||
|
"read_only": 1
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"index_web_pages_for_search": 1,
|
"index_web_pages_for_search": 1,
|
||||||
"links": [],
|
"links": [],
|
||||||
"modified": "2021-06-23 17:58:57.642873",
|
"modified": "2021-07-23 19:06:12.551633",
|
||||||
"modified_by": "Administrator",
|
"modified_by": "Administrator",
|
||||||
"module": "LMS",
|
"module": "LMS",
|
||||||
"name": "LMS Quiz",
|
"name": "LMS Quiz",
|
||||||
|
|||||||
@@ -46,34 +46,30 @@ class LMSQuiz(Document):
|
|||||||
return result[0]
|
return result[0]
|
||||||
|
|
||||||
@frappe.whitelist()
|
@frappe.whitelist()
|
||||||
def submit(quiz, result):
|
def quiz_summary(quiz, results):
|
||||||
score = 0
|
score = 0
|
||||||
answer_map = {
|
results = json.loads(results)
|
||||||
"is_correct_1": "option_1",
|
|
||||||
"is_correct_2": "option_2",
|
|
||||||
"is_correct_3": "option_3",
|
|
||||||
"is_correct_4": "option_4"
|
|
||||||
}
|
|
||||||
result = json.loads(result)
|
|
||||||
quiz_details = frappe.get_doc("LMS Quiz", quiz)
|
|
||||||
|
|
||||||
for response in result:
|
for result in results:
|
||||||
match = list(filter(lambda x: x.question == response.get("question"), quiz_details.questions))[0]
|
correct = result["is_correct"][0]
|
||||||
correct_options = quiz_details.get_correct_options(match)
|
result["question"] = frappe.db.get_value("LMS Quiz Question",
|
||||||
correct_answers = [ match.get(answer_map[option]) for option in correct_options ]
|
{"parent": quiz, "idx": result["question_index"]},
|
||||||
|
["question"])
|
||||||
|
|
||||||
if response.get("answer") == correct_answers:
|
for point in result["is_correct"]:
|
||||||
response["result"] = "Right"
|
correct = correct and point
|
||||||
score += 1
|
|
||||||
else:
|
result["result"] = "Right" if correct else "Wrong"
|
||||||
response["result"] = "Wrong"
|
score += correct
|
||||||
response["answer"] = ("").join([ ans if idx == len(response.get("answer")) -1 else ans + ", " for idx, ans in enumerate(response.get("answer")) ])
|
|
||||||
|
del result["is_correct"]
|
||||||
|
del result["question_index"]
|
||||||
|
|
||||||
frappe.get_doc({
|
frappe.get_doc({
|
||||||
"doctype": "LMS Quiz Submission",
|
"doctype": "LMS Quiz Submission",
|
||||||
"quiz": quiz,
|
"quiz": quiz,
|
||||||
"result": result,
|
"result": results,
|
||||||
"score": score
|
"score": score
|
||||||
}).save(ignore_permissions=True)
|
}).save(ignore_permissions=True)
|
||||||
update_progress(quiz_details.lesson)
|
|
||||||
return score
|
return score
|
||||||
|
|||||||
@@ -3,6 +3,39 @@
|
|||||||
|
|
||||||
# import frappe
|
# import frappe
|
||||||
import unittest
|
import unittest
|
||||||
|
import frappe
|
||||||
|
|
||||||
class TestLMSQuiz(unittest.TestCase):
|
class TestLMSQuiz(unittest.TestCase):
|
||||||
pass
|
|
||||||
|
@classmethod
|
||||||
|
def setUpClass(cls) -> None:
|
||||||
|
frappe.get_doc({
|
||||||
|
"doctype": "LMS Quiz",
|
||||||
|
"title": "Test Quiz"
|
||||||
|
}).save()
|
||||||
|
|
||||||
|
def test_with_multiple_options(self):
|
||||||
|
quiz = frappe.get_doc("LMS Quiz", "Test Quiz")
|
||||||
|
quiz.append("questions", {
|
||||||
|
"question": "Question multiple",
|
||||||
|
"option_1": "Option 1",
|
||||||
|
"is_correct_1": 1,
|
||||||
|
"option_2": "Option 2",
|
||||||
|
"is_correct_2": 1
|
||||||
|
})
|
||||||
|
quiz.save()
|
||||||
|
self.assertTrue(quiz.questions[0].multiple)
|
||||||
|
|
||||||
|
def test_with_no_correct_option(self):
|
||||||
|
quiz = frappe.get_doc("LMS Quiz", "Test Quiz")
|
||||||
|
quiz.append("questions", {
|
||||||
|
"question": "Question no correct option",
|
||||||
|
"option_1": "Option 1",
|
||||||
|
"option_2": "Option 2",
|
||||||
|
})
|
||||||
|
self.assertRaises(frappe.ValidationError, quiz.save)
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def tearDownClass(cls) -> None:
|
||||||
|
frappe.db.delete("LMS Quiz", "Test Quiz")
|
||||||
|
frappe.db.delete("LMS Quiz Question", {"parent": "Test Quiz"})
|
||||||
|
|||||||
@@ -9,15 +9,23 @@
|
|||||||
"options_section",
|
"options_section",
|
||||||
"option_1",
|
"option_1",
|
||||||
"is_correct_1",
|
"is_correct_1",
|
||||||
|
"column_break_5",
|
||||||
|
"explanation_1",
|
||||||
"section_break_5",
|
"section_break_5",
|
||||||
"option_2",
|
"option_2",
|
||||||
"is_correct_2",
|
"is_correct_2",
|
||||||
|
"column_break_10",
|
||||||
|
"explanation_2",
|
||||||
"column_break_4",
|
"column_break_4",
|
||||||
"option_3",
|
"option_3",
|
||||||
"is_correct_3",
|
"is_correct_3",
|
||||||
|
"column_break_15",
|
||||||
|
"explanation_3",
|
||||||
"section_break_11",
|
"section_break_11",
|
||||||
"option_4",
|
"option_4",
|
||||||
"is_correct_4",
|
"is_correct_4",
|
||||||
|
"column_break_20",
|
||||||
|
"explanation_4",
|
||||||
"multiple"
|
"multiple"
|
||||||
],
|
],
|
||||||
"fields": [
|
"fields": [
|
||||||
@@ -101,12 +109,52 @@
|
|||||||
{
|
{
|
||||||
"fieldname": "section_break_11",
|
"fieldname": "section_break_11",
|
||||||
"fieldtype": "Section Break"
|
"fieldtype": "Section Break"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"depends_on": "option_1",
|
||||||
|
"fieldname": "explanation_1",
|
||||||
|
"fieldtype": "Data",
|
||||||
|
"label": "Explanation"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"depends_on": "option_2",
|
||||||
|
"fieldname": "explanation_2",
|
||||||
|
"fieldtype": "Data",
|
||||||
|
"label": "Explanation"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"depends_on": "option_3",
|
||||||
|
"fieldname": "explanation_3",
|
||||||
|
"fieldtype": "Data",
|
||||||
|
"label": "Explanation"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"depends_on": "option_4",
|
||||||
|
"fieldname": "explanation_4",
|
||||||
|
"fieldtype": "Data",
|
||||||
|
"label": "Explanation"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "column_break_5",
|
||||||
|
"fieldtype": "Column Break"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "column_break_10",
|
||||||
|
"fieldtype": "Column Break"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "column_break_15",
|
||||||
|
"fieldtype": "Column Break"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"fieldname": "column_break_20",
|
||||||
|
"fieldtype": "Column Break"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"index_web_pages_for_search": 1,
|
"index_web_pages_for_search": 1,
|
||||||
"istable": 1,
|
"istable": 1,
|
||||||
"links": [],
|
"links": [],
|
||||||
"modified": "2021-06-22 16:54:13.133859",
|
"modified": "2021-07-19 19:35:28.446236",
|
||||||
"modified_by": "Administrator",
|
"modified_by": "Administrator",
|
||||||
"module": "LMS",
|
"module": "LMS",
|
||||||
"name": "LMS Quiz Question",
|
"name": "LMS Quiz Question",
|
||||||
|
|||||||
@@ -1,126 +0,0 @@
|
|||||||
"""Utilities to work with livecode service.
|
|
||||||
"""
|
|
||||||
import websocket
|
|
||||||
import json
|
|
||||||
from .svg import SVG
|
|
||||||
import frappe
|
|
||||||
from urllib.parse import urlparse
|
|
||||||
|
|
||||||
# Files to pass to livecode server
|
|
||||||
# The same code is part of livecode-canvas.js
|
|
||||||
# TODO: generate livecode-canvas.js from this file
|
|
||||||
START = '''
|
|
||||||
import sketch
|
|
||||||
code = open("main.py").read()
|
|
||||||
env = dict(sketch.__dict__)
|
|
||||||
exec(code, env)
|
|
||||||
'''
|
|
||||||
|
|
||||||
SKETCH = '''
|
|
||||||
import json
|
|
||||||
|
|
||||||
def sendmsg(msgtype, function, args):
|
|
||||||
"""Sends a message to the frontend.
|
|
||||||
|
|
||||||
The frontend will receive the specified message whenever
|
|
||||||
this function is called. The frontend can decide to some
|
|
||||||
action on each of these messages.
|
|
||||||
"""
|
|
||||||
msg = dict(msgtype=msgtype, function=function, args=args)
|
|
||||||
print("--MSG--", json.dumps(msg))
|
|
||||||
|
|
||||||
def _draw(func, **kwargs):
|
|
||||||
sendmsg(msgtype="draw", function=func, args=kwargs)
|
|
||||||
|
|
||||||
def circle(x, y, d):
|
|
||||||
"""Draws a circle of diameter d with center (x, y).
|
|
||||||
"""
|
|
||||||
_draw("circle", x=x, y=y, d=d)
|
|
||||||
|
|
||||||
def line(x1, y1, x2, y2):
|
|
||||||
"""Draws a line from point (x1, y1) to point (x2, y2).
|
|
||||||
"""
|
|
||||||
_draw("line", x1=x1, y1=y1, x2=x2, y2=y2)
|
|
||||||
|
|
||||||
def rect(x, y, w, h):
|
|
||||||
"""Draws a rectangle on the canvas.
|
|
||||||
|
|
||||||
Parameters
|
|
||||||
----------
|
|
||||||
x: x coordinate of the top-left corner of the rectangle
|
|
||||||
y: y coordinate of the top-left corner of the rectangle
|
|
||||||
w: width of the rectangle
|
|
||||||
h: height of the rectangle
|
|
||||||
"""
|
|
||||||
_draw("rect", x=x, y=y, w=w, h=h)
|
|
||||||
|
|
||||||
def clear():
|
|
||||||
_draw("clear")
|
|
||||||
|
|
||||||
# clear the canvas on start
|
|
||||||
clear()
|
|
||||||
'''
|
|
||||||
|
|
||||||
def get_livecode_url():
|
|
||||||
doc = frappe.get_cached_doc("LMS Settings")
|
|
||||||
return doc.livecode_url
|
|
||||||
|
|
||||||
def get_livecode_ws_url():
|
|
||||||
url = urlparse(get_livecode_url())
|
|
||||||
protocol = "wss" if url.scheme == "https" else "ws"
|
|
||||||
return protocol + "://" + url.netloc + "/livecode"
|
|
||||||
|
|
||||||
def livecode_to_svg(livecode_ws_url, code, *, timeout=3):
|
|
||||||
"""Renders the code as svg.
|
|
||||||
"""
|
|
||||||
if livecode_ws_url is None:
|
|
||||||
livecode_ws_url = get_livecode_ws_url()
|
|
||||||
|
|
||||||
try:
|
|
||||||
ws = websocket.WebSocket()
|
|
||||||
ws.settimeout(timeout)
|
|
||||||
ws.connect(livecode_ws_url)
|
|
||||||
|
|
||||||
msg = {
|
|
||||||
"msgtype": "exec",
|
|
||||||
"runtime": "python",
|
|
||||||
"code": code,
|
|
||||||
"files": [
|
|
||||||
{"filename": "start.py", "contents": START},
|
|
||||||
{"filename": "sketch.py", "contents": SKETCH},
|
|
||||||
],
|
|
||||||
"command": ["python", "start.py"]
|
|
||||||
}
|
|
||||||
ws.send(json.dumps(msg))
|
|
||||||
|
|
||||||
messages = _read_messages(ws)
|
|
||||||
commands = [m for m in messages if m['msgtype'] == 'draw']
|
|
||||||
img = draw_image(commands)
|
|
||||||
return img.tostring()
|
|
||||||
except websocket.WebSocketException as e:
|
|
||||||
frappe.log_error(frappe.get_traceback(), 'livecode_to_svg failed')
|
|
||||||
|
|
||||||
def _read_messages(ws):
|
|
||||||
messages = []
|
|
||||||
try:
|
|
||||||
while True:
|
|
||||||
msg = ws.recv()
|
|
||||||
if not msg:
|
|
||||||
break
|
|
||||||
messages.append(json.loads(msg))
|
|
||||||
except websocket.WebSocketTimeoutException as e:
|
|
||||||
print("Error:", e)
|
|
||||||
pass
|
|
||||||
return messages
|
|
||||||
|
|
||||||
def draw_image(commands):
|
|
||||||
img = SVG(width=300, height=300, viewBox="0 0 300 300", fill='none', stroke='black')
|
|
||||||
for c in commands:
|
|
||||||
args = c['args']
|
|
||||||
if c['function'] == 'circle':
|
|
||||||
img.circle(cx=args['x'], cy=args['y'], r=args['d']/2)
|
|
||||||
elif c['function'] == 'line':
|
|
||||||
img.line(x1=args['x1'], y1=args['y1'], x2=args['x2'], y2=args['y2'])
|
|
||||||
elif c['function'] == 'rect':
|
|
||||||
img.rect(x=args['x'], y=args['y'], width=args['w'], height=args['h'])
|
|
||||||
return img
|
|
||||||
@@ -1,105 +0,0 @@
|
|||||||
# -*- coding: utf-8 -*-
|
|
||||||
# Copyright (c) 2021, FOSS United and contributors
|
|
||||||
# For license information, please see license.txt
|
|
||||||
|
|
||||||
from __future__ import unicode_literals
|
|
||||||
import hashlib
|
|
||||||
from urllib.parse import urlparse
|
|
||||||
import frappe
|
|
||||||
from frappe.model.document import Document
|
|
||||||
from . import livecode
|
|
||||||
|
|
||||||
DEFAULT_IMAGE = """
|
|
||||||
<svg viewBox="0 0 300 300" width="300" xmlns="http://www.w3.org/2000/svg">
|
|
||||||
</svg>
|
|
||||||
"""
|
|
||||||
|
|
||||||
class LMSSketch(Document):
|
|
||||||
@property
|
|
||||||
def sketch_id(self):
|
|
||||||
"""Returns the numeric part of the name.
|
|
||||||
|
|
||||||
For example, the skech_id will be "123" for sketch with name "SKETCH-123".
|
|
||||||
"""
|
|
||||||
return self.name.replace("SKETCH-", "")
|
|
||||||
|
|
||||||
def get_owner(self):
|
|
||||||
"""Returns the owner of this sketch as a document.
|
|
||||||
"""
|
|
||||||
return frappe.get_doc("User", self.owner)
|
|
||||||
|
|
||||||
def get_owner_name(self):
|
|
||||||
return self.get_owner().full_name
|
|
||||||
|
|
||||||
def get_livecode_url(self):
|
|
||||||
doc = frappe.get_cached_doc("LMS Settings")
|
|
||||||
return doc.livecode_url
|
|
||||||
|
|
||||||
def get_livecode_ws_url(self):
|
|
||||||
url = urlparse(self.get_livecode_url())
|
|
||||||
protocol = "wss" if url.scheme == "https" else "ws"
|
|
||||||
return protocol + "://" + url.netloc + "/livecode"
|
|
||||||
|
|
||||||
def to_svg(self):
|
|
||||||
return self.svg or self.render_svg()
|
|
||||||
|
|
||||||
def render_svg(self):
|
|
||||||
h = hashlib.md5(self.code.encode('utf-8')).hexdigest()
|
|
||||||
cache = frappe.cache()
|
|
||||||
key = "sketch-" + h
|
|
||||||
value = cache.get(key)
|
|
||||||
if value:
|
|
||||||
value = value.decode('utf-8')
|
|
||||||
else:
|
|
||||||
ws_url = self.get_livecode_ws_url()
|
|
||||||
value = livecode.livecode_to_svg(ws_url, self.code)
|
|
||||||
if value:
|
|
||||||
cache.set(key, value)
|
|
||||||
return value or DEFAULT_IMAGE
|
|
||||||
|
|
||||||
@staticmethod
|
|
||||||
def get_recent_sketches(limit=100, owner=None):
|
|
||||||
"""Returns the recent sketches.
|
|
||||||
"""
|
|
||||||
filters = {}
|
|
||||||
if owner:
|
|
||||||
filters = {"owner": owner}
|
|
||||||
sketches = frappe.get_all(
|
|
||||||
"LMS Sketch",
|
|
||||||
filters=filters,
|
|
||||||
fields='*',
|
|
||||||
order_by='modified desc',
|
|
||||||
page_length=limit
|
|
||||||
)
|
|
||||||
return [frappe.get_doc(doctype='LMS Sketch', **doc) for doc in sketches]
|
|
||||||
|
|
||||||
def __repr__(self):
|
|
||||||
return f"<LMSSketch {self.name}>"
|
|
||||||
|
|
||||||
@frappe.whitelist()
|
|
||||||
def save_sketch(name, title, code):
|
|
||||||
if not name or name == "new":
|
|
||||||
doc = frappe.new_doc('LMS Sketch')
|
|
||||||
doc.title = title
|
|
||||||
doc.code = code
|
|
||||||
doc.runtime = 'python-canvas'
|
|
||||||
doc.insert(ignore_permissions=True)
|
|
||||||
status = "created"
|
|
||||||
else:
|
|
||||||
doc = frappe.get_doc("LMS Sketch", name)
|
|
||||||
|
|
||||||
if doc.owner != frappe.session.user:
|
|
||||||
return {
|
|
||||||
"ok": False,
|
|
||||||
"error": "Permission Denied"
|
|
||||||
}
|
|
||||||
doc.title = title
|
|
||||||
doc.code = code
|
|
||||||
doc.svg = ''
|
|
||||||
doc.save()
|
|
||||||
status = "updated"
|
|
||||||
return {
|
|
||||||
"ok": True,
|
|
||||||
"status": status,
|
|
||||||
"name": doc.name,
|
|
||||||
}
|
|
||||||
@@ -1,143 +0,0 @@
|
|||||||
"""SVG rendering library.
|
|
||||||
|
|
||||||
USAGE:
|
|
||||||
from svg import SVG
|
|
||||||
|
|
||||||
svg = SVG(width=200, height=200)
|
|
||||||
svg.circle(cx=100, cy=200, r=50)
|
|
||||||
print(svg.tostring())
|
|
||||||
"""
|
|
||||||
from xml.etree import ElementTree
|
|
||||||
|
|
||||||
TAGNAMES = set([
|
|
||||||
"circle", "ellipse",
|
|
||||||
"line", "path", "rect", "polygon", "polyline",
|
|
||||||
"text", "textPath", "title",
|
|
||||||
"marker", "defs",
|
|
||||||
"g"
|
|
||||||
])
|
|
||||||
|
|
||||||
class Node:
|
|
||||||
"""SVG Node"""
|
|
||||||
def __init__(self, tag, **attrs):
|
|
||||||
self.tag = tag
|
|
||||||
self.attrs = dict((k.replace('_', '-'), str(v)) for k, v in attrs.items())
|
|
||||||
self.children = []
|
|
||||||
|
|
||||||
def node(self, tag, **attrs):
|
|
||||||
n = Node(tag, **attrs)
|
|
||||||
self.children.append(n)
|
|
||||||
return n
|
|
||||||
|
|
||||||
def apply(self, func):
|
|
||||||
"""Applies a function to this node and
|
|
||||||
all the children recursively.
|
|
||||||
"""
|
|
||||||
func(self)
|
|
||||||
for n in self.children:
|
|
||||||
n.apply(func)
|
|
||||||
|
|
||||||
def clone(self):
|
|
||||||
node = Node(self.tag, **self.attrs)
|
|
||||||
node.children = [n.clone() for n in self.children]
|
|
||||||
return node
|
|
||||||
|
|
||||||
def add_node(self, node):
|
|
||||||
if not isinstance(node, Node):
|
|
||||||
node = Text(node)
|
|
||||||
self.children.append(node)
|
|
||||||
|
|
||||||
def __getattr__(self, tag):
|
|
||||||
if tag not in TAGNAMES:
|
|
||||||
raise AttributeError(tag)
|
|
||||||
return lambda **attrs: self.node(tag, **attrs)
|
|
||||||
|
|
||||||
def translate(self, x, y):
|
|
||||||
return self.g(transform="translate(%s, %s)" % (x, y))
|
|
||||||
|
|
||||||
def scale(self, *args):
|
|
||||||
return self.g(transform="scale(%s)" % ", ".join(str(a) for a in args))
|
|
||||||
|
|
||||||
def __repr__(self):
|
|
||||||
return "<%s .../>" % self.tag
|
|
||||||
|
|
||||||
def __enter__(self):
|
|
||||||
return self
|
|
||||||
|
|
||||||
def __exit__(self, type, value, traceback):
|
|
||||||
pass
|
|
||||||
|
|
||||||
def build_tree(self, builder):
|
|
||||||
builder.start(self.tag, self.attrs)
|
|
||||||
for node in self.children:
|
|
||||||
node.build_tree(builder)
|
|
||||||
return builder.end(self.tag)
|
|
||||||
|
|
||||||
def _indent(self, elem, level=0):
|
|
||||||
"""Indent etree node for prettyprinting."""
|
|
||||||
|
|
||||||
i = "\n" + level*" "
|
|
||||||
if len(elem):
|
|
||||||
if not elem.text or not elem.text.strip():
|
|
||||||
elem.text = i + " "
|
|
||||||
if not elem.tail or not elem.tail.strip():
|
|
||||||
elem.tail = i
|
|
||||||
for elem in elem:
|
|
||||||
self._indent(elem, level+1)
|
|
||||||
if not elem.tail or not elem.tail.strip():
|
|
||||||
elem.tail = i
|
|
||||||
else:
|
|
||||||
if level and (not elem.tail or not elem.tail.strip()):
|
|
||||||
elem.tail = i
|
|
||||||
|
|
||||||
def save(self, filename, encoding='utf-8'):
|
|
||||||
f = open(filename, 'w')
|
|
||||||
f.write(self.tostring())
|
|
||||||
f.close()
|
|
||||||
|
|
||||||
def tostring(self, encoding='utf-8'):
|
|
||||||
builder = ElementTree.TreeBuilder()
|
|
||||||
self.build_tree(builder)
|
|
||||||
e = builder.close()
|
|
||||||
self._indent(e)
|
|
||||||
return ElementTree.tostring(e, encoding).decode(encoding)
|
|
||||||
|
|
||||||
class Text(Node):
|
|
||||||
"""Text Node
|
|
||||||
|
|
||||||
>>> p = Node("p")
|
|
||||||
>>> p.add_node("hello, world!")
|
|
||||||
>>> p.tostring()
|
|
||||||
'<p>hello, world!</p>'
|
|
||||||
"""
|
|
||||||
def __init__(self, text):
|
|
||||||
Node.__init__(self, "__text__")
|
|
||||||
self._text = text
|
|
||||||
|
|
||||||
def build_tree(self, builder):
|
|
||||||
builder.data(str(self._text))
|
|
||||||
|
|
||||||
class SVG(Node):
|
|
||||||
"""
|
|
||||||
>>> svg = SVG(width=200, height=200)
|
|
||||||
>>> svg.rect(x=0, y=0, width=200, height=200, fill="blue")
|
|
||||||
<rect .../>
|
|
||||||
>>> with svg.translate(-50, -50) as g:
|
|
||||||
... g.rect(x=0, y=0, width=50, height=100, fill="red")
|
|
||||||
... g.rect(x=50, y=0, width=50, height=100, fill="green")
|
|
||||||
<rect .../>
|
|
||||||
<rect .../>
|
|
||||||
>>> print(svg.tostring())
|
|
||||||
<svg width="200" height="200" xmlns="http://www.w3.org/2000/svg">
|
|
||||||
<rect x="0" y="0" width="200" height="200" fill="blue" />
|
|
||||||
<g transform="translate(-50, -50)">
|
|
||||||
<rect x="0" y="0" width="50" height="100" fill="red" />
|
|
||||||
<rect x="50" y="0" width="50" height="100" fill="green" />
|
|
||||||
</g>
|
|
||||||
</svg>
|
|
||||||
|
|
||||||
"""
|
|
||||||
def __init__(self, **attrs):
|
|
||||||
attrs['xmlns'] = "http://www.w3.org/2000/svg"
|
|
||||||
Node.__init__(self, 'svg', **attrs)
|
|
||||||
|
|
||||||
@@ -93,11 +93,11 @@ class MacroInlineProcessor(InlineProcessor):
|
|||||||
macro = m.group(1)
|
macro = m.group(1)
|
||||||
arg = m.group(2)
|
arg = m.group(2)
|
||||||
html = render_macro(macro, arg)
|
html = render_macro(macro, arg)
|
||||||
html = sanitize_html(str(html))
|
html = sanitize_html(str(html), macro)
|
||||||
e = etree.fromstring(html)
|
e = etree.fromstring(html)
|
||||||
return e, m.start(0), m.end(0)
|
return e, m.start(0), m.end(0)
|
||||||
|
|
||||||
def sanitize_html(html):
|
def sanitize_html(html, macro):
|
||||||
"""Sanotize the html using BeautifulSoup.
|
"""Sanotize the html using BeautifulSoup.
|
||||||
|
|
||||||
The markdown processor request the correct markup and crashes on
|
The markdown processor request the correct markup and crashes on
|
||||||
@@ -106,4 +106,7 @@ def sanitize_html(html):
|
|||||||
"""
|
"""
|
||||||
soup = BeautifulSoup(html, features="lxml")
|
soup = BeautifulSoup(html, features="lxml")
|
||||||
nodes = soup.body.children
|
nodes = soup.body.children
|
||||||
return "<div>" + "\n".join(str(node) for node in nodes) + "</div>"
|
classname = ""
|
||||||
|
if macro == "YouTubeVideo":
|
||||||
|
classname = "lesson-video"
|
||||||
|
return "<div class='" + classname + "'>" + "\n".join(str(node) for node in nodes) + "</div>"
|
||||||
|
|||||||
@@ -1,5 +1,4 @@
|
|||||||
"""Handy module to make access to all doctypes from a single place.
|
"""Handy module to make access to all doctypes from a single place.
|
||||||
"""
|
"""
|
||||||
from .doctype.lms_course.lms_course import LMSCourse as Course
|
from .doctype.lms_course.lms_course import LMSCourse as Course
|
||||||
from .doctype.lms_sketch.lms_sketch import LMSSketch as Sketch
|
|
||||||
from .doctype.lms_batch_membership.lms_batch_membership import LMSBatchMembership as Membership
|
from .doctype.lms_batch_membership.lms_batch_membership import LMSBatchMembership as Membership
|
||||||
|
|||||||
0
community/lms/web_form/profile/__init__.py
Normal file
14
community/lms/web_form/profile/profile.js
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
frappe.ready(function () {
|
||||||
|
|
||||||
|
frappe.web_form.after_load = () => {
|
||||||
|
if (!frappe.utils.get_url_arg("name")) {
|
||||||
|
window.location.href = `/edit-profile?name=${frappe.session.user}`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
frappe.web_form.after_save = () => {
|
||||||
|
setTimeout(() => {
|
||||||
|
window.location.href = `/${frappe.web_form.get_value(["username"])}`;
|
||||||
|
})
|
||||||
|
}
|
||||||
|
})
|
||||||
212
community/lms/web_form/profile/profile.json
Normal file
@@ -0,0 +1,212 @@
|
|||||||
|
{
|
||||||
|
"accept_payment": 0,
|
||||||
|
"allow_comments": 0,
|
||||||
|
"allow_delete": 0,
|
||||||
|
"allow_edit": 1,
|
||||||
|
"allow_incomplete": 0,
|
||||||
|
"allow_multiple": 0,
|
||||||
|
"allow_print": 0,
|
||||||
|
"amount": 0.0,
|
||||||
|
"amount_based_on_field": 0,
|
||||||
|
"apply_document_permissions": 0,
|
||||||
|
"breadcrumbs": "",
|
||||||
|
"button_label": "Save",
|
||||||
|
"client_script": "",
|
||||||
|
"creation": "2021-06-30 13:48:13.682851",
|
||||||
|
"custom_css": "[data-doctype=\"Web Form\"] {\n max-width: 720px;\n margin: 6rem auto;\n}",
|
||||||
|
"doc_type": "User",
|
||||||
|
"docstatus": 0,
|
||||||
|
"doctype": "Web Form",
|
||||||
|
"idx": 0,
|
||||||
|
"is_standard": 1,
|
||||||
|
"login_required": 1,
|
||||||
|
"max_attachment_size": 0,
|
||||||
|
"modified": "2021-08-06 14:40:39.013776",
|
||||||
|
"modified_by": "Administrator",
|
||||||
|
"module": "LMS",
|
||||||
|
"name": "profile",
|
||||||
|
"owner": "Administrator",
|
||||||
|
"payment_button_label": "Buy Now",
|
||||||
|
"published": 1,
|
||||||
|
"route": "edit-profile",
|
||||||
|
"route_to_success_link": 0,
|
||||||
|
"show_attachments": 0,
|
||||||
|
"show_in_grid": 0,
|
||||||
|
"show_sidebar": 0,
|
||||||
|
"sidebar_items": [],
|
||||||
|
"success_url": "/profile",
|
||||||
|
"title": "Profile",
|
||||||
|
"web_form_fields": [
|
||||||
|
{
|
||||||
|
"allow_read_on_all_link_options": 0,
|
||||||
|
"fieldname": "first_name",
|
||||||
|
"fieldtype": "Data",
|
||||||
|
"hidden": 0,
|
||||||
|
"label": "First Name",
|
||||||
|
"max_length": 0,
|
||||||
|
"max_value": 0,
|
||||||
|
"read_only": 0,
|
||||||
|
"reqd": 1,
|
||||||
|
"show_in_filter": 0
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"allow_read_on_all_link_options": 0,
|
||||||
|
"fieldname": "middle_name",
|
||||||
|
"fieldtype": "Data",
|
||||||
|
"hidden": 0,
|
||||||
|
"label": "Middle Name (Optional)",
|
||||||
|
"max_length": 0,
|
||||||
|
"max_value": 0,
|
||||||
|
"read_only": 0,
|
||||||
|
"reqd": 0,
|
||||||
|
"show_in_filter": 0
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"allow_read_on_all_link_options": 0,
|
||||||
|
"fieldname": "last_name",
|
||||||
|
"fieldtype": "Data",
|
||||||
|
"hidden": 0,
|
||||||
|
"label": "Last Name",
|
||||||
|
"max_length": 0,
|
||||||
|
"max_value": 0,
|
||||||
|
"read_only": 0,
|
||||||
|
"reqd": 0,
|
||||||
|
"show_in_filter": 0
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"allow_read_on_all_link_options": 0,
|
||||||
|
"fieldname": "username",
|
||||||
|
"fieldtype": "Data",
|
||||||
|
"hidden": 0,
|
||||||
|
"label": "Username",
|
||||||
|
"max_length": 0,
|
||||||
|
"max_value": 0,
|
||||||
|
"read_only": 0,
|
||||||
|
"reqd": 0,
|
||||||
|
"show_in_filter": 0
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"allow_read_on_all_link_options": 0,
|
||||||
|
"description": "Get your globally recognized avatar from Gravatar.com",
|
||||||
|
"fieldname": "user_image",
|
||||||
|
"fieldtype": "Attach Image",
|
||||||
|
"hidden": 0,
|
||||||
|
"label": "User Image",
|
||||||
|
"max_length": 0,
|
||||||
|
"max_value": 0,
|
||||||
|
"read_only": 0,
|
||||||
|
"reqd": 0,
|
||||||
|
"show_in_filter": 0
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"allow_read_on_all_link_options": 0,
|
||||||
|
"fieldname": "mobile_no",
|
||||||
|
"fieldtype": "Data",
|
||||||
|
"hidden": 0,
|
||||||
|
"label": "Mobile No",
|
||||||
|
"max_length": 0,
|
||||||
|
"max_value": 0,
|
||||||
|
"options": "Phone",
|
||||||
|
"read_only": 0,
|
||||||
|
"reqd": 0,
|
||||||
|
"show_in_filter": 0
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"allow_read_on_all_link_options": 0,
|
||||||
|
"fieldname": "bio",
|
||||||
|
"fieldtype": "Small Text",
|
||||||
|
"hidden": 0,
|
||||||
|
"label": "Bio",
|
||||||
|
"max_length": 0,
|
||||||
|
"max_value": 0,
|
||||||
|
"read_only": 0,
|
||||||
|
"reqd": 0,
|
||||||
|
"show_in_filter": 0
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"allow_read_on_all_link_options": 0,
|
||||||
|
"fieldname": "linkedin",
|
||||||
|
"fieldtype": "Data",
|
||||||
|
"hidden": 0,
|
||||||
|
"label": "LinkedIn ID",
|
||||||
|
"max_length": 0,
|
||||||
|
"max_value": 0,
|
||||||
|
"read_only": 0,
|
||||||
|
"reqd": 0,
|
||||||
|
"show_in_filter": 0
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"allow_read_on_all_link_options": 0,
|
||||||
|
"fieldname": "github",
|
||||||
|
"fieldtype": "Data",
|
||||||
|
"hidden": 0,
|
||||||
|
"label": "Github ID",
|
||||||
|
"max_length": 0,
|
||||||
|
"max_value": 0,
|
||||||
|
"read_only": 0,
|
||||||
|
"reqd": 0,
|
||||||
|
"show_in_filter": 0
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"allow_read_on_all_link_options": 0,
|
||||||
|
"fieldname": "medium",
|
||||||
|
"fieldtype": "Data",
|
||||||
|
"hidden": 0,
|
||||||
|
"label": "Medium ID",
|
||||||
|
"max_length": 0,
|
||||||
|
"max_value": 0,
|
||||||
|
"read_only": 0,
|
||||||
|
"reqd": 0,
|
||||||
|
"show_in_filter": 0
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"allow_read_on_all_link_options": 0,
|
||||||
|
"fieldname": "city",
|
||||||
|
"fieldtype": "Data",
|
||||||
|
"hidden": 0,
|
||||||
|
"label": "City",
|
||||||
|
"max_length": 0,
|
||||||
|
"max_value": 0,
|
||||||
|
"read_only": 0,
|
||||||
|
"reqd": 0,
|
||||||
|
"show_in_filter": 0
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"allow_read_on_all_link_options": 0,
|
||||||
|
"fieldname": "college",
|
||||||
|
"fieldtype": "Data",
|
||||||
|
"hidden": 0,
|
||||||
|
"label": "College Name",
|
||||||
|
"max_length": 0,
|
||||||
|
"max_value": 0,
|
||||||
|
"read_only": 0,
|
||||||
|
"reqd": 0,
|
||||||
|
"show_in_filter": 0
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"allow_read_on_all_link_options": 0,
|
||||||
|
"depends_on": "college",
|
||||||
|
"fieldname": "branch",
|
||||||
|
"fieldtype": "Data",
|
||||||
|
"hidden": 0,
|
||||||
|
"label": "Branch",
|
||||||
|
"max_length": 0,
|
||||||
|
"max_value": 0,
|
||||||
|
"read_only": 0,
|
||||||
|
"reqd": 0,
|
||||||
|
"show_in_filter": 0
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"allow_read_on_all_link_options": 0,
|
||||||
|
"fieldname": "profession",
|
||||||
|
"fieldtype": "Data",
|
||||||
|
"hidden": 0,
|
||||||
|
"label": "Profession",
|
||||||
|
"max_length": 0,
|
||||||
|
"max_value": 0,
|
||||||
|
"read_only": 0,
|
||||||
|
"reqd": 0,
|
||||||
|
"show_in_filter": 0
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
5
community/lms/web_form/profile/profile.py
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
import frappe
|
||||||
|
|
||||||
|
def get_context(context):
|
||||||
|
# do your magic here
|
||||||
|
pass
|
||||||
@@ -1,67 +0,0 @@
|
|||||||
<div class="mt-5">
|
|
||||||
<a class="anchor_style" href="/courses">Courses</a> /{% if course.is_mentor(frappe.session.user) %} <a
|
|
||||||
class="anchor_style" href="/courses/{{ course.name }}"> {{ course.title }}</a> {% else %} <span class="text-muted">
|
|
||||||
{{ course.title }}</span> {% endif %}
|
|
||||||
{% set all_memberships = course.get_all_memberships(frappe.session.user) %}
|
|
||||||
{% if membership and membership.batch and all_memberships | length > 1 %}
|
|
||||||
<a class="pull-right dropdown-item border rounded" style="width: 10rem;" href="#" id="navbarDropdown" role="button"
|
|
||||||
data-toggle="dropdown" aria-haspopup="true" aria-expanded="false">
|
|
||||||
{{ membership.batch_title }}
|
|
||||||
</a>
|
|
||||||
<div class="dropdown-menu" aria-labelledby="navbarDropdown">
|
|
||||||
{% for data in all_memberships %}
|
|
||||||
{% if data.batch != membership.batch %}
|
|
||||||
<a class="dropdown-item switch-batch"
|
|
||||||
href="/courses/{{ course.name }}/home?batch={{ data.batch }}">{{ data.batch_title }}</a>
|
|
||||||
{% endif %}
|
|
||||||
{% endfor %}
|
|
||||||
</div>
|
|
||||||
{% endif %}
|
|
||||||
</div>
|
|
||||||
{% if not membership %}
|
|
||||||
{% set display_class = "hide" %}
|
|
||||||
{% else %}
|
|
||||||
{% set display_class = "" %}
|
|
||||||
{% endif %}
|
|
||||||
|
|
||||||
<ul class="nav nav-tabs mt-4">
|
|
||||||
<li class="nav-item">
|
|
||||||
<a class="nav-link" id="home" href="/courses/{{course.name}}/home{{ course.query_parameter }}">Home</a>
|
|
||||||
</li>
|
|
||||||
<li class="nav-item">
|
|
||||||
{% set lesson_index = course.get_lesson_index(membership.current_lesson) if membership and membership.current_lesson
|
|
||||||
else '1.1' %}
|
|
||||||
<a class="nav-link" id="learn"
|
|
||||||
href="{{ course.get_learn_url(lesson_index) }}{{ course.query_parameter }}">Lessons</a>
|
|
||||||
</li>
|
|
||||||
<!-- <li class="nav-item">
|
|
||||||
<a class="nav-link" id="schedule" href="/courses/{{course.name}}/schedule">Schedule</a>
|
|
||||||
</li> -->
|
|
||||||
<li class="nav-item {{ display_class }}">
|
|
||||||
<a class="nav-link" id="members" href="/courses/{{course.name}}/members{{ course.query_parameter }}">Members</a>
|
|
||||||
</li>
|
|
||||||
<!-- <li class="nav-item {{ display_class }}">
|
|
||||||
<a class="nav-link" id="discussion" href="/courses/{{course.name}}/discuss">Discussion</a>
|
|
||||||
</li> -->
|
|
||||||
<!-- <li class="nav-item">
|
|
||||||
<a class="nav-link" id="about" href="/courses/{{course.name}}/about">About</a>
|
|
||||||
</li> -->
|
|
||||||
{% if membership and membership.batch and course.is_mentor(frappe.session.user) %}
|
|
||||||
<li class="nav-item">
|
|
||||||
<a class="nav-link" id="progress" href="/courses/{{course.name}}/progress{{ course.query_parameter }}">Progress</a>
|
|
||||||
</li>
|
|
||||||
{% endif %}
|
|
||||||
</ul>
|
|
||||||
{% block script %}
|
|
||||||
<script>
|
|
||||||
frappe.ready(() => {
|
|
||||||
var selector = document.querySelector(`a[href="${decodeURIComponent(window.location.pathname)}{{ course.query_parameter }}"]`)
|
|
||||||
if (selector) {
|
|
||||||
selector.classList.add('active');
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
$("#learn").addClass('active')
|
|
||||||
}
|
|
||||||
})
|
|
||||||
</script>
|
|
||||||
{% endblock %}
|
|
||||||
21
community/lms/widgets/BreadCrumb.html
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
<div class="breadcrumb">
|
||||||
|
|
||||||
|
|
||||||
|
{% if course %}
|
||||||
|
<a class="dark-links" href="/courses">All Courses</a>
|
||||||
|
<img class="ml-1 mr-1" src="/assets/community/icons/chevron-right.svg">
|
||||||
|
{% if lesson %}
|
||||||
|
<a class="dark-links" href="/courses/{{ course.name }}">{{ course.title }}</a>
|
||||||
|
<img class="ml-1 mr-1" src="/assets/community/icons/chevron-right.svg">
|
||||||
|
<span class="muted-text"> {{ lesson.title }}</span>
|
||||||
|
{% else %}
|
||||||
|
<span class="muted-text">{{ course.title }}</span>
|
||||||
|
{% endif %}
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
{% if thread %}
|
||||||
|
<a class="dark-links" href="/discussions">Discussions</a>
|
||||||
|
<img class="ml-1 mr-1" src="/assets/community/icons/chevron-right.svg">
|
||||||
|
<span class="muted-text">{{ thread.title }}</span>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
@@ -1,28 +1,100 @@
|
|||||||
<div class="chapter-teaser">
|
<div>
|
||||||
<div class="teaser-body">
|
|
||||||
<div class="chapter-title mb-5 font-weight-bold"><span class="mr-1">{{index}}.</span> {{ chapter.title }}</div>
|
<div class="chapter-title small-title" data-target="#{{ course.get_slugified_chapter_title(chapter.title) }}"
|
||||||
<div class="chapter-description">
|
data-toggle="collapse" aria-expanded="false">
|
||||||
{{ chapter.description or "" }}
|
<img class="chapter-icon" src="/assets/community/icons/chevron-right.svg">
|
||||||
|
{{ index }}. {{ chapter.title }}
|
||||||
</div>
|
</div>
|
||||||
<div class="chapter-lessons">
|
|
||||||
{% for lesson in chapter.get_lessons() %}
|
<div class="chapter-content collapse navbar-collapse" id="{{ course.get_slugified_chapter_title(chapter.title) }}">
|
||||||
<div class="lesson-teaser">
|
|
||||||
{% if show_link or lesson.include_in_preview %}
|
{% if chapter.description %}
|
||||||
<a class="" href="{{ course.get_learn_url(course.get_lesson_index(lesson.name)) }}{{course.query_parameter}}"
|
<div class="chapter-description muted-text">
|
||||||
data-course="{{ course.name }}">{{ lesson.title }}</a>
|
{{ chapter.description }}
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
<div class="lessons">
|
||||||
|
|
||||||
|
{% for lesson in course.get_lessons(chapter) %}
|
||||||
|
|
||||||
|
<div class="lesson-info{% if membership.current_lesson == lesson.name %} active-lesson {% endif %}">
|
||||||
|
|
||||||
|
{% if membership or lesson.include_in_preview %}
|
||||||
|
<a class="lesson-links"
|
||||||
|
href="{{ course.get_learn_url(lesson.number) }}{{course.query_parameter}}"
|
||||||
|
data-course="{{ course.name }}">
|
||||||
|
{{ lesson.title }}
|
||||||
|
|
||||||
|
{% if membership %}
|
||||||
|
<img class="ml-1 lesson-progress-tick {{ course.get_progress(lesson.name) != 'Complete' and 'hide' }}"
|
||||||
|
src="/assets/community/icons/check.svg">
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
</a>
|
||||||
|
|
||||||
{% else %}
|
{% else %}
|
||||||
<div class="no-preview" title="This lesson is not available for preview">
|
<div class="no-preview" title="This lesson is not available for preview">
|
||||||
<span style="color: #2490ef;">
|
<div class="lesson-links">
|
||||||
{{ lesson.title }}
|
{{ lesson.title }}
|
||||||
</span>
|
<img class="ml-2" src="/assets/community/icons/lock.svg">
|
||||||
<i class="fa fa-lock ml-2"></i>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
{% endif %}
|
{% endif %}
|
||||||
{% if show_progress and not course.is_mentor(frappe.session.user) and lesson.get_progress() %}
|
|
||||||
<span class="ml-5 badge p-2 {{ lesson.get_slugified_class() }}"> {{ lesson.get_progress() }}</span>
|
|
||||||
{% endif %}
|
|
||||||
</div>
|
</div>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
{% if index != course.get_chapters() | length %}
|
||||||
|
<div class="card-divider"></div>
|
||||||
|
{% endif %}
|
||||||
|
|
||||||
|
<script>
|
||||||
|
frappe.ready(() => {
|
||||||
|
expand_the_active_chapter();
|
||||||
|
})
|
||||||
|
|
||||||
|
var expand_the_first_chapter = () => {
|
||||||
|
var elements = $(".collapse");
|
||||||
|
elements.each((i, element) => {
|
||||||
|
if (i <= 1) {
|
||||||
|
show_section(element);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
var expand_the_active_chapter = () => {
|
||||||
|
|
||||||
|
/* Find anchor matching the URL for course details page */
|
||||||
|
var selector = $(`a[href="${decodeURIComponent(window.location.pathname)}"]`).parent();
|
||||||
|
if (!selector.length) {
|
||||||
|
selector = $(`a[href^="${decodeURIComponent(window.location.pathname)}"]`).parent();
|
||||||
|
}
|
||||||
|
if (selector.length && $(".course-details-page").length) {
|
||||||
|
$(".lesson-info").removeClass("active-lesson")
|
||||||
|
selector.addClass("active-lesson");
|
||||||
|
show_section(selector.parent().parent());
|
||||||
|
}
|
||||||
|
|
||||||
|
/* For course home page */
|
||||||
|
else if ($(".active-lesson").length) {
|
||||||
|
selector = $(".active-lesson")
|
||||||
|
show_section(selector.parent().parent());
|
||||||
|
}
|
||||||
|
|
||||||
|
/* If no active chapter then exapand the first chapter */
|
||||||
|
else {
|
||||||
|
expand_the_first_chapter();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var show_section = (element) => {
|
||||||
|
$(element).addClass("show");
|
||||||
|
$(element).siblings(".chapter-title").children(".chapter-icon").css("transform", "rotate(90deg)");
|
||||||
|
}
|
||||||
|
|
||||||
|
</script>
|
||||||
|
|||||||
90
community/lms/widgets/CourseCard.html
Normal file
@@ -0,0 +1,90 @@
|
|||||||
|
<div class="common-card-style course-card">
|
||||||
|
<div class="course-image {% if not course.image %}default-image{% endif %}"
|
||||||
|
{% if course.image %} style="background-image: url( {{ course.image }} );" {% endif %}>
|
||||||
|
<div class="course-tags">
|
||||||
|
{% for tag in course.get_tags() %}
|
||||||
|
<div class="course-card-pills">{{ tag }}</div>
|
||||||
|
{% endfor %}
|
||||||
|
</div>
|
||||||
|
{% if not course.image %}
|
||||||
|
<div class="default-image-text">{{ course.title[0] }}</div>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
<div class="course-card-content">
|
||||||
|
<div class="course-card-meta muted-text">
|
||||||
|
{% if course.get_chapters() | length %}
|
||||||
|
<span>
|
||||||
|
{{ course.get_chapters() | length }} Chapters
|
||||||
|
</span>
|
||||||
|
{% endif %}
|
||||||
|
{% if course.get_chapters() | length and course.get_upcoming_batches() | length %}
|
||||||
|
<span class="font-weight-bold ml-3 mr-3"> . </span>
|
||||||
|
{% endif %}
|
||||||
|
{% if course.get_upcoming_batches() | length %}
|
||||||
|
<span class="">
|
||||||
|
{{ course.get_upcoming_batches() | length }} Open Batches
|
||||||
|
</span>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
<div class="course-card-title">{{ course.title }}</div>
|
||||||
|
<div class="card-divider"></div>
|
||||||
|
<div class="course-card-meta-2">
|
||||||
|
{{ widgets.Avatar(member=course.get_instructor(), avatar_class="avatar-small") }}
|
||||||
|
<span class="course-instructor">
|
||||||
|
{{ course.get_instructor().full_name }}
|
||||||
|
</span>
|
||||||
|
<span class="course-student-count">
|
||||||
|
{% if course.get_students() | length %}
|
||||||
|
<span class="mr-4">
|
||||||
|
<img class="icon-background" src="/assets/community/icons/user.svg" />
|
||||||
|
{{ course.get_students() | length }}
|
||||||
|
</span>
|
||||||
|
{% endif %}
|
||||||
|
{% set avg_rating = course.get_average_rating() %}
|
||||||
|
{% if avg_rating %}
|
||||||
|
<span>
|
||||||
|
<img class="icon-background" src="/assets/community/icons/rating.svg" />
|
||||||
|
{{ avg_rating }}
|
||||||
|
</span>
|
||||||
|
{% endif %}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
{% set membership = course.get_membership(frappe.session.user) %}
|
||||||
|
|
||||||
|
{% set lesson_index = course.get_lesson_index(membership.current_lesson) if membership and
|
||||||
|
membership.current_lesson
|
||||||
|
else '1.1' %}
|
||||||
|
|
||||||
|
{% set query_parameter = "?batch=" + membership.batch if membership and membership.batch else "" %}
|
||||||
|
|
||||||
|
{% if course.upcoming %}
|
||||||
|
<div class="view-course-link is-secondary border">
|
||||||
|
Upcoming Course <img class="ml-3" src="/assets/community/icons/black-arrow.svg" />
|
||||||
|
</div>
|
||||||
|
<a class="stretched-link" href="/courses/{{ course.name }}"></a>
|
||||||
|
|
||||||
|
{% elif membership %}
|
||||||
|
<div class="view-course-link is-primary">
|
||||||
|
Continue Course <img class="ml-3" src="/assets/community/icons/white-arrow.svg" />
|
||||||
|
</div>
|
||||||
|
<a class="stretched-link" href="{{ course.get_learn_url(lesson_index) }}{{ query_parameter }}"></a>
|
||||||
|
|
||||||
|
{% else %}
|
||||||
|
<div class="view-course-link">
|
||||||
|
View Course <img class="ml-3" src="/assets/community/icons/black-arrow.svg" />
|
||||||
|
</div>
|
||||||
|
<a class="stretched-link" href="/courses/{{ course.name }}"></a>
|
||||||
|
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<script>
|
||||||
|
frappe.ready(() => {
|
||||||
|
$(".course-card-title").each((i, element) => {
|
||||||
|
var title = $(element).text();
|
||||||
|
var length = $(window).width() <= 375 ? 60 : 65;
|
||||||
|
var suffix = title.length > length ? "..." : "";
|
||||||
|
$(element).text(title.substring(0, length) + suffix);
|
||||||
|
})
|
||||||
|
})
|
||||||
|
</script>
|
||||||
@@ -1,7 +1,12 @@
|
|||||||
<div class="mt-5">
|
{% if course.get_chapters() | length %}
|
||||||
<h3> Course Outline </h3>
|
<div class="">
|
||||||
|
<div class="course-home-headings">
|
||||||
|
Course Outline
|
||||||
|
</div>
|
||||||
|
<div class="coure-outline">
|
||||||
{% for chapter in course.get_chapters() %}
|
{% for chapter in course.get_chapters() %}
|
||||||
{{ widgets.ChapterTeaser(index=loop.index, chapter=chapter, course=course, batch=batch, show_link=show_link, show_progress=show_progress)}}
|
{{ widgets.ChapterTeaser(index=loop.index, chapter=chapter, course=course, batch=batch, membership=membership) }}
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
</div>
|
</div>
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
|||||||
15
community/lms/widgets/MemberCard.html
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
<div class="common-card-style member-card {{dimension_class}} ">
|
||||||
|
{% set avatar_class = "avatar-large" if not dimension_class else "avatar-large"%}
|
||||||
|
{{ widgets.Avatar(member=member, avatar_class=avatar_class) }}
|
||||||
|
<div class="small-title member-card-title">
|
||||||
|
{{ member.full_name }}
|
||||||
|
</div>
|
||||||
|
{% set course_count = member.get_authored_courses() | length %}
|
||||||
|
{% if show_course_count and course_count > 0 %}
|
||||||
|
{% set suffix = "Courses" if course_count > 1 else "Course" %}
|
||||||
|
<div class="small-title">
|
||||||
|
Created {{ course_count }} {{ suffix }}
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
<a class="stretched-link" href="/{{ member.username }}"></a>
|
||||||
|
</div>
|
||||||
@@ -1,30 +0,0 @@
|
|||||||
<div class="batch">
|
|
||||||
<div class="batch-details">
|
|
||||||
<div class="">Session every {{batch.sessions_on}}</div>
|
|
||||||
<div>{{frappe.utils.format_time(batch.start_time, "short")}} -
|
|
||||||
{{frappe.utils.format_time(batch.end_time, "short")}}
|
|
||||||
</div>
|
|
||||||
<div>Starting {{frappe.utils.format_date(batch.start_date, "medium")}}</div>
|
|
||||||
<div class="course-type" style="color: #888; padding: 10px 0px;">mentors</div>
|
|
||||||
|
|
||||||
{% for m in course.get_mentors(batch.name) %}
|
|
||||||
<div>
|
|
||||||
{{ widgets.Avatar(member=m, avatar_class="avatar-medium" ) }}
|
|
||||||
<span class="instructor-title">{{m.full_name}}</span>
|
|
||||||
</div>
|
|
||||||
{% endfor %}
|
|
||||||
</div>
|
|
||||||
{% if can_manage or can_join %}
|
|
||||||
<div class="cta">
|
|
||||||
<div class="">
|
|
||||||
{% if can_manage %}
|
|
||||||
<a href="/courses/{{ course.name }}/home?batch={{ batch.name }}" class="btn btn-primary manage-batch" data-batch="{{ batch.name | urlencode }}"
|
|
||||||
data-course="{{ course.name | urlencode }}">Manage</a>
|
|
||||||
{% elif can_join %}
|
|
||||||
<button class="join-batch btn btn-secondary" data-batch="{{ batch.name | urlencode }}"
|
|
||||||
data-course="{{ course.name | urlencode }}">Join this Batch</button>
|
|
||||||
{% endif %}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{% endif %}
|
|
||||||
</div>
|
|
||||||
91
community/lms/widgets/Reviews.html
Normal file
@@ -0,0 +1,91 @@
|
|||||||
|
<div class="reviews-parent">
|
||||||
|
{% set reviews = course.get_reviews() %}
|
||||||
|
{% if reviews | length or course.is_eligible_to_review(membership) %}
|
||||||
|
<div class="mb-5">
|
||||||
|
<span class="course-home-headings">Reviews</span>
|
||||||
|
{% if course.is_eligible_to_review(membership) %}
|
||||||
|
<span class="review-link button is-secondary pull-right">
|
||||||
|
Write a review
|
||||||
|
</span>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
{% if reviews | length %}
|
||||||
|
<div class="reviews-section">
|
||||||
|
{% for review in reviews %}
|
||||||
|
<div class="review-card">
|
||||||
|
<div class="common-card-style review-content small-title"> {{ review.review }} </div>
|
||||||
|
<div class="review-card-footer">
|
||||||
|
<div>
|
||||||
|
{{ widgets.Avatar(member=review.owner_details, avatar_class="avatar-medium") }}
|
||||||
|
<a class="button-links" href="/{{review.owner_details.username}}">
|
||||||
|
<span class="course-instructor">
|
||||||
|
{{ review.owner_details.full_name }}
|
||||||
|
</span>
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
<div class="rating">
|
||||||
|
{% for i in [1, 2, 3, 4, 5] %}
|
||||||
|
<svg class="icon icon-md {% if i <= review.rating %} star-click {% endif %}" data-rating="{{ i }}">
|
||||||
|
<use href="#icon-star"></use>
|
||||||
|
</svg>
|
||||||
|
{% endfor %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{% endfor %}
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="modal fade review-modal" id="review-modal" tabindex="-1" role="dialog" aria-labelledby="exampleModalLabel"
|
||||||
|
aria-hidden="true">
|
||||||
|
<div class="modal-dialog" role="document">
|
||||||
|
<div class="modal-content">
|
||||||
|
<div class="modal-header">
|
||||||
|
<div class="course-home-headings modal-headings">Review</div>
|
||||||
|
<button type="button" class="close" data-dismiss="modal" aria-label="Close">
|
||||||
|
<span aria-hidden="true">×</span>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
<div class="modal-body">
|
||||||
|
<form class="review-form" id="review-form">
|
||||||
|
<div class="form-group">
|
||||||
|
<div class="clearfix">
|
||||||
|
<label class="control-label reqd" style="padding-right: 0px;">Rating</label>
|
||||||
|
</div>
|
||||||
|
<div class="control-input-wrapper">
|
||||||
|
<div class="control-input">
|
||||||
|
<div class="rating rating-field" id="rating">
|
||||||
|
{% for i in [1, 2, 3, 4, 5] %}
|
||||||
|
<svg class="icon icon-md icon-rating" data-rating="{{ i }}">
|
||||||
|
<use href="#icon-star"></use>
|
||||||
|
</svg>
|
||||||
|
{% endfor %}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="form-group">
|
||||||
|
<div class="clearfix">
|
||||||
|
<label class="control-label reqd" style="padding-right: 0px;">Review</label>
|
||||||
|
</div>
|
||||||
|
<div class="control-input-wrapper">
|
||||||
|
<div class="control-input">
|
||||||
|
<textarea type="text" autocomplete="off" class="input-with-feedback form-control review-field"
|
||||||
|
data-fieldtype="Text" data-fieldname="feedback_comments" placeholder="" style="height: 300px;"
|
||||||
|
spellcheck="false"></textarea>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<p class="error-field muted-text"></p>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
<div class="modal-footer">
|
||||||
|
<div class="button submit-review is-primary" data-course="{{ course.name | urlencode}}" id="submit-review">
|
||||||
|
Submit</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
@@ -1,16 +0,0 @@
|
|||||||
<div class="sketch-teaser">
|
|
||||||
<div class="sketch-image">
|
|
||||||
<a href="/sketches/{{sketch.sketch_id}}">
|
|
||||||
{{ sketch.to_svg() }}
|
|
||||||
</a>
|
|
||||||
</div>
|
|
||||||
<div class="sketch-footer">
|
|
||||||
<div class="sketch-title">
|
|
||||||
<a href="sketches/{{sketch.sketch_id}}">{{sketch.title}}</a>
|
|
||||||
</div>
|
|
||||||
<div class="sketch-author">
|
|
||||||
{% set owner = sketch.get_owner() %}
|
|
||||||
by <a href="/{{owner.username}}">{{owner.full_name}}</a>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
160
community/lms/workspace/lms/lms.json
Normal file
@@ -0,0 +1,160 @@
|
|||||||
|
{
|
||||||
|
"category": "Modules",
|
||||||
|
"charts": [],
|
||||||
|
"creation": "2021-06-29 13:05:28.741459",
|
||||||
|
"developer_mode_only": 0,
|
||||||
|
"disable_user_customization": 0,
|
||||||
|
"docstatus": 0,
|
||||||
|
"doctype": "Workspace",
|
||||||
|
"extends_another_page": 0,
|
||||||
|
"hide_custom": 0,
|
||||||
|
"icon": "education",
|
||||||
|
"idx": 0,
|
||||||
|
"is_default": 0,
|
||||||
|
"is_standard": 1,
|
||||||
|
"label": "LMS",
|
||||||
|
"links": [
|
||||||
|
{
|
||||||
|
"hidden": 0,
|
||||||
|
"is_query_report": 0,
|
||||||
|
"label": "LMS",
|
||||||
|
"link_type": "DocType",
|
||||||
|
"onboard": 0,
|
||||||
|
"type": "Card Break"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"hidden": 0,
|
||||||
|
"is_query_report": 0,
|
||||||
|
"label": "LMS Course",
|
||||||
|
"link_to": "LMS Course",
|
||||||
|
"link_type": "DocType",
|
||||||
|
"onboard": 0,
|
||||||
|
"type": "Link"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"hidden": 0,
|
||||||
|
"is_query_report": 0,
|
||||||
|
"label": "LMS Batch",
|
||||||
|
"link_to": "LMS Batch",
|
||||||
|
"link_type": "DocType",
|
||||||
|
"onboard": 0,
|
||||||
|
"type": "Link"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"hidden": 0,
|
||||||
|
"is_query_report": 0,
|
||||||
|
"label": "LMS Batch Membership",
|
||||||
|
"link_to": "LMS Batch Membership",
|
||||||
|
"link_type": "DocType",
|
||||||
|
"onboard": 0,
|
||||||
|
"type": "Link"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"hidden": 0,
|
||||||
|
"is_query_report": 0,
|
||||||
|
"label": "LMS Course Mentor Mapping",
|
||||||
|
"link_to": "LMS Course Mentor Mapping",
|
||||||
|
"link_type": "DocType",
|
||||||
|
"onboard": 0,
|
||||||
|
"type": "Link"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"hidden": 0,
|
||||||
|
"is_query_report": 0,
|
||||||
|
"label": "Content",
|
||||||
|
"link_type": "DocType",
|
||||||
|
"onboard": 0,
|
||||||
|
"type": "Card Break"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"hidden": 0,
|
||||||
|
"is_query_report": 0,
|
||||||
|
"label": "Chapter",
|
||||||
|
"link_to": "Chapter",
|
||||||
|
"link_type": "DocType",
|
||||||
|
"onboard": 0,
|
||||||
|
"type": "Link"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"hidden": 0,
|
||||||
|
"is_query_report": 0,
|
||||||
|
"label": "Lesson",
|
||||||
|
"link_to": "Lesson",
|
||||||
|
"link_type": "DocType",
|
||||||
|
"onboard": 0,
|
||||||
|
"type": "Link"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"hidden": 0,
|
||||||
|
"is_query_report": 0,
|
||||||
|
"label": "Exercise",
|
||||||
|
"link_to": "Exercise",
|
||||||
|
"link_type": "DocType",
|
||||||
|
"onboard": 0,
|
||||||
|
"type": "Link"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"hidden": 0,
|
||||||
|
"is_query_report": 0,
|
||||||
|
"label": "Exercise Submission",
|
||||||
|
"link_to": "Exercise Submission",
|
||||||
|
"link_type": "DocType",
|
||||||
|
"onboard": 0,
|
||||||
|
"type": "Link"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"hidden": 0,
|
||||||
|
"is_query_report": 0,
|
||||||
|
"label": "LMS Quiz",
|
||||||
|
"link_to": "LMS Quiz",
|
||||||
|
"link_type": "DocType",
|
||||||
|
"onboard": 0,
|
||||||
|
"type": "Link"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"hidden": 0,
|
||||||
|
"is_query_report": 0,
|
||||||
|
"label": "LMS Quiz Submission",
|
||||||
|
"link_to": "LMS Quiz Submission",
|
||||||
|
"link_type": "DocType",
|
||||||
|
"onboard": 0,
|
||||||
|
"type": "Link"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"modified": "2021-06-29 15:11:07.324651",
|
||||||
|
"modified_by": "Administrator",
|
||||||
|
"module": "LMS",
|
||||||
|
"name": "LMS",
|
||||||
|
"owner": "Administrator",
|
||||||
|
"pin_to_bottom": 0,
|
||||||
|
"pin_to_top": 1,
|
||||||
|
"shortcuts": [
|
||||||
|
{
|
||||||
|
"color": "#29CD42",
|
||||||
|
"doc_view": "List",
|
||||||
|
"format": "{} Published",
|
||||||
|
"label": "Courses",
|
||||||
|
"link_to": "LMS Course",
|
||||||
|
"stats_filter": "{\"is_published\": 1}",
|
||||||
|
"type": "DocType"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"color": "#29CD42",
|
||||||
|
"doc_view": "List",
|
||||||
|
"format": "{} Active ",
|
||||||
|
"label": "Batches",
|
||||||
|
"link_to": "LMS Batch",
|
||||||
|
"stats_filter": "{\"status\": \"Active\"}",
|
||||||
|
"type": "DocType"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"color": "#39E4A5",
|
||||||
|
"doc_view": "List",
|
||||||
|
"format": "{} Students",
|
||||||
|
"label": "Memberships",
|
||||||
|
"link_to": "LMS Batch Membership",
|
||||||
|
"stats_filter": "{\"member_type\": \"Student\"}",
|
||||||
|
"type": "DocType"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
63
community/overrides/test_user.py
Normal file
@@ -0,0 +1,63 @@
|
|||||||
|
# Copyright (c) 2021, FOSS United and Contributors
|
||||||
|
# See license.txt
|
||||||
|
|
||||||
|
# import frappe
|
||||||
|
import unittest
|
||||||
|
import frappe
|
||||||
|
|
||||||
|
class TestCustomUser(unittest.TestCase):
|
||||||
|
|
||||||
|
def test_with_basic_username(self):
|
||||||
|
new_user = frappe.get_doc({
|
||||||
|
"doctype": "User",
|
||||||
|
"email": "test_with_basic_username@example.com",
|
||||||
|
"first_name": "Username"
|
||||||
|
}).insert()
|
||||||
|
self.assertEqual(new_user.username, "username")
|
||||||
|
|
||||||
|
def test_without_username(self):
|
||||||
|
""" The user in this test has the same first name as the user of the test test_with_basic_username.
|
||||||
|
In such cases frappe makes the username of the second user empty.
|
||||||
|
The condition in community app should override this and save a username. """
|
||||||
|
new_user = frappe.get_doc({
|
||||||
|
"doctype": "User",
|
||||||
|
"email": "test-without-username@example.com",
|
||||||
|
"first_name": "Username"
|
||||||
|
}).insert()
|
||||||
|
self.assertTrue(new_user.username)
|
||||||
|
|
||||||
|
def test_with_illegal_characters(self):
|
||||||
|
new_user = frappe.get_doc({
|
||||||
|
"doctype": "User",
|
||||||
|
"email": "test_with_illegal_characters@example.com",
|
||||||
|
"first_name": "Username$$"
|
||||||
|
}).insert()
|
||||||
|
self.assertEqual(new_user.username[:8], "username")
|
||||||
|
|
||||||
|
def test_with_underscore_at_end(self):
|
||||||
|
new_user = frappe.get_doc({
|
||||||
|
"doctype": "User",
|
||||||
|
"email": "test_with_underscore_at_end@example.com",
|
||||||
|
"first_name": "Username___"
|
||||||
|
}).insert()
|
||||||
|
self.assertNotEqual(new_user.username[-1], "_")
|
||||||
|
|
||||||
|
|
||||||
|
def test_with_short_first_name(self):
|
||||||
|
new_user = frappe.get_doc({
|
||||||
|
"doctype": "User",
|
||||||
|
"email": "test_with_short_first_name@example.com",
|
||||||
|
"first_name": "USN"
|
||||||
|
}).insert()
|
||||||
|
self.assertGreaterEqual(len(new_user.username), 4)
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def tearDownClass(cls) -> None:
|
||||||
|
users = [
|
||||||
|
"test_with_basic_username@example.com",
|
||||||
|
"test-without-username@example.com",
|
||||||
|
"test_with_illegal_characters@example.com",
|
||||||
|
"test_with_underscore_at_end@example.com",
|
||||||
|
"test_with_short_first_name@example.com"
|
||||||
|
]
|
||||||
|
frappe.db.delete("User", {"name": ["in", users]})
|
||||||
@@ -2,18 +2,70 @@ import frappe
|
|||||||
from frappe.core.doctype.user.user import User
|
from frappe.core.doctype.user.user import User
|
||||||
from frappe.utils import cint
|
from frappe.utils import cint
|
||||||
import hashlib
|
import hashlib
|
||||||
|
import random
|
||||||
|
import re
|
||||||
|
from frappe import _
|
||||||
|
|
||||||
class CustomUser(User):
|
class CustomUser(User):
|
||||||
|
|
||||||
def get_course_count(self) -> int:
|
def validate(self):
|
||||||
|
super(CustomUser, self).validate()
|
||||||
|
self.validate_username_characters()
|
||||||
|
|
||||||
|
def validate_username_characters(self):
|
||||||
|
if len(self.username):
|
||||||
|
underscore_condition = self.username[0] == "_" or self.username[-1] == "_"
|
||||||
|
else:
|
||||||
|
underscore_condition = ''
|
||||||
|
|
||||||
|
if self.is_new():
|
||||||
|
if not self.username:
|
||||||
|
self.username = self.get_username_from_first_name()
|
||||||
|
|
||||||
|
if self.username.find(" "):
|
||||||
|
self.username.replace(" ", "")
|
||||||
|
|
||||||
|
if not re.match("^[A-Za-z0-9_]*$", self.username) or underscore_condition:
|
||||||
|
self.username = self.remove_illegal_characters()
|
||||||
|
|
||||||
|
if len(self.username) < 4:
|
||||||
|
self.username = self.email.replace("@", "").replace(".", "")
|
||||||
|
|
||||||
|
while self.username_exists():
|
||||||
|
self.username = self.remove_illegal_characters() + str(random.randint(0, 99))
|
||||||
|
|
||||||
|
else:
|
||||||
|
if not self.username:
|
||||||
|
frappe.throw(_("Username already exists."))
|
||||||
|
|
||||||
|
if not re.match("^[A-Za-z0-9_]*$", self.username):
|
||||||
|
frappe.throw(_("Username can only contain alphabets, numbers and unedrscore."))
|
||||||
|
|
||||||
|
if underscore_condition:
|
||||||
|
frappe.throw(_("First and Last character of username cannot be Underscore(_)."))
|
||||||
|
|
||||||
|
if len(self.username) < 4:
|
||||||
|
frappe.throw(_("Username cannot be less than 4 characters"))
|
||||||
|
|
||||||
|
def get_username_from_first_name(self):
|
||||||
|
return frappe.scrub(self.first_name) + str(random.randint(0, 99))
|
||||||
|
|
||||||
|
def remove_illegal_characters(self):
|
||||||
|
return re.sub("[^\w]+", "", self.username).strip("_")
|
||||||
|
|
||||||
|
def get_authored_courses(self) -> int:
|
||||||
"""Returns the number of courses authored by this user.
|
"""Returns the number of courses authored by this user.
|
||||||
"""
|
"""
|
||||||
return frappe.db.count(
|
return frappe.get_all(
|
||||||
'LMS Course', {
|
'LMS Course', {
|
||||||
'owner': self.email
|
'owner': self.name,
|
||||||
|
'is_published': True
|
||||||
})
|
})
|
||||||
|
|
||||||
def get_palette(self):
|
def get_palette(self):
|
||||||
|
"""
|
||||||
|
Returns a color unique to each member for Avatar """
|
||||||
|
|
||||||
palette = [
|
palette = [
|
||||||
['--orange-avatar-bg', '--orange-avatar-color'],
|
['--orange-avatar-bg', '--orange-avatar-color'],
|
||||||
['--pink-avatar-bg', '--pink-avatar-color'],
|
['--pink-avatar-bg', '--pink-avatar-color'],
|
||||||
@@ -40,3 +92,35 @@ class CustomUser(User):
|
|||||||
'member_type': 'Mentor'
|
'member_type': 'Mentor'
|
||||||
})
|
})
|
||||||
|
|
||||||
|
def get_user_reviews(self):
|
||||||
|
""" Returns the reviews created by user """
|
||||||
|
return frappe.get_all("LMS Course Review",
|
||||||
|
{
|
||||||
|
"owner": self.name
|
||||||
|
})
|
||||||
|
|
||||||
|
def get_course_membership(self, member_type=None):
|
||||||
|
""" Returns all memberships of the user """
|
||||||
|
filters = {
|
||||||
|
"member": self.name
|
||||||
|
}
|
||||||
|
if member_type:
|
||||||
|
filters["member_type"] = member_type
|
||||||
|
|
||||||
|
return frappe.get_all("LMS Batch Membership", filters, ["name", "course"])
|
||||||
|
|
||||||
|
def get_mentored_courses(self):
|
||||||
|
""" Returns all courses mentored by this user """
|
||||||
|
mentored_courses = []
|
||||||
|
mapping = frappe.get_all("LMS Course Mentor Mapping",
|
||||||
|
{
|
||||||
|
"mentor": self.name,
|
||||||
|
},
|
||||||
|
["name", "course"]
|
||||||
|
)
|
||||||
|
|
||||||
|
for map in mapping:
|
||||||
|
if frappe.db.get_value("LMS Course", map.course, "is_published"):
|
||||||
|
mentored_courses.append(map)
|
||||||
|
|
||||||
|
return mentored_courses
|
||||||
|
|||||||
@@ -6,3 +6,5 @@ community.patches.replace_member_with_user_in_batch_membership
|
|||||||
community.patches.replace_member_with_user_in_course_mentor_mapping
|
community.patches.replace_member_with_user_in_course_mentor_mapping
|
||||||
community.patches.replace_member_with_user_in_lms_message
|
community.patches.replace_member_with_user_in_lms_message
|
||||||
community.patches.replace_member_with_user_in_mentor_request
|
community.patches.replace_member_with_user_in_mentor_request
|
||||||
|
community.patches.v0_0.chapter_lesson_index_table
|
||||||
|
execute:frappe.delete_doc("DocType", "LMS Message")
|
||||||
|
|||||||
50
community/patches/v0_0/chapter_lesson_index_table.py
Normal file
@@ -0,0 +1,50 @@
|
|||||||
|
from __future__ import unicode_literals
|
||||||
|
import frappe
|
||||||
|
|
||||||
|
def execute():
|
||||||
|
frappe.reload_doc("lms", "doctype", "lms_course")
|
||||||
|
frappe.reload_doc("lms", "doctype", "chapter")
|
||||||
|
frappe.reload_doc("lms", "doctype", "lesson")
|
||||||
|
frappe.reload_doc("lms", "doctype", "lessons")
|
||||||
|
frappe.reload_doc("lms", "doctype", "chapters")
|
||||||
|
|
||||||
|
update_chapters()
|
||||||
|
update_lessons()
|
||||||
|
|
||||||
|
def update_chapters():
|
||||||
|
courses = frappe.get_all("LMS Course", pluck="name")
|
||||||
|
for course in courses:
|
||||||
|
course_details = frappe.get_doc("LMS Course", course)
|
||||||
|
chapters = frappe.get_all("Chapter",
|
||||||
|
{
|
||||||
|
"course": course
|
||||||
|
},
|
||||||
|
["name"],
|
||||||
|
order_by= "index_"
|
||||||
|
)
|
||||||
|
for chapter in chapters:
|
||||||
|
course_details.append("chapters",
|
||||||
|
{
|
||||||
|
"chapter": chapter.name
|
||||||
|
})
|
||||||
|
|
||||||
|
course_details.save()
|
||||||
|
|
||||||
|
def update_lessons():
|
||||||
|
chapters = frappe.get_all("Chapter", pluck="name")
|
||||||
|
for chapter in chapters:
|
||||||
|
chapter_details = frappe.get_doc("Chapter", chapter)
|
||||||
|
lessons = frappe.get_all("Lesson",
|
||||||
|
{
|
||||||
|
"chapter": chapter
|
||||||
|
},
|
||||||
|
["name"],
|
||||||
|
order_by= "index_"
|
||||||
|
)
|
||||||
|
for lesson in lessons:
|
||||||
|
chapter_details.append("lessons",
|
||||||
|
{
|
||||||
|
"lesson": lesson.name
|
||||||
|
})
|
||||||
|
|
||||||
|
chapter_details.save()
|
||||||
@@ -67,6 +67,25 @@ class ProfileTab:
|
|||||||
"""
|
"""
|
||||||
raise NotImplementedError()
|
raise NotImplementedError()
|
||||||
|
|
||||||
|
class LiveCodeExtension(PageExtension):
|
||||||
|
def render_header(self):
|
||||||
|
livecode_url = frappe.get_value("LMS Settings", None, "livecode_url")
|
||||||
|
context = {
|
||||||
|
"livecode_url": livecode_url
|
||||||
|
}
|
||||||
|
return frappe.render_template(
|
||||||
|
"templates/livecode/extension_header.html",
|
||||||
|
context)
|
||||||
|
|
||||||
|
def render_footer(self):
|
||||||
|
livecode_url = frappe.get_value("LMS Settings", None, "livecode_url")
|
||||||
|
context = {
|
||||||
|
"livecode_url": livecode_url
|
||||||
|
}
|
||||||
|
return frappe.render_template(
|
||||||
|
"templates/livecode/extension_footer.html",
|
||||||
|
context)
|
||||||
|
|
||||||
def quiz_renderer(quiz_name):
|
def quiz_renderer(quiz_name):
|
||||||
quiz = frappe.get_doc("LMS Quiz", quiz_name)
|
quiz = frappe.get_doc("LMS Quiz", quiz_name)
|
||||||
context = dict(quiz=quiz)
|
context = dict(quiz=quiz)
|
||||||
@@ -79,7 +98,7 @@ def exercise_renderer(argument):
|
|||||||
|
|
||||||
def youtube_video_renderer(video_id):
|
def youtube_video_renderer(video_id):
|
||||||
return f"""
|
return f"""
|
||||||
<iframe width="560" height="315"
|
<iframe width="100%" height="315"
|
||||||
src="https://www.youtube.com/embed/{video_id}"
|
src="https://www.youtube.com/embed/{video_id}"
|
||||||
title="YouTube video player"
|
title="YouTube video player"
|
||||||
frameborder="0"
|
frameborder="0"
|
||||||
|
|||||||
@@ -1,4 +1,3 @@
|
|||||||
|
|
||||||
@import "./style.css";
|
@import "./style.css";
|
||||||
@import "./vars.css";
|
@import "./vars.css";
|
||||||
@import "./style.less";
|
|
||||||
|
|||||||
@@ -1,348 +0,0 @@
|
|||||||
h2 {
|
|
||||||
margin: 20px 0px;
|
|
||||||
color: black;
|
|
||||||
}
|
|
||||||
|
|
||||||
.teaser {
|
|
||||||
background: white;
|
|
||||||
border-radius: 9px;
|
|
||||||
border: 1px solid #C4C4C4;
|
|
||||||
|
|
||||||
.teaser-body {
|
|
||||||
padding: 20px;
|
|
||||||
box-shadow: 0px 5px 10px rgb(0 0 0 / 10%)
|
|
||||||
}
|
|
||||||
.teaser-footer {
|
|
||||||
padding: 20px;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.sketch-teaser {
|
|
||||||
.teaser();
|
|
||||||
width: 220px;
|
|
||||||
margin-bottom: 30px;
|
|
||||||
margin-top: 30px;
|
|
||||||
|
|
||||||
svg {
|
|
||||||
width: 200px;
|
|
||||||
height: 200px;
|
|
||||||
}
|
|
||||||
.sketch-image {
|
|
||||||
padding: 10px;
|
|
||||||
}
|
|
||||||
.sketch-footer {
|
|
||||||
border-top: 1px solid#C4C4C4;
|
|
||||||
padding: 10px;
|
|
||||||
background: #F6F6F6;
|
|
||||||
border-radius: 0px 0px 10px 10px;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.course-teaser {
|
|
||||||
.teaser();
|
|
||||||
color: #444;
|
|
||||||
margin-bottom: 20px;
|
|
||||||
margin-top: 20px;
|
|
||||||
|
|
||||||
h3, h4 {
|
|
||||||
color: black;
|
|
||||||
font-weight: bold;
|
|
||||||
}
|
|
||||||
|
|
||||||
.course-body, .course-footer {
|
|
||||||
padding: 20px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.course-body {
|
|
||||||
min-height: 8em;
|
|
||||||
}
|
|
||||||
|
|
||||||
.course-footer {
|
|
||||||
border-top: 1px solid #ddd;
|
|
||||||
}
|
|
||||||
|
|
||||||
a, a:hover {
|
|
||||||
color: inherit;
|
|
||||||
text-decoration: none;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
.anchor_style {
|
|
||||||
color: inherit;
|
|
||||||
}
|
|
||||||
|
|
||||||
.anchor_style:hover {
|
|
||||||
text-decoration: none
|
|
||||||
}
|
|
||||||
|
|
||||||
section {
|
|
||||||
padding: 60px 0px;
|
|
||||||
}
|
|
||||||
|
|
||||||
section h2 {
|
|
||||||
margin-bottom: 40px;
|
|
||||||
font-size: 48px;
|
|
||||||
line-height: 58px;
|
|
||||||
font-weight: bold;
|
|
||||||
}
|
|
||||||
|
|
||||||
section.lightgray {
|
|
||||||
background: #F6F6F6;
|
|
||||||
}
|
|
||||||
|
|
||||||
#hero .jumbotron {
|
|
||||||
background: inherit;
|
|
||||||
}
|
|
||||||
|
|
||||||
.chapter-teaser {
|
|
||||||
.teaser();
|
|
||||||
color: #444;
|
|
||||||
margin: 20px 0px;
|
|
||||||
|
|
||||||
h3, h4 {
|
|
||||||
color: black;
|
|
||||||
font-weight: bold;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.field-width {
|
|
||||||
width: 40%;
|
|
||||||
display: inline-block;
|
|
||||||
}
|
|
||||||
|
|
||||||
.footer-grouped-links {
|
|
||||||
display: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.footer-info {
|
|
||||||
border-top: 0px;
|
|
||||||
margin-top: 0px;
|
|
||||||
|
|
||||||
.footer-col-right {
|
|
||||||
padding-top: 1.8rem;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.web-footer {
|
|
||||||
border-top: 1px solid #E2E6E9;
|
|
||||||
padding: 0px;
|
|
||||||
padding: 2rem 0px;
|
|
||||||
margin-top: 2rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
.course-type {
|
|
||||||
text-transform: uppercase;
|
|
||||||
font-size: 1.0em;
|
|
||||||
font-weight: bold;
|
|
||||||
color: var(--tag-color);
|
|
||||||
}
|
|
||||||
|
|
||||||
.course-header {
|
|
||||||
margin-top: 20px;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
.course-header {
|
|
||||||
margin-top: 20px;
|
|
||||||
padding: 20px;
|
|
||||||
background: var(--header-bg);
|
|
||||||
color: var(--header-color);
|
|
||||||
border-radius: 9px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.course-author-avatar {
|
|
||||||
width: 20px;
|
|
||||||
height: 20px;
|
|
||||||
border-radius: 50%;
|
|
||||||
margin-right: 20px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.course-header h1 {
|
|
||||||
color: inherit;
|
|
||||||
}
|
|
||||||
|
|
||||||
*/
|
|
||||||
|
|
||||||
// .gray-section {
|
|
||||||
// background:#F6F6F6;
|
|
||||||
// border: 1px solid #C4C4C4;
|
|
||||||
// padding: 20px;
|
|
||||||
// margin: 20px 0px;
|
|
||||||
// }
|
|
||||||
|
|
||||||
.instructor-title {
|
|
||||||
color: black;
|
|
||||||
}
|
|
||||||
|
|
||||||
.instructor-subtitle {
|
|
||||||
font-size: 0.8em;
|
|
||||||
color: var(--text-color);
|
|
||||||
}
|
|
||||||
|
|
||||||
// .mentors-wrapper {
|
|
||||||
// .gray-section();
|
|
||||||
// }
|
|
||||||
|
|
||||||
|
|
||||||
.chapter-number {
|
|
||||||
background: var(--text-color);
|
|
||||||
color: white;
|
|
||||||
height: 24px;
|
|
||||||
min-width: 24px;
|
|
||||||
margin-right: 5px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.sidebar {
|
|
||||||
background: var(--sidebar-bg);
|
|
||||||
border: 1px solid var(--sidebar-border);
|
|
||||||
margin: 20px 0px;
|
|
||||||
border-radius: 10px;
|
|
||||||
padding: 1px 20px 20px 20px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.sidebar h3 {
|
|
||||||
margin-top: 20px;
|
|
||||||
color: black;
|
|
||||||
}
|
|
||||||
|
|
||||||
.sidebar-batch {
|
|
||||||
background: var(--sidebar-bg);
|
|
||||||
color: var(--text-color);
|
|
||||||
position: fixed;
|
|
||||||
left: 0;
|
|
||||||
height: 100%;
|
|
||||||
}
|
|
||||||
|
|
||||||
.sidebar-batch a {
|
|
||||||
padding: 16px 8px 8px 16px;
|
|
||||||
display: block;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
.sidebar .notice {
|
|
||||||
margin-top: 10px;
|
|
||||||
padding: 10px;
|
|
||||||
border-radius: 10px;
|
|
||||||
border: 1px dashed var(--text-color);
|
|
||||||
}
|
|
||||||
|
|
||||||
.sidebar .notice a {
|
|
||||||
color: inherit;
|
|
||||||
text-decoration: underline;
|
|
||||||
}
|
|
||||||
|
|
||||||
// LiveCode editor
|
|
||||||
|
|
||||||
.livecode-editor {
|
|
||||||
|
|
||||||
.CodeMirror {
|
|
||||||
border: 1px solid #ddd;
|
|
||||||
background: #ffe;
|
|
||||||
height: auto;
|
|
||||||
}
|
|
||||||
.CodeMirror-scroll {
|
|
||||||
max-height: 310px;
|
|
||||||
min-height: 310px;
|
|
||||||
}
|
|
||||||
.controls {
|
|
||||||
padding: 10px 0px;
|
|
||||||
}
|
|
||||||
canvas {
|
|
||||||
border: 5px solid #ddd;
|
|
||||||
position: relative;
|
|
||||||
z-index: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.output {
|
|
||||||
position: absolute;
|
|
||||||
z-index: 1;
|
|
||||||
width: 300px;
|
|
||||||
left: 0px;
|
|
||||||
top: 0px;
|
|
||||||
background-color: rgba(255, 255, 255, 0);
|
|
||||||
max-height: 300px;
|
|
||||||
white-space: pre-wrap;
|
|
||||||
word-wrap: break-word;
|
|
||||||
margin: 0px;
|
|
||||||
margin-left: 20px;
|
|
||||||
padding: 4px;
|
|
||||||
color: #888;
|
|
||||||
}
|
|
||||||
|
|
||||||
@media (max-width: 768px) {
|
|
||||||
.canvas-wrapper {
|
|
||||||
padding-top: 10px;
|
|
||||||
}
|
|
||||||
.code-wrapper {
|
|
||||||
min-height: 50px;
|
|
||||||
}
|
|
||||||
.CodeMirror {
|
|
||||||
min-height: 50px;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.sketch-header {
|
|
||||||
input#sketch-title {
|
|
||||||
font-weight: bold;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.chapter-description {
|
|
||||||
margin-bottom: 10px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.lesson-teaser {
|
|
||||||
line-height: 40px;
|
|
||||||
}
|
|
||||||
|
|
||||||
#hero h1 {
|
|
||||||
color: black !important;
|
|
||||||
}
|
|
||||||
|
|
||||||
.lesson-page {
|
|
||||||
margin: 20px 0px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.lesson-pagination {
|
|
||||||
clear: both;
|
|
||||||
}
|
|
||||||
|
|
||||||
.exercise-image svg {
|
|
||||||
width: 200px;
|
|
||||||
height: 200px;
|
|
||||||
border: 1px solid #ddd;
|
|
||||||
margin-bottom: 20px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.svg-200 svg {
|
|
||||||
width: 200px;
|
|
||||||
height: 200px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.livecode-editor-small .livecode-editor {
|
|
||||||
.CodeMirror-scroll {
|
|
||||||
max-height: 160px;
|
|
||||||
min-height: 160px;
|
|
||||||
}
|
|
||||||
|
|
||||||
canvas {
|
|
||||||
width: 150px;
|
|
||||||
height: 150px;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.mentor-dashboard {
|
|
||||||
margin-top: 20px;
|
|
||||||
|
|
||||||
.submission {
|
|
||||||
margin: 40px 0px 0px 20px;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
.no-preview-message {
|
|
||||||
width: fit-content;
|
|
||||||
margin: 50px 0px 50px;
|
|
||||||
color: black;
|
|
||||||
}
|
|
||||||
3
community/public/icons/black-arrow.svg
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
<svg width="13" height="8" viewBox="0 0 13 8" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<path fill-rule="evenodd" clip-rule="evenodd" d="M8.9649 0.313118C8.76964 0.117856 8.45306 0.117856 8.2578 0.313118C8.06253 0.50838 8.06253 0.824963 8.2578 1.02023L10.7374 3.49985L1.16683 3.49719C0.890684 3.49712 0.666764 3.72091 0.666687 3.99705C0.66661 4.2732 0.890406 4.49712 1.16655 4.49719L10.7377 4.49985L8.2578 6.97978C8.06253 7.17505 8.06253 7.49163 8.2578 7.68689C8.45306 7.88215 8.76964 7.88215 8.9649 7.68689L12.2982 4.35356C12.4935 4.1583 12.4935 3.84171 12.2982 3.64645L8.9649 0.313118Z" fill="#4C5A67"/>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 619 B |
3
community/public/icons/calendar.svg
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
<svg width="24" height="24" viewBox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<path fill-rule="evenodd" clip-rule="evenodd" d="M15.8947 2.47619C15.8947 2.2132 15.6885 2 15.4342 2C15.1798 2 14.9737 2.2132 14.9737 2.47619V3.90476H8.52631V2.47619C8.52631 2.2132 8.32013 2 8.06579 2C7.81145 2 7.60526 2.2132 7.60526 2.47619V3.90476H5.30263C4.03092 3.90476 3 4.97074 3 6.28571V10.0952V19.619C3 20.934 4.03092 22 5.30263 22H18.1974C19.4691 22 20.5 20.934 20.5 19.619V10.0952V6.28571C20.5 4.97074 19.4691 3.90476 18.1974 3.90476H15.8947V2.47619ZM19.579 9.61905V6.28571C19.579 5.49673 18.9604 4.85714 18.1974 4.85714H15.8947V6.28572C15.8947 6.54871 15.6885 6.76191 15.4342 6.76191C15.1798 6.76191 14.9737 6.54871 14.9737 6.28572V4.85714H8.52631V6.28572C8.52631 6.54871 8.32013 6.76191 8.06579 6.76191C7.81145 6.76191 7.60526 6.54871 7.60526 6.28572V4.85714H5.30263C4.53961 4.85714 3.92105 5.49673 3.92105 6.28571V9.61905H19.579ZM3.92105 10.5714H19.579V19.619C19.579 20.408 18.9604 21.0476 18.1974 21.0476H5.30263C4.53961 21.0476 3.92105 20.408 3.92105 19.619V10.5714Z" fill="#4C5A67"/>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 1.1 KiB |
3
community/public/icons/check.svg
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
<svg width="24" height="24" viewBox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<path fill-rule="evenodd" clip-rule="evenodd" d="M22 12C22 17.5228 17.5228 22 12 22C6.47715 22 2 17.5228 2 12C2 6.47715 6.47715 2 12 2C17.5228 2 22 6.47715 22 12ZM16.8734 10.1402C17.264 9.74969 17.264 9.11652 16.8734 8.726C16.4829 8.33547 15.8498 8.33547 15.4592 8.726L14.6259 9.55933L12.9592 11.226L10.333 13.8522L9.37345 12.8927L8.54011 12.0593C8.14959 11.6688 7.51643 11.6688 7.1259 12.0593C6.73538 12.4499 6.73538 13.083 7.1259 13.4735L7.95923 14.3069L9.6259 15.9735C9.81344 16.1611 10.0678 16.2664 10.333 16.2664C10.5982 16.2664 10.8526 16.1611 11.0401 15.9735L14.3734 12.6402L16.0401 10.9735L16.8734 10.1402Z" fill="#68D391"/>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 736 B |
3
community/public/icons/chevron-right.svg
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
<svg width="24" height="24" viewBox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<path d="M10 16L14 12L10 8" stroke="#4C5A67" stroke-linecap="round" stroke-linejoin="round"/>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 197 B |
3
community/public/icons/chevron.svg
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
<svg width="24" height="24" viewBox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<path d="M8 10L12 14L16 10" stroke="#4C5A67" stroke-miterlimit="10" stroke-linecap="round" stroke-linejoin="round"/>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 220 B |
3
community/public/icons/clock.svg
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
<svg width="24" height="24" viewBox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<path fill-rule="evenodd" clip-rule="evenodd" d="M3 11.9999C3 7.02937 7.02937 3 11.9999 3C16.9703 3 20.9997 7.02937 20.9997 11.9999C20.9997 16.9703 16.9703 20.9997 11.9999 20.9997C7.02937 20.9997 3 16.9703 3 11.9999ZM11.9999 2C6.47709 2 2 6.47709 2 11.9999C2 17.5226 6.47709 21.9997 11.9999 21.9997C17.5226 21.9997 21.9997 17.5226 21.9997 11.9999C21.9997 6.47709 17.5226 2 11.9999 2ZM12.5002 6.30027C12.5002 6.02413 12.2763 5.80027 12.0002 5.80027C11.724 5.80027 11.5002 6.02413 11.5002 6.30027V12.0002C11.5002 12.1896 11.6072 12.3627 11.7766 12.4474L15.5765 14.3474C15.8235 14.4709 16.1238 14.3708 16.2473 14.1238C16.3708 13.8768 16.2707 13.5764 16.0237 13.4529L12.5002 11.6912V6.30027Z" fill="#4C5A67"/>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 809 B |
3
community/public/icons/down-arrow.svg
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
<svg width="10" height="6" viewBox="0 0 10 6" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<path d="M1 1L5 5L9 1" stroke="#4C5A67" stroke-miterlimit="10" stroke-linecap="round" stroke-linejoin="round"/>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 213 B |
9
community/public/icons/github.svg
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
<svg width="24" height="24" viewBox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink">
|
||||||
|
<rect x="2" y="2" width="20" height="20" fill="url(#pattern0)"/>
|
||||||
|
<defs>
|
||||||
|
<pattern id="pattern0" patternContentUnits="objectBoundingBox" width="1" height="1">
|
||||||
|
<use xlink:href="#image0" transform="scale(0.05)"/>
|
||||||
|
</pattern>
|
||||||
|
<image id="image0" width="20" height="20" xlink:href="data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABQAAAAUCAYAAACNiR0NAAABiUlEQVQ4EZWU4VUCMQzHOwIjMAIb6AhuIBvgArb9QkLDh3MD3EA3wA10A9wAN0D/PdLLld5DeI+XuzT95Z9cU+cavxi7maf0GCi9BU6HwHI6//eBUhd5e9/Y1nZFSk+B5GggCqvtPq5l0aY456AqsOz+ARrAJMdI6aEJ9ZReb4INbThdtCCXOQrgeWRZepJvTeI5/ZjniHKLCCiN3SwrzaVWPdMSsFYCz07bN88p2iQ9kGWpTrU1RBPUtihEdSTHvO5Z3hUE6zl91Run3q1C7M0fyPapOKcIDX/dW2fVZaA2t7G55fIkH8qA4kvgLVPgnLMVZiB6phlgPctLS0nLh/Nn9+L4OQCsM0M3smoBrK+Hjeb8lI9UnWWAp4PnbXgmuVNQZJ77jaz6S6NcGHkUR6fDNHZ3MTXVBWBih5nWI1Myr2WhyqAY5wkNxvhpjFqUpbFqkUTXi8XmHEDyeW1SFASLUifjoc4c1Mn7ToFQNglTqQjAlwcYCdRvbVY1dQ/awPr5avZ6w9/7L3HVAW99VdWzAAAAAElFTkSuQmCC"/>
|
||||||
|
</defs>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 1.0 KiB |
3
community/public/icons/left-arrow.svg
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
<svg width="4" height="8" viewBox="0 0 4 8" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<path d="M3.33331 6.66663L0.666646 3.99996L3.33331 1.33329" stroke="#4C5A67" stroke-linecap="round" stroke-linejoin="round"/>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 225 B |
3
community/public/icons/like.svg
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
<svg width="16" height="16" viewBox="0 0 16 16" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<path d="M13.0746 3.60967C12.7814 3.31071 12.4333 3.07355 12.0501 2.91174C11.6669 2.74994 11.2562 2.66666 10.8415 2.66666C10.4267 2.66666 10.016 2.74994 9.63283 2.91174C9.24966 3.07355 8.90152 3.31071 8.60831 3.60967L7.99979 4.22983L7.39126 3.60967C6.79899 3.00607 5.9957 2.66697 5.1581 2.66697C4.32051 2.66697 3.51721 3.00607 2.92494 3.60967C2.33267 4.21327 1.99994 5.03192 1.99994 5.88554C1.99994 6.73916 2.33267 7.55782 2.92494 8.16142L3.53347 8.78158L7.99979 13.3333L12.4661 8.78158L13.0746 8.16142C13.368 7.86259 13.6007 7.5078 13.7595 7.11729C13.9182 6.72679 13.9999 6.30824 13.9999 5.88554C13.9999 5.46284 13.9182 5.04429 13.7595 4.65379C13.6007 4.26329 13.368 3.90849 13.0746 3.60967V3.60967Z" stroke="#4C5A67" stroke-linecap="round" stroke-linejoin="round"/>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 871 B |
3
community/public/icons/lock.svg
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
<svg width="24" height="24" viewBox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<path fill-rule="evenodd" clip-rule="evenodd" d="M11.7114 3.0002H11.6325C10.4336 2.98895 9.27915 3.45397 8.42278 4.29322C7.5658 5.13306 7.07747 6.27888 7.06517 7.47871L7.06514 7.47871V7.48384V8.17399H5C3.89543 8.17399 3 9.06943 3 10.174V18.9996C3 20.1041 3.89543 20.9996 5 20.9996H18.2602C19.3647 20.9996 20.2602 20.1041 20.2602 18.9996V10.174C20.2602 9.06942 19.3647 8.17399 18.2602 8.17399H16.195V7.56755C16.2063 6.36863 15.7412 5.21421 14.902 4.35784C14.0622 3.50086 12.9163 3.01253 11.7165 3.00023V3.0002H11.7114ZM15.195 8.17399V7.56514V7.56016H15.195C15.2043 6.62548 14.842 5.72537 14.1878 5.05777C13.5341 4.39076 12.6425 4.01043 11.7087 4.0002H11.6301H11.6251V4.00017C10.6904 3.99087 9.79031 4.35318 9.12271 5.00743C8.4557 5.6611 8.07537 6.55271 8.06514 7.48649V8.17399H15.195ZM5 9.17399C4.44772 9.17399 4 9.62171 4 10.174V18.9996C4 19.5518 4.44772 19.9996 5 19.9996H18.2602C18.8124 19.9996 19.2602 19.5518 19.2602 18.9996V10.174C19.2602 9.62171 18.8124 9.17399 18.2602 9.17399H5ZM11.6303 15.804C12.3026 15.804 12.8476 15.259 12.8476 14.5867C12.8476 13.9145 12.3026 13.3694 11.6303 13.3694C10.958 13.3694 10.413 13.9145 10.413 14.5867C10.413 15.259 10.958 15.804 11.6303 15.804ZM13.8476 14.5867C13.8476 15.8113 12.8549 16.804 11.6303 16.804C10.4057 16.804 9.413 15.8113 9.413 14.5867C9.413 13.3622 10.4057 12.3694 11.6303 12.3694C12.8549 12.3694 13.8476 13.3622 13.8476 14.5867Z" fill="#4C5A67"/>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 1.5 KiB |
9
community/public/icons/medium.svg
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
<svg width="24" height="24" viewBox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink">
|
||||||
|
<rect x="2" y="3" width="20.7" height="18" fill="url(#pattern0)"/>
|
||||||
|
<defs>
|
||||||
|
<pattern id="pattern0" patternContentUnits="objectBoundingBox" width="1" height="1">
|
||||||
|
<use xlink:href="#image0" transform="scale(0.0434783 0.05)"/>
|
||||||
|
</pattern>
|
||||||
|
<image id="image0" width="23" height="20" xlink:href="data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABcAAAAUCAYAAABmvqYOAAABtUlEQVRIDa2UTU7CQBSAOYJH4CgcxSNwAIPoAgSEqAmIbohR/iQIgiK0nU75q4BiKz0ALnStxoULTMZMzWvGoUBrbPLy5jUz3zd5M61nKxJ7iR0kjz0uH1HEPhErH93+7eK1m+EoobEd3Z25kbSQrAtIJjQkrMxsJQCH7EQiIuQHMJvnJADlM5UkUukNvls3GHsFJL+yUH5MJe2eGvbwULaOp9KfmXzeywpaklzlYXb1cDTWl8ITqSOSyRYqAKeHaAfi33V6KtEmhuIETjKneR/GeK2F5CkP4mtJxkTTH13As4VpQ5B2eJBdPRiNiD4xXMFJ+bJuXjs7ILxTOl0T7BqeyRZIQxCXCu4ftL/Dc+cXC+H9wcACO9p5PHn4TnfMRvXqek6AFMU8RAqFWHlbQnv72bNi6Y2Fn+SKpCmiX4Lh3diCOoYHwpEgveOlcs3PSkqVmgVvd3tzYEdtATh8RKyk3mgSQUJfY03/HzgrKZarz11VfYI28NnseTAc8wVCUYX9p8CY3znAIWuG4aMQHmy1BSbaSVbBYa2dxNw5TIDMSpzCYS0rsYXDRCqhAbWb/COZrH8Ddw70agzSigAAAAAASUVORK5CYII="/>
|
||||||
|
</defs>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 1.1 KiB |
5
community/public/icons/message.svg
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
<svg width="20" height="20" viewBox="0 0 20 20" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<path d="M6.00001 3.00001H12L15.5 3C16.6046 3 17.5 3.89543 17.5 5V9.78889V12.2556C17.5 13.3601 16.6046 14.2556 15.5 14.2556H14.5C14.2239 14.2556 14 14.4794 14 14.7556V16.4507C14 16.8715 13.5119 17.1041 13.185 16.839L10.2754 14.4789C10.0972 14.3344 9.87483 14.2556 9.64544 14.2556H4.49997C3.39539 14.2556 2.49995 13.3601 2.49997 12.2555L2.50001 9.78889V5.00001C2.50001 3.89544 3.39544 3.00001 4.50001 3.00001L6.00001 3.00001Z" stroke="#1F272E" stroke-miterlimit="10" stroke-linecap="square"/>
|
||||||
|
<path d="M6 6.5H13" stroke="#1F272E" stroke-miterlimit="10" stroke-linecap="round"/>
|
||||||
|
<path d="M6 9H10" stroke="#1F272E" stroke-miterlimit="10" stroke-linecap="round"/>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 763 B |
4
community/public/icons/minus-circle-green.svg
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
<svg width="24" height="24" viewBox="0 0 24 24" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<path d="M12 22C17.5228 22 22 17.5228 22 12C22 6.47715 17.5228 2 12 2C6.47715 2 2 6.47715 2 12C2 17.5228 6.47715 22 12 22Z" fill="#68D391" stroke="#68D391" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"/>
|
||||||
|
<path d="M8 12H16" stroke="white" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"/>
|
||||||
|
</svg>
|
||||||
|
After Width: | Height: | Size: 425 B |
1
community/public/icons/minus-circle.svg
Normal file
@@ -0,0 +1 @@
|
|||||||
|
<svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" class="feather feather-minus-circle"><circle cx="12" cy="12" r="10"></circle><line x1="8" y1="12" x2="16" y2="12"></line></svg>
|
||||||
|
After Width: | Height: | Size: 309 B |