Statistics
| Branch: | Tag: | Revision:

root / snf-astakos-app / astakos / im / tables.py @ 1e361a6d

History | View | Annotate | Download (13.5 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 aad0e329 Giorgos Korfiatis
from astakos.im.functions import (join_project_checks,
48 c1007621 Giorgos Korfiatis
                                  can_leave_request,
49 aad0e329 Giorgos Korfiatis
                                  cancel_membership_checks)
50 5550bcfb Kostas Papadimitriou
51 5550bcfb Kostas Papadimitriou
DEFAULT_DATE_FORMAT = "d/m/Y"
52 5550bcfb Kostas Papadimitriou
53 5550bcfb Kostas Papadimitriou
54 9b32e2fb Kostas Papadimitriou
class LinkColumn(tables.LinkColumn):
55 6795eb09 Kostas Papadimitriou
56 6795eb09 Kostas Papadimitriou
    def __init__(self, *args, **kwargs):
57 9b32e2fb Kostas Papadimitriou
        self.coerce = kwargs.pop('coerce', None)
58 9b32e2fb Kostas Papadimitriou
        self.append = kwargs.pop('append', None)
59 9b32e2fb Kostas Papadimitriou
        super(LinkColumn, self).__init__(*args, **kwargs)
60 9b32e2fb Kostas Papadimitriou
61 9b32e2fb Kostas Papadimitriou
    def render(self, value, record, bound_column):
62 9b32e2fb Kostas Papadimitriou
        link = super(LinkColumn, self).render(value, record, bound_column)
63 9b32e2fb Kostas Papadimitriou
        extra = ''
64 9b32e2fb Kostas Papadimitriou
        if self.append:
65 9b32e2fb Kostas Papadimitriou
            if callable(self.append):
66 9b32e2fb Kostas Papadimitriou
                extra = self.append(record, bound_column)
67 9b32e2fb Kostas Papadimitriou
            else:
68 9b32e2fb Kostas Papadimitriou
                extra = self.append
69 9b32e2fb Kostas Papadimitriou
        return mark_safe(link + extra)
70 6795eb09 Kostas Papadimitriou
71 6795eb09 Kostas Papadimitriou
    def render_link(self, uri, text, attrs=None):
72 9b32e2fb Kostas Papadimitriou
        if self.coerce:
73 9b32e2fb Kostas Papadimitriou
            text = self.coerce(text)
74 9b32e2fb Kostas Papadimitriou
        return super(LinkColumn, self).render_link(uri, text, attrs)
75 6795eb09 Kostas Papadimitriou
76 6795eb09 Kostas Papadimitriou
77 91cb6fb6 Kostas Papadimitriou
# Helper columns
78 91cb6fb6 Kostas Papadimitriou
class RichLinkColumn(tables.TemplateColumn):
79 91cb6fb6 Kostas Papadimitriou
80 91cb6fb6 Kostas Papadimitriou
    method = 'POST'
81 91cb6fb6 Kostas Papadimitriou
82 91cb6fb6 Kostas Papadimitriou
    confirm_prompt = _('Yes')
83 91cb6fb6 Kostas Papadimitriou
    cancel_prompt = _('No')
84 91cb6fb6 Kostas Papadimitriou
    confirm = True
85 91cb6fb6 Kostas Papadimitriou
86 91cb6fb6 Kostas Papadimitriou
    prompt = _('Confirm action ?')
87 91cb6fb6 Kostas Papadimitriou
88 91cb6fb6 Kostas Papadimitriou
    action_tpl = None
89 91cb6fb6 Kostas Papadimitriou
    action = _('Action')
90 91cb6fb6 Kostas Papadimitriou
    extra_context = lambda record, table, column: {}
91 91cb6fb6 Kostas Papadimitriou
92 91cb6fb6 Kostas Papadimitriou
    url = None
93 91cb6fb6 Kostas Papadimitriou
    url_args = ()
94 91cb6fb6 Kostas Papadimitriou
    resolve_func = None
95 91cb6fb6 Kostas Papadimitriou
96 91cb6fb6 Kostas Papadimitriou
    def __init__(self, *args, **kwargs):
97 91cb6fb6 Kostas Papadimitriou
        kwargs['template_name'] = kwargs.get('template_name',
98 91cb6fb6 Kostas Papadimitriou
                                             'im/table_rich_link_column.html')
99 91cb6fb6 Kostas Papadimitriou
        for attr in ['method', 'confirm_prompt',
100 91cb6fb6 Kostas Papadimitriou
                     'cancel_prompt', 'prompt', 'url',
101 91cb6fb6 Kostas Papadimitriou
                     'url_args', 'action', 'confirm',
102 91cb6fb6 Kostas Papadimitriou
                     'resolve_func', 'extra_context']:
103 91cb6fb6 Kostas Papadimitriou
            setattr(self, attr, kwargs.pop(attr, getattr(self, attr)))
104 91cb6fb6 Kostas Papadimitriou
105 91cb6fb6 Kostas Papadimitriou
        super(RichLinkColumn, self).__init__(*args, **kwargs)
106 91cb6fb6 Kostas Papadimitriou
107 91cb6fb6 Kostas Papadimitriou
    def render(self, record, table, value, bound_column, **kwargs):
108 91cb6fb6 Kostas Papadimitriou
        # If the table is being rendered using `render_table`, it hackily
109 91cb6fb6 Kostas Papadimitriou
        # attaches the context to the table as a gift to `TemplateColumn`. If
110 91cb6fb6 Kostas Papadimitriou
        # the table is being rendered via `Table.as_html`, this won't exist.
111 3f0d6293 Kostas Papadimitriou
        content = ''
112 3f0d6293 Kostas Papadimitriou
        for extra_context in self.get_template_context(record, table, value,
113 3f0d6293 Kostas Papadimitriou
                                                       bound_column, **kwargs):
114 3f0d6293 Kostas Papadimitriou
            context = getattr(table, 'context', Context())
115 3f0d6293 Kostas Papadimitriou
            context.update(extra_context)
116 3f0d6293 Kostas Papadimitriou
            try:
117 3f0d6293 Kostas Papadimitriou
                if self.template_code:
118 3f0d6293 Kostas Papadimitriou
                    content += Template(self.template_code).render(context)
119 3f0d6293 Kostas Papadimitriou
                else:
120 3f0d6293 Kostas Papadimitriou
                    content += render_to_string(self.template_name, context)
121 3f0d6293 Kostas Papadimitriou
            finally:
122 3f0d6293 Kostas Papadimitriou
                context.pop()
123 3f0d6293 Kostas Papadimitriou
124 3f0d6293 Kostas Papadimitriou
        return mark_safe(content)
125 91cb6fb6 Kostas Papadimitriou
126 91cb6fb6 Kostas Papadimitriou
    def get_confirm(self, record, table):
127 91cb6fb6 Kostas Papadimitriou
        if callable(self.confirm):
128 91cb6fb6 Kostas Papadimitriou
            return self.confirm(record, table)
129 91cb6fb6 Kostas Papadimitriou
        return self.confirm
130 91cb6fb6 Kostas Papadimitriou
131 91cb6fb6 Kostas Papadimitriou
    def resolved_url(self, record, table):
132 91cb6fb6 Kostas Papadimitriou
        if callable(self.url):
133 91cb6fb6 Kostas Papadimitriou
            return self.url(record, table)
134 91cb6fb6 Kostas Papadimitriou
135 91cb6fb6 Kostas Papadimitriou
        if not self.url:
136 91cb6fb6 Kostas Papadimitriou
            return '#'
137 91cb6fb6 Kostas Papadimitriou
138 91cb6fb6 Kostas Papadimitriou
        args = list(self.url_args)
139 91cb6fb6 Kostas Papadimitriou
        for index, arg in enumerate(args):
140 91cb6fb6 Kostas Papadimitriou
            if isinstance(arg, A):
141 91cb6fb6 Kostas Papadimitriou
                args[index] = arg.resolve(record)
142 91cb6fb6 Kostas Papadimitriou
        return reverse(self.url, args=args)
143 91cb6fb6 Kostas Papadimitriou
144 91cb6fb6 Kostas Papadimitriou
    def get_action(self, record, table):
145 91cb6fb6 Kostas Papadimitriou
        if callable(self.action):
146 91cb6fb6 Kostas Papadimitriou
            return self.action(record, table)
147 91cb6fb6 Kostas Papadimitriou
        return self.action
148 91cb6fb6 Kostas Papadimitriou
149 91cb6fb6 Kostas Papadimitriou
    def get_prompt(self, record, table):
150 91cb6fb6 Kostas Papadimitriou
        if callable(self.prompt):
151 91cb6fb6 Kostas Papadimitriou
            return self.prompt(record, table)
152 91cb6fb6 Kostas Papadimitriou
        return self.prompt
153 91cb6fb6 Kostas Papadimitriou
154 91cb6fb6 Kostas Papadimitriou
    def get_template_context(self, record, table, value, bound_column, **kwargs):
155 91cb6fb6 Kostas Papadimitriou
        context = {'default': bound_column.default,
156 91cb6fb6 Kostas Papadimitriou
                   'record': record,
157 91cb6fb6 Kostas Papadimitriou
                   'value': value,
158 91cb6fb6 Kostas Papadimitriou
                   'col': self,
159 91cb6fb6 Kostas Papadimitriou
                   'url': self.resolved_url(record, table),
160 91cb6fb6 Kostas Papadimitriou
                   'prompt': self.get_prompt(record, table),
161 91cb6fb6 Kostas Papadimitriou
                   'action': self.get_action(record, table),
162 91cb6fb6 Kostas Papadimitriou
                   'confirm': self.get_confirm(record, table)
163 91cb6fb6 Kostas Papadimitriou
                  }
164 91cb6fb6 Kostas Papadimitriou
165 f3a45fc6 Kostas Papadimitriou
        # decide whether to return dict or a list of dicts in case we want to
166 f3a45fc6 Kostas Papadimitriou
        # display multiple actions within a cell.
167 91cb6fb6 Kostas Papadimitriou
        if self.extra_context:
168 3f0d6293 Kostas Papadimitriou
            contexts = []
169 3f0d6293 Kostas Papadimitriou
            extra_contexts = self.extra_context(record, table, self)
170 3f0d6293 Kostas Papadimitriou
            if isinstance(extra_contexts, list):
171 3f0d6293 Kostas Papadimitriou
                for extra_context in extra_contexts:
172 3f0d6293 Kostas Papadimitriou
                    newcontext = dict(context)
173 3f0d6293 Kostas Papadimitriou
                    newcontext.update(extra_context)
174 3f0d6293 Kostas Papadimitriou
                    contexts.append(newcontext)
175 3f0d6293 Kostas Papadimitriou
            else:
176 3f0d6293 Kostas Papadimitriou
                context.update(extra_contexts)
177 f3a45fc6 Kostas Papadimitriou
                contexts = [context]
178 3f0d6293 Kostas Papadimitriou
        else:
179 3f0d6293 Kostas Papadimitriou
            contexts = [context]
180 91cb6fb6 Kostas Papadimitriou
181 3f0d6293 Kostas Papadimitriou
        return contexts
182 91cb6fb6 Kostas Papadimitriou
183 91cb6fb6 Kostas Papadimitriou
184 db9a498c Kostas Papadimitriou
def action_extra_context(application, table, self):
185 91cb6fb6 Kostas Papadimitriou
    user = table.user
186 91cb6fb6 Kostas Papadimitriou
    url, action, confirm, prompt = '', '', True, ''
187 91cb6fb6 Kostas Papadimitriou
    append_url = ''
188 91cb6fb6 Kostas Papadimitriou
189 aad0e329 Giorgos Korfiatis
    can_join = can_leave = can_cancel = False
190 05617ab9 Kostas Papadimitriou
    project = application.get_project()
191 05617ab9 Kostas Papadimitriou
192 c88913f7 Giorgos Korfiatis
    if project and project.is_approved():
193 05617ab9 Kostas Papadimitriou
        try:
194 ff67242a Giorgos Korfiatis
            join_project_checks(project)
195 05617ab9 Kostas Papadimitriou
            can_join = True
196 bd9af366 Kostas Papadimitriou
        except PermissionDenied, e:
197 05617ab9 Kostas Papadimitriou
            pass
198 05617ab9 Kostas Papadimitriou
199 05617ab9 Kostas Papadimitriou
        try:
200 c1007621 Giorgos Korfiatis
            can_leave = can_leave_request(project, user)
201 05617ab9 Kostas Papadimitriou
        except PermissionDenied:
202 05617ab9 Kostas Papadimitriou
            pass
203 05617ab9 Kostas Papadimitriou
204 aad0e329 Giorgos Korfiatis
        try:
205 aad0e329 Giorgos Korfiatis
            cancel_membership_checks(project)
206 aad0e329 Giorgos Korfiatis
            can_cancel = True
207 aad0e329 Giorgos Korfiatis
        except PermissionDenied:
208 aad0e329 Giorgos Korfiatis
            pass
209 aad0e329 Giorgos Korfiatis
210 aad0e329 Giorgos Korfiatis
    membership = user.get_membership(project)
211 aad0e329 Giorgos Korfiatis
    if membership is not None:
212 aad0e329 Giorgos Korfiatis
        if can_leave and membership.can_leave():
213 aad0e329 Giorgos Korfiatis
            url = 'astakos.im.views.project_leave'
214 aad0e329 Giorgos Korfiatis
            action = _('Leave')
215 aad0e329 Giorgos Korfiatis
            confirm = True
216 aad0e329 Giorgos Korfiatis
            prompt = _('Are you sure you want to leave from the project?')
217 aad0e329 Giorgos Korfiatis
        elif can_cancel and membership.can_cancel():
218 aad0e329 Giorgos Korfiatis
            url = 'astakos.im.views.project_cancel'
219 aad0e329 Giorgos Korfiatis
            action = _('Cancel')
220 aad0e329 Giorgos Korfiatis
            confirm = True
221 aad0e329 Giorgos Korfiatis
            prompt = _('Are you sure you want to cancel the join request?')
222 aad0e329 Giorgos Korfiatis
223 aad0e329 Giorgos Korfiatis
    elif can_join:
224 05617ab9 Kostas Papadimitriou
        url = 'astakos.im.views.project_join'
225 05617ab9 Kostas Papadimitriou
        action = _('Join')
226 05617ab9 Kostas Papadimitriou
        confirm = True
227 aad0e329 Giorgos Korfiatis
        prompt = _('Are you sure you want to join this project?')
228 05617ab9 Kostas Papadimitriou
    else:
229 05617ab9 Kostas Papadimitriou
        action = ''
230 05617ab9 Kostas Papadimitriou
        confirm = False
231 05617ab9 Kostas Papadimitriou
        url = None
232 db9a498c Kostas Papadimitriou
233 7184f408 Giorgos Korfiatis
    url = reverse(url, args=(application.chain, )) + append_url if url else ''
234 bd9af366 Kostas Papadimitriou
235 6401cff2 Sofia Papagiannaki
    return {'action': action,
236 6401cff2 Sofia Papagiannaki
            'confirm': confirm,
237 6401cff2 Sofia Papagiannaki
            'url': url,
238 91cb6fb6 Kostas Papadimitriou
            'prompt': prompt}
239 91cb6fb6 Kostas Papadimitriou
240 91cb6fb6 Kostas Papadimitriou
241 3f0d6293 Kostas Papadimitriou
class UserTable(tables.Table):
242 5550bcfb Kostas Papadimitriou
243 5550bcfb Kostas Papadimitriou
    def __init__(self, *args, **kwargs):
244 5550bcfb Kostas Papadimitriou
        self.user = None
245 5550bcfb Kostas Papadimitriou
246 5550bcfb Kostas Papadimitriou
        if 'request' in kwargs and kwargs.get('request').user:
247 5550bcfb Kostas Papadimitriou
            self.user = kwargs.get('request').user
248 5550bcfb Kostas Papadimitriou
249 5550bcfb Kostas Papadimitriou
        if 'user' in kwargs:
250 5550bcfb Kostas Papadimitriou
            self.user = kwargs.pop('user')
251 5550bcfb Kostas Papadimitriou
252 3f0d6293 Kostas Papadimitriou
        super(UserTable, self).__init__(*args, **kwargs)
253 3f0d6293 Kostas Papadimitriou
254 05617ab9 Kostas Papadimitriou
def project_name_append(application, column):
255 05617ab9 Kostas Papadimitriou
    if application.has_pending_modifications():
256 05617ab9 Kostas Papadimitriou
        return mark_safe("<br /><i class='tiny'>%s</i>" % \
257 3e3743f2 Giorgos Korfiatis
                             _('modifications pending'))
258 05617ab9 Kostas Papadimitriou
    return u''
259 9b32e2fb Kostas Papadimitriou
260 3f0d6293 Kostas Papadimitriou
# Table classes
261 3f0d6293 Kostas Papadimitriou
class UserProjectApplicationsTable(UserTable):
262 3f0d6293 Kostas Papadimitriou
    caption = _('My projects')
263 5550bcfb Kostas Papadimitriou
264 9b32e2fb Kostas Papadimitriou
    name = LinkColumn('astakos.im.views.project_detail',
265 9b32e2fb Kostas Papadimitriou
                      coerce=lambda x: truncatename(x, 25),
266 9b32e2fb Kostas Papadimitriou
                      append=project_name_append,
267 7184f408 Giorgos Korfiatis
                      args=(A('chain'),))
268 f31db8a5 Georgios D. Tsoukalas
    issue_date = tables.DateColumn(verbose_name=_('Application'), format=DEFAULT_DATE_FORMAT)
269 5550bcfb Kostas Papadimitriou
    start_date = tables.DateColumn(format=DEFAULT_DATE_FORMAT)
270 f31db8a5 Georgios D. Tsoukalas
    end_date = tables.DateColumn(verbose_name=_('Expiration'), format=DEFAULT_DATE_FORMAT)
271 9b32e2fb Kostas Papadimitriou
    members_count = tables.Column(verbose_name=_("Members"), default=0,
272 0cf289c2 Sofia Papagiannaki
                                  orderable=False)
273 b7c802a2 Olga Brani
    membership_status = tables.Column(verbose_name=_("Status"), empty_values=(),
274 0cf289c2 Sofia Papagiannaki
                                      orderable=False)
275 91cb6fb6 Kostas Papadimitriou
    project_action = RichLinkColumn(verbose_name=_('Action'),
276 91cb6fb6 Kostas Papadimitriou
                                    extra_context=action_extra_context,
277 0cf289c2 Sofia Papagiannaki
                                    orderable=False)
278 5550bcfb Kostas Papadimitriou
279 2743e261 Kostas Papadimitriou
280 e6d284ef Olga Brani
    def render_membership_status(self, record, *args, **kwargs):
281 7aac6b38 Giorgos Korfiatis
        if self.user.owns_application(record) or self.user.is_project_admin():
282 c88913f7 Giorgos Korfiatis
            return record.project_state_display()
283 e6d284ef Olga Brani
        else:
284 c88913f7 Giorgos Korfiatis
            try:
285 c88913f7 Giorgos Korfiatis
                project = record.project
286 c88913f7 Giorgos Korfiatis
                return self.user.membership_display(project)
287 c88913f7 Giorgos Korfiatis
            except Project.DoesNotExist:
288 8e1a5af5 Georgios D. Tsoukalas
                return _("Unknown")
289 5550bcfb Kostas Papadimitriou
290 7db8c163 Georgios D. Tsoukalas
    def render_members_count(self, record, *args, **kwargs):
291 7db8c163 Georgios D. Tsoukalas
        append = ""
292 7db8c163 Georgios D. Tsoukalas
        application = record
293 7db8c163 Georgios D. Tsoukalas
        project = application.get_project()
294 7db8c163 Georgios D. Tsoukalas
        if project is None:
295 7db8c163 Georgios D. Tsoukalas
            append = mark_safe("<i class='tiny'>%s</i>" % (_('pending'),))
296 7db8c163 Georgios D. Tsoukalas
297 7db8c163 Georgios D. Tsoukalas
        c = project.count_pending_memberships()
298 7db8c163 Georgios D. Tsoukalas
        if c > 0:
299 7db8c163 Georgios D. Tsoukalas
            append = mark_safe("<i class='tiny'> - %d %s</i>"
300 7db8c163 Georgios D. Tsoukalas
                                % (c, _('pending')))
301 7db8c163 Georgios D. Tsoukalas
302 7db8c163 Georgios D. Tsoukalas
        return mark_safe(str(record.members_count()) + append)
303 7db8c163 Georgios D. Tsoukalas
        
304 5550bcfb Kostas Papadimitriou
    class Meta:
305 5550bcfb Kostas Papadimitriou
        model = ProjectApplication
306 9b32e2fb Kostas Papadimitriou
        fields = ('name', 'membership_status', 'issue_date', 'end_date', 'members_count')
307 2743e261 Kostas Papadimitriou
        attrs = {'id': 'projects-list', 'class': 'my-projects alt-style'}
308 2743e261 Kostas Papadimitriou
        template = "im/table_render.html"
309 3f0d6293 Kostas Papadimitriou
        empty_text = _('No projects')
310 7cb08744 Kostas Papadimitriou
        exclude = ('start_date', )
311 3f0d6293 Kostas Papadimitriou
312 9b32e2fb Kostas Papadimitriou
class ProjectModificationApplicationsTable(UserProjectApplicationsTable):
313 9b32e2fb Kostas Papadimitriou
    name = LinkColumn('astakos.im.views.project_detail',
314 9b32e2fb Kostas Papadimitriou
                      verbose_name=_('Action'),
315 9b32e2fb Kostas Papadimitriou
                      coerce= lambda x: 'review',
316 9b32e2fb Kostas Papadimitriou
                      args=(A('pk'),))
317 9b32e2fb Kostas Papadimitriou
    class Meta:
318 9b32e2fb Kostas Papadimitriou
        attrs = {'id': 'projects-list', 'class': 'my-projects alt-style'}
319 9b32e2fb Kostas Papadimitriou
        fields = ('issue_date', 'membership_status')
320 9b32e2fb Kostas Papadimitriou
        exclude = ('start_date', 'end_date', 'members_count', 'project_action')
321 3f0d6293 Kostas Papadimitriou
322 3f0d6293 Kostas Papadimitriou
def member_action_extra_context(membership, table, col):
323 f3a45fc6 Kostas Papadimitriou
324 f3a45fc6 Kostas Papadimitriou
    context = []
325 6c22d993 Kostas Papadimitriou
    urls, actions, prompts, confirms = [], [], [], []
326 3f0d6293 Kostas Papadimitriou
327 d4660e00 Giorgos Korfiatis
    if membership.project.is_deactivated():
328 d4660e00 Giorgos Korfiatis
        return context
329 d4660e00 Giorgos Korfiatis
330 3f0d6293 Kostas Papadimitriou
    if membership.state == ProjectMembership.REQUESTED:
331 3f0d6293 Kostas Papadimitriou
        urls = ['astakos.im.views.project_reject_member',
332 3f0d6293 Kostas Papadimitriou
                'astakos.im.views.project_accept_member']
333 7db8c163 Georgios D. Tsoukalas
        actions = [_('Reject'), _('Accept')]
334 c1007621 Giorgos Korfiatis
        prompts = [_('Are you sure you want to reject this member?'),
335 c1007621 Giorgos Korfiatis
                   _('Are you sure you want to accept this member?')]
336 3f0d6293 Kostas Papadimitriou
        confirms = [True, True]
337 3f0d6293 Kostas Papadimitriou
338 c1007621 Giorgos Korfiatis
    if membership.state in ProjectMembership.ACTUALLY_ACCEPTED:
339 3f0d6293 Kostas Papadimitriou
        urls = ['astakos.im.views.project_remove_member']
340 3f0d6293 Kostas Papadimitriou
        actions = [_('Remove')]
341 c1007621 Giorgos Korfiatis
        prompts = [_('Are you sure you want to remove this member?')]
342 3f0d6293 Kostas Papadimitriou
        confirms = [True, True]
343 2743e261 Kostas Papadimitriou
344 2743e261 Kostas Papadimitriou
345 f3a45fc6 Kostas Papadimitriou
    for i, url in enumerate(urls):
346 f3a45fc6 Kostas Papadimitriou
        context.append(dict(url=reverse(url, args=(table.project.pk,
347 7f3c4920 Giorgos Korfiatis
                                                   membership.pk)),
348 f3a45fc6 Kostas Papadimitriou
                            action=actions[i], prompt=prompts[i],
349 f3a45fc6 Kostas Papadimitriou
                            confirm=confirms[i]))
350 f3a45fc6 Kostas Papadimitriou
    return context
351 f3a45fc6 Kostas Papadimitriou
352 3e3743f2 Giorgos Korfiatis
class ProjectMembersTable(UserTable):
353 b87429e1 Olga Brani
    email = tables.Column(accessor="person.email", verbose_name=_('Email'))    
354 2743e261 Kostas Papadimitriou
    status = tables.Column(accessor="state", verbose_name=_('Status'))
355 3f0d6293 Kostas Papadimitriou
    project_action = RichLinkColumn(verbose_name=_('Action'),
356 3f0d6293 Kostas Papadimitriou
                                    extra_context=member_action_extra_context,
357 0cf289c2 Sofia Papagiannaki
                                    orderable=False)
358 3f0d6293 Kostas Papadimitriou
359 3f0d6293 Kostas Papadimitriou
360 3f0d6293 Kostas Papadimitriou
    def __init__(self, project, *args, **kwargs):
361 3f0d6293 Kostas Papadimitriou
        self.project = project
362 3e3743f2 Giorgos Korfiatis
        super(ProjectMembersTable, self).__init__(*args, **kwargs)
363 3f0d6293 Kostas Papadimitriou
        if not self.user.owns_project(self.project):
364 3f0d6293 Kostas Papadimitriou
            self.exclude = ('project_action', )
365 2743e261 Kostas Papadimitriou
366 d4660e00 Giorgos Korfiatis
    def render_status(self, value, record, *args, **kwargs):
367 d4660e00 Giorgos Korfiatis
        return record.state_display()
368 2743e261 Kostas Papadimitriou
369 2743e261 Kostas Papadimitriou
    class Meta:
370 2743e261 Kostas Papadimitriou
        template = "im/table_render.html"
371 2743e261 Kostas Papadimitriou
        attrs = {'id': 'members-table', 'class': 'members-table alt-style'}
372 3f0d6293 Kostas Papadimitriou
        empty_text = _('No members')