2017-07-03 16:58:30 -05:00
|
|
|
# -*- coding: utf-8; -*-
|
2013-05-07 19:41:58 -05:00
|
|
|
################################################################################
|
|
|
|
#
|
|
|
|
# Rattail -- Retail Software Framework
|
2020-03-17 18:50:07 -05:00
|
|
|
# Copyright © 2010-2020 Lance Edgar
|
2013-05-07 19:41:58 -05:00
|
|
|
#
|
|
|
|
# This file is part of Rattail.
|
|
|
|
#
|
|
|
|
# Rattail is free software: you can redistribute it and/or modify it under the
|
2017-07-06 23:47:56 -05:00
|
|
|
# terms of the GNU General Public License as published by the Free Software
|
|
|
|
# Foundation, either version 3 of the License, or (at your option) any later
|
|
|
|
# version.
|
2013-05-07 19:41:58 -05:00
|
|
|
#
|
|
|
|
# Rattail is distributed in the hope that it will be useful, but WITHOUT ANY
|
|
|
|
# WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
|
2017-07-06 23:47:56 -05:00
|
|
|
# FOR A PARTICULAR PURPOSE. See the GNU General Public License for more
|
|
|
|
# details.
|
2013-05-07 19:41:58 -05:00
|
|
|
#
|
2017-07-06 23:47:56 -05:00
|
|
|
# You should have received a copy of the GNU General Public License along with
|
|
|
|
# Rattail. If not, see <http://www.gnu.org/licenses/>.
|
2013-05-07 19:41:58 -05:00
|
|
|
#
|
|
|
|
################################################################################
|
|
|
|
"""
|
2013-09-01 09:27:47 -05:00
|
|
|
Person Views
|
2013-05-07 19:41:58 -05:00
|
|
|
"""
|
|
|
|
|
2015-12-06 18:04:23 -06:00
|
|
|
from __future__ import unicode_literals, absolute_import
|
|
|
|
|
2017-08-04 11:55:53 -05:00
|
|
|
import six
|
2015-12-06 18:04:23 -06:00
|
|
|
import sqlalchemy as sa
|
2019-05-14 15:59:57 -05:00
|
|
|
from sqlalchemy import orm
|
2013-05-07 19:41:58 -05:00
|
|
|
|
2017-10-29 03:18:05 -05:00
|
|
|
from rattail.db import model, api
|
2019-08-11 17:30:08 -05:00
|
|
|
from rattail.time import localtime
|
2020-03-18 11:27:58 -05:00
|
|
|
from rattail.util import OrderedDict
|
2017-10-29 03:18:05 -05:00
|
|
|
|
2019-06-04 19:23:27 -05:00
|
|
|
import colander
|
2015-12-06 18:04:23 -06:00
|
|
|
from pyramid.httpexceptions import HTTPFound, HTTPNotFound
|
2017-07-06 20:13:42 -05:00
|
|
|
from webhelpers2.html import HTML, tags
|
2013-05-21 23:51:41 -05:00
|
|
|
|
2019-06-04 19:23:27 -05:00
|
|
|
from tailbone import forms, grids
|
2018-02-05 21:23:23 -06:00
|
|
|
from tailbone.views import MasterView, AutocompleteView
|
2017-10-24 12:09:28 -05:00
|
|
|
|
|
|
|
|
2015-12-06 18:04:23 -06:00
|
|
|
class PeopleView(MasterView):
|
|
|
|
"""
|
|
|
|
Master view for the Person class.
|
|
|
|
"""
|
|
|
|
model_class = model.Person
|
|
|
|
model_title_plural = "People"
|
|
|
|
route_prefix = 'people'
|
2019-08-02 11:30:46 -05:00
|
|
|
touchable = True
|
2017-07-05 03:07:35 -05:00
|
|
|
has_versions = True
|
2018-07-15 18:13:30 -05:00
|
|
|
supports_mobile = True
|
2020-03-16 17:47:06 -05:00
|
|
|
bulk_deletable = True
|
2020-03-17 18:50:07 -05:00
|
|
|
is_contact = True
|
2019-06-04 19:23:27 -05:00
|
|
|
manage_notes_from_profile_view = False
|
2015-12-06 18:04:23 -06:00
|
|
|
|
2020-03-17 18:50:07 -05:00
|
|
|
labels = {
|
|
|
|
'default_phone': "Phone Number",
|
|
|
|
'default_email': "Email Address",
|
|
|
|
}
|
|
|
|
|
2017-07-07 09:13:53 -05:00
|
|
|
grid_columns = [
|
|
|
|
'display_name',
|
|
|
|
'first_name',
|
|
|
|
'last_name',
|
|
|
|
'phone',
|
|
|
|
'email',
|
|
|
|
]
|
|
|
|
|
2017-12-04 18:49:52 -06:00
|
|
|
form_fields = [
|
|
|
|
'first_name',
|
|
|
|
'middle_name',
|
|
|
|
'last_name',
|
|
|
|
'display_name',
|
2020-03-17 18:50:07 -05:00
|
|
|
'default_phone',
|
|
|
|
'default_email',
|
2017-12-04 18:49:52 -06:00
|
|
|
'address',
|
|
|
|
'employee',
|
|
|
|
'customers',
|
2020-03-18 13:15:11 -05:00
|
|
|
'members',
|
2017-12-04 18:49:52 -06:00
|
|
|
'users',
|
|
|
|
]
|
|
|
|
|
2018-07-15 18:13:30 -05:00
|
|
|
mobile_form_fields = [
|
|
|
|
'first_name',
|
|
|
|
'middle_name',
|
|
|
|
'last_name',
|
|
|
|
'display_name',
|
|
|
|
'phone',
|
|
|
|
'email',
|
|
|
|
'address',
|
|
|
|
'employee',
|
|
|
|
'customers',
|
|
|
|
'users',
|
|
|
|
]
|
|
|
|
|
2017-07-07 09:13:53 -05:00
|
|
|
def configure_grid(self, g):
|
|
|
|
super(PeopleView, self).configure_grid(g)
|
|
|
|
|
2015-12-06 18:04:23 -06:00
|
|
|
g.joiners['email'] = lambda q: q.outerjoin(model.PersonEmailAddress, sa.and_(
|
|
|
|
model.PersonEmailAddress.parent_uuid == model.Person.uuid,
|
|
|
|
model.PersonEmailAddress.preference == 1))
|
|
|
|
g.joiners['phone'] = lambda q: q.outerjoin(model.PersonPhoneNumber, sa.and_(
|
|
|
|
model.PersonPhoneNumber.parent_uuid == model.Person.uuid,
|
|
|
|
model.PersonPhoneNumber.preference == 1))
|
|
|
|
|
2017-07-07 09:13:53 -05:00
|
|
|
g.filters['email'] = g.make_filter('email', model.PersonEmailAddress.address)
|
2019-04-10 14:20:36 -05:00
|
|
|
g.set_filter('phone', model.PersonPhoneNumber.number,
|
|
|
|
factory=grids.filters.AlchemyPhoneNumberFilter)
|
2015-12-06 18:04:23 -06:00
|
|
|
|
2017-07-03 16:58:30 -05:00
|
|
|
g.joiners['customer_id'] = lambda q: q.outerjoin(model.CustomerPerson).outerjoin(model.Customer)
|
2017-07-07 09:13:53 -05:00
|
|
|
g.filters['customer_id'] = g.make_filter('customer_id', model.Customer.id)
|
2017-07-03 16:58:30 -05:00
|
|
|
|
2015-12-06 18:04:23 -06:00
|
|
|
g.filters['first_name'].default_active = True
|
|
|
|
g.filters['first_name'].default_verb = 'contains'
|
|
|
|
|
|
|
|
g.filters['last_name'].default_active = True
|
|
|
|
g.filters['last_name'].default_verb = 'contains'
|
|
|
|
|
|
|
|
g.sorters['email'] = lambda q, d: q.order_by(getattr(model.PersonEmailAddress.address, d)())
|
|
|
|
g.sorters['phone'] = lambda q, d: q.order_by(getattr(model.PersonPhoneNumber.number, d)())
|
|
|
|
|
2017-12-04 22:40:10 -06:00
|
|
|
g.set_sort_defaults('display_name')
|
2015-12-06 18:04:23 -06:00
|
|
|
|
2017-07-07 09:13:53 -05:00
|
|
|
g.set_label('display_name', "Full Name")
|
|
|
|
g.set_label('phone', "Phone Number")
|
|
|
|
g.set_label('email', "Email Address")
|
|
|
|
g.set_label('customer_id', "Customer ID")
|
2013-05-21 23:51:41 -05:00
|
|
|
|
2017-08-02 20:40:02 -05:00
|
|
|
g.set_link('display_name')
|
2017-08-03 19:16:53 -05:00
|
|
|
g.set_link('first_name')
|
|
|
|
g.set_link('last_name')
|
2017-08-02 20:40:02 -05:00
|
|
|
|
2015-12-06 18:04:23 -06:00
|
|
|
def get_instance(self):
|
|
|
|
# TODO: I don't recall why this fallback check for a vendor contact
|
|
|
|
# exists here, but leaving it intact for now.
|
|
|
|
key = self.request.matchdict['uuid']
|
|
|
|
instance = self.Session.query(model.Person).get(key)
|
|
|
|
if instance:
|
|
|
|
return instance
|
|
|
|
instance = self.Session.query(model.VendorContact).get(key)
|
|
|
|
if instance:
|
|
|
|
return instance.person
|
|
|
|
raise HTTPNotFound
|
|
|
|
|
2020-09-23 16:39:44 -05:00
|
|
|
def is_person_protected(self, person):
|
|
|
|
for user in person.users:
|
|
|
|
if self.user_is_protected(user):
|
|
|
|
return True
|
|
|
|
return False
|
|
|
|
|
2017-03-19 11:21:00 -05:00
|
|
|
def editable_instance(self, person):
|
2020-09-23 16:39:44 -05:00
|
|
|
if self.request.is_root:
|
|
|
|
return True
|
|
|
|
return not self.is_person_protected(person)
|
2017-03-19 11:21:00 -05:00
|
|
|
|
|
|
|
def deletable_instance(self, person):
|
2020-09-23 16:39:44 -05:00
|
|
|
if self.request.is_root:
|
|
|
|
return True
|
|
|
|
return not self.is_person_protected(person)
|
2017-03-19 11:21:00 -05:00
|
|
|
|
2020-03-16 17:47:06 -05:00
|
|
|
def delete_instance(self, person):
|
|
|
|
"""
|
|
|
|
Supplements the default logic as follows:
|
|
|
|
|
|
|
|
Any customer associations are first deleted for the person. Once that
|
|
|
|
is complete, deletion continues as per usual.
|
|
|
|
"""
|
|
|
|
session = orm.object_session(person)
|
|
|
|
|
|
|
|
# must explicitly remove all CustomerPerson records
|
|
|
|
for cp in list(person._customers):
|
|
|
|
customer = cp.customer
|
|
|
|
session.delete(cp)
|
|
|
|
# session.flush()
|
|
|
|
customer._people.reorder()
|
|
|
|
|
|
|
|
# continue with normal logic
|
|
|
|
super(PeopleView, self).delete_instance(person)
|
|
|
|
|
2020-03-11 13:30:04 -05:00
|
|
|
def touch_instance(self, person):
|
|
|
|
"""
|
|
|
|
Supplements the default logic as follows:
|
|
|
|
|
|
|
|
In addition to "touching" the person proper, we also "touch" each
|
|
|
|
contact info record associated with them.
|
|
|
|
"""
|
|
|
|
# touch person, as per usual
|
|
|
|
super(PeopleView, self).touch_instance(person)
|
|
|
|
|
|
|
|
def touch(obj):
|
|
|
|
change = model.Change()
|
|
|
|
change.class_name = obj.__class__.__name__
|
|
|
|
change.instance_uuid = obj.uuid
|
|
|
|
change.deleted = False
|
|
|
|
self.Session.add(change)
|
|
|
|
|
|
|
|
# phone numbers
|
|
|
|
for phone in person.phones:
|
|
|
|
touch(phone)
|
|
|
|
|
|
|
|
# email addresses
|
|
|
|
for email in person.emails:
|
|
|
|
touch(email)
|
|
|
|
|
|
|
|
# mailing addresses
|
|
|
|
for address in person.addresses:
|
|
|
|
touch(address)
|
|
|
|
|
2018-07-16 20:40:29 -05:00
|
|
|
def configure_common_form(self, f):
|
|
|
|
super(PeopleView, self).configure_common_form(f)
|
2020-03-17 18:50:07 -05:00
|
|
|
person = f.model_instance
|
2017-12-04 18:49:52 -06:00
|
|
|
|
|
|
|
f.set_label('display_name', "Full Name")
|
|
|
|
|
2020-03-17 18:50:07 -05:00
|
|
|
# TODO: should remove this?
|
2017-12-04 18:49:52 -06:00
|
|
|
f.set_readonly('phone')
|
|
|
|
f.set_label('phone', "Phone Number")
|
|
|
|
|
2020-03-17 18:50:07 -05:00
|
|
|
f.set_renderer('default_phone', self.render_default_phone)
|
|
|
|
if not self.creating and person.phones:
|
|
|
|
f.set_default('default_phone', person.phones[0].number)
|
|
|
|
|
|
|
|
# TODO: should remove this?
|
2017-12-04 18:49:52 -06:00
|
|
|
f.set_readonly('email')
|
|
|
|
f.set_label('email', "Email Address")
|
|
|
|
|
2020-03-17 18:50:07 -05:00
|
|
|
f.set_renderer('default_email', self.render_default_email)
|
|
|
|
if not self.creating and person.emails:
|
|
|
|
f.set_default('default_email', person.emails[0].address)
|
|
|
|
|
2017-12-04 18:49:52 -06:00
|
|
|
f.set_readonly('address')
|
|
|
|
f.set_label('address', "Mailing Address")
|
|
|
|
|
2018-11-21 19:56:01 -06:00
|
|
|
# employee
|
|
|
|
if self.creating:
|
|
|
|
f.remove_field('employee')
|
|
|
|
else:
|
|
|
|
f.set_readonly('employee')
|
|
|
|
f.set_renderer('employee', self.render_employee)
|
|
|
|
|
|
|
|
# customers
|
|
|
|
if self.creating:
|
|
|
|
f.remove_field('customers')
|
|
|
|
else:
|
|
|
|
f.set_readonly('customers')
|
|
|
|
f.set_renderer('customers', self.render_customers)
|
|
|
|
|
2020-03-18 13:15:11 -05:00
|
|
|
# members
|
|
|
|
if self.creating:
|
|
|
|
f.remove_field('members')
|
|
|
|
else:
|
|
|
|
f.set_readonly('members')
|
|
|
|
f.set_renderer('members', self.render_members)
|
|
|
|
|
2018-11-21 19:56:01 -06:00
|
|
|
# users
|
|
|
|
if self.creating:
|
|
|
|
f.remove_field('users')
|
|
|
|
else:
|
|
|
|
f.set_readonly('users')
|
|
|
|
f.set_renderer('users', self.render_users)
|
2017-12-04 18:49:52 -06:00
|
|
|
|
|
|
|
def render_employee(self, person, field):
|
|
|
|
employee = person.employee
|
|
|
|
if not employee:
|
|
|
|
return ""
|
|
|
|
text = six.text_type(employee)
|
|
|
|
url = self.request.route_url('employees.view', uuid=employee.uuid)
|
|
|
|
return tags.link_to(text, url)
|
|
|
|
|
|
|
|
def render_customers(self, person, field):
|
|
|
|
customers = person._customers
|
|
|
|
if not customers:
|
|
|
|
return ""
|
|
|
|
items = []
|
|
|
|
for customer in customers:
|
|
|
|
customer = customer.customer
|
|
|
|
text = six.text_type(customer)
|
2020-03-18 13:15:11 -05:00
|
|
|
if customer.number:
|
|
|
|
text = "(#{}) {}".format(customer.number, text)
|
|
|
|
elif customer.id:
|
2017-12-04 18:49:52 -06:00
|
|
|
text = "({}) {}".format(customer.id, text)
|
2018-07-15 18:13:30 -05:00
|
|
|
route = '{}customers.view'.format('mobile.' if self.mobile else '')
|
|
|
|
url = self.request.route_url(route, uuid=customer.uuid)
|
2018-07-15 16:22:03 -05:00
|
|
|
items.append(HTML.tag('li', c=[tags.link_to(text, url)]))
|
2017-12-04 18:49:52 -06:00
|
|
|
return HTML.tag('ul', c=items)
|
|
|
|
|
2020-03-18 13:15:11 -05:00
|
|
|
def render_members(self, person, field):
|
|
|
|
members = person.members
|
|
|
|
if not members:
|
|
|
|
return ""
|
|
|
|
items = []
|
|
|
|
for member in members:
|
|
|
|
text = six.text_type(member)
|
|
|
|
if member.number:
|
|
|
|
text = "(#{}) {}".format(member.number, text)
|
|
|
|
elif member.id:
|
|
|
|
text = "({}) {}".format(member.id, text)
|
|
|
|
url = self.request.route_url('members.view', uuid=member.uuid)
|
|
|
|
items.append(HTML.tag('li', c=[tags.link_to(text, url)]))
|
|
|
|
return HTML.tag('ul', c=items)
|
|
|
|
|
2017-12-04 18:49:52 -06:00
|
|
|
def render_users(self, person, field):
|
2019-05-22 10:34:03 -05:00
|
|
|
use_buefy = self.get_use_buefy()
|
2017-12-04 18:49:52 -06:00
|
|
|
users = person.users
|
|
|
|
items = []
|
|
|
|
for user in users:
|
|
|
|
text = user.username
|
|
|
|
url = self.request.route_url('users.view', uuid=user.uuid)
|
2018-02-12 19:22:05 -06:00
|
|
|
items.append(HTML.tag('li', c=[tags.link_to(text, url)]))
|
2017-12-04 18:49:52 -06:00
|
|
|
if items:
|
|
|
|
return HTML.tag('ul', c=items)
|
2019-06-18 16:54:05 -05:00
|
|
|
elif self.viewing and self.request.has_perm('users.create'):
|
2019-05-22 10:34:03 -05:00
|
|
|
if use_buefy:
|
2019-08-03 16:57:13 -05:00
|
|
|
return HTML.tag('b-button', type='is-primary', c="Make User",
|
|
|
|
**{'@click': 'clickMakeUser()'})
|
2019-05-22 10:34:03 -05:00
|
|
|
else:
|
|
|
|
return HTML.tag('button', type='button', id='make-user', c="Make User")
|
2017-12-04 18:49:52 -06:00
|
|
|
else:
|
|
|
|
return ""
|
2013-05-07 19:41:58 -05:00
|
|
|
|
2017-07-05 17:16:28 -05:00
|
|
|
def get_version_child_classes(self):
|
|
|
|
return [
|
|
|
|
(model.PersonPhoneNumber, 'parent_uuid'),
|
|
|
|
(model.PersonEmailAddress, 'parent_uuid'),
|
|
|
|
(model.PersonMailingAddress, 'parent_uuid'),
|
|
|
|
(model.Employee, 'person_uuid'),
|
|
|
|
(model.CustomerPerson, 'person_uuid'),
|
|
|
|
(model.VendorContact, 'person_uuid'),
|
|
|
|
]
|
|
|
|
|
2019-04-10 16:46:16 -05:00
|
|
|
def view_profile(self):
|
|
|
|
"""
|
|
|
|
View which exposes the "full profile" for a given person, i.e. all
|
|
|
|
related customer, employee, user info etc.
|
|
|
|
"""
|
|
|
|
self.viewing = True
|
|
|
|
person = self.get_instance()
|
2019-04-12 12:55:09 -05:00
|
|
|
employee = person.employee
|
2019-04-10 16:46:16 -05:00
|
|
|
context = {
|
|
|
|
'person': person,
|
|
|
|
'instance': person,
|
|
|
|
'instance_title': self.get_instance_title(person),
|
2019-08-11 17:30:08 -05:00
|
|
|
'today': localtime(self.rattail_config).date(),
|
2020-03-17 12:28:13 -05:00
|
|
|
'person_data': self.get_context_person(person),
|
|
|
|
'customers_data': self.get_context_customers(person),
|
2020-03-18 11:27:58 -05:00
|
|
|
'members_data': self.get_context_members(person),
|
2019-04-12 12:55:09 -05:00
|
|
|
'employee': employee,
|
2019-07-11 15:17:25 -05:00
|
|
|
'employee_view_url': self.request.route_url('employees.view', uuid=employee.uuid) if employee else None,
|
2019-04-12 12:55:09 -05:00
|
|
|
'employee_history': employee.get_current_history() if employee else None,
|
2019-07-11 14:01:22 -05:00
|
|
|
'employee_history_data': self.get_context_employee_history(employee),
|
2019-04-10 16:46:16 -05:00
|
|
|
}
|
2019-07-10 22:58:05 -05:00
|
|
|
|
2019-07-11 14:01:22 -05:00
|
|
|
use_buefy = self.get_use_buefy()
|
|
|
|
template = 'view_profile_buefy' if use_buefy else 'view_profile'
|
|
|
|
return self.render_to_response(template, context)
|
|
|
|
|
2020-03-17 12:28:13 -05:00
|
|
|
def get_context_person(self, person):
|
|
|
|
return {
|
|
|
|
'uuid': person.uuid,
|
|
|
|
'first_name': person.first_name,
|
|
|
|
'last_name': person.last_name,
|
|
|
|
'display_name': person.display_name,
|
|
|
|
'view_url': self.get_action_url('view', person),
|
|
|
|
'view_profile_url': self.get_action_url('view_profile', person),
|
|
|
|
}
|
|
|
|
|
2020-03-17 18:50:07 -05:00
|
|
|
def get_context_address(self, address):
|
|
|
|
return {
|
|
|
|
'uuid': address.uuid,
|
|
|
|
'street': address.street,
|
|
|
|
'street2': address.street2,
|
|
|
|
'city': address.city,
|
|
|
|
'state': address.state,
|
|
|
|
'zipcode': address.zipcode,
|
|
|
|
'display': six.text_type(address),
|
|
|
|
}
|
|
|
|
|
2020-03-17 12:28:13 -05:00
|
|
|
def get_context_customers(self, person):
|
|
|
|
data = []
|
|
|
|
for cp in person._customers:
|
|
|
|
customer = cp.customer
|
|
|
|
data.append({
|
|
|
|
'uuid': customer.uuid,
|
|
|
|
'ordinal': cp.ordinal,
|
|
|
|
'id': customer.id,
|
|
|
|
'number': customer.number,
|
|
|
|
'name': customer.name,
|
|
|
|
'view_url': self.request.route_url('customers.view',
|
|
|
|
uuid=customer.uuid),
|
|
|
|
'people': [self.get_context_person(p)
|
|
|
|
for p in customer.people],
|
2020-03-17 18:50:07 -05:00
|
|
|
'addresses': [self.get_context_address(a)
|
|
|
|
for a in customer.addresses],
|
2020-03-17 12:28:13 -05:00
|
|
|
})
|
|
|
|
return data
|
|
|
|
|
2020-03-18 11:27:58 -05:00
|
|
|
def get_context_members(self, person):
|
|
|
|
data = OrderedDict()
|
|
|
|
|
|
|
|
for member in person.members:
|
|
|
|
data[member.uuid] = self.get_context_member(member)
|
|
|
|
|
|
|
|
for customer in person.customers:
|
|
|
|
for member in customer.members:
|
|
|
|
if member.uuid not in data:
|
|
|
|
data[member.uuid] = self.get_context_member(member)
|
|
|
|
|
|
|
|
return list(data.values())
|
|
|
|
|
|
|
|
def get_context_member(self, member):
|
2020-03-18 13:15:11 -05:00
|
|
|
profile_url = None
|
|
|
|
if member.person:
|
|
|
|
profile_url = self.request.route_url('people.view_profile',
|
|
|
|
uuid=member.person_uuid)
|
|
|
|
|
2020-03-18 11:27:58 -05:00
|
|
|
return {
|
|
|
|
'uuid': member.uuid,
|
2020-03-18 12:29:18 -05:00
|
|
|
'number': member.number,
|
2020-03-18 11:27:58 -05:00
|
|
|
'id': member.id,
|
|
|
|
'active': member.active,
|
|
|
|
'joined': six.text_type(member.joined) if member.joined else None,
|
|
|
|
'withdrew': six.text_type(member.withdrew) if member.withdrew else None,
|
2020-03-18 13:15:11 -05:00
|
|
|
'customer_uuid': member.customer_uuid,
|
|
|
|
'customer_name': member.customer.name if member.customer else None,
|
|
|
|
'person_uuid': member.person_uuid,
|
2020-03-18 11:27:58 -05:00
|
|
|
'display': six.text_type(member),
|
2020-03-18 13:15:11 -05:00
|
|
|
'person_display_name': member.person.display_name if member.person else None,
|
2020-03-18 11:27:58 -05:00
|
|
|
'view_url': self.request.route_url('members.view', uuid=member.uuid),
|
2020-03-18 13:15:11 -05:00
|
|
|
'view_profile_url': profile_url,
|
2020-03-18 11:27:58 -05:00
|
|
|
}
|
|
|
|
|
2019-07-11 14:01:22 -05:00
|
|
|
def get_context_employee_history(self, employee):
|
|
|
|
data = []
|
2019-07-10 22:58:05 -05:00
|
|
|
if employee:
|
2020-02-28 13:10:25 -06:00
|
|
|
for history in employee.sorted_history(reverse=True):
|
2019-07-10 22:58:05 -05:00
|
|
|
data.append({
|
2019-07-23 13:12:36 -05:00
|
|
|
'uuid': history.uuid,
|
2019-07-10 22:58:05 -05:00
|
|
|
'start_date': six.text_type(history.start_date),
|
|
|
|
'end_date': six.text_type(history.end_date or ''),
|
|
|
|
})
|
2019-07-11 14:01:22 -05:00
|
|
|
return data
|
2019-04-10 16:46:16 -05:00
|
|
|
|
2019-06-06 13:49:59 -05:00
|
|
|
def make_note_form(self, mode, person):
|
|
|
|
schema = NoteSchema().bind(session=self.Session(),
|
|
|
|
person_uuid=person.uuid)
|
|
|
|
if mode == 'create':
|
|
|
|
del schema['uuid']
|
|
|
|
form = forms.Form(schema=schema, request=self.request)
|
|
|
|
return form
|
|
|
|
|
|
|
|
def profile_add_note(self):
|
|
|
|
person = self.get_instance()
|
|
|
|
form = self.make_note_form('create', person)
|
|
|
|
if form.validate(newstyle=True):
|
|
|
|
note = self.create_note(person, form)
|
|
|
|
self.Session.flush()
|
|
|
|
return self.profile_add_note_success(note)
|
|
|
|
else:
|
|
|
|
return self.profile_add_note_failure(person, form)
|
2019-06-04 19:23:27 -05:00
|
|
|
|
2019-06-06 13:49:59 -05:00
|
|
|
def create_note(self, person, form):
|
2019-06-04 19:23:27 -05:00
|
|
|
note = model.PersonNote()
|
|
|
|
note.type = form.validated['note_type']
|
|
|
|
note.subject = form.validated['note_subject']
|
|
|
|
note.text = form.validated['note_text']
|
|
|
|
note.created_by = self.request.user
|
|
|
|
person.notes.append(note)
|
|
|
|
return note
|
|
|
|
|
2019-06-06 13:49:59 -05:00
|
|
|
def profile_add_note_success(self, note):
|
|
|
|
return self.redirect(self.get_action_url('view_profile', person))
|
|
|
|
|
|
|
|
def profile_add_note_failure(self, person, form):
|
|
|
|
return self.redirect(self.get_action_url('view_profile', person))
|
|
|
|
|
|
|
|
def profile_edit_note(self):
|
2019-06-04 19:23:27 -05:00
|
|
|
person = self.get_instance()
|
2019-06-06 13:49:59 -05:00
|
|
|
form = self.make_note_form('edit', person)
|
2019-06-04 19:23:27 -05:00
|
|
|
if form.validate(newstyle=True):
|
2019-06-06 13:49:59 -05:00
|
|
|
note = self.update_note(person, form)
|
2019-06-04 19:23:27 -05:00
|
|
|
self.Session.flush()
|
2019-06-06 13:49:59 -05:00
|
|
|
return self.profile_edit_note_success(note)
|
2019-06-04 19:23:27 -05:00
|
|
|
else:
|
2019-06-06 13:49:59 -05:00
|
|
|
return self.profile_edit_note_failure(person, form)
|
2019-06-04 19:23:27 -05:00
|
|
|
|
2019-06-06 13:49:59 -05:00
|
|
|
def update_note(self, person, form):
|
|
|
|
note = self.Session.query(model.PersonNote).get(form.validated['uuid'])
|
|
|
|
note.subject = form.validated['note_subject']
|
|
|
|
note.text = form.validated['note_text']
|
|
|
|
return note
|
|
|
|
|
|
|
|
def profile_edit_note_success(self, note):
|
2019-06-04 19:23:27 -05:00
|
|
|
return self.redirect(self.get_action_url('view_profile', person))
|
|
|
|
|
2019-06-06 13:49:59 -05:00
|
|
|
def profile_edit_note_failure(self, person, form):
|
|
|
|
return self.redirect(self.get_action_url('view_profile', person))
|
|
|
|
|
|
|
|
def profile_delete_note(self):
|
|
|
|
person = self.get_instance()
|
|
|
|
form = self.make_note_form('delete', person)
|
|
|
|
if form.validate(newstyle=True):
|
|
|
|
self.delete_note(person, form)
|
|
|
|
self.Session.flush()
|
|
|
|
return self.profile_delete_note_success(person)
|
|
|
|
else:
|
|
|
|
return self.profile_delete_note_failure(person, form)
|
|
|
|
|
|
|
|
def delete_note(self, person, form):
|
|
|
|
note = self.Session.query(model.PersonNote).get(form.validated['uuid'])
|
|
|
|
self.Session.delete(note)
|
|
|
|
|
|
|
|
def profile_delete_note_success(self, person):
|
|
|
|
return self.redirect(self.get_action_url('view_profile', person))
|
|
|
|
|
|
|
|
def profile_delete_note_failure(self, person, form):
|
2019-06-04 19:23:27 -05:00
|
|
|
return self.redirect(self.get_action_url('view_profile', person))
|
|
|
|
|
2017-10-29 03:18:05 -05:00
|
|
|
def make_user(self):
|
|
|
|
uuid = self.request.POST['person_uuid']
|
|
|
|
person = self.Session.query(model.Person).get(uuid)
|
|
|
|
if not person:
|
|
|
|
return self.notfound()
|
|
|
|
if person.users:
|
|
|
|
raise RuntimeError("person {} already has {} user accounts: ".format(
|
|
|
|
person.uuid, len(person.users), person))
|
|
|
|
user = model.User()
|
|
|
|
user.username = api.make_username(person)
|
|
|
|
user.person = person
|
|
|
|
user.active = False
|
|
|
|
self.Session.add(user)
|
|
|
|
self.Session.flush()
|
|
|
|
self.request.session.flash("User has been created: {}".format(user.username))
|
|
|
|
return self.redirect(self.request.route_url('users.view', uuid=user.uuid))
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def defaults(cls, config):
|
|
|
|
cls._people_defaults(config)
|
|
|
|
cls._defaults(config)
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def _people_defaults(cls, config):
|
2019-04-10 16:46:16 -05:00
|
|
|
permission_prefix = cls.get_permission_prefix()
|
2017-10-29 03:18:05 -05:00
|
|
|
route_prefix = cls.get_route_prefix()
|
|
|
|
url_prefix = cls.get_url_prefix()
|
2019-04-10 16:46:16 -05:00
|
|
|
model_key = cls.get_model_key()
|
|
|
|
model_title = cls.get_model_title()
|
|
|
|
|
2019-07-11 14:01:00 -05:00
|
|
|
# "profile" perms
|
|
|
|
# TODO: should let view class (or config) determine which of these are available
|
|
|
|
config.add_tailbone_permission_group('people_profile', "People Profile View")
|
|
|
|
config.add_tailbone_permission('people_profile', 'people_profile.toggle_employee',
|
|
|
|
"Toggle the person's Employee status")
|
2019-07-23 13:12:36 -05:00
|
|
|
config.add_tailbone_permission('people_profile', 'people_profile.edit_employee_history',
|
|
|
|
"Edit the person's Employee History records")
|
2019-07-11 14:01:00 -05:00
|
|
|
|
2019-04-10 16:46:16 -05:00
|
|
|
# view profile
|
|
|
|
config.add_tailbone_permission(permission_prefix, '{}.view_profile'.format(permission_prefix),
|
|
|
|
"View full \"profile\" for {}".format(model_title))
|
|
|
|
config.add_route('{}.view_profile'.format(route_prefix), '{}/{{{}}}/profile'.format(url_prefix, model_key),
|
|
|
|
request_method='GET')
|
|
|
|
config.add_view(cls, attr='view_profile', route_name='{}.view_profile'.format(route_prefix),
|
|
|
|
permission='{}.view_profile'.format(permission_prefix))
|
2017-10-29 03:18:05 -05:00
|
|
|
|
2019-06-04 19:23:27 -05:00
|
|
|
# manage notes from profile view
|
|
|
|
if cls.manage_notes_from_profile_view:
|
|
|
|
|
|
|
|
# add note
|
2019-10-15 16:12:56 -05:00
|
|
|
config.add_tailbone_permission('people_profile', 'people_profile.add_note',
|
|
|
|
"Add new {} Note records".format(model_title))
|
2019-06-06 13:49:59 -05:00
|
|
|
config.add_route('{}.profile_add_note'.format(route_prefix), '{}/{{{}}}/profile/new-note'.format(url_prefix, model_key),
|
2019-06-04 19:23:27 -05:00
|
|
|
request_method='POST')
|
|
|
|
config.add_view(cls, attr='profile_add_note', route_name='{}.profile_add_note'.format(route_prefix),
|
2019-10-15 16:12:56 -05:00
|
|
|
permission='people_profile.add_note')
|
2019-06-04 19:23:27 -05:00
|
|
|
|
2019-06-06 13:49:59 -05:00
|
|
|
# edit note
|
2019-10-15 16:12:56 -05:00
|
|
|
config.add_tailbone_permission('people_profile', 'people_profile.edit_note',
|
|
|
|
"Edit {} Note records".format(model_title))
|
2019-06-06 13:49:59 -05:00
|
|
|
config.add_route('{}.profile_edit_note'.format(route_prefix), '{}/{{{}}}/profile/edit-note'.format(url_prefix, model_key),
|
|
|
|
request_method='POST')
|
|
|
|
config.add_view(cls, attr='profile_edit_note', route_name='{}.profile_edit_note'.format(route_prefix),
|
2019-10-15 16:12:56 -05:00
|
|
|
permission='people_profile.edit_note')
|
2019-06-06 13:49:59 -05:00
|
|
|
|
|
|
|
# delete note
|
2019-10-15 16:12:56 -05:00
|
|
|
config.add_tailbone_permission('people_profile', 'people_profile.delete_note',
|
|
|
|
"Delete {} Note records".format(model_title))
|
2019-06-06 13:49:59 -05:00
|
|
|
config.add_route('{}.profile_delete_note'.format(route_prefix), '{}/{{{}}}/profile/delete-note'.format(url_prefix, model_key),
|
|
|
|
request_method='POST')
|
|
|
|
config.add_view(cls, attr='profile_delete_note', route_name='{}.profile_delete_note'.format(route_prefix),
|
2019-10-15 16:12:56 -05:00
|
|
|
permission='people_profile.delete_note')
|
2019-06-04 19:23:27 -05:00
|
|
|
|
2017-10-29 03:18:05 -05:00
|
|
|
# make user for person
|
|
|
|
config.add_route('{}.make_user'.format(route_prefix), '{}/make-user'.format(url_prefix),
|
|
|
|
request_method='POST')
|
|
|
|
config.add_view(cls, attr='make_user', route_name='{}.make_user'.format(route_prefix),
|
|
|
|
permission='users.create')
|
|
|
|
|
2013-05-07 19:41:58 -05:00
|
|
|
|
|
|
|
class PeopleAutocomplete(AutocompleteView):
|
|
|
|
|
2015-12-06 18:04:23 -06:00
|
|
|
mapped_class = model.Person
|
2016-12-08 12:13:45 -06:00
|
|
|
fieldname = 'display_name'
|
2013-05-07 19:41:58 -05:00
|
|
|
|
|
|
|
|
2014-07-19 20:49:00 -05:00
|
|
|
class PeopleEmployeesAutocomplete(PeopleAutocomplete):
|
|
|
|
"""
|
|
|
|
Autocomplete view for the Person model, but restricted to return only
|
|
|
|
results for people who are employees.
|
|
|
|
"""
|
|
|
|
|
|
|
|
def filter_query(self, q):
|
|
|
|
return q.join(model.Employee)
|
|
|
|
|
|
|
|
|
2019-05-14 15:59:57 -05:00
|
|
|
class PersonNoteView(MasterView):
|
|
|
|
"""
|
|
|
|
Master view for the PersonNote class.
|
|
|
|
"""
|
|
|
|
model_class = model.PersonNote
|
|
|
|
route_prefix = 'person_notes'
|
|
|
|
url_prefix = '/people/notes'
|
|
|
|
has_versions = True
|
|
|
|
|
|
|
|
grid_columns = [
|
|
|
|
'person',
|
|
|
|
'type',
|
|
|
|
'subject',
|
|
|
|
'created',
|
|
|
|
'created_by',
|
|
|
|
]
|
|
|
|
|
|
|
|
form_fields = [
|
|
|
|
'person',
|
|
|
|
'type',
|
|
|
|
'subject',
|
|
|
|
'text',
|
|
|
|
'created',
|
|
|
|
'created_by',
|
|
|
|
]
|
|
|
|
|
|
|
|
def get_instance_title(self, note):
|
|
|
|
return note.subject or "(no subject)"
|
|
|
|
|
|
|
|
def configure_grid(self, g):
|
|
|
|
super(PersonNoteView, self).configure_grid(g)
|
|
|
|
|
|
|
|
# person
|
|
|
|
g.set_joiner('person', lambda q: q.join(model.Person,
|
|
|
|
model.Person.uuid == model.PersonNote.parent_uuid))
|
|
|
|
g.set_sorter('person', model.Person.display_name)
|
|
|
|
g.set_filter('person', model.Person.display_name, label="Person Name")
|
|
|
|
|
|
|
|
# created_by
|
|
|
|
CreatorPerson = orm.aliased(model.Person)
|
|
|
|
g.set_joiner('created_by', lambda q: q.join(model.User).outerjoin(CreatorPerson,
|
|
|
|
CreatorPerson.uuid == model.User.person_uuid))
|
|
|
|
g.set_sorter('created_by', CreatorPerson.display_name)
|
|
|
|
|
2019-05-15 15:34:32 -05:00
|
|
|
g.set_sort_defaults('created', 'desc')
|
|
|
|
|
2019-05-14 15:59:57 -05:00
|
|
|
g.set_link('person')
|
|
|
|
g.set_link('subject')
|
|
|
|
g.set_link('created')
|
|
|
|
|
|
|
|
def configure_form(self, f):
|
|
|
|
super(PersonNoteView, self).configure_form(f)
|
|
|
|
|
2019-06-14 20:47:16 -05:00
|
|
|
# person
|
|
|
|
f.set_readonly('person')
|
2019-05-14 15:59:57 -05:00
|
|
|
f.set_renderer('person', self.render_person)
|
2019-06-14 20:47:16 -05:00
|
|
|
|
|
|
|
# created
|
|
|
|
f.set_readonly('created')
|
|
|
|
|
|
|
|
# created_by
|
|
|
|
f.set_readonly('created_by')
|
2019-06-05 16:04:14 -05:00
|
|
|
f.set_renderer('created_by', self.render_user)
|
2019-05-14 15:59:57 -05:00
|
|
|
|
|
|
|
|
2019-06-04 19:23:27 -05:00
|
|
|
@colander.deferred
|
2019-06-06 13:49:59 -05:00
|
|
|
def valid_note_uuid(node, kw):
|
|
|
|
session = kw['session']
|
|
|
|
person_uuid = kw['person_uuid']
|
2019-06-04 19:23:27 -05:00
|
|
|
def validate(node, value):
|
2019-06-06 13:49:59 -05:00
|
|
|
note = session.query(model.PersonNote).get(value)
|
|
|
|
if not note:
|
|
|
|
raise colander.Invalid(node, "Note not found")
|
|
|
|
if note.person.uuid != person_uuid:
|
|
|
|
raise colander.Invalid(node, "Note is for the wrong person")
|
|
|
|
return note.uuid
|
2019-06-04 19:23:27 -05:00
|
|
|
return validate
|
|
|
|
|
|
|
|
|
2019-06-06 13:49:59 -05:00
|
|
|
class NoteSchema(colander.Schema):
|
|
|
|
|
|
|
|
uuid = colander.SchemaNode(colander.String(),
|
|
|
|
validator=valid_note_uuid)
|
2019-06-04 19:23:27 -05:00
|
|
|
|
2019-06-06 13:49:59 -05:00
|
|
|
note_type = colander.SchemaNode(colander.String())
|
2019-06-04 19:23:27 -05:00
|
|
|
|
|
|
|
note_subject = colander.SchemaNode(colander.String(), missing='')
|
|
|
|
|
|
|
|
note_text = colander.SchemaNode(colander.String(), missing='')
|
|
|
|
|
|
|
|
|
2013-05-07 19:41:58 -05:00
|
|
|
def includeme(config):
|
2015-12-06 18:04:23 -06:00
|
|
|
|
|
|
|
# autocomplete
|
|
|
|
config.add_route('people.autocomplete', '/people/autocomplete')
|
2013-05-07 19:41:58 -05:00
|
|
|
config.add_view(PeopleAutocomplete, route_name='people.autocomplete',
|
2015-12-06 18:04:23 -06:00
|
|
|
renderer='json', permission='people.list')
|
|
|
|
config.add_route('people.autocomplete.employees', '/people/autocomplete/employees')
|
|
|
|
config.add_view(PeopleEmployeesAutocomplete, route_name='people.autocomplete.employees',
|
|
|
|
renderer='json', permission='people.list')
|
2017-03-24 17:29:34 -05:00
|
|
|
|
|
|
|
PeopleView.defaults(config)
|
2019-05-14 15:59:57 -05:00
|
|
|
PersonNoteView.defaults(config)
|