Statistics
| Branch: | Tag: | Revision:

root / snf-astakos-app / astakos / im / tables.py @ ff67242a

History | View | Annotate | Download (13.1 kB)

1 91cb6fb6 Kostas Papadimitriou
# Copyright 2011-2012 GRNET S.A. All rights reserved.
2 91cb6fb6 Kostas Papadimitriou
#
3 91cb6fb6 Kostas Papadimitriou
# Redistribution and use in source and binary forms, with or
4 91cb6fb6 Kostas Papadimitriou
# without modification, are permitted provided that the following
5 91cb6fb6 Kostas Papadimitriou
# conditions are met:
6 91cb6fb6 Kostas Papadimitriou
#
7 91cb6fb6 Kostas Papadimitriou
#   1. Redistributions of source code must retain the above
8 91cb6fb6 Kostas Papadimitriou
#      copyright notice, this list of conditions and the following
9 91cb6fb6 Kostas Papadimitriou
#      disclaimer.
10 91cb6fb6 Kostas Papadimitriou
#
11 91cb6fb6 Kostas Papadimitriou
#   2. Redistributions in binary form must reproduce the above
12 91cb6fb6 Kostas Papadimitriou
#      copyright notice, this list of conditions and the following
13 91cb6fb6 Kostas Papadimitriou
#      disclaimer in the documentation and/or other materials
14 91cb6fb6 Kostas Papadimitriou
#      provided with the distribution.
15 91cb6fb6 Kostas Papadimitriou
#
16 91cb6fb6 Kostas Papadimitriou
# THIS SOFTWARE IS PROVIDED BY GRNET S.A. ``AS IS'' AND ANY EXPRESS
17 91cb6fb6 Kostas Papadimitriou
# OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
18 91cb6fb6 Kostas Papadimitriou
# WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
19 91cb6fb6 Kostas Papadimitriou
# PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL GRNET S.A OR
20 91cb6fb6 Kostas Papadimitriou
# CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
21 91cb6fb6 Kostas Papadimitriou
# SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
22 91cb6fb6 Kostas Papadimitriou
# LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF
23 91cb6fb6 Kostas Papadimitriou
# USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
24 91cb6fb6 Kostas Papadimitriou
# AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
25 91cb6fb6 Kostas Papadimitriou
# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
26 91cb6fb6 Kostas Papadimitriou
# ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
27 91cb6fb6 Kostas Papadimitriou
# POSSIBILITY OF SUCH DAMAGE.
28 91cb6fb6 Kostas Papadimitriou
#
29 91cb6fb6 Kostas Papadimitriou
# The views and conclusions contained in the software and
30 91cb6fb6 Kostas Papadimitriou
# documentation are those of the authors and should not be
31 91cb6fb6 Kostas Papadimitriou
# interpreted as representing official policies, either expressed
32 91cb6fb6 Kostas Papadimitriou
# or implied, of GRNET S.A.
33 5550bcfb Kostas Papadimitriou
34 f3a45fc6 Kostas Papadimitriou
from collections import defaultdict
35 f3a45fc6 Kostas Papadimitriou
36 5550bcfb Kostas Papadimitriou
from django.utils.translation import ugettext as _
37 91cb6fb6 Kostas Papadimitriou
from django.utils.safestring import mark_safe
38 91cb6fb6 Kostas Papadimitriou
from django.template import Context, Template
39 91cb6fb6 Kostas Papadimitriou
from django.template.loader import render_to_string
40 9b32e2fb Kostas Papadimitriou
from django.core.exceptions import PermissionDenied
41 91cb6fb6 Kostas Papadimitriou
42 5550bcfb Kostas Papadimitriou
from django_tables2 import A
43 91cb6fb6 Kostas Papadimitriou
import django_tables2 as tables
44 91cb6fb6 Kostas Papadimitriou
45 5550bcfb Kostas Papadimitriou
from astakos.im.models import *
46 6795eb09 Kostas Papadimitriou
from astakos.im.templatetags.filters import truncatename
47 ff67242a Giorgos Korfiatis
from astakos.im.functions import join_project_checks, \
48 ff67242a Giorgos Korfiatis
                                 leave_project_checks
