Use GenericForeignKey for agenda related items, fix #865

This commit is contained in:
Norman Jäckel 2013-09-07 00:18:13 +02:00
parent 9084216f06
commit c800884a43
13 changed files with 127 additions and 91 deletions

View File

@ -37,7 +37,7 @@ class ItemForm(CssClassMixin, forms.ModelForm):
class Meta: class Meta:
model = Item model = Item
exclude = ('closed', 'weight', 'related_sid') exclude = ('closed', 'weight', 'content_type', 'object_id')
class RelatedItemForm(ItemForm): class RelatedItemForm(ItemForm):
@ -46,7 +46,7 @@ class RelatedItemForm(ItemForm):
""" """
class Meta: class Meta:
model = Item model = Item
exclude = ('closed', 'type', 'weight', 'related_sid', 'title', 'text') exclude = ('closed', 'type', 'weight', 'content_type', 'object_id', 'title', 'text')
class ItemOrderForm(CssClassMixin, forms.Form): class ItemOrderForm(CssClassMixin, forms.Form):

View File

@ -14,6 +14,8 @@ from datetime import datetime
from django.db import models from django.db import models
from django.contrib.auth.models import AnonymousUser from django.contrib.auth.models import AnonymousUser
from django.contrib.contenttypes.models import ContentType
from django.contrib.contenttypes import generic
from django.core.urlresolvers import reverse from django.core.urlresolvers import reverse
from django.utils.translation import ugettext_lazy, ugettext_noop, ugettext as _ from django.utils.translation import ugettext_lazy, ugettext_noop, ugettext as _
@ -87,11 +89,19 @@ class Item(MPTTModel, SlideMixin):
Weight to sort the item in the agenda. Weight to sort the item in the agenda.
""" """
related_sid = models.CharField(null=True, blank=True, max_length=63) content_type = models.ForeignKey(ContentType, null=True, blank=True)
"""
Field for generic relation to a related object. Type of the object.
""" """
Slide-ID to another object to show it in the agenda.
For example a motion or assignment. object_id = models.PositiveIntegerField(null=True, blank=True)
"""
Field for generic relation to a related object. Id of the object.
"""
content_object = generic.GenericForeignKey()
"""
Field for generic relation to a related object. General field to the related object.
""" """
speaker_list_closed = models.BooleanField( speaker_list_closed = models.BooleanField(
@ -125,53 +135,27 @@ class Item(MPTTModel, SlideMixin):
if link == 'delete': if link == 'delete':
return reverse('item_delete', args=[str(self.id)]) return reverse('item_delete', args=[str(self.id)])
def get_related_slide(self):
"""
Return the object at which the item points.
"""
# TODO: Rename it to 'get_related_object'
object = get_slide_from_sid(self.related_sid, element=True)
if object is None:
self.title = _('< Item for deleted slide (%s) >') % self.related_sid
self.related_sid = None
self.save()
return self
else:
return object
def get_related_type(self):
"""
Return the type of the releated slide.
"""
return self.get_related_slide().prefix
def print_related_type(self):
"""
Print the type of the related item.
For use in Template
??Why does {% trans item.print_related_type|capfirst %} not work??
"""
return _(self.get_related_type().capitalize())
def get_title(self): def get_title(self):
""" """
Return the title of this item. Return the title of this item.
""" """
if self.related_sid is None: if not self.content_object:
return self.title return self.title
return self.get_related_slide().get_agenda_title() try:
return self.content_object.get_agenda_title()
except AttributeError:
raise NotImplementedError('You have to provide a get_agenda_title method on your related model.')
def get_title_supplement(self): def get_title_supplement(self):
""" """
Return a supplement for the title. Return a supplement for the title.
""" """
if self.related_sid is None: if not self.content_object:
return '' return ''
try: try:
return self.get_related_slide().get_agenda_title_supplement() return self.content_object.get_agenda_title_supplement()
except AttributeError: except AttributeError:
return '(%s)' % self.print_related_type() raise NotImplementedError('You have to provide a get_agenda_title_supplement method on your related model.')
def slide(self): def slide(self):
""" """
@ -180,11 +164,11 @@ class Item(MPTTModel, SlideMixin):
There are four cases: There are four cases:
* summary slide * summary slide
* list of speakers * list of speakers
* related slide, i. e. the slide of the related object * related item, i. e. the slide of the related object
* normal slide of the item * normal slide of the item
The method returns only one of them according to the config value The method returns only one of them according to the config value
'presentation_argument' and the attribute 'related_sid'. 'presentation_argument' and the attribute 'content_object'.
""" """
if config['presentation_argument'] == 'summary': if config['presentation_argument'] == 'summary':
data = {'title': self.get_title(), data = {'title': self.get_title(),
@ -198,8 +182,8 @@ class Item(MPTTModel, SlideMixin):
'item': self, 'item': self,
'template': 'projector/agenda_list_of_speaker.html', 'template': 'projector/agenda_list_of_speaker.html',
'list_of_speakers': list_of_speakers} 'list_of_speakers': list_of_speakers}
elif self.related_sid: elif self.content_object:
data = self.get_related_slide().slide() data = self.content_object.slide()
else: else:
data = {'item': self, data = {'item': self,

View File

@ -12,6 +12,8 @@
from datetime import datetime from datetime import datetime
from django.contrib.contenttypes.models import ContentType
from django.db.models.signals import pre_delete
from django.dispatch import receiver from django.dispatch import receiver
from django import forms from django import forms
from django.utils.translation import ugettext_lazy, ugettext_noop, ugettext as _ from django.utils.translation import ugettext_lazy, ugettext_noop, ugettext as _
@ -111,3 +113,16 @@ def agenda_list_of_speakers(sender, **kwargs):
return render_to_string('agenda/overlay_speaker_projector.html', context) return render_to_string('agenda/overlay_speaker_projector.html', context)
return Overlay(name, get_widget_html, get_projector_html) return Overlay(name, get_widget_html, get_projector_html)
@receiver(pre_delete)
def listen_to_related_object_delete_signal(sender, instance, **kwargs):
"""
Receiver to listen whether a related item has been deleted.
"""
if hasattr(instance, 'get_agenda_title'):
for item in Item.objects.filter(content_type=ContentType.objects.get_for_model(sender), object_id=instance.pk):
item.title = '< Item for deleted slide (%s) >' % instance.get_agenda_title()
item.content_type = None
item.object_id = None
item.save()

View File

@ -24,9 +24,9 @@
</small> </small>
</h1> </h1>
<p> <p>
{% if item.related_sid %} {% if item.content_object %}
<a href="{% model_url item.get_related_slide 'update' %}" class="btn btn-small"> <a href="{{ item.content_object|absolute_url:'update' }}" class="btn btn-small">
{% blocktrans with type=item.get_related_type|trans name=item.get_related_slide %}Edit {{ type }} {{ name }}{% endblocktrans %} {% blocktrans with type=item.content_type.name|trans name=item.content_object %}Edit {{ type }} {{ name }}{% endblocktrans %}
</a> </a>
{% endif %} {% endif %}
</p> </p>

View File

@ -43,10 +43,10 @@
</small> </small>
</h1> </h1>
<p> <p>
{% if not item.related_sid %} {% if not item.content_object %}
{{ item.text|safe|linebreaks }} {{ item.text|safe|linebreaks }}
{% else %} {% else %}
<a href="{% model_url item.get_related_slide %}" class="btn btn-small">{% trans item.get_related_type %} {{ item.get_related_slide }}</a> <a href="{{ item.content_object|absolute_url }}" class="btn btn-small">{% trans item.content_type.name %} {{ item.content_object }}</a>
{% endif %} {% endif %}
</p> </p>

View File

@ -197,10 +197,11 @@ class ItemUpdate(UpdateView):
success_url_name = 'item_overview' success_url_name = 'item_overview'
def get_form_class(self): def get_form_class(self):
if self.object.related_sid is None: if self.object.content_object:
return ItemForm form = RelatedItemForm
else: else:
return RelatedItemForm form = ItemForm
return form
class ItemCreate(CreateView): class ItemCreate(CreateView):
@ -245,6 +246,31 @@ class ItemDelete(DeleteView):
% html_strong(self.object)) % html_strong(self.object))
class CreateRelatedAgendaItemView(SingleObjectMixin, RedirectView):
"""
View to create and agenda item for a related object.
This view is only for subclassing in views of related apps. You
have to define 'model = ....'
"""
permission_required = 'agenda.can_manage_agenda'
url_name = 'item_overview'
url_name_args = []
def get(self, request, *args, **kwargs):
"""
Set self.object to the relevant object.
"""
self.object = self.get_object()
return super(CreateRelatedAgendaItemView, self).get(request, *args, **kwargs)
def pre_redirect(self, request, *args, **kwargs):
"""
Create the agenda item.
"""
self.item = Item.objects.create(content_object=self.object)
class AgendaPDF(PDFView): class AgendaPDF(PDFView):
""" """
Create a full agenda-PDF. Create a full agenda-PDF.

View File

@ -207,6 +207,9 @@ class Assignment(models.Model, SlideMixin):
def get_agenda_title(self): def get_agenda_title(self):
return self.name return self.name
def get_agenda_title_supplement(self):
return '(%s)' % _('Assignment')
def delete(self): def delete(self):
# Remove any Agenda-Item, which is related to this assignment. # Remove any Agenda-Item, which is related to this assignment.
for item in Item.objects.filter(related_sid=self.sid): for item in Item.objects.filter(related_sid=self.sid):
@ -248,6 +251,7 @@ class Assignment(models.Model, SlideMixin):
('can_manage_assignment', ugettext_noop('Can manage assignments')), # TODO: Add plural s also to the codestring ('can_manage_assignment', ugettext_noop('Can manage assignments')), # TODO: Add plural s also to the codestring
) )
ordering = ('name',) ordering = ('name',)
verbose_name = ugettext_noop('Assignment')
register_slidemodel(Assignment) register_slidemodel(Assignment)

