thread.py 7.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242
  1. from django.core.urlresolvers import reverse
  2. from django.db import models, transaction
  3. from django.dispatch import receiver
  4. from misago.conf import settings
  5. from misago.core.shortcuts import paginate
  6. from misago.core.utils import slugify
  7. __all__ = [
  8. 'PARTICIPANT_REMOVED',
  9. 'PARTICIPANT_ACTIVE',
  10. 'PARTICIPANT_OWNER',
  11. 'Thread',
  12. 'ThreadParticipant'
  13. ]
  14. PARTICIPANT_REMOVED = 0
  15. PARTICIPANT_ACTIVE = 1
  16. PARTICIPANT_OWNER = 2
  17. class PrivateThreadMixin(object):
  18. pass
  19. class Thread(models.Model, PrivateThreadMixin):
  20. forum = models.ForeignKey('misago_forums.Forum')
  21. label = models.ForeignKey('misago_threads.Label',
  22. null=True, blank=True,
  23. on_delete=models.SET_NULL)
  24. title = models.CharField(max_length=255)
  25. slug = models.CharField(max_length=255)
  26. replies = models.PositiveIntegerField(default=0, db_index=True)
  27. has_reported_posts = models.BooleanField(default=False)
  28. has_moderated_posts = models.BooleanField(default=False)
  29. has_hidden_posts = models.BooleanField(default=False)
  30. has_events = models.BooleanField(default=False)
  31. started_on = models.DateTimeField(db_index=True)
  32. first_post = models.ForeignKey('misago_threads.Post', related_name='+',
  33. null=True, blank=True,
  34. on_delete=models.SET_NULL)
  35. starter = models.ForeignKey(settings.AUTH_USER_MODEL,
  36. null=True, blank=True,
  37. on_delete=models.SET_NULL)
  38. starter_name = models.CharField(max_length=255)
  39. starter_slug = models.CharField(max_length=255)
  40. last_post_on = models.DateTimeField(db_index=True)
  41. last_post = models.ForeignKey('misago_threads.Post', related_name='+',
  42. null=True, blank=True,
  43. on_delete=models.SET_NULL)
  44. last_poster = models.ForeignKey(settings.AUTH_USER_MODEL,
  45. related_name='last_poster_set',
  46. null=True, blank=True,
  47. on_delete=models.SET_NULL)
  48. last_poster_name = models.CharField(max_length=255, null=True, blank=True)
  49. last_poster_slug = models.CharField(max_length=255, null=True, blank=True)
  50. is_pinned = models.BooleanField(default=False, db_index=True)
  51. is_poll = models.BooleanField(default=False)
  52. is_moderated = models.BooleanField(default=False, db_index=True)
  53. is_hidden = models.BooleanField(default=False)
  54. is_closed = models.BooleanField(default=False)
  55. participants = models.ManyToManyField(settings.AUTH_USER_MODEL,
  56. related_name='private_thread_set',
  57. through='ThreadParticipant')
  58. class Meta:
  59. index_together = [
  60. ['forum', 'id'],
  61. ['forum', 'last_post_on'],
  62. ['forum', 'replies'],
  63. ]
  64. def __unicode__(self):
  65. return self.title
  66. def lock(self):
  67. return Thread.objects.select_for_update().get(id=self.id)
  68. def delete(self, *args, **kwargs):
  69. from misago.threads.signals import delete_thread
  70. delete_thread.send(sender=self)
  71. super(Thread, self).delete(*args, **kwargs)
  72. def merge(self, other_thread):
  73. if self.pk == other_thread.pk:
  74. raise ValueError("thread can't be merged with itself")
  75. from misago.threads.signals import merge_thread
  76. merge_thread.send(sender=self, other_thread=other_thread)
  77. def move(self, new_forum):
  78. from misago.threads.signals import move_thread
  79. self.forum = new_forum
  80. move_thread.send(sender=self)
  81. def synchronize(self):
  82. self.replies = self.post_set.filter(is_moderated=False).count()
  83. if self.replies > 0:
  84. self.replies -= 1
  85. reported_post_qs = self.post_set.filter(is_reported=True)
  86. self.has_reported_posts = reported_post_qs.exists()
  87. moderated_post_qs = self.post_set.filter(is_moderated=True)
  88. self.has_moderated_posts = moderated_post_qs.exists()
  89. hidden_post_qs = self.post_set.filter(is_hidden=True)[:1]
  90. self.has_hidden_posts = hidden_post_qs.exists()
  91. self.has_events = self.event_set.exists()
  92. first_post = self.post_set.order_by('id')[:1][0]
  93. self.set_first_post(first_post)
  94. self.is_moderated = first_post.is_moderated
  95. self.is_hidden = first_post.is_hidden
  96. last_post_qs = self.post_set.filter(is_moderated=False).order_by('-id')
  97. last_post = last_post_qs[:1]
  98. if last_post:
  99. self.set_last_post(last_post[0])
  100. else:
  101. self.set_last_post(first_post)
  102. @property
  103. def link_prefix(self):
  104. if self.forum.special_role == 'private_threads':
  105. return 'private_thread'
  106. else:
  107. return 'thread'
  108. def get_url_name(self, suffix=None):
  109. link = 'misago:%s' % self.link_prefix
  110. if suffix:
  111. link = '%s_%s' % (link, suffix)
  112. return link
  113. def get_url(self, suffix=None):
  114. return reverse(self.get_url_name(suffix), kwargs={
  115. 'thread_slug': self.slug,
  116. 'thread_id': self.id
  117. })
  118. def get_absolute_url(self):
  119. return self.get_url()
  120. def get_last_reply_url(self):
  121. return self.get_url('last')
  122. def get_new_reply_url(self):
  123. return self.get_url('new')
  124. def get_moderated_url(self):
  125. return self.get_url('moderated')
  126. def get_reported_url(self):
  127. return self.get_url('reported')
  128. def set_title(self, title):
  129. self.title = title
  130. self.slug = slugify(title)
  131. def set_first_post(self, post):
  132. self.started_on = post.posted_on
  133. self.first_post = post
  134. self.starter = post.poster
  135. self.starter_name = post.poster_name
  136. if post.poster:
  137. self.starter_slug = post.poster.slug
  138. else:
  139. self.starter_slug = slugify(post.poster_name)
  140. self.is_moderated = post.is_moderated
  141. self.is_hidden = post.is_hidden
  142. def set_last_post(self, post):
  143. self.last_post_on = post.posted_on
  144. self.last_post = post
  145. self.last_poster = post.poster
  146. self.last_poster_name = post.poster_name
  147. if post.poster:
  148. self.last_poster_slug = post.poster.slug
  149. else:
  150. self.last_poster_slug = slugify(post.poster_name)
  151. class ThreadParticipantManager(models.Manager):
  152. def delete_participant(self, thread, user):
  153. ThreadParticipant.objects.filter(thread=thread, user=user).delete()
  154. @transaction.atomic
  155. def set_owner(self, thread, user):
  156. thread_owner = ThreadParticipant.objects.filter(
  157. thread=thread, level=PARTICIPANT_OWNER)
  158. thread_owner.update(level=PARTICIPANT_ACTIVE)
  159. self.delete_participant(thread, user)
  160. ThreadParticipant.objects.create(
  161. thread=thread,
  162. user=user,
  163. level=PARTICIPANT_OWNER)
  164. @transaction.atomic
  165. def add_participant(self, thread, user):
  166. self.delete_participant(thread, user)
  167. ThreadParticipant.objects.create(
  168. thread=thread,
  169. user=user,
  170. level=PARTICIPANT_ACTIVE)
  171. @transaction.atomic
  172. def remove_participant(self, thread, user):
  173. self.delete_participant(thread, user)
  174. ThreadParticipant.objects.create(
  175. thread=thread,
  176. user=user,
  177. level=PARTICIPANT_REMOVED)
  178. class ThreadParticipant(models.Model):
  179. thread = models.ForeignKey(Thread)
  180. user = models.ForeignKey(settings.AUTH_USER_MODEL)
  181. level = models.PositiveIntegerField(default=PARTICIPANT_ACTIVE)
  182. objects = ThreadParticipantManager()
  183. @property
  184. def is_removed(self):
  185. return self.level == PARTICIPANT_REMOVED
  186. @property
  187. def is_active(self):
  188. return self.level == PARTICIPANT_ACTIVE
  189. @property
  190. def is_owner(self):
  191. return self.level == PARTICIPANT_OWNER