summaryrefslogtreecommitdiff
path: root/gstudio/admin/objecttype.py
diff options
context:
space:
mode:
authorgnowgi <nagarjun@gnowledge.org>2012-03-15 16:19:20 +0530
committergnowgi <nagarjun@gnowledge.org>2012-03-15 16:19:20 +0530
commit7a4f561e851fdc7246d804c3abb6748b8a4199a6 (patch)
treed2afc3463fd49625a9be482012f5c3bfcf7c42b9 /gstudio/admin/objecttype.py
downloadgnowsys-7a4f561e851fdc7246d804c3abb6748b8a4199a6.tar.gz
master trunk of gnowsys-studio
Diffstat (limited to 'gstudio/admin/objecttype.py')
-rw-r--r--gstudio/admin/objecttype.py345
1 files changed, 345 insertions, 0 deletions
diff --git a/gstudio/admin/objecttype.py b/gstudio/admin/objecttype.py
new file mode 100644
index 0000000..22ef205
--- /dev/null
+++ b/gstudio/admin/objecttype.py
@@ -0,0 +1,345 @@
+"""ObjecttypeAdmin for Gstudio"""
+from datetime import datetime
+from django.forms import Media
+from django.contrib import admin
+from django.contrib.auth.models import User
+from django.utils.html import strip_tags
+from django.utils.text import truncate_words
+from django.conf.urls.defaults import url
+from django.conf.urls.defaults import patterns
+from django.conf import settings as project_settings
+from django.utils.translation import ugettext_lazy as _
+from django.core.urlresolvers import reverse, NoReverseMatch
+
+from tagging.models import Tag
+
+import reversion
+from gstudio import settings
+from gstudio.managers import HIDDEN
+from gstudio.managers import PUBLISHED
+from gstudio.ping import DirectoryPinger
+from gstudio.admin.forms import ObjecttypeAdminForm
+
+
+
+
+class ObjecttypeAdmin(reversion.VersionAdmin):
+ """Admin for Objecttype model"""
+ form = ObjecttypeAdminForm
+ date_hierarchy = 'creation_date'
+ fieldsets = ((_('Neighbourhood'), {'fields': ('title','altnames','plural','parent','slug',
+ 'metatypes','tags','image', 'status','content')}),
+ (_('Dependency'), {'fields': ('prior_nodes', 'posterior_nodes',),
+ 'classes': ('collapse', 'collapse-closed')}),
+ (_('Options'), {'fields': ('featured', 'excerpt', 'template',
+ 'authors',
+ 'creation_date',
+ 'start_publication',
+ 'end_publication'),
+ 'classes': ('collapse', 'collapse-closed')}),
+ (_('Privacy'), {'fields': ('password', 'login_required',),
+ 'classes': ('collapse', 'collapse-closed')}),
+ (_('Discussion'), {'fields': ('comment_enabled',
+ 'pingback_enabled','sites'),'classes': ('collapse', 'collapse-closed')}),
+
+ )
+ list_filter = ('parent','metatypes', 'authors', 'status', 'featured',
+ 'login_required', 'comment_enabled', 'pingback_enabled',
+ 'creation_date', 'start_publication',
+ 'end_publication', 'sites')
+ list_display = ('get_title', 'get_authors', 'get_metatypes',
+ 'get_tags', 'get_sites',
+ 'get_comments_are_open', 'pingback_enabled',
+ 'get_is_actual', 'get_is_visible', 'get_link',
+ 'get_short_url', 'creation_date')
+ radio_fields = {'template': admin.VERTICAL}
+ filter_horizontal = ('metatypes', 'authors' )
+ prepopulated_fields = {'slug': ('title', )}
+ search_fields = ('title', 'excerpt', 'content', 'tags')
+ actions = ['make_mine', 'make_published', 'make_hidden',
+ 'close_comments', 'close_pingbacks',
+ 'ping_directories', 'make_tweet', 'put_on_top']
+ actions_on_top = True
+ actions_on_bottom = True
+
+ def __init__(self, model, admin_site):
+ self.form.admin_site = admin_site
+ super(ObjecttypeAdmin, self).__init__(model, admin_site)
+
+ # Custom Display
+ def get_title(self, nodetype):
+ """Return the title with word count and number of comments"""
+ title = _('%(title)s (%(word_count)i words)') % \
+ {'title': nodetype.title, 'word_count': nodetype.word_count}
+ comments = nodetype.comments.count()
+ if comments:
+ return _('%(title)s (%(comments)i comments)') % \
+ {'title': title, 'comments': comments}
+ return title
+ get_title.short_description = _('title')
+
+ def get_authors(self, nodetype):
+ """Return the authors in HTML"""
+ try:
+ authors = ['<a href="%s" target="blank">%s</a>' %
+ (reverse('gstudio_author_detail',
+ args=[author.username]),
+ author.username) for author in nodetype.authors.all()]
+ except NoReverseMatch:
+ authors = [author.username for author in nodetype.authors.all()]
+ return ', '.join(authors)
+ get_authors.allow_tags = True
+ get_authors.short_description = _('author(s)')
+
+ def get_metatypes(self, nodetype):
+ """Return the metatypes linked in HTML"""
+ try:
+ metatypes = ['<a href="%s" target="blank">%s</a>' %
+ (metatype.get_absolute_url(), metatype.title)
+ for metatype in nodetype.metatypes.all()]
+ except NoReverseMatch:
+ metatypes = [metatype.title for metatype in
+ nodetype.metatypes.all()]
+ return ', '.join(metatypes)
+ get_metatypes.allow_tags = True
+ get_metatypes.short_description = _('metatype(s)')
+
+ def get_tags(self, nodetype):
+ """Return the tags linked in HTML"""
+ try:
+ return ', '.join(['<a href="%s" target="blank">%s</a>' %
+ (reverse('gstudio_tag_detail',
+ args=[tag.name]), tag.name)
+ for tag in Tag.objects.get_for_object(nodetype)])
+ except NoReverseMatch:
+ return nodetype.tags
+ get_tags.allow_tags = True
+ get_tags.short_description = _('tag(s)')
+
+ def get_sites(self, nodetype):
+ """Return the sites linked in HTML"""
+ return ', '.join(
+ ['<a href="http://%(domain)s" target="blank">%(name)s</a>' %
+ site.__dict__ for site in nodetype.sites.all()])
+ get_sites.allow_tags = True
+ get_sites.short_description = _('site(s)')
+
+ def get_comments_are_open(self, nodetype):
+ """Admin wrapper for nodetype.comments_are_open"""
+ return nodetype.comments_are_open
+ get_comments_are_open.boolean = True
+ get_comments_are_open.short_description = _('comment enabled')
+
+ def get_is_actual(self, nodetype):
+ """Admin wrapper for nodetype.is_actual"""
+ return nodetype.is_actual
+ get_is_actual.boolean = True
+ get_is_actual.short_description = _('is actual')
+
+ def get_is_visible(self, nodetype):
+ """Admin wrapper for nodetype.is_visible"""
+ return nodetype.is_visible
+ get_is_visible.boolean = True
+ get_is_visible.short_description = _('is visible')
+
+ def get_link(self, nodetype):
+ """Return a formated link to the nodetype"""
+ return u'<a href="%s" target="blank">%s</a>' % (
+ nodetype.get_absolute_url(), _('View'))
+ get_link.allow_tags = True
+ get_link.short_description = _('View on site')
+
+ def get_short_url(self, nodetype):
+ """Return the short url in HTML"""
+ short_url = nodetype.short_url
+ if not short_url:
+ return _('Unavailable')
+ return '<a href="%(url)s" target="blank">%(url)s</a>' % \
+ {'url': short_url}
+ get_short_url.allow_tags = True
+ get_short_url.short_description = _('short url')
+
+ # Custom Methods
+ def save_model(self, request, nodetype, form, change):
+ """Save the authors, update time, make an excerpt"""
+ if not form.cleaned_data.get('excerpt') and nodetype.status == PUBLISHED:
+ nodetype.excerpt = truncate_words(strip_tags(nodetype.content), 50)
+
+ if nodetype.pk and not request.user.has_perm('gstudio.can_change_author'):
+ form.cleaned_data['authors'] = nodetype.authors.all()
+
+ if not form.cleaned_data.get('authors'):
+ form.cleaned_data['authors'].append(request.user)
+
+ nodetype.last_update = datetime.now()
+ nodetype.save()
+
+ def queryset(self, request):
+ """Make special filtering by user permissions"""
+ queryset = super(ObjecttypeAdmin, self).queryset(request)
+ if request.user.has_perm('gstudio.can_view_all'):
+ return queryset
+ return request.user.nodetypes.all()
+
+ def formfield_for_manytomany(self, db_field, request, **kwargs):
+ """Filters the disposable authors"""
+ if db_field.name == 'authors':
+ if request.user.has_perm('gstudio.can_change_author'):
+ kwargs['queryset'] = User.objects.filter(is_staff=True)
+ else:
+ kwargs['queryset'] = User.objects.filter(pk=request.user.pk)
+
+ return super(ObjecttypeAdmin, self).formfield_for_manytomany(
+ db_field, request, **kwargs)
+
+ def get_actions(self, request):
+ """Define user actions by permissions"""
+ actions = super(ObjecttypeAdmin, self).get_actions(request)
+ if not request.user.has_perm('gstudio.can_change_author') \
+ or not request.user.has_perm('gstudio.can_view_all'):
+ del actions['make_mine']
+ if not settings.PING_DIRECTORIES:
+ del actions['ping_directories']
+ if not settings.USE_TWITTER:
+ del actions['make_tweet']
+
+ return actions
+
+ # Custom Actions
+ def make_mine(self, request, queryset):
+ """Set the nodetypes to the user"""
+ for nodetype in queryset:
+ if request.user not in nodetype.authors.all():
+ nodetype.authors.add(request.user)
+ self.message_user(
+ request, _('The selected nodetypes now belong to you.'))
+ make_mine.short_description = _('Set the nodetypes to the user')
+
+ def make_published(self, request, queryset):
+ """Set nodetypes selected as published"""
+ queryset.update(status=PUBLISHED)
+ self.ping_directories(request, queryset, messages=False)
+ self.message_user(
+ request, _('The selected nodetypes are now marked as published.'))
+ make_published.short_description = _('Set nodetypes selected as published')
+
+ def make_hidden(self, request, queryset):
+ """Set nodetypes selected as hidden"""
+ queryset.update(status=HIDDEN)
+ self.message_user(
+ request, _('The selected nodetypes are now marked as hidden.'))
+ make_hidden.short_description = _('Set nodetypes selected as hidden')
+
+ def make_tweet(self, request, queryset):
+ """Post an update on Twitter"""
+ import tweepy
+ auth = tweepy.OAuthHandler(settings.TWITTER_CONSUMER_KEY,
+ settings.TWITTER_CONSUMER_SECRET)
+ auth.set_access_token(settings.TWITTER_ACCESS_KEY,
+ settings.TWITTER_ACCESS_SECRET)
+ api = tweepy.API(auth)
+ for nodetype in queryset:
+ short_url = nodetype.short_url
+ message = '%s %s' % (nodetype.title[:139 - len(short_url)], short_url)
+ api.update_status(message)
+ self.message_user(
+ request, _('The selected nodetypes have been tweeted.'))
+ make_tweet.short_description = _('Tweet nodetypes selected')
+
+ def close_comments(self, request, queryset):
+ """Close the comments for selected nodetypes"""
+ queryset.update(comment_enabled=False)
+ self.message_user(
+ request, _('Comments are now closed for selected nodetypes.'))
+ close_comments.short_description = _('Close the comments for '\
+ 'selected nodetypes')
+
+ def close_pingbacks(self, request, queryset):
+ """Close the pingbacks for selected nodetypes"""
+ queryset.update(pingback_enabled=False)
+ self.message_user(
+ request, _('Linkbacks are now closed for selected nodetypes.'))
+ close_pingbacks.short_description = _(
+ 'Close the linkbacks for selected nodetypes')
+
+ def put_on_top(self, request, queryset):
+ """Put the selected nodetypes on top at the current date"""
+ queryset.update(creation_date=datetime.now())
+ self.ping_directories(request, queryset, messages=False)
+ self.message_user(request, _(
+ 'The selected nodetypes are now set at the current date.'))
+ put_on_top.short_description = _(
+ 'Put the selected nodetypes on top at the current date')
+
+ def ping_directories(self, request, queryset, messages=True):
+ """Ping Directories for selected nodetypes"""
+ for directory in settings.PING_DIRECTORIES:
+ pinger = DirectoryPinger(directory, queryset)
+ pinger.join()
+ if messages:
+ success = 0
+ for result in pinger.results:
+ if not result.get('flerror', True):
+ success += 1
+ else:
+ self.message_user(request,
+ '%s : %s' % (directory,
+ result['message']))
+ if success:
+ self.message_user(
+ request,
+ _('%(directory)s directory succesfully ' \
+ 'pinged %(success)d nodetypes.') %
+ {'directory': directory, 'success': success})
+ ping_directories.short_description = _(
+ 'Ping Directories for selected nodetypes')
+
+ def get_urls(self):
+ nodetype_admin_urls = super(ObjecttypeAdmin, self).get_urls()
+ urls = patterns(
+ 'django.views.generic.simple',
+ url(r'^autocomplete_tags/$', 'direct_to_template',
+ {'template': 'admin/gstudio/nodetype/autocomplete_tags.js',
+ 'mimetype': 'application/javascript'},
+ name='gstudio_nodetype_autocomplete_tags'),
+ url(r'^wymeditor/$', 'direct_to_template',
+ {'template': 'admin/gstudio/nodetype/wymeditor.js',
+ 'mimetype': 'application/javascript'},
+ name='gstudio_nodetype_wymeditor'),
+ url(r'^markitup/$', 'direct_to_template',
+ {'template': 'admin/gstudio/nodetype/markitup.js',
+ 'mimetype': 'application/javascript'},
+ name='gstudio_nodetype_markitup'),)
+ return urls + nodetype_admin_urls
+
+ def _media(self):
+ STATIC_URL = '%sgstudio/' % project_settings.STATIC_URL
+ media = super(ObjecttypeAdmin, self).media + Media(
+ css={'all': ('%scss/jquery.autocomplete.css' % STATIC_URL,)},
+ js=('%sjs/jquery.js' % STATIC_URL,
+ '%sjs/jquery.bgiframe.js' % STATIC_URL,
+ '%sjs/jquery.autocomplete.js' % STATIC_URL,
+ reverse('admin:gstudio_nodetype_autocomplete_tags'),))
+
+ if settings.WYSIWYG == 'wymeditor':
+ media += Media(
+ js=('%sjs/wymeditor/jquery.wymeditor.pack.js' % STATIC_URL,
+ '%sjs/wymeditor/plugins/hovertools/'
+ 'jquery.wymeditor.hovertools.js' % STATIC_URL,
+ reverse('admin:gstudio_nodetype_wymeditor')))
+ elif settings.WYSIWYG == 'tinymce':
+ from tinymce.widgets import TinyMCE
+ media += TinyMCE().media + Media(
+ js=(reverse('tinymce-js', args=('admin/gstudio/nodetype',)),))
+ elif settings.WYSIWYG == 'markitup':
+ media += Media(
+ js=('%sjs/markitup/jquery.markitup.js' % STATIC_URL,
+ '%sjs/markitup/sets/%s/set.js' % (
+ STATIC_URL, settings.MARKUP_LANGUAGE),
+ reverse('admin:gstudio_nodetype_markitup')),
+ css={'all': (
+ '%sjs/markitup/skins/django/style.css' % STATIC_URL,
+ '%sjs/markitup/sets/%s/style.css' % (
+ STATIC_URL, settings.MARKUP_LANGUAGE))})
+ return media
+ media = property(_media)