49 5550bcfb Kostas Papadimitriou
50 5550bcfb Kostas Papadimitriou
DEFAULT_DATE_FORMAT = "d/m/Y"
51 5550bcfb Kostas Papadimitriou
52 5550bcfb Kostas Papadimitriou
53 9b32e2fb Kostas Papadimitriou
class LinkColumn(tables.LinkColumn):
54 6795eb09 Kostas Papadimitriou
55 6795eb09 Kostas Papadimitriou
    def __init__(self, *args, **kwargs):
56 9b32e2fb Kostas Papadimitriou
        self.coerce = kwargs.pop('coerce', None)
57 9b32e2fb Kostas Papadimitriou
        self.append = kwargs.pop('append', None)
58 9b32e2fb Kostas Papadimitriou
        super(LinkColumn, self).__init__(*args, **kwargs)
59 9b32e2fb Kostas Papadimitriou
60 9b32e2fb Kostas Papadimitriou
    def render(self, value, record, bound_column):
61 9b32e2fb Kostas Papadimitriou
        link = super(LinkColumn, self).render(value, record, bound_column)
62 9b32e2fb Kostas Papadimitriou
        extra = ''
63 9b32e2fb Kostas Papadimitriou
        if self.append:
64 9b32e2fb Kostas Papadimitriou
            if callable(self.append):
65 9b32e2fb Kostas Papadimitriou
                extra = self.append(record, bound_column)
66 9b32e2fb Kostas Papadimitriou
            else:
67 9b32e2fb Kostas Papadimitriou
                extra = self.append
68 9b32e2fb Kostas Papadimitriou
        return mark_safe(link + extra)
69 6795eb09 Kostas Papadimitriou
70 6795eb09 Kostas Papadimitriou
    def render_link(self, uri, text, attrs=None):
71 9b32e2fb Kostas Papadimitriou
        if self.coerce:
72 9b32e2fb Kostas Papadimitriou
            text = self.coerce(text)
73 9b32e2fb Kostas Papadimitriou
        return super(LinkColumn, self).render_link(uri, text, attrs)
74 6795eb09 Kostas Papadimitriou
75 6795eb09 Kostas Papadimitriou
76 91cb6fb6 Kostas Papadimitriou
# Helper columns
77 91cb6fb6 Kostas Papadimitriou
class RichLinkColumn(tables.TemplateColumn):
78 91cb6fb6 Kostas Papadimitriou
79 91cb6fb6 Kostas Papadimitriou
    method = 'POST'
80 91cb6fb6 Kostas Papadimitriou
81 91cb6fb6 Kostas Papadimitriou
    confirm_prompt = _('Yes')
82 91cb6fb6 Kostas Papadimitriou
    cancel_prompt = _('No')
83 91cb6fb6 Kostas Papadimitriou
    confirm = True
84 91cb6fb6 Kostas Papadimitriou
85 91cb6fb6 Kostas Papadimitriou
    prompt = _('Confirm action ?')
86 91cb6fb6 Kostas Papadimitriou
87 91cb6fb6 Kostas Papadimitriou
    action_tpl = None
88 91cb6fb6 Kostas Papadimitriou
    action = _('Action')
89 91cb6fb6 Kostas Papadimitriou
    extra_context = lambda record, table, column: {}
90 91cb6fb6 Kostas Papadimitriou
91 91cb6fb6 Kostas Papadimitriou
    url = None
92 91cb6fb6 Kostas Papadimitriou
    url_args = ()
93 91cb6fb6 Kostas Papadimitriou
    resolve_func = None
94 91cb6fb6 Kostas Papadimitriou
95 91cb6fb6 Kostas Papadimitriou
    def __init__(self, *args, **kwargs):
96 91cb6fb6 Kostas Papadimitriou
        kwargs['template_name'] = kwargs.get('template_name',
97 91cb6fb6 Kostas Papadimitriou
                                             'im/table_rich_link_column.html')
98 91cb6fb6 Kostas Papadimitriou
        for attr in ['method', 'confirm_prompt',
99 91cb6fb6 Kostas Papadimitriou
                     'cancel_prompt', 'prompt', 'url',
100 91cb6fb6 Kostas Papadimitriou
                     'url_args', 'action', 'confirm',
101 91cb6fb6 Kostas Papadimitriou
                     'resolve_func', 'extra_context']:
102 91cb6fb6 Kostas Papadimitriou
            setattr(self, attr, kwargs.pop(attr, getattr(self, attr)))
