Skip to content

Commit 38d41bb

Browse files
Paul BoesPaul Boes
Paul Boes
authored and
Paul Boes
committed
improved dashboard looks, changed url path structure for dashboards and curate sites in general, changed fa icons for curate, changed model return to python 3
1 parent 8c91be9 commit 38d41bb

File tree

28 files changed

+606
-142
lines changed

28 files changed

+606
-142
lines changed

conf/settings/base.py

+2-1
Original file line numberDiff line numberDiff line change
@@ -139,7 +139,8 @@
139139
CONTROLCENTER_CHARTIST_COLORS = 'material'
140140

141141
STATIC_URL = '/static/'
142-
LOGIN_REDIRECT_URL = '/profile'
142+
LOGIN_REDIRECT_URL = '/curate/dashboard/1'
143+
LOGIN_URL = '/login'
143144

144145
LOGGING = {
145146
'version': 1,

conf/settings/local.py

+4-4
Original file line numberDiff line numberDiff line change
@@ -33,8 +33,8 @@
3333
'PORT': ''}}
3434

3535

36-
# STATICFILES_DIRS += [
37-
# os.path.join(BASE_DIR, "static"),
38-
# ]
36+
STATICFILES_DIRS += [
37+
os.path.join(BASE_DIR, "static"),
38+
]
3939

40-
STATIC_ROOT = os.path.join(BASE_DIR, 'static/')
40+
# STATIC_ROOT = os.path.join(BASE_DIR, 'static/')

conf/urls.py

+2-1
Original file line numberDiff line numberDiff line change
@@ -23,9 +23,10 @@
2323
url(r'^', include('django.contrib.auth.urls')),
2424
url(r'^captcha/', include('captcha.urls')),
2525
url(r'^admin/', include(admin.site.urls)),
26-
url(r'^admin/dashboard/', include(controlcenter.urls)),
26+
#url(r'^admin/dashboard/', include(controlcenter.urls)),
2727
# url(r'^admin/dashboard/(?P<query_pk>\d+)/', include(controlcenter.urls)),
2828
url(r'^curate/', include('curate.urls',namespace="curate")),
29+
url(r'^curate/dashboard/', include(controlcenter.urls)),
2930
url(r'^explore/', include('explore.urls',namespace="explore")),
3031
url(r'^', include('homepage.urls', namespace='homepage')),
3132
url(r'^misc/', include('misc.urls', namespace='misc')),

controlcenter/dashboards.py

+6-1
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,7 @@
1313

1414
class Dashboard(six.with_metaclass(MediaDefiningClass, BaseModel)):
1515
pk = None
16+
customer_key = None
1617
widgets = ()
1718

1819
class Media:
@@ -35,9 +36,13 @@ class Media:
3536
def __init__(self, pk):
3637
super(Dashboard, self).__init__()
3738
self.pk = self.id = pk
39+
40+
41+
def add_customer_key(self, customer_key):
42+
self.customer_key = customer_key
3843

3944
def get_absolute_url(self):
40-
return reverse('controlcenter:dashboard', kwargs={'pk': self.pk})
45+
return reverse('curate:controlcenter_with_key:dashboard', kwargs={'pk': self.pk, 'customer_key':self.customer_key})#
4146

4247
def get_widgets(self, request, **options):
4348
# TODO: permission check

controlcenter/static/controlcenter/css/all.css

+9-7
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
.controlcenter {
2-
line-height: 1.4;
2+
/* line-height: 1.4;
33
font-size: 12px;
4-
color: #333;
4+
color: #333;*/
55
width: 100%;
66
}
77

@@ -40,7 +40,7 @@
4040
background-color: #f2f2f2;
4141
background-repeat: repeat-x, repeat;
4242
background-size: auto 35px, auto;
43-
background-image: url("data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAAEAAABGCAAAAAAoI6GBAAAAOElEQVR4ATWNhQEAMAjDoP/fO3cJkwrmTU1dprmfX/TLwfefA6J2xWEfh4wJ1NJLLTtuNbuYEpUWQ5QZZ928F4cAAAAASUVORK5CYII="), url("data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABgAAAAYCAAAAADFHGIkAAAAQElEQVR4AXXIOQHAAACEMPzbxMEp6NafjMG7XYiXeImXeImXeImXeImXeImXeImX+BE/4kf8iB/xI37Ej/gRvwPQEA5XmMlaIAAAAABJRU5ErkJggg==");
43+
/*background-image: url("data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAAEAAABGCAAAAAAoI6GBAAAAOElEQVR4ATWNhQEAMAjDoP/fO3cJkwrmTU1dprmfX/TLwfefA6J2xWEfh4wJ1NJLLTtuNbuYEpUWQ5QZZ928F4cAAAAASUVORK5CYII="), url("data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABgAAAAYCAAAAADFHGIkAAAAQElEQVR4AXXIOQHAAACEMPzbxMEp6NafjMG7XYiXeImXeImXeImXeImXeImXeImX+BE/4kf8iB/xI37Ej/gRvwPQEA5XmMlaIAAAAABJRU5ErkJggg==");*/
4444
}
4545

