Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Event views #147

Merged
merged 36 commits into from
Dec 11, 2015
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
36 commits
Select commit Hold shift + click to select a range
bc91d68
added the view layer to manage participants
shreyas-satish Oct 15, 2015
5cd8bf7
resolved conflicts
shreyas-satish Oct 19, 2015
b66d557
make and return qrcode in-memory instead of saving to disk
shreyas-satish Oct 19, 2015
fb3ca39
use the form object for checkin
shreyas-satish Oct 19, 2015
9a5b045
clean up views
shreyas-satish Oct 19, 2015
8bef554
fixed upsert call
shreyas-satish Oct 20, 2015
0f8da74
added qrcode to requirements
shreyas-satish Oct 20, 2015
83a7562
added ability to add and edit events
shreyas-satish Oct 20, 2015
cfc17dd
fix background job
shreyas-satish Oct 20, 2015
4e751c8
fix refs
shreyas-satish Oct 20, 2015
779050c
separate event and ticket type lists
shreyas-satish Oct 20, 2015
effeca7
Merge branch 'master' into event_views
shreyas-satish Nov 6, 2015
f17f519
added more forms
shreyas-satish Nov 10, 2015
fa23c9a
fixed background job
shreyas-satish Nov 10, 2015
9bdba30
fix routes, minimal admin UI
shreyas-satish Nov 10, 2015
40829a3
rm bulk import for now. will be included in a later release
shreyas-satish Nov 12, 2015
c5830de
cleanup
shreyas-satish Nov 12, 2015
14ffe84
rm file validator. later release
shreyas-satish Nov 12, 2015
530f958
moved tablesearch to scripts.js.
shreyas-satish Nov 12, 2015
dd33ed8
cleanup
shreyas-satish Nov 12, 2015
50d72a1
Adding footable to ext_requires and UI changes for responsive partici…
vidya-ram Nov 16, 2015
53f59d9
Merge pull request #148 from hasgeek/add-footable
shreyas-satish Nov 17, 2015
cced982
rm filter_by_ticket_types. use a simpler approach directly in view
shreyas-satish Dec 2, 2015
dff5bd6
use funnel for q name
shreyas-satish Dec 2, 2015
b31e770
Merge branch 'event_views' of github.com:hasgeek/funnel into event_views
shreyas-satish Dec 2, 2015
7c18e6d
use verb-noun for permission names
shreyas-satish Dec 2, 2015
db2d538
added null case. return unicode
shreyas-satish Dec 2, 2015
deebee3
use cStringIO if available
shreyas-satish Dec 2, 2015
3311616
rm unnecessary param. fix q name
shreyas-satish Dec 2, 2015
da70d6e
use add_and_commit
shreyas-satish Dec 2, 2015
8e9f6d3
use coaster's getbool
shreyas-satish Dec 2, 2015
5c3766e
fix form name
shreyas-satish Dec 2, 2015
dd332a9
use jsonify instead of jsonp
shreyas-satish Dec 11, 2015
815d81a
back to StringIO, marginally faster and accepts unicode
shreyas-satish Dec 11, 2015
125b852
droidcon 2015 badge template
vidya-ram Dec 11, 2015
046529f
Change page rule for droidconbadge template
vidya-ram Dec 11, 2015
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 2 additions & 2 deletions funnel/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -45,8 +45,8 @@ def init_for(env):
lastuser.init_app(app)
lastuser.init_usermanager(UserManager(db, models.User, models.Team))
baseframe.init_app(app, requires=['funnel'], ext_requires=[
('codemirror-markdown', 'pygments'), 'toastr', 'baseframe-bs3', 'fontawesome>=4.0.0'
])
('codemirror-markdown', 'pygments'), 'toastr', 'baseframe-bs3', 'fontawesome>=4.0.0',
'footable'])
app.assets.register('js_fullcalendar',
Bundle(assets.require('!jquery.js', 'jquery.fullcalendar.js', 'spectrum.js'),
output='js/fullcalendar.packed.js', filters='uglipyjs'))
Expand Down
1 change: 1 addition & 0 deletions funnel/forms/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -9,3 +9,4 @@
from .venue import *
from .session import *
from .profile import *
from .participant import *
28 changes: 28 additions & 0 deletions funnel/forms/participant.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,28 @@
# -*- coding: utf-8 -*-
import os
import wtforms
import baseframe.forms as forms
from baseframe import __
from wtforms.ext.sqlalchemy.fields import QuerySelectMultipleField
from wtforms.widgets import CheckboxInput, ListWidget