103 91cb6fb6 Kostas Papadimitriou
104 91cb6fb6 Kostas Papadimitriou
        super(RichLinkColumn, self).__init__(*args, **kwargs)
105 91cb6fb6 Kostas Papadimitriou
106 91cb6fb6 Kostas Papadimitriou
    def render(self, record, table, value, bound_column, **kwargs):
107 91cb6fb6 Kostas Papadimitriou
        # If the table is being rendered using `render_table`, it hackily
108 91cb6fb6 Kostas Papadimitriou
        # attaches the context to the table as a gift to `TemplateColumn`. If
109 91cb6fb6 Kostas Papadimitriou
        # the table is being rendered via `Table.as_html`, this won't exist.
110 3f0d6293 Kostas Papadimitriou
        content = ''
111 3f0d6293 Kostas Papadimitriou
        for extra_context in self.get_template_context(record, table, value,
112 3f0d6293 Kostas Papadimitriou
                                                       bound_column, **kwargs):
113 3f0d6293 Kostas Papadimitriou
            context = getattr(table, 'context', Context())
114 3f0d6293 Kostas Papadimitriou
            context.update(extra_context)
115 3f0d6293 Kostas Papadimitriou
            try:
116 3f0d6293 Kostas Papadimitriou
                if self.template_code:
117 3f0d6293 Kostas Papadimitriou
                    content += Template(self.template_code).render(context)
118 3f0d6293 Kostas Papadimitriou
                else:
119 3f0d6293 Kostas Papadimitriou
                    content += render_to_string(self.template_name, context)
120 3f0d6293 Kostas Papadimitriou
            finally:
121 3f0d6293 Kostas Papadimitriou
                context.pop()
122 3f0d6293 Kostas Papadimitriou
123 3f0d6293 Kostas Papadimitriou
        return mark_safe(content)
124 91cb6fb6 Kostas Papadimitriou
125 91cb6fb6 Kostas Papadimitriou
    def get_confirm(self, record, table):
126 91cb6fb6 Kostas Papadimitriou
        if callable(self.confirm):
127 91cb6fb6 Kostas Papadimitriou
            return self.confirm(record, table)
128 91cb6fb6 Kostas Papadimitriou
        return self.confirm
129 91cb6fb6 Kostas Papadimitriou
130 91cb6fb6 Kostas Papadimitriou
    def resolved_url(self, record, table):
131 91cb6fb6 Kostas Papadimitriou
        if callable(self.url):
132 91cb6fb6 Kostas Papadimitriou
            return self.url(record, table)
133 91cb6fb6 Kostas Papadimitriou
134 91cb6fb6 Kostas Papadimitriou
        if not self.url:
135 91cb6fb6 Kostas Papadimitriou
            return '#'
136 91cb6fb6 Kostas Papadimitriou
137 91cb6fb6 Kostas Papadimitriou
        args = list(self.url_args)
138 91cb6fb6 Kostas Papadimitriou
        for index, arg in enumerate(args):
139 91cb6fb6 Kostas Papadimitriou
            if isinstance(arg, A):
140 91cb6fb6 Kostas Papadimitriou
                args[index] = arg.resolve(record)
141 91cb6fb6 Kostas Papadimitriou
        return reverse(self.url, args=args)
142 91cb6fb6 Kostas Papadimitriou
143 91cb6fb6 Kostas Papadimitriou
    def get_action(self, record, table):
144 91cb6fb6 Kostas Papadimitriou
        if callable(self.action):
145 91cb6fb6 Kostas Papadimitriou
            return self.action(record, table)
146 91cb6fb6 Kostas Papadimitriou
        return self.action
147 91cb6fb6 Kostas Papadimitriou
148 91cb6fb6 Kostas Papadimitriou
    def get_prompt(self, record, table):
149 91cb6fb6 Kostas Papadimitriou
        if callable(self.prompt):
150 91cb6fb6 Kostas Papadimitriou
            return self.prompt(record, table)
151 91cb6fb6 Kostas Papadimitriou
        return self.prompt
152 91cb6fb6 Kostas Papadimitriou
153 91cb6fb6 Kostas Papadimitriou
    def get_template_context(self, record, table, value, bound_column, **kwargs):
