├── .gitignore ├── MANIFEST.in ├── README.md ├── setup.cfg ├── setup.py ├── simplemde ├── __init__.py ├── fields.py ├── static │ └── simplemde │ │ ├── simplemde.init.js │ │ ├── simplemde.min.css │ │ └── simplemde.min.js ├── utils.py └── widgets.py └── testproject ├── manage.py ├── markdown ├── __init__.py ├── admin.py ├── migrations │ ├── 0001_initial.py │ └── __init__.py ├── models.py ├── templates │ └── markdown │ │ └── index.html ├── tests.py ├── urls.py └── views.py ├── simplemde └── testproject ├── __init__.py ├── settings.py ├── urls.py └── wsgi.py /.gitignore: -------------------------------------------------------------------------------- 1 | led / optimized / DLL files 2 | __pycache__/ 3 | *.py[cod] 4 | *$py.class 5 | 6 | # C extensions 7 | *.so 8 | 9 | # Distribution / packaging 10 | .Python 11 | env/ 12 | build/ 13 | develop-eggs/ 14 | dist/ 15 | downloads/ 16 | eggs/ 17 | .eggs/ 18 | lib/ 19 | lib64/ 20 | parts/ 21 | sdist/ 22 | var/ 23 | *.egg-info/ 24 | .installed.cfg 25 | *.egg 26 | 27 | # PyInstaller 28 | # Usually these files are written by a python script from a template 29 | # before PyInstaller builds the exe, so as to inject date/other infos into it. 30 | *.manifest 31 | *.spec 32 | 33 | # Installer logs 34 | pip-log.txt 35 | pip-delete-this-directory.txt 36 | 37 | # Unit test / coverage reports 38 | htmlcov/ 39 | .tox/ 40 | .coverage 41 | .coverage.* 42 | .cache 43 | nosetests.xml 44 | coverage.xml 45 | *,cover 46 | .hypothesis/ 47 | 48 | # Translations 49 | *.mo 50 | *.pot 51 | 52 | # Django stuff: 53 | *.log 54 | 55 | # Sphinx documentation 56 | docs/_build/ 57 | 58 | # PyBuilder 59 | target/ 60 | 61 | # sqlite file 62 | *.sqlite3 63 | .DS_Store 64 | -------------------------------------------------------------------------------- /MANIFEST.in: -------------------------------------------------------------------------------- 1 | include README.md 2 | recursive-include simplemde * 3 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # A markdown editor(with preview) for Django 2 | Use markdown editor https://github.com/sparksuite/simplemde-markdown-editor in django project, this project is inspired by https://github.com/douglasmiranda/django-wysiwyg-redactor/ 3 | 4 | # Getting started 5 | * install django-simplemde 6 | ``` 7 | pip install django-simplemde 8 | ``` 9 | 10 | * add 'simplemde' to INSTALLED_APPS. 11 | 12 | ```python 13 | INSTALLED_APPS = ( 14 | # ... 15 | 'simplemde', 16 | # ... 17 | ) 18 | ``` 19 | 20 | # Using in models 21 | ```python 22 | from django.db import models 23 | from simplemde.fields import SimpleMDEField 24 | 25 | class Entry(models.Model): 26 | title = models.CharField(max_length=250, verbose_name=u'Title') 27 | content = SimpleMDEField(verbose_name=u'mardown content') 28 | ``` 29 | 30 | # SimpleMDE options 31 | You could set SimpleMDE options in settings.py like this: 32 | 33 | ```python 34 | SIMPLEMDE_OPTIONS = { 35 | 'placeholder': 'haha', 36 | 'status': False, 37 | 'autosave': { 38 | 'enabled': True 39 | } 40 | } 41 | ``` 42 | 43 | Right now this plugin supports [SimpleMDE Configurations](https://github.com/sparksuite/simplemde-markdown-editor#configuration), but only the static ones(don't support js configurations like ```previewRender```) 44 | 45 | ***for autosave option, you dont need to set it, this plugin will generate uniqueId with python's uuid.uuid4 automatically*** 46 | 47 | # Get SimpleMDE instance from DOM 48 | 49 | After SimpleMDE initialized, you could get SimpleMDE instance from dom element like this: 50 | 51 | ```javascript 52 | $('.simplemde-box')[0].SimpleMDE 53 | ``` 54 | -------------------------------------------------------------------------------- /setup.cfg: -------------------------------------------------------------------------------- 1 | [metadata] 2 | description-file = README.md 3 | -------------------------------------------------------------------------------- /setup.py: -------------------------------------------------------------------------------- 1 | import os 2 | from setuptools import setup 3 | 4 | f = open(os.path.join(os.path.dirname(__file__), 'README.md')) 5 | readme = f.read() 6 | f.close() 7 | 8 | setup( 9 | name='django-simplemde', 10 | version='0.1.4', 11 | description='django-simplemde is a WYSIWYG markdown editor for Django', 12 | long_description=readme, 13 | long_description_content_type="text/markdown", 14 | author="Siyuan Zhang", 15 | author_email='onepill@gmail.com', 16 | url='https://github.com/onepill/django-simplemde', 17 | license='MIT', 18 | packages=['simplemde'], 19 | include_package_data=True, 20 | install_requires=['setuptools'], 21 | zip_safe=False, 22 | classifiers=[ 23 | 'Development Status :: 5 - Production/Stable', 24 | 'Environment :: Web Environment', 25 | 'Framework :: Django', 26 | 'Intended Audience :: Developers', 27 | 'License :: OSI Approved :: BSD License', 28 | 'Operating System :: OS Independent', 29 | 'Programming Language :: Python', 30 | ], 31 | keywords='django,admin,wysiwyg,markdown,editor,simplemde', 32 | ) 33 | -------------------------------------------------------------------------------- /simplemde/__init__.py: -------------------------------------------------------------------------------- 1 | __version__ = '0.1.3' 2 | -------------------------------------------------------------------------------- /simplemde/fields.py: -------------------------------------------------------------------------------- 1 | from django.db.models import TextField 2 | from django.conf import settings 3 | from django.contrib.admin import widgets as admin_widgets 4 | from .widgets import SimpleMDEEditor 5 | 6 | 7 | class SimpleMDEField(TextField): 8 | def __init__(self, *args, **kwargs): 9 | options = kwargs.pop('simplemde_options', {}) 10 | self.widget = SimpleMDEEditor( 11 | simplemde_options=options, 12 | ) 13 | super(SimpleMDEField, self).__init__(*args, **kwargs) 14 | 15 | def formfield(self, **kwargs): 16 | defaults = {'widget': self.widget} 17 | defaults.update(kwargs) 18 | 19 | if defaults['widget'] == admin_widgets.AdminTextareaWidget: 20 | defaults['widget'] = self.widget 21 | return super(SimpleMDEField, self).formfield(**defaults) 22 | 23 | 24 | if 'south' in settings.INSTALLED_APPS: 25 | from south.modelsinspector import add_introspection_rules 26 | add_introspection_rules([], ["^simplemde\.fields\.SimpleMDEField"]) 27 | -------------------------------------------------------------------------------- /simplemde/static/simplemde/simplemde.init.js: -------------------------------------------------------------------------------- 1 | var simplemdeJQuery = null; 2 | 3 | if (typeof jQuery !== 'undefined') { 4 | simplemdeJQuery = jQuery; 5 | } else if (typeof django !== 'undefined') { 6 | //use jQuery come with django admin 7 | simplemdeJQuery = django.jQuery 8 | } else { 9 | console.error('cant find jQuery, please make sure your have jQuery imported before this script'); 10 | } 11 | 12 | if (!!simplemdeJQuery) { 13 | simplemdeJQuery(function() { 14 | simplemdeJQuery.each(simplemdeJQuery('.simplemde-box'), function(i, elem) { 15 | var options = JSON.parse(simplemdeJQuery(elem).attr('data-simplemde-options')); 16 | options['element'] = elem; 17 | var simplemde = new SimpleMDE(options); 18 | elem.SimpleMDE = simplemde; 19 | }); 20 | }); 21 | } -------------------------------------------------------------------------------- /simplemde/static/simplemde/simplemde.min.css: -------------------------------------------------------------------------------- 1 | /** 2 | * simplemde v1.11.2 3 | * Copyright Next Step Webs, Inc. 4 | * @link https://github.com/sparksuite/simplemde-markdown-editor 5 | * @license MIT 6 | */ 7 | .CodeMirror{color:#000}.CodeMirror-lines{padding:4px 0}.CodeMirror pre{padding:0 4px}.CodeMirror-gutter-filler,.CodeMirror-scrollbar-filler{background-color:#fff}.CodeMirror-gutters{border-right:1px solid #ddd;background-color:#f7f7f7;white-space:nowrap}.CodeMirror-linenumber{padding:0 3px 0 5px;min-width:20px;text-align:right;color:#999;white-space:nowrap}.CodeMirror-guttermarker{color:#000}.CodeMirror-guttermarker-subtle{color:#999}.CodeMirror-cursor{border-left:1px solid #000;border-right:none;width:0}.CodeMirror div.CodeMirror-secondarycursor{border-left:1px solid silver}.cm-fat-cursor .CodeMirror-cursor{width:auto;border:0!important;background:#7e7}.cm-fat-cursor div.CodeMirror-cursors{z-index:1}.cm-animate-fat-cursor{width:auto;border:0;-webkit-animation:blink 1.06s steps(1) infinite;-moz-animation:blink 1.06s steps(1) infinite;animation:blink 1.06s steps(1) infinite;background-color:#7e7}@-moz-keyframes blink{50%{background-color:transparent}}@-webkit-keyframes blink{50%{background-color:transparent}}@keyframes blink{50%{background-color:transparent}}.cm-tab{display:inline-block;text-decoration:inherit}.CodeMirror-ruler{border-left:1px solid #ccc;position:absolute}.cm-s-default .cm-header{color:#00f}.cm-s-default .cm-quote{color:#090}.cm-negative{color:#d44}.cm-positive{color:#292}.cm-header,.cm-strong{font-weight:700}.cm-em{font-style:italic}.cm-link{text-decoration:underline}.cm-strikethrough{text-decoration:line-through}.cm-s-default .cm-keyword{color:#708}.cm-s-default .cm-atom{color:#219}.cm-s-default .cm-number{color:#164}.cm-s-default .cm-def{color:#00f}.cm-s-default .cm-variable-2{color:#05a}.cm-s-default .cm-variable-3{color:#085}.cm-s-default .cm-comment{color:#a50}.cm-s-default .cm-string{color:#a11}.cm-s-default .cm-string-2{color:#f50}.cm-s-default .cm-meta,.cm-s-default .cm-qualifier{color:#555}.cm-s-default .cm-builtin{color:#30a}.cm-s-default .cm-bracket{color:#997}.cm-s-default .cm-tag{color:#170}.cm-s-default .cm-attribute{color:#00c}.cm-s-default .cm-hr{color:#999}.cm-s-default .cm-link{color:#00c}.cm-invalidchar,.cm-s-default .cm-error{color:red}.CodeMirror-composing{border-bottom:2px solid}div.CodeMirror span.CodeMirror-matchingbracket{color:#0f0}div.CodeMirror span.CodeMirror-nonmatchingbracket{color:#f22}.CodeMirror-matchingtag{background:rgba(255,150,0,.3)}.CodeMirror-activeline-background{background:#e8f2ff}.CodeMirror{position:relative;overflow:hidden;background:#fff}.CodeMirror-scroll{overflow:scroll!important;margin-bottom:-30px;margin-right:-30px;padding-bottom:30px;height:100%;outline:0;position:relative}.CodeMirror-sizer{position:relative;border-right:30px solid transparent}.CodeMirror-gutter-filler,.CodeMirror-hscrollbar,.CodeMirror-scrollbar-filler,.CodeMirror-vscrollbar{position:absolute;z-index:6;display:none}.CodeMirror-vscrollbar{right:0;top:0;overflow-x:hidden;overflow-y:scroll}.CodeMirror-hscrollbar{bottom:0;left:0;overflow-y:hidden;overflow-x:scroll}.CodeMirror-scrollbar-filler{right:0;bottom:0}.CodeMirror-gutter-filler{left:0;bottom:0}.CodeMirror-gutters{position:absolute;left:0;top:0;min-height:100%;z-index:3}.CodeMirror-gutter{white-space:normal;height:100%;display:inline-block;vertical-align:top;margin-bottom:-30px}.CodeMirror-gutter-wrapper{position:absolute;z-index:4;background:0 0!important;border:none!important;-webkit-user-select:none;-moz-user-select:none;user-select:none}.CodeMirror-gutter-background{position:absolute;top:0;bottom:0;z-index:4}.CodeMirror-gutter-elt{position:absolute;cursor:default;z-index:4}.CodeMirror-lines{cursor:text;min-height:1px}.CodeMirror pre{-moz-border-radius:0;-webkit-border-radius:0;border-radius:0;border-width:0;background:0 0;font-family:inherit;font-size:inherit;margin:0;white-space:pre;word-wrap:normal;line-height:inherit;color:inherit;z-index:2;position:relative;overflow:visible;-webkit-tap-highlight-color:transparent;-webkit-font-variant-ligatures:none;font-variant-ligatures:none}.CodeMirror-wrap pre{word-wrap:break-word;white-space:pre-wrap;word-break:normal}.CodeMirror-linebackground{position:absolute;left:0;right:0;top:0;bottom:0;z-index:0}.CodeMirror-linewidget{position:relative;z-index:2;overflow:auto}.CodeMirror-code{outline:0}.CodeMirror-gutter,.CodeMirror-gutters,.CodeMirror-linenumber,.CodeMirror-scroll,.CodeMirror-sizer{-moz-box-sizing:content-box;box-sizing:content-box}.CodeMirror-measure{position:absolute;width:100%;height:0;overflow:hidden;visibility:hidden}.CodeMirror-cursor{position:absolute}.CodeMirror-measure pre{position:static}div.CodeMirror-cursors{visibility:hidden;position:relative;z-index:3}.CodeMirror-focused div.CodeMirror-cursors,div.CodeMirror-dragcursors{visibility:visible}.CodeMirror-selected{background:#d9d9d9}.CodeMirror-focused .CodeMirror-selected,.CodeMirror-line::selection,.CodeMirror-line>span::selection,.CodeMirror-line>span>span::selection{background:#d7d4f0}.CodeMirror-crosshair{cursor:crosshair}.CodeMirror-line::-moz-selection,.CodeMirror-line>span::-moz-selection,.CodeMirror-line>span>span::-moz-selection{background:#d7d4f0}.cm-searching{background:#ffa;background:rgba(255,255,0,.4)}.cm-force-border{padding-right:.1px}@media print{.CodeMirror div.CodeMirror-cursors{visibility:hidden}}.cm-tab-wrap-hack:after{content:''}span.CodeMirror-selectedtext{background:0 0}.CodeMirror{height:auto;min-height:300px;border:1px solid #ddd;border-bottom-left-radius:4px;border-bottom-right-radius:4px;padding:10px;font:inherit;z-index:1}.CodeMirror-scroll{min-height:300px}.CodeMirror-fullscreen{background:#fff;position:fixed!important;top:50px;left:0;right:0;bottom:0;height:auto;z-index:9}.CodeMirror-sided{width:50%!important}.editor-toolbar{position:relative;opacity:.6;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;-o-user-select:none;user-select:none;padding:0 10px;border-top:1px solid #bbb;border-left:1px solid #bbb;border-right:1px solid #bbb;border-top-left-radius:4px;border-top-right-radius:4px}.editor-toolbar:after,.editor-toolbar:before{display:block;content:' ';height:1px}.editor-toolbar:before{margin-bottom:8px}.editor-toolbar:after{margin-top:8px}.editor-toolbar:hover,.editor-wrapper input.title:focus,.editor-wrapper input.title:hover{opacity:.8}.editor-toolbar.fullscreen{width:100%;height:50px;overflow-x:auto;overflow-y:hidden;white-space:nowrap;padding-top:10px;padding-bottom:10px;box-sizing:border-box;background:#fff;border:0;position:fixed;top:0;left:0;opacity:1;z-index:9}.editor-toolbar.fullscreen::before{width:20px;height:50px;background:-moz-linear-gradient(left,rgba(255,255,255,1) 0,rgba(255,255,255,0) 100%);background:-webkit-gradient(linear,left top,right top,color-stop(0,rgba(255,255,255,1)),color-stop(100%,rgba(255,255,255,0)));background:-webkit-linear-gradient(left,rgba(255,255,255,1) 0,rgba(255,255,255,0) 100%);background:-o-linear-gradient(left,rgba(255,255,255,1) 0,rgba(255,255,255,0) 100%);background:-ms-linear-gradient(left,rgba(255,255,255,1) 0,rgba(255,255,255,0) 100%);background:linear-gradient(to right,rgba(255,255,255,1) 0,rgba(255,255,255,0) 100%);position:fixed;top:0;left:0;margin:0;padding:0}.editor-toolbar.fullscreen::after{width:20px;height:50px;background:-moz-linear-gradient(left,rgba(255,255,255,0) 0,rgba(255,255,255,1) 100%);background:-webkit-gradient(linear,left top,right top,color-stop(0,rgba(255,255,255,0)),color-stop(100%,rgba(255,255,255,1)));background:-webkit-linear-gradient(left,rgba(255,255,255,0) 0,rgba(255,255,255,1) 100%);background:-o-linear-gradient(left,rgba(255,255,255,0) 0,rgba(255,255,255,1) 100%);background:-ms-linear-gradient(left,rgba(255,255,255,0) 0,rgba(255,255,255,1) 100%);background:linear-gradient(to right,rgba(255,255,255,0) 0,rgba(255,255,255,1) 100%);position:fixed;top:0;right:0;margin:0;padding:0}.editor-toolbar a{display:inline-block;text-align:center;text-decoration:none!important;color:#2c3e50!important;width:30px;height:30px;margin:0;border:1px solid transparent;border-radius:3px;cursor:pointer}.editor-toolbar a.active,.editor-toolbar a:hover{background:#fcfcfc;border-color:#95a5a6}.editor-toolbar a:before{line-height:30px}.editor-toolbar i.separator{display:inline-block;width:0;border-left:1px solid #d9d9d9;border-right:1px solid #fff;color:transparent;text-indent:-10px;margin:0 6px}.editor-toolbar a.fa-header-x:after{font-family:Arial,"Helvetica Neue",Helvetica,sans-serif;font-size:65%;vertical-align:text-bottom;position:relative;top:2px}.editor-toolbar a.fa-header-1:after{content:"1"}.editor-toolbar a.fa-header-2:after{content:"2"}.editor-toolbar a.fa-header-3:after{content:"3"}.editor-toolbar a.fa-header-bigger:after{content:"▲"}.editor-toolbar a.fa-header-smaller:after{content:"▼"}.editor-toolbar.disabled-for-preview a:not(.no-disable){pointer-events:none;background:#fff;border-color:transparent;text-shadow:inherit}@media only screen and (max-width:700px){.editor-toolbar a.no-mobile{display:none}}.editor-statusbar{padding:8px 10px;font-size:12px;color:#959694;text-align:right}.editor-statusbar span{display:inline-block;min-width:4em;margin-left:1em}.editor-preview,.editor-preview-side{padding:10px;background:#fafafa;overflow:auto;display:none;box-sizing:border-box}.editor-statusbar .lines:before{content:'lines: '}.editor-statusbar .words:before{content:'words: '}.editor-statusbar .characters:before{content:'characters: '}.editor-preview{position:absolute;width:100%;height:100%;top:0;left:0;z-index:7}.editor-preview-side{position:fixed;bottom:0;width:50%;top:50px;right:0;z-index:9;border:1px solid #ddd}.editor-preview-active,.editor-preview-active-side{display:block}.editor-preview-side>p,.editor-preview>p{margin-top:0}.editor-preview pre,.editor-preview-side pre{background:#eee;margin-bottom:10px}.editor-preview table td,.editor-preview table th,.editor-preview-side table td,.editor-preview-side table th{border:1px solid #ddd;padding:5px}.CodeMirror .CodeMirror-code .cm-tag{color:#63a35c}.CodeMirror .CodeMirror-code .cm-attribute{color:#795da3}.CodeMirror .CodeMirror-code .cm-string{color:#183691}.CodeMirror .CodeMirror-selected{background:#d9d9d9}.CodeMirror .CodeMirror-code .cm-header-1{font-size:200%;line-height:200%}.CodeMirror .CodeMirror-code .cm-header-2{font-size:160%;line-height:160%}.CodeMirror .CodeMirror-code .cm-header-3{font-size:125%;line-height:125%}.CodeMirror .CodeMirror-code .cm-header-4{font-size:110%;line-height:110%}.CodeMirror .CodeMirror-code .cm-comment{background:rgba(0,0,0,.05);border-radius:2px}.CodeMirror .CodeMirror-code .cm-link{color:#7f8c8d}.CodeMirror .CodeMirror-code .cm-url{color:#aab2b3}.CodeMirror .CodeMirror-code .cm-strikethrough{text-decoration:line-through}.CodeMirror .CodeMirror-placeholder{opacity:.5}.CodeMirror .cm-spell-error:not(.cm-url):not(.cm-comment):not(.cm-tag):not(.cm-word){background:rgba(255,0,0,.15)} 8 | -------------------------------------------------------------------------------- /simplemde/utils.py: -------------------------------------------------------------------------------- 1 | from django.core.exceptions import ImproperlyConfigured 2 | from importlib import import_module 3 | 4 | try: 5 | # django > 3 6 | from django.utils.encoding import force_str 7 | except ImportError: 8 | # django 2 9 | from django.utils.encoding import force_unicode as force_str 10 | 11 | from django.utils.functional import Promise 12 | 13 | import json 14 | 15 | 16 | class LazyEncoder(json.JSONEncoder): 17 | 18 | def default(self, obj): 19 | if isinstance(obj, Promise): 20 | return force_str(obj) 21 | return super(LazyEncoder, self).default(obj) 22 | 23 | 24 | def json_dumps(data): 25 | return json.dumps(data, cls=LazyEncoder) 26 | -------------------------------------------------------------------------------- /simplemde/widgets.py: -------------------------------------------------------------------------------- 1 | from __future__ import unicode_literals 2 | import uuid 3 | from django import forms 4 | from django.forms import widgets 5 | from django.utils.safestring import mark_safe 6 | from django.conf import settings 7 | 8 | try: 9 | from django.core.urlresolvers import reverse_lazy 10 | except ImportError: 11 | from django.urls import reverse_lazy 12 | 13 | from .utils import json_dumps 14 | 15 | 16 | GLOBAL_OPTIONS = getattr(settings, 'SIMPLEMDE_OPTIONS', {}) 17 | 18 | 19 | class SimpleMDEEditor(widgets.Textarea): 20 | def __init__(self, *args, **kwargs): 21 | self.custom_options = kwargs.pop('simplemde_options', {}) 22 | super(SimpleMDEEditor, self).__init__(*args, **kwargs) 23 | 24 | @property 25 | def options(self): 26 | options = GLOBAL_OPTIONS.copy() 27 | if 'autosave' in options and options['autosave'].get('enabled', False): 28 | options['autosave']['uniqueId'] = str(uuid.uuid4()) 29 | options.update(self.custom_options) 30 | return options 31 | 32 | def render(self, name, value, attrs=None, renderer=None): 33 | if 'class' not in attrs.keys(): 34 | attrs['class'] = '' 35 | 36 | attrs['class'] += ' simplemde-box' 37 | 38 | attrs['data-simplemde-options'] = json_dumps(self.options) 39 | 40 | html = super(SimpleMDEEditor, self).render(name, value, attrs, renderer=renderer) 41 | 42 | # insert this style tag to fix the label from breaking into the toolbar 43 | html += "" % name 44 | 45 | return mark_safe(html) 46 | 47 | def _media(self): 48 | js = ( 49 | 'simplemde/simplemde.min.js', 50 | 'simplemde/simplemde.init.js' 51 | ) 52 | 53 | css = { 54 | 'all': ( 55 | 'simplemde/simplemde.min.css', 56 | ) 57 | } 58 | return forms.Media(css=css, js=js) 59 | media = property(_media) 60 | -------------------------------------------------------------------------------- /testproject/manage.py: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env python 2 | import os 3 | import sys 4 | 5 | if __name__ == "__main__": 6 | os.environ.setdefault("DJANGO_SETTINGS_MODULE", "testproject.settings") 7 | 8 | from django.core.management import execute_from_command_line 9 | 10 | execute_from_command_line(sys.argv) 11 | -------------------------------------------------------------------------------- /testproject/markdown/__init__.py: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/onepill/django-simplemde/cd8512c9544b34356dde5332c6e51701044bb459/testproject/markdown/__init__.py -------------------------------------------------------------------------------- /testproject/markdown/admin.py: -------------------------------------------------------------------------------- 1 | from django.contrib import admin 2 | from .models import Markdown 3 | # Register your models here. 4 | class MarkdownAdmin(admin.ModelAdmin): 5 | pass 6 | 7 | admin.site.register(Markdown, MarkdownAdmin) -------------------------------------------------------------------------------- /testproject/markdown/migrations/0001_initial.py: -------------------------------------------------------------------------------- 1 | # -*- coding: utf-8 -*- 2 | # Generated by Django 1.10.3 on 2016-11-18 17:51 3 | from __future__ import unicode_literals 4 | 5 | from django.db import migrations, models 6 | import simplemde.fields 7 | 8 | 9 | class Migration(migrations.Migration): 10 | 11 | initial = True 12 | 13 | dependencies = [ 14 | ] 15 | 16 | operations = [ 17 | migrations.CreateModel( 18 | name='Markdown', 19 | fields=[ 20 | ('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')), 21 | ('title', models.CharField(max_length=32)), 22 | ('content', simplemde.fields.SimpleMDEField()), 23 | ], 24 | ), 25 | ] 26 | -------------------------------------------------------------------------------- /testproject/markdown/migrations/__init__.py: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/onepill/django-simplemde/cd8512c9544b34356dde5332c6e51701044bb459/testproject/markdown/migrations/__init__.py -------------------------------------------------------------------------------- /testproject/markdown/models.py: -------------------------------------------------------------------------------- 1 | from django.db import models 2 | from simplemde.fields import SimpleMDEField 3 | 4 | 5 | class Markdown(models.Model): 6 | title = models.CharField(max_length=32) 7 | content = SimpleMDEField() 8 | -------------------------------------------------------------------------------- /testproject/markdown/templates/markdown/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 |
4 | 5 |