Statistics
| Branch: | Tag: | Revision:

root / cloudcmsblog / models.py @ 913e2640

History | View | Annotate | Download (7.8 kB)

1
# Copyright 2012 GRNET S.A. All rights reserved.
2
#
3
# Redistribution and use in source and binary forms, with or
4
# without modification, are permitted provided that the following
5
# conditions are met:
6
#
7
#   1. Redistributions of source code must retain the above
8
#      copyright notice, this list of conditions and the following
9
#      disclaimer.
10
#
11
#   2. Redistributions in binary form must reproduce the above
12
#      copyright notice, this list of conditions and the following
13
#      disclaimer in the documentation and/or other materials
14
#      provided with the distribution.
15
#
16
# THIS SOFTWARE IS PROVIDED BY GRNET S.A. ``AS IS'' AND ANY EXPRESS
17
# OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
18
# WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
19
# PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL GRNET S.A OR
20
# CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
21
# SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
22
# LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF
23
# USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
24
# AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
25
# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
26
# ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
27
# POSSIBILITY OF SUCH DAMAGE.
28
#
29
# The views and conclusions contained in the software and
30
# documentation are those of the authors and should not be
31
# interpreted as representing official policies, either expressed
32
# or implied, of GRNET S.A.
33

    
34

    
35
from datetime import datetime
36

    
37
from django.db import models
38
from django.conf import settings
39
from django.contrib.auth.models import User
40
from django.utils.translation import ugettext_lazy as _, ugettext, ungettext
41
from django.template.loader import render_to_string
42
from django.core import urlresolvers
43

    
44
from feincms import translations
45
from feincms.models import Base
46
from feincms.module.page.models import Page
47
from feincms.content.richtext.models import RichTextContent
48
from feincms.content.section.models import SectionContent
49
from feincms.content.application.models import reverse
50
from feincms.module.medialibrary.fields import MediaFileForeignKey
51
from feincms.module.medialibrary.models import MediaFile
52
from feincms.module.page.extensions.navigation import NavigationExtension
53
from feincms.module.page.extensions.navigation import PagePretender
54
from feincms.content.application.models import ApplicationContent
55

    
56
from cloudcms.models import Application
57
from cloudcms.cms_utils import get_app_page
58

    
59
# monkeypatch django reverse (feincms 1.5+ solves this issue)
60
urlresolvers.reverse = reverse
61

    
62
class Category(models.Model, translations.TranslatedObjectMixin):
63
    """
64
    Blog entry category. Each blog post may belong to multiple categories.
65
    """
66

    
67
    ordering = models.SmallIntegerField(_('ordering'), default=0)
68
    display_on_menu = models.BooleanField(default=False)
69

    
70
    class Meta:
71
        verbose_name = _('category')
72
        verbose_name_plural = _('categories')
73
        ordering = ['-ordering',]
74

    
75
    objects = translations.TranslatedObjectManager()
76

    
77
    def __unicode__(self):
78
        trans = translations.TranslatedObjectMixin.__unicode__(self)
79
        return trans or _('Unnamed category')
80

    
81

    
82
class CategoryTranslation(translations.Translation(Category)):
83
    """
84
    Category translation
85
    """
86
    title = models.CharField(_('category title'), max_length=100)
87
    slug = models.SlugField(_('slug'), unique=True)
88
    description = models.CharField(_('description'), max_length=250, blank=True)
89

    
90
    class Meta:
91
        verbose_name = _('category translation')
92
        verbose_name_plural = _('category translations')
93
        ordering = ['title']
94

    
95
    def __unicode__(self):
96
        return self.title
97

    
98
    def get_absolute_url(self):
99
        return reverse('cloudcmsblog_entries_archive', 'cloudcmsblog.urls', (), {
100
            'category': self.slug,
101
        })
102

    
103
    def save(self, *args, **kwargs):
104
        if not self.slug:
105
            self.slug = slugify(self.title)
106

    
107
        super(CategoryTranslation, self).save(*args, **kwargs)
108

    
109

    
110
class EntryManager(models.Manager):
111

    
112
    def active(self):