__all__ = ['ParticipantForm', 'ParticipantBadgeForm']


class ParticipantForm(forms.Form):
fullname = forms.StringField(__("Full Name"), validators=[forms.validators.DataRequired()])
email = forms.EmailField(__("Email"), validators=[forms.validators.DataRequired(), forms.validators.Length(max=80)])
phone = forms.StringField(__("Phone number"), validators=[forms.validators.Length(max=80)])
city = forms.StringField(__("City"), validators=[forms.validators.Length(max=80)])
company = forms.StringField(__("Company"), validators=[forms.validators.Length(max=80)])
job_title = forms.StringField(__("Job Title"), validators=[forms.validators.Length(max=80)])
twitter = forms.StringField(__("Twitter"), validators=[forms.validators.Length(max=15)])
events = QuerySelectMultipleField(__("Events"),
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Doesn't this need a model or query factory?

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Because of the concerns in hasgeek/baseframe#112, I'd move this context into this file itself if possible.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The events need to be scoped by the proposal space, which is why they're set in the view.

widget=ListWidget(), option_widget=CheckboxInput(),
get_label='title',
validators=[forms.validators.DataRequired(u"Select at least one event")])


class ParticipantBadgeForm(forms.Form):
choices = [('', "Badge printing status"), ('t', "Printed"), ('f', "Not printed")]
badge_printed = forms.SelectField("", choices=[(val_title[0], val_title[1]) for val_title in choices])
22 changes: 21 additions & 1 deletion funnel/forms/space.py
Original file line number Diff line number Diff line change
Expand Up @@ -2,13 +2,15 @@

import re
from coaster.utils import sorted_timezones
from wtforms.widgets import CheckboxInput, ListWidget
from wtforms.ext.sqlalchemy.fields import QuerySelectMultipleField
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think we should expose the SQLAlchemy fields in Baseframe forms so they don't need specific import here.

from baseframe import _, __
import baseframe.forms as forms
from baseframe.forms.sqlalchemy import AvailableName, QuerySelectField
from .profile import profile_teams
from ..models import RSVP_STATUS

__all__ = ['ProposalSpaceForm', 'RsvpForm']
__all__ = ['ProposalSpaceForm', 'RsvpForm', 'EventForm', 'TicketTypeForm', 'TicketClientForm']


valid_color_re = re.compile("^[a-fA-F\d]{6}|[a-fA-F\d]{3}$")
Expand Down Expand Up @@ -74,3 +76,21 @@ def validate_bg_color(self, field):

class RsvpForm(forms.Form):
status = forms.RadioField("Status", choices=[(k, RSVP_STATUS[k].title) for k in RSVP_STATUS.USER_CHOICES])


class EventForm(forms.Form):
title = forms.StringField(__("Title"), validators=[forms.validators.DataRequired()])


class TicketClientForm(forms.Form):
name = forms.StringField(__("Name"), validators=[forms.validators.DataRequired()])
clientid = forms.StringField(__("Client id"), validators=[forms.validators.DataRequired()])
client_eventid = forms.StringField(__("Client event id"), validators=[forms.validators.DataRequired()])
client_secret = forms.StringField(__("Client event secret"), validators=[forms.validators.DataRequired()])
client_access_token = forms.StringField(__("Client access token"), validators=[forms.validators.DataRequired()])


class TicketTypeForm(forms.Form):
title = forms.StringField(__("Title"), validators=[forms.validators.DataRequired()])
events = QuerySelectMultipleField(__("Events"),
widget=ListWidget(), option_widget=CheckboxInput(), allow_blank=True, get_label='title', query_factory=lambda: [])
3 changes: 3 additions & 0 deletions funnel/jobs/__init__.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,3 @@
# -*- coding: utf-8 -*-

from .jobs import *
16 changes: 16 additions & 0 deletions funnel/jobs/jobs.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,16 @@
from ..models import (db, TicketClient, SyncTicket)
from ..extapi.explara import ExplaraAPI
from flask.ext.rq import job
from funnel import app