154 91cb6fb6 Kostas Papadimitriou
        context = {'default': bound_column.default,
155 91cb6fb6 Kostas Papadimitriou
                   'record': record,
156 91cb6fb6 Kostas Papadimitriou
                   'value': value,
157 91cb6fb6 Kostas Papadimitriou
                   'col': self,
158 91cb6fb6 Kostas Papadimitriou
                   'url': self.resolved_url(record, table),
159 91cb6fb6 Kostas Papadimitriou
                   'prompt': self.get_prompt(record, table),
160 91cb6fb6 Kostas Papadimitriou
                   'action': self.get_action(record, table),
161 91cb6fb6 Kostas Papadimitriou
                   'confirm': self.get_confirm(record, table)
162 91cb6fb6 Kostas Papadimitriou
                  }
163 91cb6fb6 Kostas Papadimitriou
164 f3a45fc6 Kostas Papadimitriou
        # decide whether to return dict or a list of dicts in case we want to
165 f3a45fc6 Kostas Papadimitriou
        # display multiple actions within a cell.
166 91cb6fb6 Kostas Papadimitriou
        if self.extra_context:
167 3f0d6293 Kostas Papadimitriou
            contexts = []
168 3f0d6293 Kostas Papadimitriou
            extra_contexts = self.extra_context(record, table, self)
169 3f0d6293 Kostas Papadimitriou
            if isinstance(extra_contexts, list):
170 3f0d6293 Kostas Papadimitriou
                for extra_context in extra_contexts:
171 3f0d6293 Kostas Papadimitriou
                    newcontext = dict(context)
172 3f0d6293 Kostas Papadimitriou
                    newcontext.update(extra_context)
173 3f0d6293 Kostas Papadimitriou
                    contexts.append(newcontext)
174 3f0d6293 Kostas Papadimitriou
            else:
175 3f0d6293 Kostas Papadimitriou
                context.update(extra_contexts)
176 f3a45fc6 Kostas Papadimitriou
                contexts = [context]
177 3f0d6293 Kostas Papadimitriou
        else:
178 3f0d6293 Kostas Papadimitriou
            contexts = [context]
179 91cb6fb6 Kostas Papadimitriou
180 3f0d6293 Kostas Papadimitriou
        return contexts
181 91cb6fb6 Kostas Papadimitriou
182 91cb6fb6 Kostas Papadimitriou
183 db9a498c Kostas Papadimitriou
def action_extra_context(application, table, self):
184 91cb6fb6 Kostas Papadimitriou
    user = table.user
185 91cb6fb6 Kostas Papadimitriou
    url, action, confirm, prompt = '', '', True, ''
186 91cb6fb6 Kostas Papadimitriou
    append_url = ''
187 91cb6fb6 Kostas Papadimitriou
188 05617ab9 Kostas Papadimitriou
    can_join = can_leave = False
189 05617ab9 Kostas Papadimitriou
    project = application.get_project()
190 05617ab9 Kostas Papadimitriou
191 05617ab9 Kostas Papadimitriou
    if project:
192 05617ab9 Kostas Papadimitriou
        try:
193 ff67242a Giorgos Korfiatis
            join_project_checks(project)
194 05617ab9 Kostas Papadimitriou
            can_join = True
195 bd9af366 Kostas Papadimitriou
        except PermissionDenied, e:
196 05617ab9 Kostas Papadimitriou
            pass
197 05617ab9 Kostas Papadimitriou
198 05617ab9 Kostas Papadimitriou
        try:
199 ff67242a Giorgos Korfiatis
            leave_project_checks(project)
200 05617ab9 Kostas Papadimitriou
            can_leave = True
201 05617ab9 Kostas Papadimitriou
        except PermissionDenied:
202 05617ab9 Kostas Papadimitriou
            pass
203 05617ab9 Kostas Papadimitriou
204 f5991951 Kostas Papadimitriou
    if can_leave and user.is_project_member(application):
205 05617ab9 Kostas Papadimitriou
        url = 'astakos.im.views.project_leave'
206 05617ab9 Kostas Papadimitriou
        action = _('Leave')
207 05617ab9 Kostas Papadimitriou
        confirm = True
208 05617ab9 Kostas Papadimitriou
        prompt = _('Are you sure you want to leave from the project ?')
209 bd9af366 Kostas Papadimitriou
    elif can_join and not user.is_project_member(application):
210 05617ab9 Kostas Papadimitriou
        url = 'astakos.im.views.project_join'