113
        return self.filter(is_active=True)
114

    
115
    def latest(self, limit=3):
116
        return self.filter()[:limit]
117

    
118

    
119
def get_blog_page():
120
    """
121
    Returns Page model that has been associated with blog application
122
    """
123
    return get_app_page(Page, "cloudcmsblog")
124

    
125
class Entry(Base):
126
    """
127
    Blog post entry
128
    """
129
    is_active = models.BooleanField(_('is active'), default=True)
130
    is_featured = models.BooleanField(_('is featured'), default=False)
131

    
132
    title = models.CharField(_('title'), max_length=100)
133
    slug = models.SlugField(_('slug'), max_length=100, unique_for_date='published_on')
134
    author = models.ForeignKey(User, related_name='blogentries', verbose_name=_('author'))
135
    language = models.CharField(max_length=255, choices=settings.LANGUAGES)
136

    
137
    intro_text = models.TextField(max_length=255,
138
            help_text="Displayed in list views", blank=True)
139
    image = MediaFileForeignKey(MediaFile, null=True, blank=True)
140
    application = models.ManyToManyField(Application,
141
            related_name="blogentries",
142
            verbose_name=_('application'))
143

    
144
    published_on = models.DateTimeField(_('published on'), blank=True, null=True, default=datetime.now,
145
        help_text=_('Will be filled in automatically when entry gets published.'))
146
    last_changed = models.DateTimeField(_('last change'), auto_now=True, editable=False)
147

    
148
    categories = models.ManyToManyField(Category, verbose_name=_('categories'),
149
        related_name='blogentries', null=True, blank=True)
150

    
151
    objects = EntryManager()
152

    
153
    class Meta:
154
        get_latest_by = 'published_on'
155
        ordering = ['-published_on']
156
        verbose_name = _('entry')
157
        verbose_name_plural = _('entries')
158

    
159
    def __unicode__(self):
160
        return self.title
161

    
162
    def get_absolute_url(self):
163
        try:
164
            r = reverse('cloudcmsblog_entry_detail', 'cloudcmsblog.urls', (),
165
                    {
166
                    'year': self.published_on.strftime('%Y'),
167
                    'month': self.published_on.strftime('%m'),
168
                    'day': self.published_on.strftime('%d'),
169
                    'slug': self.slug,
170
                    })
171
        except Exception,e:
172
            pass
173

    
174
        # ugly hack to fix proper application reverse url
175
        BLOG_URL = ""
176
        try:
177
            BLOG_URL = get_blog_page().get_navigation_url()
178
        except Exception, e:
179
            print e
180

    
181
        if r.startswith(BLOG_URL):
182
            return r
183
        else:
184
            return BLOG_URL + r.lstrip('/')
185

    
186
    def back_url(self):
187
        return get_blog_page().get_navigation_url()
188

    
189

    
190
# Feincms navigation extension
191
class BlogCategoriesNavigationExtension(NavigationExtension):
192
    """
193
    Navigation extension for FeinCMS which lists all categories that user
194
    wants to include in global site navigation.
195
    """
196

    
197
    name = _('blog categories')
198

    
199
    def children(self, page, **kwargs):
200
        for category in Category.objects.filter(display_on_menu=True):
201
            url='%scategory/%s/' % (page.get_absolute_url(), category.translation.slug)
202
            yield PagePretender(
203
                title=category.translation.title,
204
                tree_id=page.tree_id,
205
                url=url,
206
                lft=0,
207
                rght=0,
208
                slug=category.translation.slug,
209
            )
210

    
211

    
212
# Feincms content abstract models
213
class LatestEntries(models.Model):
214
    title = models.CharField(max_length=255)
215
    limit = models.PositiveIntegerField(default=3)
216
    display_text = models.BooleanField(default=False)
217

    
218
    class Meta:
219
        abstract = True
220
        verbose_name = _('Latest blog entries')
221
        verbose_name_plural = _('Latest blog entries')
222

    
223
    def render(self, **kwargs):
224
        return render_to_string(['content/latest_blog.html'], {'posts':
225
            Entry.objects.latest(self.limit), 'content': self})
226