@job('funnel')
def import_tickets(ticket_client_id):
with app.test_request_context():
ticket_client = TicketClient.query.get(ticket_client_id)
if ticket_client and ticket_client.name == u'explara':
ticket_list = ExplaraAPI(access_token=ticket_client.client_access_token).get_tickets(ticket_client.client_eventid)
# cancelled tickets are excluded from the list returned by get_tickets
cancel_list = SyncTicket.exclude(ticket_client, [ticket.get('ticket_no') for ticket in ticket_list]).all()
ticket_client.import_from_list(ticket_list, cancel_list=cancel_list)
db.session.commit()
1 change: 1 addition & 0 deletions funnel/models/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -16,3 +16,4 @@
from .venue import *
from .rsvp import *
from .event import *
from .contact_exchange import *
21 changes: 18 additions & 3 deletions funnel/models/event.py
Original file line number Diff line number Diff line change
Expand Up @@ -144,6 +144,12 @@ def remove_events(self, events):
if event in self.events:
self.events.remove(event)

@classmethod
def checkin_list(cls, event):
participant_attendee_join = db.join(Participant, Attendee, Participant.id == Attendee.participant_id)
stmt = db.select([Participant.id, Participant.fullname, Participant.email, Participant.company, Participant.twitter, Participant.puk, Participant.key, Attendee.checked_in, Participant.badge_printed]).select_from(participant_attendee_join).where(Attendee.event_id == event.id).order_by(Participant.fullname)
return db.session.execute(stmt).fetchall()


class Attendee(BaseMixin, db.Model):
"""
Expand All @@ -162,6 +168,10 @@ class Attendee(BaseMixin, db.Model):

__table_args__ = (db.UniqueConstraint('event_id', 'participant_id'),)

@classmethod
def get(cls, event, participant):
return cls.query.filter_by(event=event, participant=participant).one_or_none()


class TicketClient(BaseMixin, db.Model):
__tablename__ = 'ticket_client'
Expand All @@ -174,7 +184,7 @@ class TicketClient(BaseMixin, db.Model):
proposal_space = db.relationship(ProposalSpace,
backref=db.backref('ticket_clients', cascade='all, delete-orphan'))

def import_from_list(self, space, ticket_list, cancel_list=[]):
def import_from_list(self, ticket_list, cancel_list=[]):
"""
Batch upserts the tickets and its associated ticket types and participants.
Cancels the tickets in cancel_list.
Expand All @@ -183,9 +193,9 @@ def import_from_list(self, space, ticket_list, cancel_list=[]):
ticket.participant.remove_events(ticket.ticket_type.events)

for ticket_dict in ticket_list:
ticket_type = TicketType.upsert(space, current_title=ticket_dict['ticket_type'])
ticket_type = TicketType.upsert(self.proposal_space, current_title=ticket_dict['ticket_type'])

participant = Participant.upsert(space, ticket_dict['email'],
participant = Participant.upsert(self.proposal_space, ticket_dict['email'],
fullname=ticket_dict['fullname'],
phone=ticket_dict['phone'],
twitter=ticket_dict['twitter'],
Expand Down Expand Up @@ -243,3 +253,8 @@ def upsert(cls, ticket_client, order_no, ticket_no, **fields):
db.session.add(ticket)

return ticket

@classmethod
def exclude(cls, ticket_client, ticket_nos):
return cls.query.filter_by(ticket_client=ticket_client
).filter(~cls.ticket_no.in_(ticket_nos))
28 changes: 28 additions & 0 deletions funnel/models/space.py
Original file line number Diff line number Diff line change
Expand Up @@ -168,6 +168,12 @@ def permissions(self, user, inherited=None):
'view-rsvps',
'new-session',
'edit-session',
'new-event',
'new-ticket-type',
'new-ticket-client',
'edit-ticket-client',
'edit-event',
'admin'
])
if self.review_team and user in self.review_team.users:
perms.update([
Expand All @@ -180,6 +186,12 @@ def permissions(self, user, inherited=None):
'edit-schedule',
'new-session',
'edit-session',
'checkin-event',
'view-event',
'view-ticket-type',
'edit-participant',
'view-participant',
'new-participant'
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We'll need to be consistent with verb-noun or noun-verb order with permissions. Our convention appears to be verb-noun for permission names but noun-verb for view names (which in itself is a confusing difference).

])
return perms

