Skip to content

Commit 88bbc1e

Browse files
authored
Migration + Front End for Learning Den Support (#1106)
* added front end and models * added migration py with a few courses, not sure if working * model is saving to db but not showing on ui * front end works when run commands from shell * added try catch, migrate not running after initial * removed console log * migration works and edited styling * added 5px more padding * styled logo * removed search result persist * added a period to learning den description * accidently left in static results * passes tests and syntax
1 parent c2578eb commit 88bbc1e

File tree

9 files changed

+175
-31
lines changed

9 files changed

+175
-31
lines changed

integrations/tests.py

+19-12
Original file line numberDiff line numberDiff line change
@@ -31,23 +31,28 @@ class IntegrationsGetAddTest(APITestCase):
3131
}
3232

3333
def setUp(self):
34-
Integration.objects.create(id=1, name='myint')
35-
Course.objects.create(id=1, school='uoft', code='SEA101', name='Intro')
36-
CourseIntegration.objects.create(course_id=1, integration_id=1, json='oldstuff')
34+
#get or create
35+
self.integration = Integration.objects.create(name='myint')
36+
self.integrationIdStr = str(self.integration.id)
37+
self.course = Course.objects.create(school='uoft', code='SEA101', name='Intro')
38+
self.courseIdStr = str(self.course.id)
39+
self.courseIntegration = CourseIntegration.objects.create(course_id=self.course.id, integration_id=self.integration.id, json='oldstuff')
3740

3841
def test_get_existing_integration(self):
39-
response = self.client.get('/integrations/1/course/1/', format='json', **self.request_headers)
42+
response = self.client.get('/integrations/' + self.integrationIdStr + '/course/' + self.courseIdStr + '/', format='json', **self.request_headers)
4043
self.assertEqual(response.status_code, status.HTTP_200_OK)
4144

4245
def test_get_nonexistent_integration(self):
43-
response = self.client.get('/integrations/5/course/3/', format='json', **self.request_headers)
46+
nonexistentIntegrationIdStr = str(self.integration.id + 1)
47+
nonexistentCourseIdStr = str(self.course.id + 1)
48+
response = self.client.get('/integrations/' + nonexistentIntegrationIdStr + '/course/' + nonexistentCourseIdStr + '/', format='json', **self.request_headers)
4449
self.assertEqual(response.status_code, status.HTTP_404_NOT_FOUND)
4550

4651
def test_add_integration(self):
4752
data = {'json': 'newstuff'}
48-
response = self.client.post('/integrations/1/course/1/', data, format='json', **self.request_headers)
53+
response = self.client.post('/integrations/' + self.integrationIdStr + '/course/' + self.courseIdStr + '/', data, format='json', **self.request_headers)
4954
self.assertEqual(response.status_code, status.HTTP_204_NO_CONTENT)
50-
CourseIntegration.objects.get(course_id=1, integration_id=1, **data)
55+
CourseIntegration.objects.get(course_id=self.course.id, integration_id=self.integration.id, **data)
5156

5257

5358
class IntegrationsDeleteTest(APITestCase):
@@ -56,11 +61,13 @@ class IntegrationsDeleteTest(APITestCase):
5661
}
5762

5863
def setUp(self):
59-
Integration.objects.create(id=1, name='myint')
60-
Course.objects.create(id=1, school='uoft', code='SEA101', name='Intro')
61-
CourseIntegration.objects.create(course_id=1, integration_id=1, json='oldstuff')
64+
self.integration = Integration.objects.create(name='myint')
65+
self.integrationIdStr = str(self.integration.id)
66+
self.course = Course.objects.create(school='uoft', code='SEA101', name='Intro')
67+
self.courseIdStr = str(self.course.id)
68+
self.courseIntegration = CourseIntegration.objects.create(course_id=self.course.id, integration_id=self.integration.id, json='oldstuff')
6269

6370
def test_delete_integration(self):
64-
response = self.client.delete('/integrations/1/course/1/', **self.request_headers)
71+
response = self.client.delete('/integrations/' + self.integrationIdStr + '/course/' + self.courseIdStr + '/', **self.request_headers)
6572
self.assertEqual(response.status_code, status.HTTP_204_NO_CONTENT)
66-
self.assertFalse(CourseIntegration.objects.filter(course_id=1, integration_id=1).exists())
73+
self.assertFalse(CourseIntegration.objects.filter(course_id=self.course.id, integration_id=self.integration.id).exists())

static/css/timetable/partials/course_modal.scss