4646
.controlcenter__widget__tab {
@@ -56,7 +56,7 @@
5656
position: relative;
5757
cursor: pointer;
5858
text-transform: uppercase;
59-
font-size: 11px;
59+
font-size: 13px;
6060
font-weight: bold;
6161
color: #aaa;
6262
}
@@ -128,7 +128,7 @@
128128
background: #f2f2f2;
129129
line-height: 26px;
130130
padding: 0 15px;
131-
font-size: 12px;
131+
font-size: 14px;
132132
margin-right: -1px;
133133
}
134134

@@ -307,15 +307,17 @@
307307
}
308308

309309
.controlcenter__masonry__block--w2 {
310-
width: 33.333333333333336%;
310+
width: 33%;
311+
/*.333333333333336%;*/
311312
}
312313

313314
.controlcenter__masonry__block--w3 {
314315
width: 50%;
315316
}
316317

317318
.controlcenter__masonry__block--w4 {
318-
width: 66.66666666666667%;
319+
width: 66%;
320+
/*.66666666666667%;*/
319321
}
320322

321323
.controlcenter__masonry__block--w5 {

controlcenter/templates/controlcenter/dashboard.html

+1-1
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,7 @@
77
{{ dashboard.media }}
88
{% endblock %}
99
{% block content %}
10-
<div class="controlcenter" id="{{ dashboard.slug }}">
10+
<div class="controlcenter dashwidth" id="{{ dashboard.slug }}">
1111
{% if dashboards|length > 1 %}
1212
<nav class="controlcenter__nav">
1313
{% for item in dashboards %}

controlcenter/templates/controlcenter/widgets/itemlist.html

+1-1
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,7 @@
1616
<tbody class="controlcenter__table__tbody">
1717
{% for obj in widget.values %}
1818
<tr class="controlcenter__table__tr">
19-
{% change_url widget obj as url %}
19+
{% change_url widget obj as url %}
2020
{% for attr in widget.list_display %}
2121
<td class="controlcenter__table__td controlcenter__table__td--{% if attr == sharp %}row-counter{% else %}{{ attr }}{% endif %}">
2222
{% if attr == sharp %}

controlcenter/views.py

+51-3
Original file line numberDiff line numberDiff line change
@@ -1,16 +1,23 @@
11
from django.conf.urls import url
22
from django.contrib import admin
33
from django.contrib.admin.views.decorators import staff_member_required
4+
from django.contrib.auth.decorators import user_passes_test
45
from django.core.exceptions import ImproperlyConfigured
56
from django.http import Http404
67
from django.utils.decorators import method_decorator
78
from django.utils.module_loading import import_string
89
from django.views.generic.base import TemplateView
10+
from django.contrib.auth.mixins import UserPassesTestMixin
11+
from django.contrib.auth.mixins import LoginRequiredMixin
12+
from django.shortcuts import redirect, render
13+
from django.contrib.auth.decorators import login_required
914

15+
from scope.models import UserProfile
1016
from . import app_settings
1117

1218

1319
class ControlCenter(object):
20+
1421
def __init__(self, view_class):
1522
self.view_class = view_class
1623

@@ -35,29 +42,70 @@ def get_urls(self):
3542
def urls(self):
3643
return self.get_urls(), 'controlcenter', 'controlcenter'
3744

45+
@property
46+
def urls_with_key(self):
47+
return self.get_urls(), 'controlcenter_with_key', 'controlcenter_with_key'
48+
49+
50+
51+
# def user_corresponds_to_customer_key(user):
52+
# user_profile = UserProfile.objects.get(user=request.user)
53+
# return customer_key == user_profile.customer.customer_key:
54+
# pass
55+
56+
57+
class DashboardView(LoginRequiredMixin, TemplateView): #UserPassesTestMixin,
3858

39-
class DashboardView(TemplateView):
4059
dashboards = NotImplemented
4160
template_name = 'controlcenter/dashboard.html'
61+
# def determine_login_path(self):
62+
# return redirect('/login/?next=%s' % self.request.path)
63+
64+
# login_url =
65+
# @method_decorator(staff_member_required)
66+
# def test_func(self):
67+
# try:
68+
# user_profile = UserProfile.objects.get(user=self.request.user)
69+
# customer_key = self.kwargs['customer_key']
70+
# return customer_key == user_profile.customer.customer_key
71+
# except:
72+
# return False
4273

43-
@method_decorator(staff_member_required)
4474
def dispatch(self, *args, **kwargs):
4575
return super(DashboardView, self).dispatch(*args, **kwargs)
4676

4777
def get(self, request, *args, **kwargs):
4878
pk = int(self.kwargs['pk'])
79+
if self.request.user.is_superuser:
80+
try:
81+
customer_key = self.kwargs['customer_key']
82+
except:
83+
return redirect('/admin')
84+
else:
85+
user_profile = UserProfile.objects.get(user=self.request.user)
86+
customer_key = user_profile.customer.customer_key
87+
88+
for dashboard in self.dashboards:
89+
dashboard.add_customer_key(customer_key)
4990
try:
5091
self.dashboard = self.dashboards[pk]
5192
except IndexError:
5293
raise Http404('Dashboard not found.')
5394
return super(DashboardView, self).get(request, *args, **kwargs)
5495

5596
def get_context_data(self, **kwargs):
97+
if self.request.user.is_superuser:
98+
customer_key = self.kwargs['customer_key']
99+
else:
100+
user_profile = UserProfile.objects.get(user=self.request.user)
101+
customer_key = user_profile.customer.customer_key
102+
options = {'customer_key': customer_key}
56103
context = {
104+
'customer_key': customer_key,
57105
'title': self.dashboard.title,
58106
'dashboard': self.dashboard,
59107
'dashboards': self.dashboards,
60-
'groups': self.dashboard.get_widgets(self.request),
108+
'groups': self.dashboard.get_widgets(self.request, **options),
61109
'sharp': app_settings.SHARP,
62110
}
63111

curate/convenience/functions.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -30,7 +30,7 @@ def create_customer_from_config_file(customer_key):
3030

3131
# Create Curate Customer
3232
curate_customer, created_customer = Curate_Customer.objects.get_or_create(
33-
customer=customer, expires=date.today())
33+
customer=customer, default={'expires':date.today()})
3434