Expand Down Expand Up @@ -226,6 +238,22 @@ def url_for(self, action='view', _external=False):
return url_for('rsvp', profile=self.profile.name, space=self.name)
elif action == 'rsvp-list':
return url_for('rsvp_list', profile=self.profile.name, space=self.name)
elif action == 'admin':
return url_for('admin', profile=self.profile.name, space=self.name)
elif action == 'events':
return url_for('events', profile=self.profile.name, space=self.name)
elif action == 'participants':
return url_for('participants', profile=self.profile.name, space=self.name)
elif action == 'new-participant':
return url_for('new_participant', profile=self.profile.name, space=self.name)
elif action == 'new-ticket-type-participant':
return url_for('new_ticket_type_participant', profile=self.profile.name, space=self.name)
elif action == 'new-event':
return url_for('new_event', profile=self.profile.name, space=self.name)
elif action == 'new-ticket-type':
return url_for('new_ticket_type', profile=self.profile.name, space=self.name)
elif action == 'new-ticket-client':
return url_for('new_ticket_client', profile=self.profile.name, space=self.name)
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hopefully we'll clean this up in hasgeek/coaster#77. :)


@classmethod
def all(cls):
Expand Down
58 changes: 58 additions & 0 deletions funnel/static/css/app.css
Original file line number Diff line number Diff line change
Expand Up @@ -658,3 +658,61 @@ div.markdown-field a.button.selected {
.rsvp-header {
padding: 1px 0px 1px 15px;
}

.footable > thead > tr > th {
background-color: #e8e8e8;
border: 1px solid #b7b7b7;
}

.list-inline > li.badge-print-status-btn {
display: block;
width: 200px;
}

@media (min-width: 768px) {
.list-inline > li.badge-print-status-btn {
display: inline-block;
width: auto;
}
}
.status-btn-list {
margin: 0;
padding: 0;
}

.status-btn-list li {
padding-left: 0;
padding-right: 10px;
}

.status-btn-list .controls {
padding-left: 0;
padding-right: 10px;
}

.status-btn-list li, .status-btn-list .controls, .search-participant {
margin-bottom: 10px;
}

td .status-btn-list li:last-child {
margin: 0;
}

@media (min-width: 992px) {
.status-btn-list .controls {
margin-bottom: 0;
}

td .status-btn-list li {
margin: 0 0 5px;
}
}
@media (min-width: 1200px) {
.status-btn-list {
float: right;
}

.status-btn-list li:last-child {
padding-right: 0;
}
}
58 changes: 58 additions & 0 deletions funnel/static/js/scripts.js
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@

function radioHighlight(radioName, highlightClass) {
var selector = "input[name='" + radioName + "']";
$(selector + ":checked").parent().addClass(highlightClass);
Expand Down Expand Up @@ -89,3 +90,60 @@ $(function() {
}
});
});

window.TableSearch = function(tableId){
// a little library that takes a table id
// and provides a method to search the table's rows for a given query.
// the row's td must contain the class 'js-searchable' to be considered
// for searching.
// Eg:
// var tableSearch = new TableSearch('tableId');
// var hits = tableSearch.searchRows('someQuery');
// 'hits' is a list of ids of the table's rows which contained 'someQuery'
this.tableId = tableId;
this.rowData = [];
this.allMatchedIds = [];
};

window.TableSearch.prototype.getRows = function(){
return $('#' + this.tableId +' tbody tr');
};

window.TableSearch.prototype.setRowData = function(rowD){
// Builds a list of objects and sets it the object's rowData
var rowMap = [];
$.each(this.getRows(), function(rowIndex, row){
rowMap.push({
'rid': '#' + $(row).attr('id'),
'text': $(row).find('td.js-searchable').text().toLowerCase()
});
});
this.rowData = rowMap;
};

window.TableSearch.prototype.setAllMatchedIds = function(ids) {
this.allMatchedIds = ids;
};

window.TableSearch.prototype.searchRows = function(q){
// Search the rows of the table for a supplied query.
// reset data collection on first search or if table has changed
if (this.rowData.length !== this.getRows().length) {
this.setRowData();
}
// return cached matched ids if query is blank
if (q === '' && this.allMatchedIds.length !== 0) {
return this.allMatchedIds;
}
var matchedIds = [];
for (var i = this.rowData.length - 1; i >= 0; i--) {
if (this.rowData[i].text.indexOf(q.toLowerCase()) !== -1) {
matchedIds.push(this.rowData[i]['rid']);
}
}
// cache ids if query is blank
if (q === '') {
this.setAllMatchedIds(matchedIds);
}
return matchedIds;
};
Loading