211 05617ab9 Kostas Papadimitriou
        action = _('Join')
212 05617ab9 Kostas Papadimitriou
        confirm = True
213 05617ab9 Kostas Papadimitriou
        prompt = _('Are you sure you want to join this project ?')
214 05617ab9 Kostas Papadimitriou
    else:
215 05617ab9 Kostas Papadimitriou
        action = ''
216 05617ab9 Kostas Papadimitriou
        confirm = False
217 05617ab9 Kostas Papadimitriou
        url = None
218 db9a498c Kostas Papadimitriou
219 db9a498c Kostas Papadimitriou
    url = reverse(url, args=(application.pk, )) + append_url if url else ''
220 bd9af366 Kostas Papadimitriou
221 6401cff2 Sofia Papagiannaki
    return {'action': action,
222 6401cff2 Sofia Papagiannaki
            'confirm': confirm,
223 6401cff2 Sofia Papagiannaki
            'url': url,
224 91cb6fb6 Kostas Papadimitriou
            'prompt': prompt}
225 91cb6fb6 Kostas Papadimitriou
226 91cb6fb6 Kostas Papadimitriou
227 3f0d6293 Kostas Papadimitriou
class UserTable(tables.Table):
228 5550bcfb Kostas Papadimitriou
229 5550bcfb Kostas Papadimitriou
    def __init__(self, *args, **kwargs):
230 5550bcfb Kostas Papadimitriou
        self.user = None
231 5550bcfb Kostas Papadimitriou
232 5550bcfb Kostas Papadimitriou
        if 'request' in kwargs and kwargs.get('request').user:
233 5550bcfb Kostas Papadimitriou
            self.user = kwargs.get('request').user
234 5550bcfb Kostas Papadimitriou
235 5550bcfb Kostas Papadimitriou
        if 'user' in kwargs:
236 5550bcfb Kostas Papadimitriou
            self.user = kwargs.pop('user')
237 5550bcfb Kostas Papadimitriou
238 3f0d6293 Kostas Papadimitriou
        super(UserTable, self).__init__(*args, **kwargs)
239 3f0d6293 Kostas Papadimitriou
240 3f0d6293 Kostas Papadimitriou
241 05617ab9 Kostas Papadimitriou
def project_name_append(application, column):
242 05617ab9 Kostas Papadimitriou
    if application.has_pending_modifications():
243 05617ab9 Kostas Papadimitriou
        return mark_safe("<br /><i class='tiny'>%s</i>" % \
244 05617ab9 Kostas Papadimitriou
                                                _('modifications pending'))
245 05617ab9 Kostas Papadimitriou
    return u''
246 9b32e2fb Kostas Papadimitriou
247 3f0d6293 Kostas Papadimitriou
# Table classes
248 3f0d6293 Kostas Papadimitriou
class UserProjectApplicationsTable(UserTable):
249 3f0d6293 Kostas Papadimitriou
    caption = _('My projects')
250 5550bcfb Kostas Papadimitriou
251 9b32e2fb Kostas Papadimitriou
    name = LinkColumn('astakos.im.views.project_detail',
252 9b32e2fb Kostas Papadimitriou
                      coerce=lambda x: truncatename(x, 25),
253 9b32e2fb Kostas Papadimitriou
                      append=project_name_append,
254 9b32e2fb Kostas Papadimitriou
                      args=(A('pk'),))
255 f31db8a5 Georgios D. Tsoukalas
    issue_date = tables.DateColumn(verbose_name=_('Application'), format=DEFAULT_DATE_FORMAT)
256 5550bcfb Kostas Papadimitriou
    start_date = tables.DateColumn(format=DEFAULT_DATE_FORMAT)
257 f31db8a5 Georgios D. Tsoukalas
    end_date = tables.DateColumn(verbose_name=_('Expiration'), format=DEFAULT_DATE_FORMAT)
258 9b32e2fb Kostas Papadimitriou
    members_count = tables.Column(verbose_name=_("Members"), default=0,
259 0cf289c2 Sofia Papagiannaki
                                  orderable=False)
260 b7c802a2 Olga Brani
    membership_status = tables.Column(verbose_name=_("Status"), empty_values=(),
261 0cf289c2 Sofia Papagiannaki
                                      orderable=False)