3535
# Create curate_query
3636
query = Curate_Query(

curate/methods/curate_process.py

+15-3
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@
44
from scope.methods.semantics import document_embedding
55
from scope.methods.graphs import clustering_methods
66
import scope.methods.dataprovider.provider as provider
7+
from scope.methods.semantics import summarizer
78

89
from scope.models import Customer
910

@@ -99,18 +100,29 @@ def _produce_cluster_dict(self, labels):
99100
return articles_dict
100101

101102
def produce_and_save_clusters(self, labels):
103+
words, rep = self.produce_keywords_and_summaries(labels)
102104
articles_dict = self._produce_cluster_dict(labels)
103105
counter = 1
104106
for key in articles_dict:
105-
cluster = Curate_Query_Cluster(rank=counter, center=key)
106-
cluster.save()
107+
cluster, created = Curate_Query_Cluster.objects.get_or_create(rank=counter, center=key, defaults={'keywords':words[counter-1], 'summary':rep[counter-1]})
108+
cluster.cluster_articles.clear()
107109
for instance in articles_dict[key]:
108110
cluster.cluster_articles.add(instance)
109111
cluster.save()
110112
counter += 1
111113
self.query.no_clusters = counter
112114
self.query.save()
113115

116+
def produce_keywords_and_summaries(self, cluster_articles):
117+
nlp = spacy.load(self.language)
118+
representative_model = summarizer.Summarizer(self.language, nlp)
119+
120+
summary_max_len_words = 40
121+
122+
words = representative_model.get_keywords(cluster_articles)
123+
rep = representative_model.text_rank(cluster_articles, max_size=summary_max_len_words)
124+
return words, rep
125+
114126
def _process(self, filtered_articles):
115127

116128
# filtered_articles = remove_duplicate_articles_for_processing(filtered_articles)
@@ -125,7 +137,7 @@ def _process(self, filtered_articles):
125137

126138
print([a.title for a in selected_articles])
127139

128-
# you can generate the dict at this point actually.
140+
#you can generate the dict at this point actually.
129141
self.produce_and_save_clusters(cluster_articles)
130142
else:
131143
selected_articles = []
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,35 @@
1+
# -*- coding: utf-8 -*-
2+
# Generated by Django 1.11.1 on 2017-06-01 19:22
3+
from __future__ import unicode_literals
4+
5+
from django.db import migrations, models
6+
7+
8+
class Migration(migrations.Migration):
9+
10+
dependencies = [
11+
('curate', '0024_auto_20170522_2054'),
12+
]
13+
14+
operations = [
15+
migrations.AlterField(
16+
model_name='curate_customer_selection',
17+
name='color',
18+
field=models.CharField(default='#fff', max_length=100),
19+
),
20+
migrations.AlterField(
21+
model_name='curate_customer_selection',
22+
name='kind',
23+
field=models.CharField(choices=[('sel', 'selection'), ('mis', 'mistake'), ('oth', 'other')], default='sel', max_length=3),
24+
),
25+
migrations.AlterField(
26+
model_name='curate_query_cluster',
27+
name='summary',
28+
field=models.CharField(max_length=500),
29+
),
30+
migrations.AlterField(
31+
model_name='curate_rejection_reasons',
32+
name='kind',
33+
field=models.CharField(choices=[('sou', 'bad source'), ('con', 'bad content'), ('frq', 'too frequent'), ('oth', 'other')], default='con', max_length=3),
34+
),
35+
]

curate/models.py

+6-7
Original file line numberDiff line numberDiff line change
@@ -11,10 +11,9 @@ class Curate_Customer(models.Model):
1111
expires = models.DateField(blank=True)
1212
bad_source = models.ManyToManyField(Source, blank=True)
1313

14-
def __unicode__(self): # __unicode__ on Python 2
14+
def __str__(self):
1515
return self.customer.name
1616

17-
1817
class Curate_Customer_Selection(models.Model):
1918
curate_customer = models.ForeignKey(Curate_Customer)
2019
kind = models.CharField(max_length=3, choices=(
@@ -25,7 +24,7 @@ class Curate_Customer_Selection(models.Model):
2524
)
2625
name = models.CharField(max_length=100)
2726
color = models.CharField(max_length=100, default="#fff")
28-
def __unicode__(self): # __unicode__ on Python 2
27+
def __str__(self): # __str__ on Python 2
2928
return self.name + ', ' + self.curate_customer.customer.name
3029

3130
def human_readable_name(self):
@@ -42,7 +41,7 @@ class Curate_Rejection_Reasons(models.Model):
4241
), default="con"
4342
)
4443
current_members = models.ManyToManyField('Article_Curate_Query', blank=True)
45-
def __unicode__(self): # __unicode__ on Python 2
44+
def __str__(self): # __str__ on Python 2
4645
return self.name + ', ' + self.selection.name + ', ' + self.selection.curate_customer.customer.name
4746
def human_readable_name(self):
4847
return self.name.replace('_', ' ')
@@ -56,7 +55,7 @@ class Curate_Query(models.Model):
5655
no_clusters = models.IntegerField(null=True, blank=True)
5756
selection_made = models.BooleanField(default=False)
5857