View File

@ -13,7 +13,7 @@
from django.conf.urls import url, patterns from django.conf.urls import url, patterns
from openslides.assignment.views import (ViewPoll, AssignmentPDF, from openslides.assignment.views import (ViewPoll, AssignmentPDF,
AssignmentPollPDF, AssignmentPollDelete, CreateAgendaItem) AssignmentPollPDF, AssignmentPollDelete, CreateRelatedAgendaItemView)
urlpatterns = patterns('openslides.assignment.views', urlpatterns = patterns('openslides.assignment.views',
url(r'^$', url(r'^$',
@ -70,8 +70,8 @@ urlpatterns = patterns('openslides.assignment.views',
name='print_assignment_poll', name='print_assignment_poll',
), ),
url(r'^(?P<assignment_id>\d+)/agenda/$', url(r'^(?P<pk>\d+)/agenda/$',
CreateAgendaItem.as_view(), CreateRelatedAgendaItemView.as_view(),
name='assignment_create_agenda', name='assignment_create_agenda',
), ),

View File

@ -34,7 +34,7 @@ from openslides.config.api import config
from openslides.participant.models import User, Group from openslides.participant.models import User, Group
from openslides.projector.projector import Widget from openslides.projector.projector import Widget
from openslides.poll.views import PollFormView from openslides.poll.views import PollFormView
from openslides.agenda.models import Item from openslides.agenda.views import CreateRelatedAgendaItemView as _CreateRelatedAgendaItemView
from openslides.assignment.models import Assignment, AssignmentPoll from openslides.assignment.models import Assignment, AssignmentPoll
from openslides.assignment.forms import AssignmentForm, AssignmentRunForm from openslides.assignment.forms import AssignmentForm, AssignmentRunForm
@ -487,16 +487,11 @@ class AssignmentPDF(PDFView):
'<br/>'), stylesheet['Paragraph'])) '<br/>'), stylesheet['Paragraph']))
class CreateAgendaItem(RedirectView): class CreateRelatedAgendaItemView(_CreateRelatedAgendaItemView):
permission_required = 'agenda.can_manage_agenda' """
View to create and agenda item for an assignment.
def pre_redirect(self, request, *args, **kwargs): """
self.assignment = Assignment.objects.get(pk=kwargs['assignment_id']) model = Assignment
self.item = Item(related_sid=self.assignment.sid)
self.item.save()
def get_redirect_url(self, **kwargs):
return reverse('item_overview')
class AssignmentPollPDF(PDFView): class AssignmentPollPDF(PDFView):