+2
Original file line numberDiff line numberDiff line change
@@ -585,6 +585,7 @@
585585
li {
586586
list-style-type: none;
587587
padding-top: 5px;
588+
padding-bottom: 5px;
588589
}
589590

590591
.integration-image {
@@ -607,6 +608,7 @@
607608
a {
608609
float: right;
609610
font-size: 12px;
611+
line-height: 22px;
610612
}
611613

612614
p {

static/css/timetable/partials/search_bar.scss

+16
Original file line numberDiff line numberDiff line change
@@ -221,6 +221,8 @@
221221
}
222222

223223
.integration {
224+
margin: 1px 5px 0 0;
225+
224226
a {
225227
float: left;
226228
font-size: 12px;
@@ -319,6 +321,20 @@
319321
width: 43px;
320322
}
321323

324+
// If a course has an den integration turned on
325+
.has-den {
326+
background-position: center;
327+
background-repeat: no-repeat;
328+
background-size: contain;
329+
float: left;
330+
height: 15px;
331+
margin-right: 5px;
332+
opacity: 1;
333+
transition: visibility .3s, opacity .3s;
334+
visibility: visible;
335+
width: 100px;
336+
}
337+
322338
// Section Listings per Course
323339
.search-sections {
324340
opacity: 1;
14.2 KB
Loading
62.6 KB
Loading
3.62 KB
Loading

static/js/redux/ui/modals/course_modal_body.jsx

+35-15
Original file line numberDiff line numberDiff line change
@@ -214,23 +214,43 @@ class CourseModalBody extends React.Component {
214214
<h3 className="modal-module-header">{this.props.schoolSpecificInfo.areasName}</h3>
215215
<p>{ this.props.data.areas || 'None' }</p>
216216
</div>);
217-
const integrationDivStyle = {
217+
const pilotLogoImg = {
218218
backgroundImage: 'url(/static/img/integrations/pilot.png)',
219219
};
220-
const academicSupportDisplay = integrationList.indexOf('Pilot') > -1 ?
221-
(<div className="modal-module academic-support">
222-
<h3 className="modal-module-header">Academic Support</h3>
223-
<li className="cf">
224-
<span className="integration-image" style={integrationDivStyle} />
225-
<h4>Pilot</h4>
226-
<a href="http://academicsupport.jhu.edu/pilot-learning/" target="_blank" rel="noopener noreferrer">
227-
Learn More
228-
</a>
229-
<p>In the PILOT program, students are organized into study teams consisting of
230-
6-10 members who meet
231-
weekly to work problems together.</p>
232-
</li>
233-
</div>) : null;
220+
const pilotDisplay = integrationList.indexOf('Pilot') > -1 ?
221+
(<li className="cf">
222+
<span className="integration-image" style={pilotLogoImg} />
223+
<h4>Pilot</h4>
224+
<a href="http://academicsupport.jhu.edu/pilot-learning/" target="_blank" rel="noopener noreferrer">
225+
Learn More
226+
</a>
227+
<p>In the PILOT program, students are organized into study teams consisting of
228+
6-10 members who meet
229+
weekly to work problems together.</p>
230+
</li>) : null;
231+
const learningDenLogoImg = {
232+
backgroundImage: 'url(/static/img/integrations/learningDen_books.png)',
233+
};
234+
const learningDenDisplay = integrationList.indexOf('LearningDen') > -1 ?
235+
(<li className="cf">
236+
<span className="integration-image" style={learningDenLogoImg} />
237+
<h4>Learning Den</h4>
238+
<a
239+
href="https://advising.jhu.edu/tutoring-mentoring/learning-den-tutoring-services/" target="_blank"
240+
rel="noopener noreferrer"
241+
>
242+
Learn More
243+
</a>
244+
<p>The Learning Den is a peer-to-peer, small group tutoring program that
245+
helps students to improve their understanding of course materials,
246+
and prepare for exams.</p>
247+
</li>) : null;
248+
const academicSupportDisplay = integrationList.indexOf('LearningDen') > -1 || integrationList.indexOf('Pilot') > -1 ?
249+
(<div className="modal-module academic-support">
250+
<h3 className="modal-module-header">Academic Support</h3>
251+
{ pilotDisplay }
252+
{ learningDenDisplay }
253+
</div>) : null;
234254
let friendCircles = (<div className="loading"><span className="img-icon"><div
235255
className="loader"
236256
/></span><p>

static/js/redux/ui/search_result.jsx

+11-4
Original file line numberDiff line numberDiff line change
@@ -138,13 +138,19 @@ class SearchResult extends React.Component {
138138
info = !inRoster ? 'Add this course to your timetable' :
139139
'Remove this course from your timetable';
140140
}
141-
const integrationLogoImageUrl = {
141+
const pilotLogoImg = {
142142
backgroundImage: 'url(/static/img/integrations/pilotLogo.png)',
143143
};
144-
const integrationLogo = course.integrations.indexOf('Pilot') > -1 ?
144+
const pilotLogo = course.integrations.indexOf('Pilot') > -1 ?
145145
(<div className="label integration">
146-
<span className="has-pilot" style={integrationLogoImageUrl} />
146+
<span className="has-pilot" style={pilotLogoImg} />
147147
</div>) : null;
148+
const learningDenLogoImg = {
149+
backgroundImage: 'url(/static/img/integrations/learningDen.png)',
150+
};
151+
const learningDenLogo = course.integrations.indexOf('LearningDen') > -1 ? (<div className="label integration">
152+
<span className="has-den" style={learningDenLogoImg} />
153+
</div>) : null;
148154
const waitlistOnlyFlag = this.hasOnlyWaitlistedSections() ?
149155
<h4 className="label flag">Waitlist Only</h4> : null;
150156
return (
@@ -171,7 +177,8 @@ class SearchResult extends React.Component {
171177
<h4
172178
className={classNames('label', 'bubble')}
173179
>{this.props.campuses[course.campus]}</h4>
174-
{ integrationLogo }
180+
{ pilotLogo }
181+
{ learningDenLogo }
175182
{ waitlistOnlyFlag }
176183
</div>
177184
</li>);
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,92 @@
1+
"""
2+
Copyright (C) 2017 Semester.ly Technologies, LLC
3+
Semester.ly is free software: you can redistribute it and/or modify
4+
it under the terms of the GNU General Public License as published by
5+
the Free Software Foundation, either version 3 of the License, or
6+
(at your option) any later version.
7+
Semester.ly is distributed in the hope that it will be useful,
8+
but WITHOUT ANY WARRANTY; without even the implied warranty of
9+
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
10+
GNU General Public License for more details.
11+
"""
12+
13+
from __future__ import unicode_literals
14+
15+
from django.db import migrations
16+
from timetable.models import Integration, CourseIntegration, Course
17+
18+
19+
def add_learning_den(apps, schema_editor):
20+
""" Learning Den classes for Spring 2018 """
21+
learning_den_codes = [
22+
"AS.020.152",
23+
"AS.020.306",
24+
"AS.030.102",
25+
"AS.030.103",
26+
"AS.030.206",
27+
"AS.050.203",
28+
"AS.080.203",
29+
"AS.080.306",
30+
"AS.110.106",
31+
"AS.110.107",
32+
"AS.110.109",
33+
"AS.110.302",
34+
"AS.171.101",
35+
"AS.171.102",
36+
"AS.171.104",
37+
"AS.171.108",
38+
"AS.180.102",
39+
"AS.180.242",
40+
"AS.180.302",
41+
"AS.180.334",
42+
"AS.210.102",
43+
"AS.210.111",
44+
"AS.200.101",
45+
"AS.200.141",
46+
"AS.210.112",
47+
"AS.210.202",
48+
"AS.210.211",
49+
"AS.210.212",
50+
"AS.210.301",
51+
"AS.210.302",
52+
"AS.210.311",
53+
"AS.210.312",
54+
"AS.280.350",
55+
"AS.375.116",
56+
"AS.375.216",
57+
"EN.510.312",
58+
"EN.510.314",
59+
"EN.530.105",
60+
"EN.540.202",
61+
"EN.540.203",
62+
"EN.540.303",
63+
"EN.553.111",
64+
"EN.553.112",
65+
"EN.553.171",
66+
"EN.553.211",
67+
"EN.553.310",
68+
"EN.580.222",
69+
"EN.580.223",
70+
"EN.540.422",
71+
"EN.601.107",
72+
"EN.660.203",
73+
]
74+
75+
integration, created = Integration.objects.get_or_create(name="LearningDen")
76+
integration.save()
77+
78+
for code in learning_den_codes:
79+
if Course.objects.filter(school="jhu",code=code).exists():
80+
course = Course.objects.filter(school="jhu",code=code)
81+
courseIntegration, created = CourseIntegration.objects.get_or_create(course=course[0],integration=integration,json='')
82+
courseIntegration.save()
83+
84+
class Migration(migrations.Migration):
85+
86+
dependencies = [
87+
('timetable', '0022_auto_20170821_1305'),
88+
]
89+
90+
operations = [
91+
migrations.RunPython(add_learning_den)
92+
]

0 commit comments

Comments
 (0)