262 91cb6fb6 Kostas Papadimitriou
    project_action = RichLinkColumn(verbose_name=_('Action'),
263 91cb6fb6 Kostas Papadimitriou
                                    extra_context=action_extra_context,
264 0cf289c2 Sofia Papagiannaki
                                    orderable=False)
265 5550bcfb Kostas Papadimitriou
266 2743e261 Kostas Papadimitriou
267 e6d284ef Olga Brani
    def render_membership_status(self, record, *args, **kwargs):
268 9b32e2fb Kostas Papadimitriou
        if self.user.owns_project(record):
269 9b32e2fb Kostas Papadimitriou
            return record.state_display()
270 e6d284ef Olga Brani
        else:
271 9b32e2fb Kostas Papadimitriou
            status = record.user_status(self.user)
272 9b32e2fb Kostas Papadimitriou
            return record.user_status_display(self.user)
273 5550bcfb Kostas Papadimitriou
274 7db8c163 Georgios D. Tsoukalas
    def render_members_count(self, record, *args, **kwargs):
275 7db8c163 Georgios D. Tsoukalas
        append = ""
276 7db8c163 Georgios D. Tsoukalas
        application = record
277 7db8c163 Georgios D. Tsoukalas
        project = application.get_project()
278 7db8c163 Georgios D. Tsoukalas
        if project is None:
279 7db8c163 Georgios D. Tsoukalas
            append = mark_safe("<i class='tiny'>%s</i>" % (_('pending'),))
280 7db8c163 Georgios D. Tsoukalas
281 7db8c163 Georgios D. Tsoukalas
        c = project.count_pending_memberships()
282 7db8c163 Georgios D. Tsoukalas
        if c > 0:
283 7db8c163 Georgios D. Tsoukalas
            append = mark_safe("<i class='tiny'> - %d %s</i>"
284 7db8c163 Georgios D. Tsoukalas
                                % (c, _('pending')))
285 7db8c163 Georgios D. Tsoukalas
286 7db8c163 Georgios D. Tsoukalas
        return mark_safe(str(record.members_count()) + append)
287 7db8c163 Georgios D. Tsoukalas
        
288 5550bcfb Kostas Papadimitriou
    class Meta:
289 5550bcfb Kostas Papadimitriou
        model = ProjectApplication
290 9b32e2fb Kostas Papadimitriou
        fields = ('name', 'membership_status', 'issue_date', 'end_date', 'members_count')
291 2743e261 Kostas Papadimitriou
        attrs = {'id': 'projects-list', 'class': 'my-projects alt-style'}
292 2743e261 Kostas Papadimitriou
        template = "im/table_render.html"
293 3f0d6293 Kostas Papadimitriou
        empty_text = _('No projects')
294 7cb08744 Kostas Papadimitriou
        exclude = ('start_date', )
295 3f0d6293 Kostas Papadimitriou
296 9b32e2fb Kostas Papadimitriou
class ProjectModificationApplicationsTable(UserProjectApplicationsTable):
297 9b32e2fb Kostas Papadimitriou
    name = LinkColumn('astakos.im.views.project_detail',
298 9b32e2fb Kostas Papadimitriou
                      verbose_name=_('Action'),
299 9b32e2fb Kostas Papadimitriou
                      coerce= lambda x: 'review',
300 9b32e2fb Kostas Papadimitriou
                      args=(A('pk'),))
301 9b32e2fb Kostas Papadimitriou
    class Meta:
302 9b32e2fb Kostas Papadimitriou
        attrs = {'id': 'projects-list', 'class': 'my-projects alt-style'}
303 9b32e2fb Kostas Papadimitriou
        fields = ('issue_date', 'membership_status')
304 9b32e2fb Kostas Papadimitriou
        exclude = ('start_date', 'end_date', 'members_count', 'project_action')
305 3f0d6293 Kostas Papadimitriou
306 3f0d6293 Kostas Papadimitriou
def member_action_extra_context(membership, table, col):
307 f3a45fc6 Kostas Papadimitriou
308 f3a45fc6 Kostas Papadimitriou
    context = []
309 6c22d993 Kostas Papadimitriou
    urls, actions, prompts, confirms = [], [], [], []
310 3f0d6293 Kostas Papadimitriou
311 3f0d6293 Kostas Papadimitriou
    if membership.state == ProjectMembership.REQUESTED:
