Statistics
| Branch: | Tag: | Revision:

root / cloudcmsfaq / models.py @ f1ff020b

History | View | Annotate | Download (6.9 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
    Question category.
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('cloudcmsfaq_category_archive', 'cloudcmsfaq.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 QuestionManager(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_faq_page():
120
    """
121
    Returns Page model that has been associated with faq application
122
    """
123
    return get_app_page(Page, "cloudcmsfaq")
124

    
125
class Question(Base):
126
    """
127
    Question/answer 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='faqs', verbose_name=_('author'))
135
    language = models.CharField(max_length=255, choices=settings.LANGUAGES)
136

    
137
    application = models.ManyToManyField(Application,
138
            related_name="faqs",
139
            verbose_name=_('application'))
140

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

    
145
    category = models.ForeignKey(Category, verbose_name=_('category'),
146
        related_name='faqs', null=False, blank=False)
147

    
148
    objects = QuestionManager()
149

    
150
    class Meta:
151
        get_latest_by = 'published_on'
152
        ordering = ['-published_on']
153
        verbose_name = _('faq')
154
        verbose_name_plural = _('faqs')
155

    
156
    def __unicode__(self):
157
        return self.title
158

    
159
    def get_absolute_url(self):
160
        try:
161
            r = reverse('cloudcmsfaq_question_detail', 'cloudcmsfaq.urls', (),
162
                    {
163
                     'category': self.category.translation.slug,
164
                     'slug': self.slug,
165
                    })
166
        except Exception, e:
167
            pass
168

    
169
        # ugly hack to fix proper application reverse url
170
        FAQ_URL = ""
171
        try:
172
            FAQ_URL = get_faq_page().get_navigation_url()
173
        except Exception, e:
174
            pass
175

    
176
        if r.startswith(FAQ_URL):
177
            return r
178
        else:
179
            return FAQ_URL + r.lstrip('/')
180

    
181
    def back_url(self):
182
        return get_faq_page().get_navigation_url()
183

    
184

    
185
# Feincms navigation extension
186
class FaqCategoriesNavigationExtension(NavigationExtension):
187
    """
188
    Navigation extension for FeinCMS which lists all categories that user
189
    wants to include in global site navigation.
190
    """
191

    
192
    name = _('faq categories')
193

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