Source code for djangocodemirror.widgets

# -*- coding: utf-8 -*-
"""
Form field widget
=================

"""
import json

from django import forms

from django.conf import settings
from django.utils.safestring import mark_safe

from djangocodemirror.manifest import CodeMirrorManifest


[docs]class CodeMirrorWidget(forms.Textarea): """ Widget to add a CodeMirror or DjangoCodeMirror instance on a textarea Take the same arguments than ``forms.Textarea`` and accepts one suplementary optionnal arguments : Arguments: config_name (string): A Codemirror config name available in ``settings.CODEMIRROR_SETTINGS``. Default is ``empty``. embed_config (bool): If ``True`` will add Codemirror Javascript config just below the input. Default is ``False``. Attributes: config_name (string): For given config name. """ codemirror_field_js = settings.CODEMIRROR_FIELD_INIT_JS def __init__(self, *args, **kwargs): self.config_name = kwargs.pop('config_name', 'empty') self.embed_config = kwargs.pop('embed_config', False) super(CodeMirrorWidget, self).__init__(*args, **kwargs)
[docs] def init_manifest(self, name): """ Initialize a manifest instance Arguments: name (string): Config name to register. Returns: CodeMirrorManifest: A manifest instance where config (from ``config_name`` attribute) is registred. """ manifesto = CodeMirrorManifest() manifesto.register(name) return manifesto
[docs] def get_codemirror_field_js(self): """ Return CodeMirror HTML template from ``CodeMirrorWidget.codemirror_field_js``. Returns: string: HTML template string. """ return self.codemirror_field_js
[docs] def codemirror_config(self): """ Shortcut to get Codemirror parameters. Returns: dict: CodeMirror parameters. """ return self.editor_manifest.get_codemirror_parameters(self.config_name)
[docs] def codemirror_script(self, final_attrs): """ Build CodeMirror HTML script tag which contains CodeMirror init. Arguments: final_attrs (dict): Widget attributes that must contains the ``id`` attribute. Returns: string: HTML for field CodeMirror instance. """ inputid = final_attrs['id'] varname = "{}_codemirror".format(inputid) html = self.get_codemirror_field_js() opts = self.codemirror_config() return html.format(varname=varname, inputid=inputid, settings=json.dumps(opts, sort_keys=True))
[docs] def render(self, name, value, attrs=None): """ Render widget HTML Arguments: name (string): Field input name. value (string): Field input value. Keyword Arguments: attrs (dict): Optional field widget attributes. Default to ``None``. Returns: string: Widget HTML representation. """ if not hasattr(self, "editor_manifest"): self.editor_manifest = self.init_manifest(self.config_name) config = self.editor_manifest.get_config(self.config_name) final_attrs = self.build_attrs(attrs, name=name) # Widget allways need an id to be able to set CodeMirror Javascript # config if 'id' not in final_attrs: final_attrs['id'] = 'id_{}'.format(name) html = [super(CodeMirrorWidget, self).render(name, value, attrs)] # Append HTML for CodeMirror Javascript config just below the textarea if self.embed_config or config.get('embed_config'): html.append(self.codemirror_script(final_attrs)) return mark_safe(u'\n'.join(html))
@property def media(self): """ Adds necessary files (Js/CSS) to the widget's medias. Returns: django.forms.Media: Media object with all assets from registered config. """ if not hasattr(self, "editor_manifest"): self.editor_manifest = self.init_manifest(self.config_name) return forms.Media( css={"all": self.editor_manifest.css()}, js=self.editor_manifest.js() )
[docs]class CodeMirrorAdminWidget(CodeMirrorWidget): """ CodeMirror widget suited for usage in models admins. Act like CodeMirrorWidget but allways embed Codemirror Javascript config. """ def __init__(self, *args, **kwargs): kwargs['embed_config'] = True super(CodeMirrorAdminWidget, self).__init__(*args, **kwargs)