Mercurial > public > sg101
changeset 1001:c6c3ba5cf6eb
V2 news stories use forums for comments.
author | Brian Neal <bgneal@gmail.com> |
---|---|
date | Thu, 26 Nov 2015 00:27:42 -0600 (2015-11-26) |
parents | abd4c02aefdb |
children | 28d68f97cb26 |
files | forums/tools.py news/admin.py news/fixtures/news_categories.json news/migrations/0006_category_forum_slug.py news/migrations/0007_auto_20151125_2300.py news/models.py news/templatetags/news_tags.py news/utils.py news/views.py sg101/templates/news/story.html sg101/templates/news/story_summary.html |
diffstat | 11 files changed, 265 insertions(+), 137 deletions(-) [+] |
line wrap: on
line diff
--- a/forums/tools.py Tue Nov 24 22:55:18 2015 -0600 +++ b/forums/tools.py Thu Nov 26 00:27:42 2015 -0600 @@ -88,6 +88,7 @@ 'sticky' - if True, the post will be stickied 'locked' - if True, the post will be locked + The new topic is returned. """ try: forum = Forum.objects.get(slug=forum_slug) @@ -110,6 +111,7 @@ notify_new_topic(topic) notify_new_post(post) + return topic def auto_favorite(post):
--- a/news/admin.py Tue Nov 24 22:55:18 2015 -0600 +++ b/news/admin.py Thu Nov 26 00:27:42 2015 -0600 @@ -4,6 +4,7 @@ from django.contrib import admin from django.conf import settings +from forums.tools import create_topic from news.models import PendingStory from news.models import Story from news.models import Category @@ -11,9 +12,14 @@ import ftfy +COMMENT_THREAD_BODY = ("This topic was automatically created to discuss the " + "news story [{title}]({url}).") + + class CategoryAdmin(admin.ModelAdmin): prepopulated_fields = {'slug': ("title", )} - list_display = ['title', 'slug'] + list_display = ['title', 'slug', 'forum_slug'] + list_editable = ['forum_slug'] class PendingStoryAdmin(admin.ModelAdmin): @@ -21,7 +27,7 @@ list_filter = ['date_submitted'] search_fields = ['title', 'short_text', 'long_text'] date_hierarchy = 'date_submitted' - actions = ['approve_story'] + actions = ['approve_stories'] readonly_fields = ['update_date', 'version'] raw_id_fields = ['submitter'] @@ -48,31 +54,43 @@ }), ] - def approve_story(self, request, qs): + def approve_stories(self, request, qs): for pending_story in qs: - story = Story( - title=pending_story.title, - submitter=pending_story.submitter, - category=pending_story.category, - short_text=pending_story.short_text, - long_text=pending_story.long_text, - date_submitted=pending_story.date_submitted, - allow_comments=pending_story.allow_comments, - tags=pending_story.tags, - front_page_expiration=pending_story.front_page_expiration, - priority=pending_story.priority, - meta_description=pending_story.meta_description, - short_markup=pending_story.short_markup, - long_markup=pending_story.long_markup, - admin_content=pending_story.admin_content) - story.save() - pending_story.delete() + self._approve_story(pending_story) count = len(qs) msg = "1 story" if count == 1 else "%d stories" % count self.message_user(request, "%s approved." % msg) - approve_story.short_description = "Approve selected pending stories" + approve_stories.short_description = "Approve selected pending stories" + + def _approve_story(self, pending_story): + story = Story( + title=pending_story.title, + submitter=pending_story.submitter, + category=pending_story.category, + short_text=pending_story.short_text, + long_text=pending_story.long_text, + date_submitted=pending_story.date_submitted, + allow_comments=pending_story.allow_comments, + tags=pending_story.tags, + front_page_expiration=pending_story.front_page_expiration, + priority=pending_story.priority, + meta_description=pending_story.meta_description, + short_markup=pending_story.short_markup, + long_markup=pending_story.long_markup, + admin_content=pending_story.admin_content) + story.save() + pending_story.delete() + + # Create comment thread if configured to do so. + forum_slug = story.category.forum_slug + if story.allow_comments and forum_slug: + post_body = COMMENT_THREAD_BODY.format(title=story.title, + url=story.get_absolute_url()) + topic = create_topic(forum_slug, story.submitter, story.title, post_body) + story.forums_topic = topic + story.save() class Media: js = ['js/news_admin.js'] + settings.GPP_THIRD_PARTY_JS['tiny_mce']
--- a/news/fixtures/news_categories.json Tue Nov 24 22:55:18 2015 -0600 +++ b/news/fixtures/news_categories.json Thu Nov 26 00:27:42 2015 -0600 @@ -1,101 +1,112 @@ [ - { - "pk": 2, - "model": "news.category", - "fields": { - "icon": "news/categories/Articles.png", - "slug": "articles", - "title": "Articles" - } - }, - { - "pk": 3, - "model": "news.category", - "fields": { - "icon": "news/categories/Bands.png", - "slug": "bands", - "title": "Bands" - } - }, - { - "pk": 11, - "model": "news.category", - "fields": { - "icon": "news/categories/Featured_Video.png", - "slug": "featured-videos", - "title": "Featured Videos" - } - }, - { - "pk": 4, - "model": "news.category", - "fields": { - "icon": "news/categories/Gear.png", - "slug": "gear", - "title": "Gear" - } - }, - { - "pk": 5, - "model": "news.category", - "fields": { - "icon": "news/categories/Interviews.png", - "slug": "interviews", - "title": "Interviews" - } - }, - { - "pk": 6, - "model": "news.category", - "fields": { - "icon": "news/categories/Reviews.png", - "slug": "reviews", - "title": "Reviews" - } - }, - { - "pk": 7, - "model": "news.category", - "fields": { - "icon": "news/categories/Show_Announcements.png", - "slug": "show-announcements", - "title": "Show Announcements" - } - }, - { - "pk": 8, - "model": "news.category", - "fields": { - "icon": "news/categories/Show_Reports.png", - "slug": "show-reports", - "title": "Show Reports" - } - }, - { - "pk": 1, - "model": "news.category", - "fields": { - "icon": "news/categories/Site_News.png", - "slug": "site-news", - "title": "Site News" - } - }, - { - "pk": 9, - "model": "news.category", - "fields": { - "icon": "news/categories/Surf_Scene_News.png", - "slug": "surf-scene-news", - "title": "Surf Scene News" - } - }, - { - "pk": 10, - "model": "news.category", - "fields": { - "icon": "news/categories/Guitar_Tabs.png", - "slug": "tablature", - "title": "Tablature" - } - } -] \ No newline at end of file +{ + "fields": { + "forum_slug": "surfguitar101-website", + "icon": "news/categories/Site_News.png", + "slug": "site-news", + "title": "Site News" + }, + "model": "news.category", + "pk": 1 +}, +{ + "fields": { + "forum_slug": "surf-music", + "icon": "news/categories/Articles.png", + "slug": "articles", + "title": "Articles" + }, + "model": "news.category", + "pk": 2 +}, +{ + "fields": { + "forum_slug": "surf-music", + "icon": "news/categories/Bands.png", + "slug": "bands", + "title": "Bands" + }, + "model": "news.category", + "pk": 3 +}, +{ + "fields": { + "forum_slug": "gear", + "icon": "news/categories/Gear.png", + "slug": "gear", + "title": "Gear" + }, + "model": "news.category", + "pk": 4 +}, +{ + "fields": { + "forum_slug": "surf-music", + "icon": "news/categories/Interviews.png", + "slug": "interviews", + "title": "Interviews" + }, + "model": "news.category", + "pk": 5 +}, +{ + "fields": { + "forum_slug": "music-reviews", + "icon": "news/categories/Reviews.png", + "slug": "reviews", + "title": "Reviews" + }, + "model": "news.category", + "pk": 6 +}, +{ + "fields": { + "forum_slug": "gigs", + "icon": "news/categories/Show_Announcements.png", + "slug": "show-announcements", + "title": "Show Announcements" + }, + "model": "news.category", + "pk": 7 +}, +{ + "fields": { + "forum_slug": "surf-music", + "icon": "news/categories/Show_Reports.png", + "slug": "show-reports", + "title": "Show Reports" + }, + "model": "news.category", + "pk": 8 +}, +{ + "fields": { + "forum_slug": "surf-music", + "icon": "news/categories/Surf_Scene_News.png", + "slug": "surf-scene-news", + "title": "Surf Scene News" + }, + "model": "news.category", + "pk": 9 +}, +{ + "fields": { + "forum_slug": "surf-musician", + "icon": "news/categories/Guitar_Tabs.png", + "slug": "tablature", + "title": "Tablature" + }, + "model": "news.category", + "pk": 10 +}, +{ + "fields": { + "forum_slug": "surf-videos", + "icon": "news/categories/Featured_Video.png", + "slug": "featured-videos", + "title": "Featured Videos" + }, + "model": "news.category", + "pk": 11 +} +]
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/news/migrations/0006_category_forum_slug.py Thu Nov 26 00:27:42 2015 -0600 @@ -0,0 +1,20 @@ +# -*- coding: utf-8 -*- +from __future__ import unicode_literals + +from django.db import models, migrations + + +class Migration(migrations.Migration): + + dependencies = [ + ('news', '0005_auto_20151121_1445'), + ] + + operations = [ + migrations.AddField( + model_name='category', + name='forum_slug', + field=models.CharField(default=b'', help_text=b'Identifies the forum to create comment threads in for stories in this category', max_length=80, blank=True), + preserve_default=True, + ), + ]
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/news/migrations/0007_auto_20151125_2300.py Thu Nov 26 00:27:42 2015 -0600 @@ -0,0 +1,28 @@ +# -*- coding: utf-8 -*- +from __future__ import unicode_literals + +from django.db import models, migrations +import django.db.models.deletion + + +class Migration(migrations.Migration): + + dependencies = [ + ('forums', '__first__'), + ('news', '0006_category_forum_slug'), + ] + + operations = [ + migrations.AddField( + model_name='story', + name='forums_topic', + field=models.OneToOneField(null=True, on_delete=django.db.models.deletion.SET_NULL, blank=True, to='forums.Topic', help_text=b'Forum topic used for comments', db_index=False), + preserve_default=True, + ), + migrations.AlterField( + model_name='category', + name='forum_slug', + field=models.CharField(default=b'', help_text=b'Identifies the forum to create comment threads in for stories in this category. If blank, no comment threads will be created for stories in this category.', max_length=80, blank=True), + preserve_default=True, + ), + ]
--- a/news/models.py Tue Nov 24 22:55:18 2015 -0600 +++ b/news/models.py Thu Nov 26 00:27:42 2015 -0600 @@ -19,6 +19,13 @@ title = models.CharField(max_length=64) slug = models.SlugField(max_length=64) icon = models.ImageField(upload_to='news/categories/', blank=True) + forum_slug = models.CharField( + max_length=80, + default='', + blank=True, + help_text=("Identifies the forum to create comment threads in for " + "stories in this category. If blank, no comment threads " + "will be created for stories in this category.")) def __unicode__(self): return self.title @@ -85,6 +92,10 @@ class Story(StoryBase): """Model for news stories""" + forums_topic = models.OneToOneField('forums.Topic', blank=True, null=True, + on_delete=models.SET_NULL, + db_index=False, + help_text="Forum topic used for comments") @models.permalink def get_absolute_url(self): @@ -99,10 +110,17 @@ verbose_name_plural = 'news stories' def can_comment_on(self): + # Only used for version 0 stories now = datetime.datetime.now() delta = now - self.date_submitted return self.allow_comments and delta.days < 30 + def forums_comment_count(self): + """Returns number of comments for V2 news stories.""" + if self.forums_topic: + return max(0, self.forums_topic.post_count - 1) + return 0 + def search_title(self): return self.title
--- a/news/templatetags/news_tags.py Tue Nov 24 22:55:18 2015 -0600 +++ b/news/templatetags/news_tags.py Thu Nov 26 00:27:42 2015 -0600 @@ -17,9 +17,10 @@ def current_news(): # Defer the tags field because we are going to get all the # tags out in 1 query later... - stories = Story.objects.defer('tags').exclude( - front_page_expiration__lt=datetime.date.today()).order_by( - '-priority', '-date_submitted')[:10] + stories = Story.objects.defer('tags')\ + .exclude(front_page_expiration__lt=datetime.date.today())\ + .select_related('forums_topic')\ + .order_by('-priority', '-date_submitted')[:10] attach_extra_attrs(stories)
--- a/news/utils.py Tue Nov 24 22:55:18 2015 -0600 +++ b/news/utils.py Thu Nov 26 00:27:42 2015 -0600 @@ -28,17 +28,25 @@ for story in stories_dict.values(): story.tag_list = [] - story.comment_count = 0 + if story.version == 0: + story.comment_count = 0 + elif story.forums_topic: + # for convenience/consistency with old models... + story.comment_count = story.forums_comment_count() # attach tags for item in tagged_items: stories_dict[item.object_id].tag_list.append(item.tag.name) - # Now get all the comment counts out in one fell swoop + # Now get all the comment counts out in one fell swoop. This is only needed + # for older news stories... - story_ids = Comment.objects.filter(content_type=ct, - object_id__in=story_ids).values_list('object_id', flat=True) + story_ids = [pk for pk in story_ids if stories_dict[pk].version == 0] - # compute comment_count - for story_id in story_ids: - stories_dict[story_id].comment_count += 1 + if story_ids: + story_ids = Comment.objects.filter(content_type=ct, + object_id__in=story_ids).values_list('object_id', flat=True) + + # compute comment_count + for story_id in story_ids: + stories_dict[story_id].comment_count += 1
--- a/news/views.py Tue Nov 24 22:55:18 2015 -0600 +++ b/news/views.py Thu Nov 26 00:27:42 2015 -0600 @@ -42,7 +42,8 @@ def index(request): # Defer the tags field because we are going to get all the # tags out in 1 query later... - stories = Story.objects.all().defer('tags').select_related() + stories = Story.objects.all().defer('tags').select_related( + 'submitter', 'category', 'forums_topic') paginator = create_paginator(stories) page = get_page(request.GET) @@ -75,7 +76,8 @@ def archive(request, year, month): stories = Story.objects.defer('tags').filter(date_submitted__year=year, - date_submitted__month=month).select_related() + date_submitted__month=month).select_related( + 'submitter', 'category', 'forums_topic') paginator = create_paginator(stories) page = get_page(request.GET) try: @@ -110,7 +112,8 @@ def category(request, slug): category = get_object_or_404(Category, slug=slug) - stories = Story.objects.defer('tags').filter(category=category).select_related() + stories = Story.objects.defer('tags').filter(category=category).select_related( + 'submitter', 'category', 'forums_topic') paginator = create_paginator(stories) page = get_page(request.GET) try: @@ -129,7 +132,8 @@ ####################################################################### def story(request, story_id): - story = get_object_or_404(Story, pk=story_id) + story = get_object_or_404(Story.objects.select_related( + 'submitter', 'category', 'forums_topic'), pk=story_id) return render_to_response('news/story.html', { 'story': story, }, @@ -174,7 +178,8 @@ def tag(request, tag_name): tag = get_object_or_404(Tag, name=tag_name) stories = TaggedItem.objects.get_by_model( - Story.objects.defer('tags').select_related(), tag) + Story.objects.defer('tags').select_related( + 'submitter', 'category', 'forums_topic'), tag) paginator = create_paginator(stories) page = get_page(request.GET) try:
--- a/sg101/templates/news/story.html Tue Nov 24 22:55:18 2015 -0600 +++ b/sg101/templates/news/story.html Thu Nov 26 00:27:42 2015 -0600 @@ -55,6 +55,8 @@ {% endif %} {% social_sharing story.title story.get_absolute_url %} </div> + +{% if story.version == 0 %} {% get_comment_count for story as comment_count %} <p>This story has <span id="comment-count">{{ comment_count }}</span> comment{{ comment_count|pluralize }}.</p> <hr /> @@ -66,5 +68,12 @@ <p>Comments are closed for this story. If you'd like to share your thoughts on this story with the site staff, you can <a href="{% url 'contact-form' %}">contact us directly</a>.</p> {% endif %} +{% else %} + {% if story.forums_topic %} + <a href="{{ story.forums_topic.get_absolute_url }}">{{ story.forums_comment_count }} comment{{ story.forums_comment_count|pluralize }}</a> + {% else %} + <p><em>Comments are disabled.</em></p> + {% endif %} +{% endif %} </div> {% endblock %}
--- a/sg101/templates/news/story_summary.html Tue Nov 24 22:55:18 2015 -0600 +++ b/sg101/templates/news/story_summary.html Thu Nov 26 00:27:42 2015 -0600 @@ -30,7 +30,15 @@ <p> Category: <a href="{% url 'news-category' slug=story.category.slug %}">{{ story.category.title }}</a> <img src="{{ STATIC_URL }}icons/comments.png" alt="Comments" title="Comments" /> -<a href="{{ story.get_absolute_url }}">{{ story.comment_count }} comment{{ story.comment_count|pluralize }}</a> +{% if story.version == 0 %} + <a href="{{ story.get_absolute_url }}">{{ story.comment_count }} comment{{ story.comment_count|pluralize }}</a> +{% else %} + {% if story.forums_topic %} + <a href="{{ story.forums_topic.get_absolute_url }}">{{ story.comment_count }} comment{{ story.comment_count|pluralize }}</a> + {% else %} + <em>Comments are disabled</em> + {% endif %} +{% endif %} <a href="{{ story.get_absolute_url }}"><img src="{{ STATIC_URL }}icons/link.png" alt="Permalink" title="Permalink" /></a> {% if user.is_authenticated %} <a href="{% url 'news.views.email_story' story.id %}"><img src="{{ STATIC_URL }}icons/email_go.png"