59-
def __unicode__(self): # __unicode__ on Python 2
58+
def __str__(self): # __str__ on Python 2
6059
return self.time_stamp.isoformat() + ', ' + self.curate_customer.customer.name
6160

6261
def selected_articles(self):
@@ -75,15 +74,15 @@ class Article_Curate_Query(models.Model):
7574
agent = models.ForeignKey(Agent, null=True, blank=True)
7675
newsletter = models.ForeignKey(Newsletter, null=True, blank=True)
7776
bad_article = models.BooleanField(default=False)
78-
def __unicode__(self): # __unicode__ on Python 2
77+
def __str__(self): # __str__ on Python 2
7978
return self.article.title + ', ' + self.curate_query.time_stamp.isoformat() + ', ' + self.curate_query.curate_customer.customer.name
8079

8180
class Curate_Query_Cluster(models.Model):
8281
rank = models.IntegerField(null=True, blank=True)
8382
center = models.ForeignKey(Article_Curate_Query)
8483
cluster_articles = models.ManyToManyField(Article_Curate_Query, blank=True, related_name='cluster_articles')
8584
keywords = models.CharField(max_length=50)
86-
summary = models.CharField(max_length=300)
85+
summary = models.CharField(max_length=500)
8786

8887

8988

0 commit comments

Comments
 (0)