PageRenderTime 41ms CodeModel.GetById 30ms RepoModel.GetById 0ms app.codeStats 1ms

/cms/templatetags/cms_tags.py

https://github.com/brente/django-cms
Python | 418 lines | 405 code | 7 blank | 6 comment | 12 complexity | 197d744cb7e4c966f640603c85170768 MD5 | raw file
  1. # -*- coding: utf-8 -*-
  2. from classytags.arguments import Argument, MultiValueArgument
  3. from classytags.core import Options, Tag
  4. from classytags.helpers import InclusionTag
  5. from classytags.parser import Parser
  6. from cms.models import Page
  7. from cms.plugin_rendering import render_plugins, render_placeholder
  8. from cms.plugins.utils import get_plugins
  9. from cms.utils import get_language_from_request
  10. from cms.utils.moderator import get_cmsplugin_queryset, get_page_queryset
  11. from cms.utils.placeholder import validate_placeholder_name
  12. from django import template
  13. from django.conf import settings
  14. from django.contrib.sites.models import Site
  15. from django.core.cache import cache
  16. from django.core.mail import mail_managers
  17. from django.utils.safestring import mark_safe
  18. from django.utils.translation import ugettext_lazy as _
  19. from itertools import chain
  20. import re
  21. register = template.Library()
  22. def get_site_id(site):
  23. if site:
  24. if isinstance(site, Site):
  25. site_id = site.id
  26. elif isinstance(site, int) or (isinstance(site, basestring) and site.isdigit()):
  27. site_id = int(site)
  28. else:
  29. site_id = settings.SITE_ID
  30. else:
  31. site_id = settings.SITE_ID
  32. return site_id
  33. def has_permission(page, request):
  34. return page.has_change_permission(request)
  35. register.filter(has_permission)
  36. CLEAN_KEY_PATTERN = re.compile(r'[^a-zA-Z0-9_-]')
  37. def _clean_key(key):
  38. return CLEAN_KEY_PATTERN.sub('-', key)
  39. def _get_cache_key(name, page_lookup, lang, site_id):
  40. if isinstance(page_lookup, Page):
  41. page_key = str(page_lookup.pk)
  42. else:
  43. page_key = str(page_lookup)
  44. page_key = _clean_key(page_key)
  45. return name+'__page_lookup:'+page_key+'_site:'+str(site_id)+'_lang:'+str(lang)
  46. def _get_page_by_untyped_arg(page_lookup, request, site_id):
  47. """
  48. The `page_lookup` argument can be of any of the following types:
  49. - Integer: interpreted as `pk` of the desired page
  50. - String: interpreted as `reverse_id` of the desired page
  51. - `dict`: a dictionary containing keyword arguments to find the desired page
  52. (for instance: `{'pk': 1}`)
  53. - `Page`: you can also pass a Page object directly, in which case there will be no database lookup.
  54. - `None`: the current page will be used
  55. """
  56. if page_lookup is None:
  57. return request.current_page
  58. if isinstance(page_lookup, Page):
  59. return page_lookup
  60. if isinstance(page_lookup, basestring):
  61. page_lookup = {'reverse_id': page_lookup}
  62. elif isinstance(page_lookup, (int, long)):
  63. page_lookup = {'pk': page_lookup}
  64. elif not isinstance(page_lookup, dict):
  65. raise TypeError('The page_lookup argument can be either a Dictionary, Integer, Page, or String.')
  66. page_lookup.update({'site': site_id})
  67. try:
  68. return get_page_queryset(request).get(**page_lookup)
  69. except Page.DoesNotExist:
  70. site = Site.objects.get_current()
  71. subject = _('Page not found on %(domain)s') % {'domain':site.domain}
  72. body = _("A template tag couldn't find the page with lookup arguments `%(page_lookup)s\n`. "
  73. "The URL of the request was: http://%(host)s%(path)s") \
  74. % {'page_lookup': repr(page_lookup), 'host': site.domain, 'path': request.path}
  75. if settings.DEBUG:
  76. raise Page.DoesNotExist(body)
  77. else:
  78. if settings.SEND_BROKEN_LINK_EMAILS:
  79. mail_managers(subject, body, fail_silently=True)
  80. return None
  81. class PageUrl(InclusionTag):
  82. template = 'cms/content.html'
  83. name = 'page_url'
  84. options = Options(
  85. Argument('page_lookup'),
  86. Argument('lang', required=False, default=None),
  87. Argument('site', required=False, default=None),
  88. )
  89. def get_context(self, context, page_lookup, lang, site):
  90. site_id = get_site_id(site)
  91. request = context.get('request', False)
  92. if not request:
  93. return {'content': ''}
  94. if request.current_page == "dummy":
  95. return {'content': ''}
  96. if lang is None:
  97. lang = get_language_from_request(request)
  98. cache_key = _get_cache_key('page_url', page_lookup, lang, site_id)+'_type:absolute_url'
  99. url = cache.get(cache_key)
  100. if not url:
  101. page = _get_page_by_untyped_arg(page_lookup, request, site_id)
  102. if page:
  103. url = page.get_absolute_url(language=lang)
  104. cache.set(cache_key, url, settings.CMS_CACHE_DURATIONS['content'])
  105. if url:
  106. return {'content': url}
  107. return {'content': ''}
  108. register.tag(PageUrl)
  109. register.tag('page_id_url', PageUrl)
  110. def _get_placeholder(current_page, page, context, name):
  111. placeholder_cache = getattr(current_page, '_tmp_placeholders_cache', {})
  112. if page.pk in placeholder_cache:
  113. return placeholder_cache[page.pk].get(name, None)
  114. placeholder_cache[page.pk] = {}
  115. placeholders = page.placeholders.all()
  116. for placeholder in placeholders:
  117. placeholder_cache[page.pk][placeholder.slot] = placeholder
  118. current_page._tmp_placeholders_cache = placeholder_cache
  119. return placeholder_cache[page.pk].get(name, None)
  120. def get_placeholder_content(context, request, current_page, name, inherit):
  121. pages = [current_page]
  122. if inherit:
  123. pages = chain([current_page], current_page.get_cached_ancestors(ascending=True))
  124. for page in pages:
  125. placeholder = _get_placeholder(current_page, page, context, name)
  126. if placeholder is None:
  127. continue
  128. if not get_plugins(request, placeholder):
  129. continue
  130. content = render_placeholder(placeholder, context, name)
  131. if content:
  132. return content
  133. placeholder = _get_placeholder(current_page, current_page, context, name)
  134. return render_placeholder(placeholder, context, name)
  135. class PlaceholderParser(Parser):
  136. def parse_blocks(self):
  137. for bit in getattr(self.kwargs['extra_bits'], 'value', self.kwargs['extra_bits']):
  138. if getattr(bit, 'value', bit.var.value) == 'or':
  139. return super(PlaceholderParser, self).parse_blocks()
  140. return
  141. class PlaceholderOptions(Options):
  142. def get_parser_class(self):
  143. return PlaceholderParser
  144. class Placeholder(Tag):
  145. """
  146. This template node is used to output page content and
  147. is also used in the admin to dynamically generate input fields.
  148. eg: {% placeholder "placeholder_name" %}
  149. {% placeholder "sidebar" inherit %}
  150. {% placeholder "footer" inherit or %}
  151. <a href="/about/">About us</a>
  152. {% endplaceholder %}
  153. Keyword arguments:
  154. name -- the name of the placeholder
  155. width -- additional width attribute (integer) which gets added to the plugin context
  156. (deprecated, use `{% with 320 as width %}{% placeholder "foo"}{% endwith %}`)
  157. inherit -- optional argument which if given will result in inheriting
  158. the content of the placeholder with the same name on parent pages
  159. or -- optional argument which if given will make the template tag a block
  160. tag whose content is shown if the placeholder is empty
  161. """
  162. name = 'placeholder'
  163. options = PlaceholderOptions(
  164. Argument('name', resolve=False),
  165. MultiValueArgument('extra_bits', required=False, resolve=False),
  166. blocks=[
  167. ('endplaceholder', 'nodelist'),
  168. ]
  169. )
  170. def render_tag(self, context, name, extra_bits, nodelist=None):
  171. validate_placeholder_name(name)
  172. width = None
  173. inherit = False
  174. for bit in extra_bits:
  175. if bit == 'inherit':
  176. inherit = True
  177. elif bit.isdigit():
  178. width = int(bit)
  179. import warnings
  180. warnings.warn(
  181. "The width parameter for the placeholder tag is deprecated.",
  182. DeprecationWarning
  183. )
  184. if not 'request' in context:
  185. return ''
  186. request = context['request']
  187. if width:
  188. context.update({'width': width})
  189. page = request.current_page
  190. if not page or page == 'dummy':
  191. return ''
  192. content = get_placeholder_content(context, request, page, name, inherit)
  193. if not content and nodelist:
  194. return nodelist.render(context)
  195. return content
  196. def get_name(self):
  197. return self.kwargs['name'].var.value.strip('"').strip("'")
  198. register.tag(Placeholder)
  199. class PageAttribute(Tag):
  200. """
  201. This template node is used to output attribute from a page such
  202. as its title or slug.
  203. Synopsis
  204. {% page_attribute "field-name" %}
  205. {% page_attribute "field-name" page_lookup %}
  206. Example
  207. {# Output current page's page_title attribute: #}
  208. {% page_attribute "page_title" %}
  209. {# Output page_title attribute of the page with reverse_id "the_page": #}
  210. {% page_attribute "page_title" "the_page" %}
  211. {# Output slug attribute of the page with pk 10: #}
  212. {% page_attribute "slug" 10 %}
  213. Keyword arguments:
  214. field-name -- the name of the field to output. Use one of:
  215. - title
  216. - menu_title
  217. - page_title
  218. - slug
  219. - meta_description
  220. - meta_keywords
  221. page_lookup -- lookup argument for Page, if omitted field-name of current page is returned.
  222. See _get_page_by_untyped_arg() for detailed information on the allowed types and their interpretation
  223. for the page_lookup argument.
  224. """
  225. name = 'page_attribute'
  226. options = Options(
  227. Argument('name', resolve=False),
  228. Argument('page_lookup', required=False, default=None)
  229. )
  230. valid_attributes = [
  231. "title",
  232. "slug",
  233. "meta_description",
  234. "meta_keywords",
  235. "page_title",
  236. "menu_title"
  237. ]
  238. def render_tag(self, context, name, page_lookup):
  239. if not 'request' in context:
  240. return ''
  241. name = name.lower()
  242. request = context['request']
  243. lang = get_language_from_request(request)
  244. page = _get_page_by_untyped_arg(page_lookup, request, get_site_id(None))
  245. if page == "dummy":
  246. return ''
  247. if page and name in self.valid_attributes:
  248. f = getattr(page, "get_%s" % name)
  249. return f(language=lang, fallback=True)
  250. return ''
  251. register.tag(PageAttribute)
  252. class CleanAdminListFilter(InclusionTag):
  253. template = 'admin/filter.html'
  254. name = 'clean_admin_list_filter'
  255. options = Options(
  256. Argument('cl'),
  257. Argument('spec'),
  258. )
  259. def get_context(self, context, cl, spec):
  260. choices = sorted(list(spec.choices(cl)), key=lambda k: k['query_string'])
  261. query_string = None
  262. unique_choices = []
  263. for choice in choices:
  264. if choice['query_string'] != query_string:
  265. unique_choices.append(choice)
  266. query_string = choice['query_string']
  267. return {'title': spec.title(), 'choices' : unique_choices}
  268. def _show_placeholder_for_page(context, placeholder_name, page_lookup, lang=None,
  269. site=None, cache_result=True):
  270. """
  271. Shows the content of a page with a placeholder name and given lookup
  272. arguments in the given language.
  273. This is useful if you want to have some more or less static content that is
  274. shared among many pages, such as a footer.
  275. See _get_page_by_untyped_arg() for detailed information on the allowed types
  276. and their interpretation for the page_lookup argument.
  277. """
  278. validate_placeholder_name(placeholder_name)
  279. request = context.get('request', False)
  280. site_id = get_site_id(site)
  281. if not request:
  282. return {'content': ''}
  283. if lang is None:
  284. lang = get_language_from_request(request)
  285. content = None
  286. if cache_result:
  287. cache_key = _get_cache_key('_show_placeholder_for_page', page_lookup, lang, site_id)+'_placeholder:'+placeholder_name
  288. content = cache.get(cache_key)
  289. if not content:
  290. page = _get_page_by_untyped_arg(page_lookup, request, site_id)
  291. if not page:
  292. return {'content': ''}
  293. try:
  294. placeholder = page.placeholders.get(slot=placeholder_name)
  295. except Placeholder.DoesNotExist:
  296. if settings.DEBUG:
  297. raise
  298. return {'content': ''}
  299. baseqs = get_cmsplugin_queryset(request)
  300. plugins = baseqs.filter(
  301. placeholder=placeholder,
  302. language=lang,
  303. placeholder__slot__iexact=placeholder_name,
  304. parent__isnull=True
  305. ).order_by('position').select_related()
  306. c = render_plugins(plugins, context, placeholder)
  307. content = "".join(c)
  308. if cache_result:
  309. cache.set(cache_key, content, settings.CMS_CACHE_DURATIONS['content'])
  310. if content:
  311. return {'content': mark_safe(content)}
  312. return {'content': ''}
  313. class ShowPlaceholderById(InclusionTag):
  314. template = 'cms/content.html'
  315. name = 'show_placeholder_by_id'
  316. options = Options(
  317. Argument('placeholder_name'),
  318. Argument('reverse_id'),
  319. Argument('lang', required=False, default=None),
  320. Argument('site', required=False, default=None),
  321. )
  322. def get_context(self, *args, **kwargs):
  323. return _show_placeholder_for_page(**self.get_kwargs(*args, **kwargs))
  324. def get_kwargs(self, context, placeholder_name, reverse_id, lang, site):
  325. return {
  326. 'context': context,
  327. 'placeholder_name': placeholder_name,
  328. 'page_lookup': reverse_id,
  329. 'lang': lang,
  330. 'site': site
  331. }
  332. register.tag(ShowPlaceholderById)
  333. register.tag('show_placeholder', ShowPlaceholderById)
  334. class ShowUncachedPlaceholderById(ShowPlaceholderById):
  335. name = 'show_uncached_placeholder_by_id'
  336. def get_kwargs(self, *args, **kwargs):
  337. kwargs = super(ShowUncachedPlaceholderById, self).get_kwargs(*args, **kwargs)
  338. kwargs['cache_result'] = False
  339. return kwargs
  340. register.tag(ShowUncachedPlaceholderById)
  341. register.tag('show_uncached_placeholder', ShowUncachedPlaceholderById)
  342. class CMSToolbar(InclusionTag):
  343. template = 'cms/toolbar/toolbar.html'
  344. name = 'cms_toolbar'
  345. def render(self, context):
  346. request = context.get('request', None)
  347. if not request:
  348. return ''
  349. toolbar = getattr(request, 'toolbar', None)
  350. if not toolbar:
  351. return ''
  352. if not toolbar.show_toolbar:
  353. return ''
  354. return super(CMSToolbar, self).render(context)
  355. def get_context(self, context):
  356. context['CMS_TOOLBAR_CONFIG'] = context['request'].toolbar.as_json(context)
  357. return context
  358. register.tag(CMSToolbar)