View File

@ -95,6 +95,7 @@ class Motion(SlideMixin, models.Model):
('can_manage_motion', ugettext_noop('Can manage motions')), ('can_manage_motion', ugettext_noop('Can manage motions')),
) )
ordering = ('identifier', ) ordering = ('identifier', )
verbose_name = ugettext_noop('Motion')
def __unicode__(self): def __unicode__(self):
""" """
@ -475,7 +476,7 @@ class Motion(SlideMixin, models.Model):
def get_agenda_title(self): def get_agenda_title(self):
""" """
Return a title for the Agenda. Return a title for the agenda.
""" """
return self.title return self.title

View File

@ -18,7 +18,6 @@ from django.db import transaction
from django.db.models import Model from django.db.models import Model
from django.utils.text import slugify from django.utils.text import slugify
from django.utils.translation import ugettext as _, ugettext_lazy, ugettext_noop from django.utils.translation import ugettext as _, ugettext_lazy, ugettext_noop
from django.views.generic.detail import SingleObjectMixin
from django.http import Http404, HttpResponseRedirect from django.http import Http404, HttpResponseRedirect
from reportlab.platypus import SimpleDocTemplate from reportlab.platypus import SimpleDocTemplate
@ -33,7 +32,7 @@ from openslides.poll.views import PollFormView
from openslides.projector.api import get_active_slide from openslides.projector.api import get_active_slide
from openslides.projector.projector import Widget, SLIDE from openslides.projector.projector import Widget, SLIDE
from openslides.config.api import config from openslides.config.api import config
from openslides.agenda.models import Item from openslides.agenda.views import CreateRelatedAgendaItemView as _CreateRelatedAgendaItemView
from .models import (Motion, MotionSubmitter, MotionSupporter, MotionPoll, from .models import (Motion, MotionSubmitter, MotionSupporter, MotionPoll,
MotionVersion, State, WorkflowError, Category) MotionVersion, State, WorkflowError, Category)
@ -674,30 +673,20 @@ set_state = MotionSetStateView.as_view()
reset_state = MotionSetStateView.as_view(reset=True) reset_state = MotionSetStateView.as_view(reset=True)
class CreateAgendaItemView(SingleObjectMixin, RedirectView): class CreateRelatedAgendaItemView(_CreateRelatedAgendaItemView):
""" """
View to create and agenda item for a motion. View to create and agenda item for a motion.
""" """
permission_required = 'agenda.can_manage_agenda'
model = Motion model = Motion
url_name = 'item_overview'
url_name_args = []
def get(self, request, *args, **kwargs):
"""
Set self.object to a motion.
"""
self.object = self.get_object()
return super(CreateAgendaItemView, self).get(request, *args, **kwargs)
def pre_redirect(self, request, *args, **kwargs): def pre_redirect(self, request, *args, **kwargs):
""" """
Create the agenda item. Create the agenda item.
""" """
self.item = Item.objects.create(related_sid=self.object.sid) super(CreateRelatedAgendaItemView, self).pre_redirect(request, *args, **kwargs)
self.object.write_log([ugettext_noop('Agenda item created')], self.request.user) self.object.write_log([ugettext_noop('Agenda item created')], self.request.user)
create_agenda_item = CreateAgendaItemView.as_view() create_agenda_item = CreateRelatedAgendaItemView.as_view()
class MotionPDFView(SingleObjectMixin, PDFView): class MotionPDFView(SingleObjectMixin, PDFView):

