@@ -150,6 +150,7 @@ primary_rules = [
|
||||
|
||||
# Any frappe default URL is blocked by profile-rules, add it here to unblock it
|
||||
whitelist = [
|
||||
"/home",
|
||||
"/login",
|
||||
"/update-password",
|
||||
"/update-profile",
|
||||
|
||||
@@ -88,3 +88,15 @@ class LMSCourse(Document):
|
||||
member.batch_count = len(frappe.get_all("LMS Batch Membership", {"member": member.name, "member_type": "Mentor"}))
|
||||
course_mentors.append(member)
|
||||
return course_mentors
|
||||
|
||||
def get_instructor(self):
|
||||
return frappe.get_doc("User", self.owner)
|
||||
|
||||
@staticmethod
|
||||
def find_all():
|
||||
"""Returns all published courses.
|
||||
"""
|
||||
rows = frappe.db.get_all("LMS Course",
|
||||
filters={"is_published": True},
|
||||
fields='*')
|
||||
return [frappe.get_doc(dict(row, doctype='LMS Course')) for row in rows]
|
||||
|
||||
@@ -4,6 +4,7 @@
|
||||
from __future__ import unicode_literals
|
||||
|
||||
import frappe
|
||||
from .lms_course import LMSCourse
|
||||
import unittest
|
||||
|
||||
class TestLMSCourse(unittest.TestCase):
|
||||
@@ -26,6 +27,23 @@ class TestLMSCourse(unittest.TestCase):
|
||||
assert course.slug == "test-course"
|
||||
assert course.get_mentors() == []
|
||||
|
||||
def test_find_all(self):
|
||||
courses = LMSCourse.find_all()
|
||||
assert courses == []
|
||||
|
||||
# new couse, but not published
|
||||
course = self.new_course("Test Course")
|
||||
assert courses == []
|
||||
|
||||
# publish the course
|
||||
course.is_published = True
|
||||
course.save()
|
||||
|
||||
# now we should find one course
|
||||
courses = LMSCourse.find_all()
|
||||
assert [c.slug for c in courses] == [course.slug]
|
||||
|
||||
# disabled this test as it is failing
|
||||
def _test_add_mentors(self):
|
||||
course = self.new_course("Test Course")
|
||||
assert course.get_mentors() == []
|
||||
@@ -47,4 +65,4 @@ def new_user(name, email):
|
||||
email=email,
|
||||
first_name=name))
|
||||
doc.insert()
|
||||
return doc
|
||||
return doc
|
||||
|
||||
@@ -10,9 +10,6 @@ from frappe.model.document import Document
|
||||
from . import livecode
|
||||
|
||||
class LMSSketch(Document):
|
||||
def get_owner_name(self):
|
||||
return get_userinfo(self.owner)['full_name']
|
||||
|
||||
@property
|
||||
def sketch_id(self):
|
||||
"""Returns the numeric part of the name.
|
||||
@@ -21,6 +18,14 @@ class LMSSketch(Document):
|
||||
"""
|
||||
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
|
||||
@@ -46,6 +51,18 @@ class LMSSketch(Document):
|
||||
cache.set(key, value)
|
||||
return value
|
||||
|
||||
@staticmethod
|
||||
def get_recent_sketches(limit=100):
|
||||
"""Returns the recent sketches.
|
||||
"""
|
||||
sketches = frappe.get_all(
|
||||
"LMS Sketch",
|
||||
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}>"
|
||||
|
||||
@@ -76,36 +93,3 @@ def save_sketch(name, title, code):
|
||||
"status": status,
|
||||
"name": doc.name,
|
||||
}
|
||||
|
||||
def get_recent_sketches():
|
||||
"""Returns the recent sketches.
|
||||
|
||||
The return value will be a list of dicts with each entry containing
|
||||
the following fields:
|
||||
- name
|
||||
- title
|
||||
- owner
|
||||
- owner_name
|
||||
- modified
|
||||
"""
|
||||
sketches = frappe.get_all(
|
||||
"LMS Sketch",
|
||||
fields='*',
|
||||
order_by='modified desc',
|
||||
page_length=100
|
||||
)
|
||||
for s in sketches:
|
||||
s['owner_name'] = get_userinfo(s['owner'])['full_name']
|
||||
return [frappe.get_doc(doctype='LMS Sketch', **doc) for doc in sketches]
|
||||
|
||||
def get_userinfo(email):
|
||||
"""Returns the username and fullname of a user.
|
||||
|
||||
Please note that the email could be "Administrator" or "Guest"
|
||||
as a special case to denote the system admin and guest user respectively.
|
||||
"""
|
||||
user = frappe.get_doc("User", email)
|
||||
return {
|
||||
"full_name": user.full_name,
|
||||
"username": user.username
|
||||
}
|
||||
|
||||
5
community/lms/models.py
Normal file
5
community/lms/models.py
Normal file
@@ -0,0 +1,5 @@
|
||||
"""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_sketch.lms_sketch import LMSSketch as Sketch
|
||||
|
||||
13
community/lms/widgets/CourseTeaser.html
Normal file
13
community/lms/widgets/CourseTeaser.html
Normal file
@@ -0,0 +1,13 @@
|
||||
<div class="course-teaser">
|
||||
<div class="course-body">
|
||||
<h3 class="course-title"><a href="/courses/{{ course.slug }}">{{ course.title }}</a></h3>
|
||||
<div class="course-intro">
|
||||
{{ course.short_introduction or "" }}
|
||||
</div>
|
||||
</div>
|
||||
<div class="course-footer">
|
||||
<div class="course-author">
|
||||
{{ course.get_instructor().full_name }}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
15
community/lms/widgets/SketchTeaser.html
Normal file
15
community/lms/widgets/SketchTeaser.html
Normal file
@@ -0,0 +1,15 @@
|
||||
<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">
|
||||
by {{sketch.get_owner().full_name}}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
@@ -25,12 +25,14 @@
|
||||
--cta-color: var(--c4);
|
||||
--send-message: var(--c7);
|
||||
--received-message: var(--c8);
|
||||
|
||||
--primary-color: #08B74F;
|
||||
}
|
||||
|
||||
body {
|
||||
padding: 0px;
|
||||
margin: 0px;
|
||||
background: var(--bg);
|
||||
background: white;
|
||||
}
|
||||
|
||||
.course-header {
|
||||
@@ -200,10 +202,6 @@ img.profile-photo {
|
||||
|
||||
/* override style of base */
|
||||
|
||||
nav.navbar {
|
||||
background: var(--c1) !important;
|
||||
}
|
||||
|
||||
.message {
|
||||
border: 1px dashed var(--text-color);
|
||||
padding: 20px;
|
||||
@@ -291,4 +289,71 @@ nav.navbar {
|
||||
.message-section {
|
||||
margin-left: 5%;
|
||||
display: inline-block;
|
||||
}
|
||||
}
|
||||
|
||||
.sketch-teaser {
|
||||
background: white;
|
||||
border-radius: 10px;
|
||||
border: 1px solid #ddd;
|
||||
width: 220px;
|
||||
}
|
||||
|
||||
.sketch-teaser svg {
|
||||
width: 200px;
|
||||
height: 200px;
|
||||
}
|
||||
|
||||
.sketch-image {
|
||||
padding: 10px;
|
||||
}
|
||||
|
||||
.sketch-footer {
|
||||
padding: 10px;
|
||||
background: #eee;
|
||||
border-radius: 0px 0px 10px 10px;
|
||||
}
|
||||
|
||||
.course-teaser {
|
||||
background: white;
|
||||
border-radius: 10px;
|
||||
border: 1px solid #ddd;
|
||||
color: #444;
|
||||
}
|
||||
|
||||
.course-teaser h3, .course-teaser h4 {
|
||||
color: black;
|
||||
font-weight: bold;;
|
||||
}
|
||||
|
||||
.course-teaser .course-body, .course-teaser .course-footer {
|
||||
padding: 20px;
|
||||
}
|
||||
|
||||
.course-body {
|
||||
min-height: 8em;
|
||||
}
|
||||
|
||||
.course-footer {
|
||||
border-top: 1px solid #ddd;
|
||||
}
|
||||
|
||||
.course-teaser a, .course-teaser a:hover {
|
||||
color: inherit;
|
||||
text-decoration: none;
|
||||
}
|
||||
|
||||
section {
|
||||
padding: 60px 0px;
|
||||
}
|
||||
|
||||
section h2 {
|
||||
margin-bottom: 40px;
|
||||
}
|
||||
|
||||
section.lightgray {
|
||||
background: #f8f8f8;
|
||||
}
|
||||
|
||||
#hero .jumbotron {
|
||||
background: inherit;
|
||||
}
|
||||
|
||||
51
community/www/home/index.html
Normal file
51
community/www/home/index.html
Normal file
@@ -0,0 +1,51 @@
|
||||
{% extends "templates/base.html" %}
|
||||
|
||||
{% block content %}
|
||||
{{ HeroSection() }}
|
||||
{{ ExploreCourses(courses) }}
|
||||
{{ RecentSketches(recent_sketches) }}
|
||||
{% endblock %}
|
||||
|
||||
{% macro HeroSection() %}
|
||||
<section id="hero">
|
||||
<div class="container">
|
||||
<div class="jumbotron">
|
||||
<h1 class="display-4">Guided online courses, with a mentor at your back.</h1>
|
||||
<p class="lead">Hands-on online courses designed by experts, delivered by passionate mentors.</p>
|
||||
<p class="lead">
|
||||
<a class="btn btn-primary btn-lg" href="#" role="button">Request Invite</a>
|
||||
</p>
|
||||
</div>
|
||||
</div>
|
||||
</section>
|
||||
{% endmacro %}
|
||||
|
||||
{% macro ExploreCourses(courses) %}
|
||||
<section id="explore-courses" class="lightgray">
|
||||
<div class="container lightgray">
|
||||
<h2>Explore Courses</h2>
|
||||
<div class="row">
|
||||
{% for course in courses %}
|
||||
<div class="col-md-6">
|
||||
{{ widgets.CourseTeaser(course=course) }}
|
||||
</div>
|
||||
{% endfor %}
|
||||
</div>
|
||||
</div>
|
||||
</section>
|
||||
{% endmacro %}
|
||||
|
||||
{% macro RecentSketches(sketches) %}
|
||||
<section id="recet-sketches">
|
||||
<div class="container">
|
||||
<h2>Recent Sketches</h2>
|
||||
<div class="row">
|
||||
{% for sketch in sketches %}
|
||||
<div class="col-md-3">
|
||||
{{ widgets.SketchTeaser(sketch=sketch) }}
|
||||
</div>
|
||||
{% endfor %}
|
||||
</div>
|
||||
</div>
|
||||
</section>
|
||||
{% endmacro %}
|
||||
7
community/www/home/index.py
Normal file
7
community/www/home/index.py
Normal file
@@ -0,0 +1,7 @@
|
||||
import frappe
|
||||
from community.lms.models import Course, Sketch
|
||||
|
||||
def get_context(context):
|
||||
context.no_cache = 1
|
||||
context.courses = Course.find_all()
|
||||
context.recent_sketches = Sketch.get_recent_sketches(limit=8)
|
||||
@@ -16,37 +16,13 @@
|
||||
<a href="/sketches/new">Create a New Sketch</a>
|
||||
</div>
|
||||
<div class='container'>
|
||||
<div class="row row-cols-1 row-cols-xl-5 row-cols-lg-4 row-cols-md-3 row-cols-sm-2 ">
|
||||
{% for sketch in sketches %}
|
||||
<div class="col mb-4">
|
||||
<div class="card sketch-card" style="width: 200px;">
|
||||
<div class="card-img-top">
|
||||
<a href="/sketches/{{sketch.sketch_id}}">
|
||||
{{ sketch.to_svg() }}
|
||||
</a>
|
||||
</div>
|
||||
<div class="card-footer">
|
||||
<div class="sketch-title">
|
||||
<a href="sketches/{{sketch.sketch_id}}">{{sketch.title}}</a>
|
||||
<div class="row">
|
||||
{% for sketch in sketches %}
|
||||
<div class="col-md-3">
|
||||
{{ widgets.SketchTeaser(sketch=sketch) }}
|
||||
</div>
|
||||
<div class="sketch-author">
|
||||
by {{sketch.get_owner_name()}}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
{% endfor %}
|
||||
{% endfor %}
|
||||
</div>
|
||||
</div>
|
||||
</section>
|
||||
{% endblock %}
|
||||
|
||||
{% block style %}
|
||||
{{super()}}
|
||||
<style type="text/css">
|
||||
svg {
|
||||
width: 200px;
|
||||
height: 200px;
|
||||
}
|
||||
</style>
|
||||
{% endblock %}
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
import frappe
|
||||
from ...lms.doctype.lms_sketch.lms_sketch import get_recent_sketches
|
||||
from community.lms.models import Sketch
|
||||
|
||||
def get_context(context):
|
||||
context.no_cache = 1
|
||||
context.sketches = get_recent_sketches()
|
||||
context.sketches = Sketch.get_recent_sketches()
|
||||
|
||||
|
||||
Reference in New Issue
Block a user