Statistics
| Branch: | Tag: | Revision:

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

History | View | Annotate | Download (11.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 91cb6fb6 Kostas Papadimitriou
41 5550bcfb Kostas Papadimitriou
from django_tables2 import A
42 91cb6fb6 Kostas Papadimitriou
import django_tables2 as tables
43 91cb6fb6 Kostas Papadimitriou
44 5550bcfb Kostas Papadimitriou
from astakos.im.models import *
45 6795eb09 Kostas Papadimitriou
from astakos.im.templatetags.filters import truncatename
46 5550bcfb Kostas Papadimitriou
47 5550bcfb Kostas Papadimitriou
DEFAULT_DATE_FORMAT = "d/m/Y"
48 5550bcfb Kostas Papadimitriou
49 5550bcfb Kostas Papadimitriou
50 5550bcfb Kostas Papadimitriou
MEMBER_STATUS_DISPLAY = {
51 5550bcfb Kostas Papadimitriou
    100: _('Owner'),
52 5550bcfb Kostas Papadimitriou
      0: _('Requested'),
53 5550bcfb Kostas Papadimitriou
      1: _('Pending'),
54 5550bcfb Kostas Papadimitriou
      2: _('Accepted'),
55 5550bcfb Kostas Papadimitriou
      3: _('Removing'),
56 5550bcfb Kostas Papadimitriou
      4: _('Removed'),
57 5550bcfb Kostas Papadimitriou
     -1: _('Unregistered'),
58 5550bcfb Kostas Papadimitriou
}
59 5550bcfb Kostas Papadimitriou
60 6795eb09 Kostas Papadimitriou
class TruncatedLinkColumn(tables.LinkColumn):
61 6795eb09 Kostas Papadimitriou
62 6795eb09 Kostas Papadimitriou
    def __init__(self, *args, **kwargs):
63 6795eb09 Kostas Papadimitriou
        self.truncate_chars = kwargs.pop('truncate_chars', 10)
64 6795eb09 Kostas Papadimitriou
        super(TruncatedLinkColumn, self).__init__(*args, **kwargs)
65 6795eb09 Kostas Papadimitriou
66 6795eb09 Kostas Papadimitriou
67 6795eb09 Kostas Papadimitriou
    def render_link(self, uri, text, attrs=None):
68 6795eb09 Kostas Papadimitriou
        text = truncatename(text, self.truncate_chars)
69 6795eb09 Kostas Papadimitriou
        return super(TruncatedLinkColumn, self).render_link(uri, text, attrs)
70 6795eb09 Kostas Papadimitriou
71 6795eb09 Kostas Papadimitriou
72 91cb6fb6 Kostas Papadimitriou
# Helper columns
73 91cb6fb6 Kostas Papadimitriou
class RichLinkColumn(tables.TemplateColumn):
74 91cb6fb6 Kostas Papadimitriou
75 91cb6fb6 Kostas Papadimitriou
    method = 'POST'
76 91cb6fb6 Kostas Papadimitriou
77 91cb6fb6 Kostas Papadimitriou
    confirm_prompt = _('Yes')
78 91cb6fb6 Kostas Papadimitriou
    cancel_prompt = _('No')
79 91cb6fb6 Kostas Papadimitriou
    confirm = True
80 91cb6fb6 Kostas Papadimitriou
81 91cb6fb6 Kostas Papadimitriou
    prompt = _('Confirm action ?')
82 91cb6fb6 Kostas Papadimitriou
83 91cb6fb6 Kostas Papadimitriou
    action_tpl = None
84 91cb6fb6 Kostas Papadimitriou
    action = _('Action')
85 91cb6fb6 Kostas Papadimitriou
    extra_context = lambda record, table, column: {}
86 91cb6fb6 Kostas Papadimitriou
87 91cb6fb6 Kostas Papadimitriou
    url = None
88 91cb6fb6 Kostas Papadimitriou
    url_args = ()
89 91cb6fb6 Kostas Papadimitriou
    resolve_func = None
90 91cb6fb6 Kostas Papadimitriou
91 91cb6fb6 Kostas Papadimitriou
    def __init__(self, *args, **kwargs):
92 91cb6fb6 Kostas Papadimitriou
        kwargs['template_name'] = kwargs.get('template_name',
93 91cb6fb6 Kostas Papadimitriou
                                             'im/table_rich_link_column.html')
94 91cb6fb6 Kostas Papadimitriou
        for attr in ['method', 'confirm_prompt',
95 91cb6fb6 Kostas Papadimitriou
                     'cancel_prompt', 'prompt', 'url',
96 91cb6fb6 Kostas Papadimitriou
                     'url_args', 'action', 'confirm',
97 91cb6fb6 Kostas Papadimitriou
                     'resolve_func', 'extra_context']:
98 91cb6fb6 Kostas Papadimitriou
            setattr(self, attr, kwargs.pop(attr, getattr(self, attr)))
99 91cb6fb6 Kostas Papadimitriou
100 91cb6fb6 Kostas Papadimitriou
        super(RichLinkColumn, self).__init__(*args, **kwargs)
101 91cb6fb6 Kostas Papadimitriou
102 91cb6fb6 Kostas Papadimitriou
    def render(self, record, table, value, bound_column, **kwargs):
103 91cb6fb6 Kostas Papadimitriou
        # If the table is being rendered using `render_table`, it hackily
104 91cb6fb6 Kostas Papadimitriou
        # attaches the context to the table as a gift to `TemplateColumn`. If
105 91cb6fb6 Kostas Papadimitriou
        # the table is being rendered via `Table.as_html`, this won't exist.
106 3f0d6293 Kostas Papadimitriou
        content = ''
107 3f0d6293 Kostas Papadimitriou
        for extra_context in self.get_template_context(record, table, value,
108 3f0d6293 Kostas Papadimitriou
                                                       bound_column, **kwargs):
109 3f0d6293 Kostas Papadimitriou
            context = getattr(table, 'context', Context())
110 3f0d6293 Kostas Papadimitriou
            context.update(extra_context)
111 3f0d6293 Kostas Papadimitriou
            try:
112 3f0d6293 Kostas Papadimitriou
                if self.template_code:
113 3f0d6293 Kostas Papadimitriou
                    content += Template(self.template_code).render(context)
114 3f0d6293 Kostas Papadimitriou
                else:
115 3f0d6293 Kostas Papadimitriou
                    content += render_to_string(self.template_name, context)
116 3f0d6293 Kostas Papadimitriou
            finally:
117 3f0d6293 Kostas Papadimitriou
                context.pop()
118 3f0d6293 Kostas Papadimitriou
119 3f0d6293 Kostas Papadimitriou
        return mark_safe(content)
120 91cb6fb6 Kostas Papadimitriou
121 91cb6fb6 Kostas Papadimitriou
    def get_confirm(self, record, table):
122 91cb6fb6 Kostas Papadimitriou
        if callable(self.confirm):
123 91cb6fb6 Kostas Papadimitriou
            return self.confirm(record, table)
124 91cb6fb6 Kostas Papadimitriou
        return self.confirm
125 91cb6fb6 Kostas Papadimitriou
126 91cb6fb6 Kostas Papadimitriou
    def resolved_url(self, record, table):
127 91cb6fb6 Kostas Papadimitriou
        if callable(self.url):
128 91cb6fb6 Kostas Papadimitriou
            return self.url(record, table)
129 91cb6fb6 Kostas Papadimitriou
130 91cb6fb6 Kostas Papadimitriou
        if not self.url:
131 91cb6fb6 Kostas Papadimitriou
            return '#'
132 91cb6fb6 Kostas Papadimitriou
133 91cb6fb6 Kostas Papadimitriou
        args = list(self.url_args)
134 91cb6fb6 Kostas Papadimitriou
        for index, arg in enumerate(args):
135 91cb6fb6 Kostas Papadimitriou
            if isinstance(arg, A):
136 91cb6fb6 Kostas Papadimitriou
                args[index] = arg.resolve(record)
137 91cb6fb6 Kostas Papadimitriou
        return reverse(self.url, args=args)
138 91cb6fb6 Kostas Papadimitriou
139 91cb6fb6 Kostas Papadimitriou
    def get_action(self, record, table):
140 91cb6fb6 Kostas Papadimitriou
        if callable(self.action):
141 91cb6fb6 Kostas Papadimitriou
            return self.action(record, table)
142 91cb6fb6 Kostas Papadimitriou
        return self.action
143 91cb6fb6 Kostas Papadimitriou
144 91cb6fb6 Kostas Papadimitriou
    def get_prompt(self, record, table):
145 91cb6fb6 Kostas Papadimitriou
        if callable(self.prompt):
146 91cb6fb6 Kostas Papadimitriou
            return self.prompt(record, table)
147 91cb6fb6 Kostas Papadimitriou
        return self.prompt
148 91cb6fb6 Kostas Papadimitriou
149 91cb6fb6 Kostas Papadimitriou
    def get_template_context(self, record, table, value, bound_column, **kwargs):
150 91cb6fb6 Kostas Papadimitriou
        context = {'default': bound_column.default,
151 91cb6fb6 Kostas Papadimitriou
                   'record': record,
152 91cb6fb6 Kostas Papadimitriou
                   'value': value,
153 91cb6fb6 Kostas Papadimitriou
                   'col': self,
154 91cb6fb6 Kostas Papadimitriou
                   'url': self.resolved_url(record, table),
155 91cb6fb6 Kostas Papadimitriou
                   'prompt': self.get_prompt(record, table),
156 91cb6fb6 Kostas Papadimitriou
                   'action': self.get_action(record, table),
157 91cb6fb6 Kostas Papadimitriou
                   'confirm': self.get_confirm(record, table)
158 91cb6fb6 Kostas Papadimitriou
                  }
159 91cb6fb6 Kostas Papadimitriou
160 f3a45fc6 Kostas Papadimitriou
        # decide whether to return dict or a list of dicts in case we want to
161 f3a45fc6 Kostas Papadimitriou
        # display multiple actions within a cell.
162 91cb6fb6 Kostas Papadimitriou
        if self.extra_context:
163 3f0d6293 Kostas Papadimitriou
            contexts = []
164 3f0d6293 Kostas Papadimitriou
            extra_contexts = self.extra_context(record, table, self)
165 3f0d6293 Kostas Papadimitriou
            if isinstance(extra_contexts, list):
166 3f0d6293 Kostas Papadimitriou
                for extra_context in extra_contexts:
167 3f0d6293 Kostas Papadimitriou
                    newcontext = dict(context)
168 3f0d6293 Kostas Papadimitriou
                    newcontext.update(extra_context)
169 3f0d6293 Kostas Papadimitriou
                    contexts.append(newcontext)
170 3f0d6293 Kostas Papadimitriou
            else:
171 3f0d6293 Kostas Papadimitriou
                context.update(extra_contexts)
172 f3a45fc6 Kostas Papadimitriou
                contexts = [context]
173 3f0d6293 Kostas Papadimitriou
        else:
174 3f0d6293 Kostas Papadimitriou
            contexts = [context]
175 91cb6fb6 Kostas Papadimitriou
176 3f0d6293 Kostas Papadimitriou
        return contexts
177 91cb6fb6 Kostas Papadimitriou
178 91cb6fb6 Kostas Papadimitriou
179 91cb6fb6 Kostas Papadimitriou
def action_extra_context(project, table, self):
180 91cb6fb6 Kostas Papadimitriou
    user = table.user
181 91cb6fb6 Kostas Papadimitriou
    url, action, confirm, prompt = '', '', True, ''
182 91cb6fb6 Kostas Papadimitriou
    append_url = ''
183 91cb6fb6 Kostas Papadimitriou
184 91cb6fb6 Kostas Papadimitriou
    if user.owns_project(project):
185 91cb6fb6 Kostas Papadimitriou
        url = 'astakos.im.views.project_update'
186 91cb6fb6 Kostas Papadimitriou
        action = _('Update')
187 91cb6fb6 Kostas Papadimitriou
        confirm = False
188 91cb6fb6 Kostas Papadimitriou
        prompt = ''
189 6401cff2 Sofia Papagiannaki
    elif user.is_project_accepted_member(project):
190 91cb6fb6 Kostas Papadimitriou
        url = 'astakos.im.views.project_leave'
191 91cb6fb6 Kostas Papadimitriou
        action = _('- Leave')
192 91cb6fb6 Kostas Papadimitriou
        confirm = True
193 91cb6fb6 Kostas Papadimitriou
        prompt = _('Are you sure you want to leave from the project ?')
194 6401cff2 Sofia Papagiannaki
    elif not user.is_project_member(project):
195 91cb6fb6 Kostas Papadimitriou
        url = 'astakos.im.views.project_join'
196 91cb6fb6 Kostas Papadimitriou
        action = _('+ Join')
197 91cb6fb6 Kostas Papadimitriou
        confirm = True
198 91cb6fb6 Kostas Papadimitriou
        prompt = _('Are you sure you want to join this project ?')
199 6401cff2 Sofia Papagiannaki
    else:
200 3f0d6293 Kostas Papadimitriou
        action = ''
201 3f0d6293 Kostas Papadimitriou
        confirm = False
202 3f0d6293 Kostas Papadimitriou
        url = None
203 91cb6fb6 Kostas Papadimitriou
204 6401cff2 Sofia Papagiannaki
    url = reverse(url, args=(project.pk, )) + append_url if url else ''
205 6401cff2 Sofia Papagiannaki
    return {'action': action,
206 6401cff2 Sofia Papagiannaki
            'confirm': confirm,
207 6401cff2 Sofia Papagiannaki
            'url': url,
208 91cb6fb6 Kostas Papadimitriou
            'prompt': prompt}
209 91cb6fb6 Kostas Papadimitriou
210 91cb6fb6 Kostas Papadimitriou
211 3f0d6293 Kostas Papadimitriou
class UserTable(tables.Table):
212 5550bcfb Kostas Papadimitriou
213 5550bcfb Kostas Papadimitriou
    def __init__(self, *args, **kwargs):
214 5550bcfb Kostas Papadimitriou
        self.user = None
215 5550bcfb Kostas Papadimitriou
216 5550bcfb Kostas Papadimitriou
        if 'request' in kwargs and kwargs.get('request').user:
217 5550bcfb Kostas Papadimitriou
            self.user = kwargs.get('request').user
218 5550bcfb Kostas Papadimitriou
219 5550bcfb Kostas Papadimitriou
        if 'user' in kwargs:
220 5550bcfb Kostas Papadimitriou
            self.user = kwargs.pop('user')
221 5550bcfb Kostas Papadimitriou
222 3f0d6293 Kostas Papadimitriou
        super(UserTable, self).__init__(*args, **kwargs)
223 3f0d6293 Kostas Papadimitriou
224 3f0d6293 Kostas Papadimitriou
225 3f0d6293 Kostas Papadimitriou
# Table classes
226 3f0d6293 Kostas Papadimitriou
class UserProjectApplicationsTable(UserTable):
227 3f0d6293 Kostas Papadimitriou
    caption = _('My projects')
228 5550bcfb Kostas Papadimitriou
229 6795eb09 Kostas Papadimitriou
    name = TruncatedLinkColumn('astakos.im.views.project_detail',
230 6795eb09 Kostas Papadimitriou
                                                truncate_chars=25,
231 6795eb09 Kostas Papadimitriou
                                                args=(A('pk'),))
232 5550bcfb Kostas Papadimitriou
    issue_date = tables.DateColumn(format=DEFAULT_DATE_FORMAT)
233 5550bcfb Kostas Papadimitriou
    start_date = tables.DateColumn(format=DEFAULT_DATE_FORMAT)
234 b7c802a2 Olga Brani
    end_date = tables.DateColumn(format=DEFAULT_DATE_FORMAT)
235 2743e261 Kostas Papadimitriou
    members_count = tables.Column(verbose_name=_("Enrolled"), default=0,
236 a5cef8d0 Kostas Papadimitriou
                                  sortable=False)
237 b7c802a2 Olga Brani
    membership_status = tables.Column(verbose_name=_("Status"), empty_values=(),
238 b7c802a2 Olga Brani
                                      sortable=False)
239 91cb6fb6 Kostas Papadimitriou
    project_action = RichLinkColumn(verbose_name=_('Action'),
240 91cb6fb6 Kostas Papadimitriou
                                    extra_context=action_extra_context,
241 91cb6fb6 Kostas Papadimitriou
                                    sortable=False)
242 5550bcfb Kostas Papadimitriou
243 2743e261 Kostas Papadimitriou
244 e6d284ef Olga Brani
    def render_membership_status(self, record, *args, **kwargs):
245 f38084ce Kostas Papadimitriou
        status = record.user_status(self.user)
246 e6d284ef Olga Brani
        if status == 100:
247 e6d284ef Olga Brani
            return record.state
248 e6d284ef Olga Brani
        else:
249 e6d284ef Olga Brani
            return MEMBER_STATUS_DISPLAY.get(status, 'Unknown')
250 5550bcfb Kostas Papadimitriou
251 5550bcfb Kostas Papadimitriou
    class Meta:
252 5550bcfb Kostas Papadimitriou
        model = ProjectApplication
253 b7c802a2 Olga Brani
        fields = ('name', 'membership_status', 'issue_date', 'start_date','end_date', 'members_count')
254 2743e261 Kostas Papadimitriou
        attrs = {'id': 'projects-list', 'class': 'my-projects alt-style'}
255 2743e261 Kostas Papadimitriou
        template = "im/table_render.html"
256 3f0d6293 Kostas Papadimitriou
        empty_text = _('No projects')
257 3f0d6293 Kostas Papadimitriou
258 3f0d6293 Kostas Papadimitriou
259 3f0d6293 Kostas Papadimitriou
def member_action_extra_context(membership, table, col):
260 f3a45fc6 Kostas Papadimitriou
261 f3a45fc6 Kostas Papadimitriou
    context = []
262 6c22d993 Kostas Papadimitriou
    urls, actions, prompts, confirms = [], [], [], []
263 3f0d6293 Kostas Papadimitriou
264 3f0d6293 Kostas Papadimitriou
    if membership.state == ProjectMembership.REQUESTED:
265 3f0d6293 Kostas Papadimitriou
        urls = ['astakos.im.views.project_reject_member',
266 3f0d6293 Kostas Papadimitriou
                'astakos.im.views.project_accept_member']
267 3f0d6293 Kostas Papadimitriou
        actions = [_('Reject'), _('Approve')]
268 3f0d6293 Kostas Papadimitriou
        prompts = [_('Are you sure you want to reject this member ?'),
269 3f0d6293 Kostas Papadimitriou
                   _('Are you sure you want to approve this member ?')]
270 3f0d6293 Kostas Papadimitriou
        confirms = [True, True]
271 3f0d6293 Kostas Papadimitriou
272 3f0d6293 Kostas Papadimitriou
    if membership.state == ProjectMembership.ACCEPTED:
273 3f0d6293 Kostas Papadimitriou
        urls = ['astakos.im.views.project_remove_member']
274 3f0d6293 Kostas Papadimitriou
        actions = [_('Remove')]
275 3f0d6293 Kostas Papadimitriou
        prompts = [_('Are you sure you want to remove this member ?')]
276 3f0d6293 Kostas Papadimitriou
        confirms = [True, True]
277 2743e261 Kostas Papadimitriou
278 2743e261 Kostas Papadimitriou
279 f3a45fc6 Kostas Papadimitriou
    for i, url in enumerate(urls):
280 f3a45fc6 Kostas Papadimitriou
        context.append(dict(url=reverse(url, args=(table.project.pk,
281 f3a45fc6 Kostas Papadimitriou
                                                   membership.person.pk)),
282 f3a45fc6 Kostas Papadimitriou
                            action=actions[i], prompt=prompts[i],
283 f3a45fc6 Kostas Papadimitriou
                            confirm=confirms[i]))
284 f3a45fc6 Kostas Papadimitriou
    return context
285 f3a45fc6 Kostas Papadimitriou
286 3f0d6293 Kostas Papadimitriou
class ProjectApplicationMembersTable(UserTable):
287 2743e261 Kostas Papadimitriou
    name = tables.Column(accessor="person.last_name", verbose_name=_('Name'))
288 2743e261 Kostas Papadimitriou
    status = tables.Column(accessor="state", verbose_name=_('Status'))
289 3f0d6293 Kostas Papadimitriou
    project_action = RichLinkColumn(verbose_name=_('Action'),
290 3f0d6293 Kostas Papadimitriou
                                    extra_context=member_action_extra_context,
291 3f0d6293 Kostas Papadimitriou
                                    sortable=False)
292 3f0d6293 Kostas Papadimitriou
293 3f0d6293 Kostas Papadimitriou
294 3f0d6293 Kostas Papadimitriou
    def __init__(self, project, *args, **kwargs):
295 3f0d6293 Kostas Papadimitriou
        self.project = project
296 3f0d6293 Kostas Papadimitriou
        super(ProjectApplicationMembersTable, self).__init__(*args, **kwargs)
297 3f0d6293 Kostas Papadimitriou
        if not self.user.owns_project(self.project):
298 3f0d6293 Kostas Papadimitriou
            self.exclude = ('project_action', )
299 2743e261 Kostas Papadimitriou
300 2743e261 Kostas Papadimitriou
301 2743e261 Kostas Papadimitriou
    def render_name(self, value, record, *args, **kwargs):
302 5d87753f Kostas Papadimitriou
        return record.person.realname
303 2743e261 Kostas Papadimitriou
304 2743e261 Kostas Papadimitriou
    def render_status(self, value, *args, **kwargs):
305 2743e261 Kostas Papadimitriou
        return MEMBER_STATUS_DISPLAY.get(value, 'Unknown')
306 2743e261 Kostas Papadimitriou
307 2743e261 Kostas Papadimitriou
    class Meta:
308 2743e261 Kostas Papadimitriou
        template = "im/table_render.html"
309 2743e261 Kostas Papadimitriou
        model = ProjectMembership
310 2743e261 Kostas Papadimitriou
        fields = ('name', 'status')
311 2743e261 Kostas Papadimitriou
        attrs = {'id': 'members-table', 'class': 'members-table alt-style'}
312 3f0d6293 Kostas Papadimitriou
        empty_text = _('No members')