View File

@ -9,6 +9,9 @@ class ReleatedItem(SlideMixin, models.Model):
name = models.CharField(max_length='255') name = models.CharField(max_length='255')
class Meta:
verbose_name = 'Releated Item CHFNGEJ5634DJ34F'
def get_agenda_title(self): def get_agenda_title(self):
return self.name return self.name
@ -19,4 +22,11 @@ class ReleatedItem(SlideMixin, models.Model):
return '/absolute-url-here/' return '/absolute-url-here/'
class BadReleatedItem(SlideMixin, models.Model):
prefix = 'badreleateditem'
name = models.CharField(max_length='255')
register_slidemodel(ReleatedItem) register_slidemodel(ReleatedItem)
register_slidemodel(BadReleatedItem)

View File

@ -6,7 +6,7 @@
Unit test for the agenda app. Unit test for the agenda app.
:copyright: 2011, 2012 by OpenSlides team, see AUTHORS. :copyright: 2011-2013 by OpenSlides team, see AUTHORS.
:license: GNU GPL, see LICENSE for more details. :license: GNU GPL, see LICENSE for more details.
""" """
@ -19,7 +19,7 @@ from openslides.participant.models import User
from openslides.agenda.models import Item from openslides.agenda.models import Item
from openslides.agenda.slides import agenda_show from openslides.agenda.slides import agenda_show
from .models import ReleatedItem from .models import ReleatedItem, BadReleatedItem # TODO: Rename releated to related
class ItemTest(TestCase): class ItemTest(TestCase):
@ -28,8 +28,8 @@ class ItemTest(TestCase):
self.item2 = Item.objects.create(title='item2') self.item2 = Item.objects.create(title='item2')
self.item3 = Item.objects.create(title='item1A', parent=self.item1) self.item3 = Item.objects.create(title='item1A', parent=self.item1)
self.item4 = Item.objects.create(title='item1Aa', parent=self.item3) self.item4 = Item.objects.create(title='item1Aa', parent=self.item3)
self.releated = ReleatedItem.objects.create(name='foo') self.releated = ReleatedItem.objects.create(name='ekdfjen458gj1siek45nv')
self.item5 = Item.objects.create(title='item5', related_sid=self.releated.sid) self.item5 = Item.objects.create(title='item5', content_object=self.releated)
def testClosed(self): def testClosed(self):
self.assertFalse(self.item1.closed) self.assertFalse(self.item1.closed)
@ -61,10 +61,6 @@ class ItemTest(TestCase):
self.assertEqual(initial['parent'], 0) self.assertEqual(initial['parent'], 0)
self.assertEqual(initial['weight'], item.weight) self.assertEqual(initial['weight'], item.weight)
def testRelated_sid(self):
self.item1.related_sid = 'foobar'
self.assertFalse(self.item1.get_related_slide() is None)
def test_title_supplement(self): def test_title_supplement(self):
self.assertEqual(self.item1.get_title_supplement(), '') self.assertEqual(self.item1.get_title_supplement(), '')
@ -91,8 +87,24 @@ class ItemTest(TestCase):
def test_releated_item(self): def test_releated_item(self):
self.assertEqual(self.item5.get_title(), self.releated.name) self.assertEqual(self.item5.get_title(), self.releated.name)
self.assertEqual(self.item5.get_title_supplement(), 'test item') self.assertEqual(self.item5.get_title_supplement(), 'test item')
self.assertEqual(self.item5.get_related_type(), 'releateditem') self.assertEqual(self.item5.content_type.name, 'Releated Item CHFNGEJ5634DJ34F')
self.assertEqual(self.item5.print_related_type(), 'Releateditem')
def test_deleted_releated_item(self):
self.releated.delete()
self.assertFalse(ReleatedItem.objects.all().exists())
self.assertEqual(Item.objects.get(pk=self.item5.pk).title, '< Item for deleted slide (ekdfjen458gj1siek45nv) >')
def test_bad_releated_item(self):
bad = BadReleatedItem.objects.create(name='dhfne94irkgl2047fzvb')
item = Item.objects.create(title='item_jghfndzrh46w738kdmc', content_object=bad)
self.assertRaisesMessage(
NotImplementedError,
'You have to provide a get_agenda_title method on your related model.',
item.get_title)
self.assertRaisesMessage(
NotImplementedError,
'You have to provide a get_agenda_title_supplement method on your related model.',
item.get_title_supplement)
class ViewTest(TestCase): class ViewTest(TestCase):