312 3f0d6293 Kostas Papadimitriou
        urls = ['astakos.im.views.project_reject_member',
313 3f0d6293 Kostas Papadimitriou
                'astakos.im.views.project_accept_member']
314 7db8c163 Georgios D. Tsoukalas
        actions = [_('Reject'), _('Accept')]
315 3f0d6293 Kostas Papadimitriou
        prompts = [_('Are you sure you want to reject this member ?'),
316 7db8c163 Georgios D. Tsoukalas
                   _('Are you sure you want to accept this member ?')]
317 3f0d6293 Kostas Papadimitriou
        confirms = [True, True]
318 3f0d6293 Kostas Papadimitriou
319 3f0d6293 Kostas Papadimitriou
    if membership.state == ProjectMembership.ACCEPTED:
320 3f0d6293 Kostas Papadimitriou
        urls = ['astakos.im.views.project_remove_member']
321 3f0d6293 Kostas Papadimitriou
        actions = [_('Remove')]
322 db9a498c Kostas Papadimitriou
        if table.user == membership.person:
323 db9a498c Kostas Papadimitriou
            actions = [_('Leave')]
324 3f0d6293 Kostas Papadimitriou
        prompts = [_('Are you sure you want to remove this member ?')]
325 3f0d6293 Kostas Papadimitriou
        confirms = [True, True]
326 2743e261 Kostas Papadimitriou
327 2743e261 Kostas Papadimitriou
328 f3a45fc6 Kostas Papadimitriou
    for i, url in enumerate(urls):
329 f3a45fc6 Kostas Papadimitriou
        context.append(dict(url=reverse(url, args=(table.project.pk,
330 f3a45fc6 Kostas Papadimitriou
                                                   membership.person.pk)),
331 f3a45fc6 Kostas Papadimitriou
                            action=actions[i], prompt=prompts[i],
332 f3a45fc6 Kostas Papadimitriou
                            confirm=confirms[i]))
333 f3a45fc6 Kostas Papadimitriou
    return context
334 f3a45fc6 Kostas Papadimitriou
335 3f0d6293 Kostas Papadimitriou
class ProjectApplicationMembersTable(UserTable):
336 2743e261 Kostas Papadimitriou
    name = tables.Column(accessor="person.last_name", verbose_name=_('Name'))
337 2743e261 Kostas Papadimitriou
    status = tables.Column(accessor="state", verbose_name=_('Status'))
338 3f0d6293 Kostas Papadimitriou
    project_action = RichLinkColumn(verbose_name=_('Action'),
339 3f0d6293 Kostas Papadimitriou
                                    extra_context=member_action_extra_context,
340 0cf289c2 Sofia Papagiannaki
                                    orderable=False)
341 3f0d6293 Kostas Papadimitriou
342 3f0d6293 Kostas Papadimitriou
343 3f0d6293 Kostas Papadimitriou
    def __init__(self, project, *args, **kwargs):
344 3f0d6293 Kostas Papadimitriou
        self.project = project
345 3f0d6293 Kostas Papadimitriou
        super(ProjectApplicationMembersTable, self).__init__(*args, **kwargs)
346 3f0d6293 Kostas Papadimitriou
        if not self.user.owns_project(self.project):
347 3f0d6293 Kostas Papadimitriou
            self.exclude = ('project_action', )
348 2743e261 Kostas Papadimitriou
349 2743e261 Kostas Papadimitriou
350 2743e261 Kostas Papadimitriou
    def render_name(self, value, record, *args, **kwargs):
351 5d87753f Kostas Papadimitriou
        return record.person.realname
352 2743e261 Kostas Papadimitriou
353 2743e261 Kostas Papadimitriou
    def render_status(self, value, *args, **kwargs):
354 db9a498c Kostas Papadimitriou
        return USER_STATUS_DISPLAY.get(value, 'Unknown')
355 2743e261 Kostas Papadimitriou
356 2743e261 Kostas Papadimitriou
    class Meta:
357 2743e261 Kostas Papadimitriou
        template = "im/table_render.html"
358 2743e261 Kostas Papadimitriou
        model = ProjectMembership
359 2743e261 Kostas Papadimitriou
        fields = ('name', 'status')
360 2743e261 Kostas Papadimitriou
        attrs = {'id': 'members-table', 'class': 'members-table alt-style'}
361 3f0d6293 Kostas Papadimitriou
        empty_text = _('No members')