thread.html 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266
  1. {% extends "sora/layout.html" %}
  2. {% load i18n %}
  3. {% load url from future %}
  4. {% import "_forms.html" as form_theme with context %}
  5. {% import "sora/editor.html" as editor with context %}
  6. {% import "sora/macros.html" as macros with context %}
  7. {% block title %}{{ macros.page_title(title=thread.name,parent=forum.name,page=pagination['page']) }}{% endblock %}
  8. {% block breadcrumb %}{{ super() }} <span class="divider">/</span></li>
  9. {% for parent in parents %}
  10. <li><a href="{{ parent.type|url(forum=forum.pk, slug=forum.slug) }}">{{ parent.name }}</a> <span class="divider">/</span></li>
  11. {% endfor %}
  12. <li class="active">{{ thread.name }}
  13. {%- endblock %}
  14. {% block content %}
  15. <div class="page-header">
  16. <ul class="breadcrumb">
  17. {{ self.breadcrumb() }}</li>
  18. </ul>
  19. <h1>{{ thread.name }}</h1>
  20. <ul class="unstyled thread-info">
  21. {% if thread.moderated %}<li><i class="icon-eye-close"></i> {% trans %}Not Reviewed{% endtrans %}</li>{% endif %}
  22. <li><i class="icon-time"></i> {{ thread.last|reltimesince }}</li>
  23. <li><i class="icon-user"></i> {% if thread.start_poster_id %}<a href="{% url 'user' user=thread.start_poster_id, username=thread.start_poster_slug %}">{{ thread.start_poster_name }}</a>{% else %}{{ thread.start_poster_name }}{% endif %}</li>
  24. <li><i class="icon-comment"></i> {% if thread.replies > 0 -%}
  25. {% trans count=thread.replies, replies=thread.replies|intcomma %}One reply{% pluralize %}{{ replies }} replies{% endtrans %}
  26. {%- else -%}
  27. {% trans %}No replies{% endtrans %}
  28. {%- endif %}</li>
  29. </ul>
  30. </div>
  31. {% if message %}{{ macros.draw_message(message) }}{% endif %}
  32. <div class="list-nav">
  33. {{ pager() }}
  34. {% if user.is_authenticated() %}
  35. <ul class="nav nav-pills pull-right">
  36. <li class="info"><a href="{% url 'thread_new' forum=forum.pk, slug=forum.slug %}"><i class="icon-ok"></i> {% trans %}Watch Thread{% endtrans %}</a></li>{% if acl.threads.can_reply(forum, thread) %}
  37. <li class="primary"><a href="{% url 'thread_reply' thread=thread.pk, slug=thread.slug %}"><i class="icon-plus"></i> {% trans %}Reply{% endtrans %}</a></li>{% endif %}
  38. </ul>
  39. {% endif %}
  40. </div>
  41. <div class="posts-list">
  42. {% for post in posts %}
  43. {% if post.message %}{{ macros.draw_message(post.message) }}{% endif %}
  44. <div id="post-{{ post.pk }}" class="well well-post{% if post.user and post.user.rank and post.user.rank.style %} {{ post.user.rank.style }}{% endif %}">
  45. <div class="post-author">
  46. <img src="{% if post.user_id %}{{ post.user.get_avatar(80) }}{% else %}{{ macros.avatar_guest(80) }}{% endif %}" alt="" class="avatar-normal">
  47. <div class="post-bit">
  48. {% if post.user_id %}
  49. <a href="{% url 'user' user=post.user.pk, username=post.user.username_slug %}" class="lead">{{ post.user.username }}</a>
  50. {% if post.user.get_title() %}<p class="user-title">{{ _(post.user.get_title()) }}</p>{% endif %}
  51. {% else %}
  52. <span class="lead">{{ post.user_name }}</span>
  53. <p class="user-title">{% trans %}Unregistered{% endtrans %}</p>
  54. {% endif %}
  55. <p class="post-date">{{ post.date|reltimesince|low }}</p>
  56. </div>
  57. </div>
  58. <div class="post-extra">
  59. {% if user.is_authenticated() and posts_form %}
  60. <label class="checkbox post-checkbox"><input form="posts_form" name="{{ posts_form['list_items']['html_name'] }}" type="checkbox" class="checkbox-member" value="{{ post.pk }}"{% if posts_form['list_items']['has_value'] and ('' ~ post.pk) in posts_form['list_items']['value'] %} checked="checked"{% endif %}></label>
  61. {% endif %}
  62. <a href="{% if pagination['page'] > 1 -%}
  63. {% url 'thread' thread=thread.pk, slug=thread.slug, page=pagination['page'] %}
  64. {%- else -%}
  65. {% url 'thread' thread=thread.pk, slug=thread.slug %}
  66. {%- endif %}#post-{{ post.pk }}" class="post-perma pull-right tooltip-left" title="{% trans %}Direct link to this post{% endtrans %}">#{{ pagination['start'] + loop.index }}</a>
  67. {% if not post.is_read %}
  68. <span class="label label-warning pull-right">
  69. {% trans %}New{% endtrans %}
  70. </span>
  71. {% endif %}
  72. {% if post.moderated %}
  73. <span class="label label-purple pull-right">
  74. {% trans %}Unreviewed{% endtrans %}
  75. </span>
  76. {% endif %}
  77. {% if post.reported %}
  78. <span class="label label-important pull-right">
  79. {% trans %}Reported{% endtrans %}
  80. </span>
  81. {% endif %}
  82. {% if post.deleted %}
  83. <span class="label label-inverse pull-right">
  84. {% trans %}Deleted{% endtrans %}
  85. </span>
  86. {% endif %}
  87. {% if post.protected and acl.threads.can_protect(forum) %}
  88. <span class="label label-info pull-right">
  89. {% trans %}Protected{% endtrans %}
  90. </span>
  91. {% endif %}
  92. </div>
  93. <div class="post-content">
  94. <div class="markdown">
  95. {{ post.post_preparsed|safe }}
  96. </div>
  97. {% if post.user.signature %}
  98. <div class="post-foot">
  99. {# <p class='lead'><a href="{% url 'user' user=post.user.pk, username=post.user.username_slug %}" class="lead">{{ post.user.username }}</a>, {{ post.date|reltimesince|low }}</p> #}
  100. {% if post.user.signature %}
  101. <div class="signature">
  102. <div class="markdown">
  103. {{ post.user.signature_preparsed|safe }}
  104. </div>
  105. </div>
  106. {% endif %}
  107. </div>
  108. {% endif %}
  109. </div>
  110. <div class="post-nav">
  111. {% if post.edits %}
  112. {% if acl.threads.can_see_changelog(user, forum, post) %}
  113. <a href="{% url 'changelog' thread=thread.pk, slug=thread.slug, post=post.pk %}" class="changelog tooltip-bottom" title="{% trans %}Show changelog{% endtrans %}">{% trans count=post.edits %}One edit{% pluralize %}{{ count }} edits{% endtrans %}</a>
  114. {% else %}
  115. <span class="changelog">{% trans count=post.edits %}One edit{% pluralize %}{{ count }} edits{% endtrans %}</span>
  116. {% endif %}
  117. {% endif %}
  118. {% if user.is_authenticated() %}
  119. <ul class="nav nav-pills pull-right">
  120. <li class="tooltip-top" title="{% trans %}Delete this thread for good{% endtrans %}"><button class="btn danger"><i class="icon-remove"></i></button></li>
  121. <li><button class="btn danger"><i class="icon-trash"></i> {% trans %}Delete{% endtrans %}</button></li>
  122. {% if 1 == 2 %}
  123. <li><a href="#"><i class="icon-info-sign"></i> {% trans %}Info{% endtrans %}</a></li>
  124. <li><button class="btn danger"><i class="icon-remove"></i> {% trans %}Delete{% endtrans %}</button></li>
  125. {% endif %}
  126. {% if acl.threads.can_edit_thread(user, forum, thread, post) and thread.start_post_id == post.pk -%}
  127. <li><a href="{% url 'thread_edit' thread=thread.pk, slug=thread.slug %}"><i class="icon-edit"></i> {% trans %}Edit{% endtrans %}</a></li>
  128. {% elif acl.threads.can_edit_reply(user, forum, thread, post) %}
  129. <li><a href="{% url 'post_edit' thread=thread.pk, slug=thread.slug, post=post.pk %}"><i class="icon-edit"></i> {% trans %}Edit{% endtrans %}</a></li>
  130. {%- endif %}
  131. {% if acl.threads.can_reply(forum, thread) %}<li><a href="{% url 'thread_reply' thread=thread.pk, slug=thread.slug, quote=post.pk %}"><i class="icon-comment"></i> {% trans %}Reply{% endtrans %}</a></li>{% endif %}
  132. </ul>
  133. {% endif %}
  134. </div>
  135. </div>
  136. {% if post.checkpoint_set.all() %}
  137. <div class="post-checkpoints">
  138. {% for checkpoint in post.checkpoint_set.all() %}
  139. <div class="checkpoint">
  140. <hr>
  141. <span>
  142. {%- if checkpoint.action == 'limit' -%}
  143. <i class="icon-lock"></i> {% trans %}This thread has reached its post limit and has been closed.{% endtrans %}
  144. {%- elif checkpoint.action == 'accepted' -%}
  145. <i class="icon-ok"></i> {% trans user=checkpoint_user(checkpoint), date=checkpoint.date|reltimesince|low %}{{ user }} accepted this thread {{ date }}{% endtrans %}
  146. {%- elif checkpoint.action == 'closed' -%}
  147. <i class="icon-lock"></i> {% trans user=checkpoint_user(checkpoint), date=checkpoint.date|reltimesince|low %}{{ user }} closed this thread {{ date }}{% endtrans %}
  148. {%- elif checkpoint.action == 'opened' -%}
  149. <i class="icon-lock"></i> {% trans user=checkpoint_user(checkpoint), date=checkpoint.date|reltimesince|low %}{{ user }} opened this thread {{ date }}{% endtrans %}
  150. {%- elif checkpoint.action == 'deleted' -%}
  151. <i class="icon-trash"></i> {% trans user=checkpoint_user(checkpoint), date=checkpoint.date|reltimesince|low %}{{ user }} deleted this thread {{ date }}{% endtrans %}
  152. {%- elif checkpoint.action == 'undeleted' -%}
  153. <i class="icon-trash"></i> {% trans user=checkpoint_user(checkpoint), date=checkpoint.date|reltimesince|low %}{{ user }} restored this thread {{ date }}{% endtrans %}
  154. {%- endif -%}
  155. </span>
  156. </div>
  157. {% endfor %}
  158. </div>
  159. {% endif %}
  160. {% endfor %}
  161. </div>
  162. {% if user.is_authenticated() and (thread_form or posts_form) %}
  163. <div class="form-actions table-footer mod-actions">
  164. {% if thread_form%}
  165. <form id="thread_form" class="form-inline pull-left" action="{% url 'thread' slug=thread.slug, thread=thread.id, page=pagination['page'] %}" method="POST">
  166. <input type="hidden" name="{{ csrf_id }}" value="{{ csrf_token }}">
  167. <input type="hidden" name="origin" value="thread_form">
  168. {{ form_theme.input_select(thread_form['thread_action'],width=3) }}
  169. <button type="submit" class="btn btn-primary">{% trans %}Go{% endtrans %}</button>
  170. </form>
  171. {% endif %}
  172. {% if posts_form%}
  173. <form id="posts_form" class="form-inline pull-right" action="{% url 'thread' slug=thread.slug, thread=thread.id, page=pagination['page'] %}" method="POST">
  174. <input type="hidden" name="{{ csrf_id }}" value="{{ csrf_token }}">
  175. <input type="hidden" name="origin" value="posts_form">
  176. {{ form_theme.input_select(posts_form['list_action'],width=3) }}
  177. <button type="submit" class="btn btn-primary">{% trans %}Go{% endtrans %}</button>
  178. </form>
  179. {% endif %}
  180. </div>
  181. {% endif %}
  182. <div class="list-nav last">
  183. {{ pager(false) }}
  184. {% if user.is_authenticated() and acl.threads.can_reply(forum, thread) %}
  185. <ul class="nav nav-pills pull-right">
  186. <li class="primary"><a href="{% url 'thread_reply' thread=thread.pk, slug=thread.slug %}"><i class="icon-plus"></i> {% trans %}Reply{% endtrans %}</a></li>
  187. </ul>
  188. {% endif %}
  189. </div>
  190. {% if user.is_authenticated() and acl.threads.can_reply(forum, thread) %}
  191. <div class="quick-reply">
  192. <form action="{% url 'thread_reply' thread=thread.pk, slug=thread.slug %}" method="post">
  193. <input type="hidden" name="{{ csrf_id }}" value="{{ csrf_token }}">
  194. <input type="hidden" name="quick_reply" value="1">
  195. <img src="{{ user.get_avatar() }}" alt="{% trans %}Your Avatar{% endtrans %}" class="avatar-big">
  196. <div class="arrow"></div>
  197. {{ editor.editor(quick_reply.post, _('Post Reply')) }}
  198. </form>
  199. </div>
  200. {% endif %}
  201. {% endblock %}
  202. {% macro pager(extra=true) %}
  203. <ul class="pager pull-left">
  204. {%- if pagination['prev'] > 1 %}<li><a href="{% url 'thread' slug=thread.slug, thread=thread.id %}" class="tooltip-top" title="{% trans %}Go to first page{% endtrans %}"><i class="icon-chevron-left"></i> {% trans %}First{% endtrans %}</a></li>{% endif -%}
  205. {%- if pagination['prev'] > 0 %}<li><a href="{%- if pagination['prev'] > 1 %}{% url 'thread' slug=thread.slug, thread=thread.id, page=pagination['prev'] %}{% else %}{% url 'thread' slug=thread.slug, thread=thread.id %}{% endif %}" class="tooltip-top" title="{% trans %}Older Posts{% endtrans %}"><i class="icon-chevron-left"></i></a></li>{% endif -%}
  206. {%- if pagination['next'] > 0 %}<li><a href="{% url 'thread' slug=thread.slug, thread=thread.id, page=pagination['next'] %}" class="tooltip-top" title="{% trans %}Newest Posts{% endtrans %}"><i class="icon-chevron-right"></i></a></li>{% endif -%}
  207. {%- if pagination['next'] > 0 and pagination['next'] < pagination['total'] %}<li><a href="{% url 'thread' slug=thread.slug, thread=thread.id, page=pagination['total'] %}" class="tooltip-top" title="{% trans %}Go to last page{% endtrans %}">{% trans %}Last{% endtrans %} <i class="icon-chevron-right"></i></a></li>{% endif -%}
  208. <li class="count">
  209. {%- trans current_page=pagination['page'], pages=pagination['total'] -%}
  210. Page {{ current_page }} of {{ pages }}
  211. {%- endtrans -%}
  212. </li>
  213. {% if extra and user.is_authenticated() %}
  214. {% if not is_read %}<li class="unread"><a href="{% url 'thread_new' slug=thread.slug, thread=thread.id %}" class="tooltip-top" title="{% trans %}Go to first unread{% endtrans %}"><i class="icon-star"></i> {% trans %}First Unread{% endtrans %}</a></li>{% endif %}
  215. {% if thread.replies_moderated > 0 and acl.threads.can_approve(forum) %}<li class="moderated"><a href="{% url 'thread_moderated' slug=thread.slug, thread=thread.id %}" class="tooltip-top" title="{% trans %}Go to first post awaiting review{% endtrans %}"><i class="icon-eye-close"></i> {% trans %}First Unreviewed{% endtrans %}</a></li>{% endif %}
  216. {% if thread.replies_reported > 0 and acl.threads.can_mod_posts(thread) %}<li class="reported"><a href="{% url 'thread_reported' slug=thread.slug, thread=thread.id %}" class="tooltip-top" title="{% trans %}Go to first reported post{% endtrans %}"><i class="icon-fire"></i> {% trans %}First Reported{% endtrans %}</a></li>{% endif %}
  217. {% endif %}
  218. </ul>
  219. {% endmacro %}
  220. {% macro checkpoint_user(checkpoint) -%}
  221. {{ ('<a href="' ~ 'user'|url(user=checkpoint.user_id, username=checkpoint.user_slug) ~ '">')|safe ~ (checkpoint.user_name) ~ ("</a>")|safe }}
  222. {%- endmacro %}
  223. {% block javascripts -%}
  224. {{ super() }}
  225. {%- if user.is_authenticated() and acl.threads.can_start_threads(forum) %}
  226. <script type="text/javascript">
  227. $(function () {
  228. $('#thread_form').submit(function() {
  229. if ($('#id_thread_action').val() == 'hard') {
  230. var decision = confirm("{% trans %}Are you sure you want to delete this thread? This action is not reversible!{% endtrans %}");
  231. return decision;
  232. }
  233. return true;
  234. });
  235. $('#posts_form').submit(function() {
  236. if ($('.post-checkbox[]:checked').length == 0) {
  237. alert("{% trans %}You have to select at least one post.{% endtrans %}");
  238. return false;
  239. }
  240. if ($('#id_list_action').val() == 'merge') {
  241. if ($('.post-checkbox[]:checked').length < 2) {
  242. alert("{% trans %}You have to select at least two posts you want to merge.{% endtrans %}");
  243. return false;
  244. }
  245. var decision = confirm("{% trans %}Are you sure you want to merge selected posts? This action is not reversible!{% endtrans %}");
  246. return decision;
  247. }
  248. if ($('#id_list_action').val() == 'hard') {
  249. var decision = confirm("{% trans %}Are you sure you want to delete selected posts? This action is not reversible!{% endtrans %}");
  250. return decision;
  251. }
  252. return true;
  253. });
  254. });
  255. </script>{% endif %}
  256. {%- endblock %}