├── django_kss ├── __init__.py ├── pykss │ ├── contrib │ │ ├── __init__.py │ │ └── django │ │ │ ├── __init__.py │ │ │ ├── models.py │ │ │ ├── templatetags │ │ │ └── __init__.py │ │ │ ├── views.py │ │ │ ├── templates │ │ │ └── pykss │ │ │ │ └── styleguideblock.html │ │ │ └── static │ │ │ └── pykss │ │ │ └── js │ │ │ └── kss.js │ ├── __init__.py │ ├── exceptions.py │ ├── modifier.py │ ├── parser.py │ ├── section.py │ └── comment.py ├── templatetags │ ├── __init__.py │ ├── kss_extra.py │ └── pykss.py ├── templates │ ├── prototype │ │ └── hello.html │ ├── styleguide-full-content.html │ ├── styleguide-inline-content.html │ ├── styleguide_base.html │ ├── pykss │ │ └── styleguideblock.html │ └── styleguide.html ├── models.py ├── tests.py ├── admin.py ├── static │ ├── css │ │ ├── forms.css │ │ ├── buttons.css │ │ └── layout.css │ └── js │ │ └── kss.js ├── urls.py ├── setup.py ├── utils.py └── views.py ├── django_kss_project ├── sample │ ├── __init__.py │ ├── migrations │ │ └── __init__.py │ ├── models.py │ ├── tests.py │ ├── admin.py │ ├── views.py │ ├── styleguide.py │ └── static │ │ └── css │ │ └── form.css ├── sample2 │ ├── __init__.py │ ├── migrations │ │ └── __init__.py │ ├── templates │ │ └── prototype │ │ │ └── ya.html │ ├── models.py │ ├── tests.py │ ├── admin.py │ ├── views.py │ ├── styleguide.py │ └── static │ │ └── cxx │ │ └── form.scss ├── django_kss_project │ ├── __init__.py │ ├── wsgi.py │ ├── urls.py │ └── settings.py ├── static │ └── css │ │ ├── forms.css │ │ └── buttons.css └── manage.py ├── Makefile ├── requirements.txt ├── pictures └── screenshot.png ├── MANIFEST.in ├── .gitignore ├── .idea └── inspectionProfiles │ ├── profiles_settings.xml │ └── Project_Default.xml ├── setup.py └── README.rst /django_kss/__init__.py: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /django_kss/pykss/contrib/__init__.py: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /django_kss_project/sample/__init__.py: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /django_kss_project/sample2/__init__.py: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /django_kss/pykss/contrib/django/__init__.py: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /django_kss/pykss/contrib/django/models.py: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /django_kss_project/sample/migrations/__init__.py: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /django_kss_project/django_kss_project/__init__.py: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /django_kss_project/sample2/migrations/__init__.py: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /django_kss/pykss/contrib/django/templatetags/__init__.py: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /django_kss/templatetags/__init__.py: -------------------------------------------------------------------------------- 1 | __author__ = 'tim' 2 | -------------------------------------------------------------------------------- /Makefile: -------------------------------------------------------------------------------- 1 | upload: 2 | python setup.py sdist bdist upload 3 | -------------------------------------------------------------------------------- /django_kss/templates/prototype/hello.html: -------------------------------------------------------------------------------- 1 |

hellow

2 | -------------------------------------------------------------------------------- /django_kss/pykss/__init__.py: -------------------------------------------------------------------------------- 1 | from .parser import Parser # NOQA 2 | -------------------------------------------------------------------------------- /requirements.txt: -------------------------------------------------------------------------------- 1 | Django==1.7.1 2 | Pygments==2.0.1 3 | wsgiref==0.1.2 4 | -------------------------------------------------------------------------------- /django_kss_project/sample2/templates/prototype/ya.html: -------------------------------------------------------------------------------- 1 |

hellowx

2 | -------------------------------------------------------------------------------- /django_kss/pykss/exceptions.py: -------------------------------------------------------------------------------- 1 | class SectionDoesNotExist(Exception): 2 | pass 3 | -------------------------------------------------------------------------------- /django_kss/models.py: -------------------------------------------------------------------------------- 1 | from django.db import models 2 | 3 | # Create your models here. 4 | -------------------------------------------------------------------------------- /django_kss/tests.py: -------------------------------------------------------------------------------- 1 | from django.test import TestCase 2 | 3 | # Create your tests here. 4 | -------------------------------------------------------------------------------- /django_kss/admin.py: -------------------------------------------------------------------------------- 1 | from django.contrib import admin 2 | 3 | # Register your models here. 4 | -------------------------------------------------------------------------------- /pictures/screenshot.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/dorm7/django_kss/HEAD/pictures/screenshot.png -------------------------------------------------------------------------------- /django_kss_project/sample/models.py: -------------------------------------------------------------------------------- 1 | from django.db import models 2 | 3 | # Create your models here. 4 | -------------------------------------------------------------------------------- /django_kss_project/sample/tests.py: -------------------------------------------------------------------------------- 1 | from django.test import TestCase 2 | 3 | # Create your tests here. 4 | -------------------------------------------------------------------------------- /django_kss_project/sample2/models.py: -------------------------------------------------------------------------------- 1 | from django.db import models 2 | 3 | # Create your models here. 4 | -------------------------------------------------------------------------------- /django_kss_project/sample2/tests.py: -------------------------------------------------------------------------------- 1 | from django.test import TestCase 2 | 3 | # Create your tests here. 4 | -------------------------------------------------------------------------------- /django_kss_project/sample/admin.py: -------------------------------------------------------------------------------- 1 | from django.contrib import admin 2 | 3 | # Register your models here. 4 | -------------------------------------------------------------------------------- /django_kss_project/sample/views.py: -------------------------------------------------------------------------------- 1 | from django.shortcuts import render 2 | 3 | # Create your views here. 4 | -------------------------------------------------------------------------------- /django_kss_project/sample2/admin.py: -------------------------------------------------------------------------------- 1 | from django.contrib import admin 2 | 3 | # Register your models here. 4 | -------------------------------------------------------------------------------- /django_kss_project/sample2/views.py: -------------------------------------------------------------------------------- 1 | from django.shortcuts import render 2 | 3 | # Create your views here. 4 | -------------------------------------------------------------------------------- /MANIFEST.in: -------------------------------------------------------------------------------- 1 | include README.rst 2 | recursive-include django_kss/static * 3 | recursive-include django_kss/templates * 4 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | *.pyc 2 | VENV 3 | venv 4 | dist/ 5 | django_kss.egg-info/ 6 | build/ 7 | django_kss_project/db.sqlite3 8 | *.swp 9 | -------------------------------------------------------------------------------- /django_kss_project/sample/styleguide.py: -------------------------------------------------------------------------------- 1 | styleguide = { 2 | 'source_dir': 'static/css', 3 | 'verbose_name': 'Sample APP', 4 | } -------------------------------------------------------------------------------- /django_kss/templates/styleguide-full-content.html: -------------------------------------------------------------------------------- 1 | {% extends 'styleguide_base.html' %} 2 | 3 | {% block content %} 4 | {% include inline_content %} 5 | {% endblock %} -------------------------------------------------------------------------------- /django_kss_project/sample2/styleguide.py: -------------------------------------------------------------------------------- 1 | styleguide = { 2 | 'source_dir': 'static/cxx', 3 | 'verbose_name': 'Sample APP2', 4 | 'target_files': 'static/cxx/form.scss' 5 | } -------------------------------------------------------------------------------- /django_kss/templates/styleguide-inline-content.html: -------------------------------------------------------------------------------- 1 | {% extends 'styleguide.html' %} 2 | 3 | {% block section %} 4 | Full Page Version 5 | {% include inline_content %} 6 | {% endblock %} -------------------------------------------------------------------------------- /.idea/inspectionProfiles/profiles_settings.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 7 | -------------------------------------------------------------------------------- /django_kss_project/static/css/forms.css: -------------------------------------------------------------------------------- 1 | /* 2 | Form 3 | 4 | Styleguide 2 5 | */ 6 | 7 | /* 8 | Form input. 9 | 10 | .error - When form error. 11 | 12 | Example: 13 | 14 | 15 | Styleguide 2.1 16 | */ 17 | input.error { 18 | border: 1px solid red; 19 | } 20 | 21 | 22 | 23 | 24 | 25 | 26 | 27 | -------------------------------------------------------------------------------- /django_kss_project/manage.py: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env python 2 | import os 3 | import sys 4 | 5 | dirname = os.path.dirname(os.path.dirname( os.path.abspath( __file__))) 6 | sys.path.append( 7 | dirname 8 | ) 9 | if __name__ == "__main__": 10 | os.environ.setdefault("DJANGO_SETTINGS_MODULE", "django_kss_project.settings") 11 | from django.core.management import execute_from_command_line 12 | execute_from_command_line(sys.argv) 13 | -------------------------------------------------------------------------------- /django_kss/pykss/modifier.py: -------------------------------------------------------------------------------- 1 | class Modifier(object): 2 | 3 | def __init__(self, name, description): 4 | self.name = name 5 | self.description = description 6 | self.example = '' 7 | 8 | @property 9 | def class_name(self): 10 | return self.name.replace('.', ' ').replace(':', ' pseudo-class-').strip() 11 | 12 | def add_example(self, example): 13 | self.example = example.replace('$modifier_class', '%s' % self.class_name) 14 | -------------------------------------------------------------------------------- /django_kss/static/css/forms.css: -------------------------------------------------------------------------------- 1 | /* 2 | Form input. 3 | 4 | .error - When form error. 5 | 6 | Example: 7 | 8 | 9 | Styleguide 2.1 10 | */ 11 | input.error { 12 | border: 1px solid red; } 13 | 14 | 15 | 16 | /* 17 | Form input. 18 | 19 | .error2 - When form error. 20 | 21 | Example: 22 | 23 | 24 | Styleguide 2.2 25 | */ 26 | input.error2 { 27 | border: 1px solid red; } 28 | -------------------------------------------------------------------------------- /django_kss_project/django_kss_project/wsgi.py: -------------------------------------------------------------------------------- 1 | """ 2 | WSGI config for django_kss_project project. 3 | 4 | It exposes the WSGI callable as a module-level variable named ``application``. 5 | 6 | For more information on this file, see 7 | https://docs.djangoproject.com/en/1.6/howto/deployment/wsgi/ 8 | """ 9 | 10 | import os 11 | os.environ.setdefault("DJANGO_SETTINGS_MODULE", "django_kss_project.settings") 12 | 13 | from django.core.wsgi import get_wsgi_application 14 | application = get_wsgi_application() 15 | -------------------------------------------------------------------------------- /django_kss_project/sample/static/css/form.css: -------------------------------------------------------------------------------- 1 | /* 2 | Form input sample. 3 | 4 | .error - When form error. 5 | 6 | Example: 7 | 8 | 9 | Styleguide 2.1 10 | */ 11 | input.error { 12 | border: 1px solid red; 13 | } 14 | 15 | 16 | 17 | /* 18 | 19 | Form input. 20 | 21 | .error2 - When form error. 22 | 23 | Example: 24 | 25 | 26 | Styleguide 2.2 27 | 28 | */ 29 | input.error2 { 30 | border: 1px solid red; 31 | } 32 | -------------------------------------------------------------------------------- /django_kss_project/sample2/static/cxx/form.scss: -------------------------------------------------------------------------------- 1 | /* 2 | Form input sample. 3 | 4 | .error - When form error. 5 | 6 | Example: 7 | 8 | 9 | Styleguide 1.1 10 | */ 11 | 12 | $color: yellow; 13 | input.error { 14 | border: 1px solid $color; 15 | } 16 | 17 | 18 | 19 | /* 20 | Form input. 21 | 22 | .error2 - When form error. 23 | 24 | Example: 25 | 26 | 27 | Styleguide 1.2 28 | */ 29 | input.error2 { 30 | border: 1px solid $color; 31 | } 32 | 33 | 34 | h1{ 35 | color: green; 36 | } 37 | -------------------------------------------------------------------------------- /django_kss_project/django_kss_project/urls.py: -------------------------------------------------------------------------------- 1 | from django.conf.urls import patterns, include, url 2 | from django.conf.urls.static import static 3 | from django.contrib.staticfiles.urls import staticfiles_urlpatterns 4 | from django.contrib import admin 5 | 6 | import django_kss.urls 7 | 8 | admin.autodiscover() 9 | 10 | urlpatterns = patterns( 11 | '', 12 | # Examples: 13 | # url(r'^$', 'django_kss_project.views.home', name='home'), 14 | # url(r'^blog/', include('blog.urls')), 15 | 16 | url(r'^admin/', include(admin.site.urls)), 17 | url(r'^', include(django_kss.urls)), 18 | ) 19 | -------------------------------------------------------------------------------- /django_kss/pykss/contrib/django/views.py: -------------------------------------------------------------------------------- 1 | from django.conf import settings 2 | from django.views.generic.base import TemplateView 3 | 4 | import pykss 5 | 6 | 7 | class StyleguideMixin(object): 8 | 9 | def get_styleguide(self): 10 | dirs = getattr(settings, 'PYKSS_DIRS', []) 11 | return pykss.Parser(*dirs) 12 | 13 | def get_context_data(self, **kwargs): 14 | context = super(StyleguideMixin, self).get_context_data(**kwargs) 15 | context.update({'styleguide': self.get_styleguide()}) 16 | return context 17 | 18 | 19 | class StyleguideView(StyleguideMixin, TemplateView): 20 | pass 21 | -------------------------------------------------------------------------------- /django_kss/urls.py: -------------------------------------------------------------------------------- 1 | from django.conf.urls import patterns, include, url 2 | from .views import AutoStyleGuideView, FullTemplateStyleGuideView, InlineTemplateStyleGuideView 3 | from . import utils 4 | 5 | def make_style_guide_pattern(template_name='styleguide.html'): 6 | view = AutoStyleGuideView.as_view(template_name=template_name) 7 | return patterns( 8 | '', 9 | url(r'^$', view, name='styleguide'), 10 | url(r'^full/(?P.*)/(?P.*\.html)/$', FullTemplateStyleGuideView.as_view(), name='prototype'), 11 | url(r'^(?P.*)/(?P.*\.html)/$', InlineTemplateStyleGuideView.as_view(), name='inline_prototype'), 12 | url(r'^(?P.*)/(?P
.*)/$', view, name='styleguide'), 13 | url(r'^(?P.*)/$', view, name='styleguide'), 14 | ) 15 | 16 | urlpatterns = make_style_guide_pattern() 17 | -------------------------------------------------------------------------------- /django_kss/templatetags/kss_extra.py: -------------------------------------------------------------------------------- 1 | from django import template 2 | from pygments import highlight 3 | from pygments.lexers import get_lexer_by_name 4 | from pygments.formatters import HtmlFormatter 5 | 6 | from pygments.util import ClassNotFound 7 | 8 | register = template.Library() 9 | 10 | # Don't use this in the future, just include pykss is okay 11 | 12 | @register.filter(name='highlight_code') 13 | def highlight_code(code, lang): 14 | if code is not None: 15 | try: 16 | lexer = get_lexer_by_name(lang, encoding='utf-8', stripall=True, startinline=True) 17 | except ClassNotFound: 18 | lexer = get_lexer_by_name('text') 19 | formatter = HtmlFormatter(encoding='utf-8', style='colorful', cssclass='highlight', 20 | lineanchors="line") 21 | return highlight(code, lexer, formatter) 22 | else: 23 | return code -------------------------------------------------------------------------------- /.idea/inspectionProfiles/Project_Default.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 23 | -------------------------------------------------------------------------------- /django_kss/setup.py: -------------------------------------------------------------------------------- 1 | import os 2 | from setuptools import setup 3 | 4 | BASE_DIR = os.path.dirname(__file__) 5 | 6 | with open(os.path.join(BASE_DIR, 'README.rst')) as readme: 7 | README = readme.read() 8 | 9 | # allow setup.py to be run from any path 10 | os.chdir(os.path.normpath(os.path.join(os.path.abspath(__file__), os.pardir))) 11 | 12 | setup( 13 | name='django-kss', 14 | version='0.1', 15 | packages=['django_kss'], 16 | include_package_data=True, 17 | license='BSD License', # example license 18 | description='A simple Django app to make styleguide', 19 | long_description=README, 20 | install_requires=['pykss'], 21 | url='http://www.example.com/', 22 | author='Tim Hsu', 23 | author_email='tim.yellow@gmail.com', 24 | classifiers=[ 25 | 'Environment :: Web Environment', 26 | 'Framework :: Django', 27 | 'Intended Audience :: Developers', 28 | 'License :: OSI Approved :: BSD License', 29 | 'Operating System :: OS Independent', 30 | 'Programming Language :: Python', 31 | # Replace these appropriately if you are stuck on Python 2. 32 | 'Topic :: Internet :: WWW/HTTP', 33 | 'Topic :: Internet :: WWW/HTTP :: Dynamic Content', 34 | ], 35 | ) 36 | -------------------------------------------------------------------------------- /django_kss/pykss/parser.py: -------------------------------------------------------------------------------- 1 | import os 2 | 3 | from .comment import CommentParser 4 | from .exceptions import SectionDoesNotExist 5 | from .section import Section 6 | 7 | 8 | class Parser(object): 9 | 10 | def __init__(self, *paths): 11 | self.paths = paths 12 | 13 | def parse(self): 14 | sections = {} 15 | 16 | filenames = [os.path.join(subpath, filename) 17 | for path in self.paths 18 | for subpath, dirs, files in os.walk(path) 19 | for filename in files] 20 | 21 | for filename in filenames: 22 | parser = CommentParser(filename) 23 | for block in parser.blocks: 24 | section = Section(block, os.path.basename(filename)) 25 | if section.section: 26 | sections[section.section] = section 27 | 28 | return sections 29 | 30 | @property 31 | def sections(self): 32 | if not hasattr(self, '_sections'): 33 | self._sections = self.parse() 34 | return self._sections 35 | 36 | def section(self, reference): 37 | try: 38 | return self.sections[reference] 39 | except KeyError: 40 | raise SectionDoesNotExist('Section "%s" does not exist.' % reference) 41 | -------------------------------------------------------------------------------- /setup.py: -------------------------------------------------------------------------------- 1 | import os 2 | from setuptools import setup, find_packages 3 | 4 | with open(os.path.join(os.path.dirname(__file__), 'README.rst')) as readme: 5 | README = readme.read() 6 | 7 | # allow setup.py to be run from any path 8 | os.chdir(os.path.normpath(os.path.join(os.path.abspath(__file__), os.pardir))) 9 | 10 | setup( 11 | name='django-kss-styleguide', 12 | version='0.5.8', 13 | packages=find_packages() , 14 | include_package_data=True, 15 | license='BSD License', # example license 16 | description='A simple Django app to make styleguide', 17 | long_description=README, 18 | url='https://github.com/timtan/django_kss', 19 | author='Tim Hsu', 20 | author_email='tim.yellow@gmail.com', 21 | install_requires = ['Pygments', 'django_compressor', 'django-libsass'], 22 | classifiers=[ 23 | 'Environment :: Web Environment', 24 | 'Framework :: Django', 25 | 'Intended Audience :: Developers', 26 | 'License :: OSI Approved :: BSD License', 27 | 'Operating System :: OS Independent', 28 | 'Programming Language :: Python', 29 | # Replace these appropriately if you are stuck on Python 2. 30 | 'Topic :: Internet :: WWW/HTTP', 31 | 'Topic :: Internet :: WWW/HTTP :: Dynamic Content', 32 | ], 33 | ) 34 | -------------------------------------------------------------------------------- /django_kss/pykss/contrib/django/templates/pykss/styleguideblock.html: -------------------------------------------------------------------------------- 1 |
2 | {% block styleguide_header %} 3 |

{{ section.section }} {{ section.filename }}

4 | {% endblock %} 5 | 6 | {% block styleguide_description %} 7 |
8 |

{{ section.description }}

9 | {% if section.modifiers %} 10 |
    11 | {% for modifier in section.modifiers %} 12 |
  • {{ modifier.name }} - {{ modifier.description }}
  • 13 | {% endfor %} 14 |
15 | {% endif %} 16 |
17 | {% endblock %} 18 | 19 | {% block styleguide_element %} 20 |
21 | {{ section.example|safe }} 22 |
23 | {% endblock %} 24 | 25 | {% block styleguide_modifier_elements %} 26 | {% for modifier in section.modifiers %} 27 |
28 | {{ modifier.name }} 29 | {{ modifier.example|safe }} 30 |
31 | {% endfor %} 32 | {% endblock %} 33 | 34 | {% block styleguide_code %} 35 |
{{ section.example|escape }}
36 | {% endblock %} 37 |
38 | -------------------------------------------------------------------------------- /django_kss/templates/styleguide_base.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | {% load static %} 5 | {% load pykss %} 6 | {% load compress %} 7 | 8 | 9 | 10 | 11 | 12 | Styleguide Example 13 | 14 | 15 | {% for user_css_file, is_scss in css_source_files %} 16 | {% if is_scss%} 17 | {% compress css %} 18 | 19 | {% endcompress %} 20 | {% else %} 21 | 22 | {% endif %} 23 | {% endfor %} 24 | 25 | {% block style %} {% endblock %} 26 | 29 | 30 | 31 | 32 | 33 | {% block content %} {% endblock %} 34 | {% block buttom %} {% endblock %} 35 | 36 | 37 | 38 | 39 | 40 | 41 | -------------------------------------------------------------------------------- /django_kss/templates/pykss/styleguideblock.html: -------------------------------------------------------------------------------- 1 | {% load kss_extra %} 2 | 3 |
4 | {% block styleguide_header %} 5 |

{{ section.section }} {{ section.filename }}

6 | {% endblock %} 7 | 8 | {% block styleguide_description %} 9 |
10 |

{{ section.description }}

11 | {% if section.modifiers %} 12 |
    13 | {% for modifier in section.modifiers %} 14 |
  • {{ modifier.name }} - {{ modifier.description }}
  • 15 | {% endfor %} 16 |
17 | {% endif %} 18 |
19 | {% endblock %} 20 | 21 | {% block styleguide_element %} 22 |
23 | {{ section.example|safe }} 24 |
25 | {% endblock %} 26 | 27 | {% block styleguide_modifier_elements %} 28 | {% for modifier in section.modifiers %} 29 |
30 | {{ modifier.name }} 31 | {{ modifier.example|safe }} 32 |
33 | {% endfor %} 34 | {% endblock %} 35 | 36 | 37 | {% block styleguide_code %} 38 | {%if section.example %} 39 |
40 | {{ section.example| highlight_code:'html' | safe}} 41 |
42 | {%endif%} 43 | {% endblock %} 44 | 45 | 46 |
47 | -------------------------------------------------------------------------------- /django_kss/templates/styleguide.html: -------------------------------------------------------------------------------- 1 | {% extends 'styleguide_base.html' %} 2 | {% load static %} 3 | {% load pykss %} 4 | {% load compress %} 5 | 6 | {% block content %} 7 | 8 |
9 |
10 |
11 | {% block header %} Django Style Guide {% endblock %} 12 |
13 |
    14 | {% for name, setting in styleguide_settings %} 15 |
  • {{ setting.verbose_name}}
  • 16 | {% endfor %} 17 |
18 |
19 |
20 | 21 | 22 |
23 | 24 | 25 |
    26 | {% for file_name in file_names %} 27 |
  • 28 | 29 | {{ file_name }} 30 | 31 |
  • 32 | {% endfor %} 33 | {% for html in htmls %} 34 |
  • 35 | 36 | {{ html }} 37 | 38 |
  • 39 | {% endfor %} 40 |
41 | 42 | 43 | {% block section %} 44 | {% renderstyleguide styleguide current_section %} 45 | {% endblock %} 46 | 47 | 48 | 49 |
50 | 51 | 52 | 53 | 54 | 55 | {% endblock %} 56 | -------------------------------------------------------------------------------- /django_kss/pykss/contrib/django/static/pykss/js/kss.js: -------------------------------------------------------------------------------- 1 | (function() { 2 | var KssStateGenerator; 3 | 4 | KssStateGenerator = (function() { 5 | var pseudo_selectors; 6 | 7 | pseudo_selectors = ['hover', 'enabled', 'disabled', 'active', 'visited', 'focus', 'target', 'checked', 'empty', 'first-of-type', 'last-of-type', 'first-child', 'last-child']; 8 | 9 | function KssStateGenerator() { 10 | var idx, idxs, pseudos, replaceRule, rule, stylesheet, _i, _len, _len2, _ref, _ref2; 11 | pseudos = new RegExp("(\\:" + (pseudo_selectors.join('|\\:')) + ")", "g"); 12 | try { 13 | _ref = document.styleSheets; 14 | for (_i = 0, _len = _ref.length; _i < _len; _i++) { 15 | stylesheet = _ref[_i]; 16 | if (stylesheet.href && stylesheet.href.indexOf(document.domain) >= 0) { 17 | idxs = []; 18 | _ref2 = stylesheet.cssRules; 19 | for (idx = 0, _len2 = _ref2.length; idx < _len2; idx++) { 20 | rule = _ref2[idx]; 21 | if ((rule.type === CSSRule.STYLE_RULE) && pseudos.test(rule.selectorText)) { 22 | replaceRule = function(matched, stuff) { 23 | return matched.replace(/\:/g, '.pseudo-class-'); 24 | }; 25 | this.insertRule(rule.cssText.replace(pseudos, replaceRule)); 26 | } 27 | pseudos.lastIndex = 0; 28 | } 29 | } 30 | } 31 | } catch (_error) {} 32 | } 33 | 34 | KssStateGenerator.prototype.insertRule = function(rule) { 35 | var headEl, styleEl; 36 | headEl = document.getElementsByTagName('head')[0]; 37 | styleEl = document.createElement('style'); 38 | styleEl.type = 'text/css'; 39 | if (styleEl.styleSheet) { 40 | styleEl.styleSheet.cssText = rule; 41 | } else { 42 | styleEl.appendChild(document.createTextNode(rule)); 43 | } 44 | return headEl.appendChild(styleEl); 45 | }; 46 | 47 | return KssStateGenerator; 48 | 49 | })(); 50 | 51 | new KssStateGenerator; 52 | 53 | }).call(this); -------------------------------------------------------------------------------- /django_kss/static/js/kss.js: -------------------------------------------------------------------------------- 1 | /** 2 | * Created by tim on 14/12/23. 3 | */ 4 | (function() { 5 | var KssStateGenerator; 6 | 7 | KssStateGenerator = (function() { 8 | var pseudo_selectors; 9 | 10 | pseudo_selectors = ['hover', 'enabled', 'disabled', 'active', 'visited', 'focus', 'target', 'checked', 'empty', 'first-of-type', 'last-of-type', 'first-child', 'last-child']; 11 | 12 | function KssStateGenerator() { 13 | var idx, idxs, pseudos, replaceRule, rule, stylesheet, _i, _len, _len2, _ref, _ref2; 14 | pseudos = new RegExp("(\\:" + (pseudo_selectors.join('|\\:')) + ")", "g"); 15 | try { 16 | _ref = document.styleSheets; 17 | for (_i = 0, _len = _ref.length; _i < _len; _i++) { 18 | stylesheet = _ref[_i]; 19 | if (stylesheet.href && stylesheet.href.indexOf(document.domain) >= 0) { 20 | idxs = []; 21 | _ref2 = stylesheet.cssRules; 22 | for (idx = 0, _len2 = _ref2.length; idx < _len2; idx++) { 23 | rule = _ref2[idx]; 24 | if ((rule.type === CSSRule.STYLE_RULE) && pseudos.test(rule.selectorText)) { 25 | replaceRule = function(matched, stuff) { 26 | return matched.replace(/\:/g, '.pseudo-class-'); 27 | }; 28 | this.insertRule(rule.cssText.replace(pseudos, replaceRule)); 29 | } 30 | pseudos.lastIndex = 0; 31 | } 32 | } 33 | } 34 | } catch (_error) {} 35 | } 36 | 37 | KssStateGenerator.prototype.insertRule = function(rule) { 38 | var headEl, styleEl; 39 | headEl = document.getElementsByTagName('head')[0]; 40 | styleEl = document.createElement('style'); 41 | styleEl.type = 'text/css'; 42 | if (styleEl.styleSheet) { 43 | styleEl.styleSheet.cssText = rule; 44 | } else { 45 | styleEl.appendChild(document.createTextNode(rule)); 46 | } 47 | return headEl.appendChild(styleEl); 48 | }; 49 | 50 | return KssStateGenerator; 51 | 52 | })(); 53 | 54 | new KssStateGenerator; 55 | 56 | }).call(this); -------------------------------------------------------------------------------- /django_kss/static/css/buttons.css: -------------------------------------------------------------------------------- 1 | /* 2 | Buttons 3 | 4 | Styleguide 1 5 | */ 6 | 7 | /* 8 | Your standard form button. 9 | 10 | :hover - Highlights when hovering. 11 | :disabled - Dims the button when disabled. 12 | .primary - Indicates button is the primary action. 13 | .smaller - A smaller button 14 | 15 | Styleguide 1.1 16 | */ 17 | button { 18 | padding: 5px 15px; 19 | line-height: normal; 20 | font-family: "Helvetica Neue", Helvetica; 21 | font-size: 12px; 22 | font-weight: bold; 23 | color: #666; 24 | text-shadow: 0 1px rgba(255, 255, 255, 0.9); 25 | border-radius: 3px; 26 | border: 1px solid #ddd; 27 | border-bottom-color: #bbb; 28 | background: #f5f5f5; 29 | filter: progid:DXImageTransform.Microsoft.gradient(GradientType=0, startColorstr='$start', endColorstr='$end'); 30 | background: -webkit-gradient(linear, left top, left bottom, from(#f5f5f5), to(#e5e5e5)); 31 | background: -moz-linear-gradient(top, #f5f5f5, #e5e5e5); 32 | box-shadow: 0 1px 4px rgba(0, 0, 0, 0.15); 33 | cursor: pointer; } 34 | button.primary, button.primary:hover { 35 | color: #fff; 36 | text-shadow: 0 -1px 0 rgba(0, 0, 0, 0.4); 37 | border-color: #74bb5a; 38 | border-bottom-color: #509338; 39 | background: #8add6d; 40 | filter: progid:DXImageTransform.Microsoft.gradient(GradientType=0, startColorstr='$start', endColorstr='$end'); 41 | background: -webkit-gradient(linear, left top, left bottom, from(#8add6d), to(#60b044)); 42 | background: -moz-linear-gradient(top, #8add6d, #60b044); 43 | box-shadow: 0 1px 4px rgba(0, 0, 0, 0.2); } 44 | button.smaller { 45 | font-size: 11px; 46 | padding: 4px 7px; } 47 | button:hover { 48 | color: #337797; 49 | background: #f0f7fa; 50 | filter: progid:DXImageTransform.Microsoft.gradient(GradientType=0, startColorstr='$start', endColorstr='$end'); 51 | background: -webkit-gradient(linear, left top, left bottom, from(#f0f7fa), to(#d8eaf2)); 52 | background: -moz-linear-gradient(top, #f0f7fa, #d8eaf2); 53 | border-color: #cbe3ee; 54 | border-bottom-color: #97c7dd; } 55 | button:disabled { 56 | opacity: 0.5; } 57 | -------------------------------------------------------------------------------- /django_kss_project/static/css/buttons.css: -------------------------------------------------------------------------------- 1 | /* 2 | Buttons 3 | 4 | Styleguide 1 5 | */ 6 | 7 | 8 | /* 9 | Your standard form button. 10 | 11 | :hover - Highlights when hovering. 12 | :disabled - Dims the button when disabled. 13 | .primary - Indicates button is the primary action. 14 | .smaller - A smaller button 15 | 16 | Example: 17 | 18 | 19 | Styleguide 1.1 20 | */ 21 | button { 22 | padding: 5px 15px; 23 | line-height: normal; 24 | font-family: "Helvetica Neue", Helvetica; 25 | font-size: 12px; 26 | font-weight: bold; 27 | color: #666; 28 | text-shadow: 0 1px rgba(255, 255, 255, 0.9); 29 | border-radius: 3px; 30 | border: 1px solid #ddd; 31 | border-bottom-color: #bbb; 32 | background: #f5f5f5; 33 | filter: progid:DXImageTransform.Microsoft.gradient(GradientType=0, startColorstr='$start', endColorstr='$end'); 34 | background: -webkit-gradient(linear, left top, left bottom, from(#f5f5f5), to(#e5e5e5)); 35 | background: -moz-linear-gradient(top, #f5f5f5, #e5e5e5); 36 | box-shadow: 0 1px 4px rgba(0, 0, 0, 0.15); 37 | cursor: pointer; } 38 | button.primary, button.primary:hover { 39 | color: #fff; 40 | text-shadow: 0 -1px 0 rgba(0, 0, 0, 0.4); 41 | border-color: #74bb5a; 42 | border-bottom-color: #509338; 43 | background: #8add6d; 44 | filter: progid:DXImageTransform.Microsoft.gradient(GradientType=0, startColorstr='$start', endColorstr='$end'); 45 | background: -webkit-gradient(linear, left top, left bottom, from(#8add6d), to(#60b044)); 46 | background: -moz-linear-gradient(top, #8add6d, #60b044); 47 | box-shadow: 0 1px 4px rgba(0, 0, 0, 0.2); } 48 | button.smaller { 49 | font-size: 11px; 50 | padding: 4px 7px; } 51 | button:hover { 52 | color: #337797; 53 | background: #f0f7fa; 54 | filter: progid:DXImageTransform.Microsoft.gradient(GradientType=0, startColorstr='$start', endColorstr='$end'); 55 | background: -webkit-gradient(linear, left top, left bottom, from(#f0f7fa), to(#d8eaf2)); 56 | background: -moz-linear-gradient(top, #f0f7fa, #d8eaf2); 57 | border-color: #cbe3ee; 58 | border-bottom-color: #97c7dd; } 59 | button:disabled { 60 | opacity: 0.5; } 61 | -------------------------------------------------------------------------------- /django_kss/pykss/section.py: -------------------------------------------------------------------------------- 1 | import re 2 | 3 | from .modifier import Modifier 4 | 5 | 6 | CLASS_MODIFIER = '.' 7 | PSEUDO_CLASS_MODIFIER = ':' 8 | MODIFIER_DESCRIPTION_SEPARATOR = ' - ' 9 | EXAMPLE_START = 'Example:' 10 | REFERENCE_START = 'Styleguide' 11 | 12 | reference_re = re.compile(r'%s ([\d\.]+)' % REFERENCE_START) 13 | optional_re = re.compile(r'\[(.*)\]\?') 14 | 15 | 16 | class Section(object): 17 | 18 | def __init__(self, comment=None, filename=None): 19 | self.comment = comment or '' 20 | self.filename = filename 21 | 22 | def parse(self): 23 | self._description_lines = [] 24 | self._modifiers = [] 25 | self._example_lines = [] 26 | self._reference = None 27 | 28 | in_example = False 29 | 30 | for line in self.comment.splitlines(): 31 | if line.startswith(CLASS_MODIFIER) or line.startswith(PSEUDO_CLASS_MODIFIER): 32 | try: 33 | modifier, description = line.split(MODIFIER_DESCRIPTION_SEPARATOR) 34 | except ValueError: 35 | pass 36 | else: 37 | self._modifiers.append(Modifier(modifier.strip(), description.strip())) 38 | 39 | elif line.startswith(EXAMPLE_START): 40 | in_example = True 41 | 42 | elif line.startswith(REFERENCE_START): 43 | in_example = False 44 | self._reference = reference_re.match(line).groups()[0].rstrip('.') 45 | 46 | elif in_example is True: 47 | self._example_lines.append(line) 48 | 49 | else: 50 | self._description_lines.append(line) 51 | 52 | self._description = '\n'.join(self._description_lines).strip() 53 | self.add_example('\n'.join(self._example_lines).strip()) 54 | 55 | @property 56 | def description(self): 57 | if not hasattr(self, '_description'): 58 | self.parse() 59 | return self._description 60 | 61 | @property 62 | def modifiers(self): 63 | if not hasattr(self, '_modifiers'): 64 | self.parse() 65 | return self._modifiers 66 | 67 | @property 68 | def example(self): 69 | if not hasattr(self, '_modifiers'): 70 | self.parse() 71 | return self._example 72 | 73 | @property 74 | def section(self): 75 | if not hasattr(self, '_reference'): 76 | self.parse() 77 | return self._reference 78 | 79 | def add_example(self, example): 80 | self._example = optional_re.sub('', example).replace('$modifier_class', '') 81 | for modifier in self._modifiers: 82 | modifier.add_example(optional_re.sub(r'\1', example)) 83 | -------------------------------------------------------------------------------- /django_kss_project/django_kss_project/settings.py: -------------------------------------------------------------------------------- 1 | """ 2 | Django settings for django_kss_project project. 3 | 4 | For more information on this file, see 5 | https://docs.djangoproject.com/en/1.6/topics/settings/ 6 | 7 | For the full list of settings and their values, see 8 | https://docs.djangoproject.com/en/1.6/ref/settings/ 9 | """ 10 | 11 | # Build paths inside the project like this: os.path.join(BASE_DIR, ...) 12 | import os 13 | BASE_DIR = os.path.dirname(os.path.dirname(__file__)) 14 | 15 | 16 | # Quick-start development settings - unsuitable for production 17 | # See https://docs.djangoproject.com/en/1.6/howto/deployment/checklist/ 18 | 19 | # SECURITY WARNING: keep the secret key used in production secret! 20 | SECRET_KEY = 'o&_8@#1cw8$2+%gr%%g_t$5y7+v((0#qzsc-f_x)4j#fiy(+75' 21 | 22 | # SECURITY WARNING: don't run with debug turned on in production! 23 | DEBUG = True 24 | 25 | TEMPLATE_DEBUG = True 26 | 27 | ALLOWED_HOSTS = ['*'] 28 | 29 | 30 | TEMPLATE_CONTEXT_PROCESSORS = ( 31 | 'django.contrib.auth.context_processors.auth', 32 | 'django.core.context_processors.debug', 33 | 'django.core.context_processors.i18n', 34 | 'django.core.context_processors.media', 35 | 'django.core.context_processors.static', 36 | 'django.core.context_processors.tz', 37 | 'django.contrib.messages.context_processors.messages', 38 | 'django.core.context_processors.request', 39 | ) 40 | # Application definition 41 | 42 | INSTALLED_APPS = ( 43 | 'django.contrib.admin', 44 | 'django.contrib.auth', 45 | 'django.contrib.contenttypes', 46 | 'django.contrib.sessions', 47 | 'django.contrib.messages', 48 | 'django.contrib.staticfiles', 49 | "compressor", 50 | 'django_kss', 51 | 'sample', 52 | 'sample2', 53 | ) 54 | 55 | MIDDLEWARE_CLASSES = ( 56 | 'django.contrib.sessions.middleware.SessionMiddleware', 57 | 'django.middleware.common.CommonMiddleware', 58 | 'django.middleware.csrf.CsrfViewMiddleware', 59 | 'django.contrib.auth.middleware.AuthenticationMiddleware', 60 | 'django.contrib.messages.middleware.MessageMiddleware', 61 | 'django.middleware.clickjacking.XFrameOptionsMiddleware', 62 | ) 63 | 64 | ROOT_URLCONF = 'django_kss_project.urls' 65 | 66 | WSGI_APPLICATION = 'django_kss_project.wsgi.application' 67 | 68 | 69 | # Database 70 | # https://docs.djangoproject.com/en/1.6/ref/settings/#databases 71 | 72 | DATABASES = { 73 | 'default': { 74 | 'ENGINE': 'django.db.backends.sqlite3', 75 | 'NAME': os.path.join(BASE_DIR, 'db.sqlite3'), 76 | } 77 | } 78 | 79 | # Internationalization 80 | # https://docs.djangoproject.com/en/1.6/topics/i18n/ 81 | 82 | LANGUAGE_CODE = 'en-us' 83 | 84 | TIME_ZONE = 'UTC' 85 | 86 | USE_I18N = True 87 | 88 | USE_L10N = True 89 | 90 | USE_TZ = True 91 | 92 | 93 | # Static files (CSS, JavaScript, Images) 94 | # https://docs.djangoproject.com/en/1.6/howto/static-files/ 95 | 96 | STATICFILES_DIRS = ( 97 | os.path.join(BASE_DIR, "static"), 98 | ) 99 | 100 | STATIC_URL = '/static/' 101 | 102 | 103 | COMPRESS_PRECOMPILERS = ( 104 | ('text/x-scss', 'django_libsass.SassCompiler'), 105 | ) 106 | STATICFILES_FINDERS = ( 107 | 'django.contrib.staticfiles.finders.FileSystemFinder', 108 | 'django.contrib.staticfiles.finders.AppDirectoriesFinder', 109 | 'compressor.finders.CompressorFinder', 110 | ) 111 | # Django Compressor for development. so it can put image to correct place 112 | COMPRESS_ENABLED = True 113 | COMPRESS_REBUILD_TIMEOUT = 0 114 | 115 | STATIC_ROOT = '/tmp/root' -------------------------------------------------------------------------------- /django_kss/pykss/comment.py: -------------------------------------------------------------------------------- 1 | import re 2 | 3 | 4 | single_line_re = re.compile(r'^\s*\/\/') 5 | single_line_strip_re = re.compile(r'\s*\/\/') 6 | 7 | multi_line_start_re = re.compile(r'^\s*\/\*') 8 | multi_line_end_re = re.compile(r'.*\*\/') 9 | multi_line_start_strip_re = re.compile(r'\s*\/\*') 10 | multi_line_end_strip_re = re.compile(r'\*\/') 11 | multi_line_middle_strip_re = re.compile(r'^(\s*\*+)') 12 | 13 | preceding_white_space_re = re.compile(r'^\s*') 14 | 15 | 16 | def is_single_line_comment(line): 17 | return single_line_re.match(line) is not None 18 | 19 | 20 | def is_multi_line_comment_start(line): 21 | return multi_line_start_re.match(line) is not None 22 | 23 | 24 | def is_multi_line_comment_end(line): 25 | if is_single_line_comment(line): 26 | return False 27 | return multi_line_end_re.match(line) is not None 28 | 29 | 30 | def parse_single_line(line): 31 | return single_line_strip_re.sub('', line).rstrip() 32 | 33 | 34 | def parse_multi_line(line): 35 | cleaned = multi_line_start_strip_re.sub('', line) 36 | return multi_line_end_strip_re.sub('', cleaned).rstrip() 37 | 38 | 39 | def normalize(lines): 40 | cleaned = [] 41 | indents = [] 42 | 43 | for line in lines: 44 | line = multi_line_middle_strip_re.sub('', line) 45 | cleaned.append(line) 46 | match = preceding_white_space_re.match(line) 47 | if line: 48 | indents.append(len(match.group())) 49 | 50 | indent = min(indents) if indents else 0 51 | 52 | return '\n'.join([line[indent:] for line in cleaned]).strip() 53 | 54 | 55 | class CommentParser(object): 56 | 57 | def __init__(self, filename): 58 | self.filename = filename 59 | 60 | def parse(self): 61 | blocks = [] 62 | current_block = [] 63 | inside_single_line_block = False 64 | inside_multi_line_block = False 65 | 66 | with open(self.filename) as fileobj: 67 | for line in fileobj: 68 | # Parse single-line style 69 | if is_single_line_comment(line): 70 | parsed = parse_single_line(line) 71 | 72 | if inside_single_line_block: 73 | current_block.append(parsed) 74 | else: 75 | current_block = [parsed] 76 | inside_single_line_block = True 77 | 78 | # Prase multi-line style 79 | if is_multi_line_comment_start(line) or inside_multi_line_block: 80 | parsed = parse_multi_line(line) 81 | 82 | if inside_multi_line_block: 83 | current_block.append(parsed) 84 | else: 85 | current_block = [parsed] 86 | inside_multi_line_block = True 87 | 88 | # End a multi-line block if detected 89 | if is_multi_line_comment_end(line): 90 | inside_multi_line_block = False 91 | 92 | # Store the current block if we're done 93 | if is_single_line_comment(line) is False and inside_multi_line_block is False: 94 | if current_block: 95 | blocks.append(normalize(current_block)) 96 | 97 | inside_single_line_block = False 98 | current_block = [] 99 | 100 | return blocks 101 | 102 | @property 103 | def blocks(self): 104 | if not hasattr(self, '_blocks'): 105 | self._blocks = self.parse() 106 | return self._blocks 107 | -------------------------------------------------------------------------------- /README.rst: -------------------------------------------------------------------------------- 1 | 2 | ************ 3 | introduction 4 | ************ 5 | 6 | Help you launch your style guide site with django in KSS syntax. you can integrate the style giude in your site easily. 7 | I use it every day. feel free to give me any suggestion. 8 | 9 | I now change the name to django-kss-styleguide https://pypi.python.org/pypi/django-kss-styleguide/ 10 | 11 | .. image:: pictures/screenshot.png 12 | 13 | ===== 14 | start 15 | ===== 16 | 17 | pip install django-kss-styleguide 18 | 19 | 20 | 21 | ======== 22 | Settings 23 | ======== 24 | 25 | in yout settings.py 26 | 27 | Add the app, 28 | 29 | .. code-block:: python 30 | 31 | INSTALLED_APPS += ( 32 | "compressor", 33 | "django_kss", 34 | ) 35 | 36 | 37 | ================ 38 | Related Settings 39 | ================ 40 | 41 | in settings.py 42 | 43 | because scss is very common, we support it via djagno compressor 44 | Add setting about django compressor. 45 | 46 | .. code-block:: python 47 | 48 | COMPRESS_PRECOMPILERS = ( 49 | ('text/x-scss', 'django_libsass.SassCompiler'), 50 | ) 51 | STATICFILES_FINDERS = ( 52 | 'django.contrib.staticfiles.finders.FileSystemFinder', 53 | 'django.contrib.staticfiles.finders.AppDirectoriesFinder', 54 | 'compressor.finders.CompressorFinder', 55 | ) 56 | # Django Compressor for development. so it can put image to correct place 57 | COMPRESS_ENABLED = True 58 | COMPRESS_REBUILD_TIMEOUT = 0 59 | 60 | STATIC_ROOT = '/tmp/root' 61 | 62 | in your app. 63 | 64 | add filename called styleguide.py in your app. 65 | 66 | .. code-block:: python 67 | 68 | styleguide = { 69 | 'source_dir': 'static/css', 70 | 'verbose_name': 'Sample APP2', #Optional 71 | 'target_files': 'static/css/form.scss' # optional 72 | } 73 | 74 | 75 | source_dir: Where you write you kss comment and css files 76 | 77 | verbose_name: Your app name 78 | 79 | target_files: If you use scss, put the scss file you want to compile 80 | 81 | 82 | urls.py settings 83 | ================ 84 | 85 | Routing, add the following two lines in your project's urls.py 86 | 87 | import: 88 | 89 | .. code-block:: python 90 | 91 | import django_kss.urls 92 | 93 | add the url patterns: 94 | 95 | .. code-block:: python 96 | 97 | url(r'^$', include(django_kss.urls)), 98 | 99 | 100 | 101 | html 102 | ==== 103 | 104 | for F2E or designer 105 | 106 | put your complete html in templates/prototype/ 107 | 108 | you can view it automatically in the site 109 | 110 | 111 | Writing The KSS in your scss/less/css file 112 | ========================================== 113 | 114 | 115 | .. code-block:: scss 116 | 117 | /* 118 | Buttons 119 | 120 | Styleguide 1 121 | */ 122 | 123 | 124 | /* 125 | Your standard form button. 126 | 127 | .btn-danger - danger 128 | .btn-warning - warning 129 | .btn-info - info 130 | 131 | 132 | Example: 133 | 134 | 135 | Styleguide 1.1 136 | */ 137 | 138 | .liftedBtn{ 139 | @extend .btn; 140 | position: relative; 141 | border-width: 0; 142 | letter-spacing: 1px; 143 | border-bottom-color: rgba(30,30,30,0.3); 144 | border-bottom-width: 0; 145 | transition: all 0.2s; 146 | bottom: 0; 147 | &:hover{ 148 | bottom: $strong-border-width; 149 | border-bottom-width: $strong-border-width; 150 | } 151 | } 152 | 153 | 154 | 155 | 156 | =============== 157 | One More Things 158 | =============== 159 | 160 | the base template is already integrate livereload. 161 | 162 | to utilize livereload, 163 | 164 | 165 | .. code-block:: bash 166 | 167 | sudo pip install livereload 168 | 169 | 170 | and type 'livereload .' in you repository root. 171 | 172 | you can see all the braowser will reflect your change 173 | 174 | 175 | 176 | -------------------------------------------------------------------------------- /django_kss/utils.py: -------------------------------------------------------------------------------- 1 | from __future__ import print_function 2 | import imp 3 | import logging 4 | import os 5 | import os.path 6 | import sys 7 | from importlib import import_module 8 | from django.conf import settings 9 | 10 | 11 | logger = logging.getLogger(__name__) 12 | 13 | 14 | def _get_styleguide_module(module_name='styleguide'): 15 | 16 | style_guides = [] 17 | modules_to_check = [] 18 | 19 | # Hackish: do this in case we have some project top-level 20 | # (homepage, etc) urls defined project-level instead of app-level. 21 | settings_module = settings.SETTINGS_MODULE 22 | if settings_module: 23 | if "." in settings_module: 24 | # strip off '.settings" from end of module 25 | # (want project module, if possible) 26 | settings_module = settings_module.split(".", 1)[0] 27 | modules_to_check += [settings_module, ] 28 | 29 | # INSTALLED_APPS that aren't the project itself (also ignoring this 30 | # django_medusa module) 31 | modules_to_check += filter( 32 | lambda x: (x != "django_kss") and (x != settings_module), 33 | settings.INSTALLED_APPS 34 | ) 35 | 36 | for app in modules_to_check: 37 | try: 38 | import_module(app) 39 | app_path = sys.modules[app].__path__ 40 | except AttributeError: 41 | logger.debug("Skipping app '%s'... (Not found)", app) 42 | continue 43 | try: 44 | imp.find_module(module_name, app_path) 45 | except ImportError: 46 | logger.debug("Skipping app '%s'... (No 'styleguide.py')" , app) 47 | continue 48 | try: 49 | app_styleguide_module = import_module('%s.%s' % (app, module_name)) 50 | if hasattr(app_styleguide_module, "styleguide"): 51 | style_guides.append(app_styleguide_module) 52 | else: 53 | logger.debug("Skipping app '%s'... ('%s.styleguide' does not contain "\ 54 | "'styleguide' var (list of styleguide classes)" , (app, app)) 55 | except AttributeError: 56 | logger.debug("Skipping app '%s'... (Error importing '%s.styleguide')" , ( 57 | app, app 58 | )) 59 | continue 60 | logger.debug("Found styleguide for '%s'..." , app) 61 | return style_guides 62 | 63 | 64 | def _remove_static(target_file): 65 | if target_file.startswith('static/'): 66 | return target_file.replace('static/', '') 67 | return target_file 68 | 69 | 70 | def _complete_setting(m): 71 | app_path = os.path.dirname(m.__file__) 72 | 73 | try: 74 | source_dir = m.styleguide['source_dir'] 75 | except KeyError: 76 | source_dir = 'static/css' 77 | abs_source_dir = os.path.join(app_path, source_dir) 78 | print(source_dir) 79 | 80 | try: 81 | verbose_name = m.styleguide['verbose_name'] 82 | except KeyError: 83 | verbose_name = m.__name__ 84 | 85 | try: 86 | target_files = m.styleguide['target_files'] 87 | if isinstance(target_files, str): 88 | target_files = [target_files] 89 | except KeyError: 90 | target_files = os.listdir(abs_source_dir) 91 | target_files = map(lambda x: os.path.join(source_dir, x), target_files) 92 | 93 | try: 94 | htmls = os.listdir(os.path.join(app_path, 'templates', 'prototype')) 95 | except OSError: 96 | htmls = [] 97 | 98 | target_files = map(_remove_static, target_files) 99 | 100 | styleguide = m.styleguide.copy() 101 | styleguide['source_dir'] = abs_source_dir 102 | styleguide['app_name'] = m.__name__ 103 | styleguide['verbose_name'] = verbose_name 104 | styleguide['target_files'] = target_files 105 | styleguide['htmls'] = htmls 106 | return styleguide 107 | 108 | 109 | def get_styleguide(): 110 | safe_settings = map(_complete_setting, _get_styleguide_module()) 111 | setting_map = {} 112 | for setting in safe_settings: 113 | setting_map[setting['app_name']] = setting 114 | return setting_map 115 | 116 | 117 | -------------------------------------------------------------------------------- /django_kss/views.py: -------------------------------------------------------------------------------- 1 | import os.path 2 | from pygments.formatters import HtmlFormatter 3 | from django.utils.functional import cached_property 4 | 5 | from django import conf 6 | from django.views.generic.base import TemplateView 7 | from django.shortcuts import render 8 | 9 | 10 | from . import utils 11 | from . import pykss 12 | 13 | 14 | 15 | def render_prototype(request, html): 16 | prototype_directory = dirs = getattr(conf.settings, 'PROTOTYPR_DIR', "prototype") 17 | return render(request, "styleguide-full-content.html", 18 | { 'inline_content': os.path.join(prototype_directory, html)}) 19 | 20 | 21 | class StyleguideMixin(object): 22 | def get_styleguide(self): 23 | dirs = getattr(conf.settings, 'PYKSS_DIRS', []) 24 | return pykss.Parser(*dirs) 25 | 26 | def get_context_data(self, **kwargs): 27 | context = super(StyleguideMixin, self).get_context_data(**kwargs) 28 | context.update({'styleguide': self.get_styleguide()}) 29 | return context 30 | 31 | 32 | class StyleguideView(StyleguideMixin, TemplateView): 33 | pass 34 | 35 | 36 | class AutoStyleGuideView(TemplateView): 37 | 38 | def styleguide_settings(self, settings): 39 | return settings.items() 40 | 41 | def domain(self): 42 | host = self.request.META['HTTP_HOST'] 43 | domain = host.split(":")[0] 44 | return domain 45 | 46 | def _get_settings(self): 47 | return utils.get_styleguide() 48 | 49 | def _get_setting(self, settings): 50 | try: 51 | app_name = self.kwargs['app_name'] 52 | return settings[app_name] 53 | except KeyError: 54 | return settings.values()[0] 55 | 56 | def get_styleguide(self, settings): 57 | setting = self._get_setting(settings) 58 | return pykss.Parser(setting['source_dir']) 59 | 60 | def css_source_files(self, settings): 61 | return map(lambda fn: [fn, fn.endswith('scss')], self._get_setting(settings)['target_files']) 62 | 63 | @cached_property 64 | def pygament_style(self): 65 | return HtmlFormatter().get_style_defs('.highlight') 66 | 67 | def get_context_data(self, **kwargs): 68 | 69 | context = super(AutoStyleGuideView, self).get_context_data(**kwargs) 70 | 71 | settings = self._get_settings() 72 | context.update({'styleguide': self.get_styleguide(settings)}) 73 | context.update({'styleguide_settings': self.styleguide_settings(settings)}) 74 | context.update({'css_source_files': self.css_source_files(settings)}) 75 | 76 | styleguide = context["styleguide"] 77 | file_names = set(map(lambda section: section.filename, styleguide.sections.values())) 78 | 79 | if 'section' not in self.kwargs or not self.kwargs['section']: 80 | current_section = styleguide.sections.values()[0].section 81 | else: 82 | target_filename = self.kwargs['section'] 83 | possible = filter(lambda section: section.filename == target_filename, styleguide.sections.values()) 84 | current_section = possible[0].section 85 | current_section_prefix = current_section.split(".")[0] 86 | 87 | context.update({'app_name': self._get_setting(settings)['app_name']}) 88 | context.update({'current_section': current_section_prefix}) 89 | context.update({'file_names': file_names}) 90 | context.update({'htmls': self._get_setting(settings)['htmls']}) 91 | return context 92 | 93 | 94 | class InlineTemplateStyleGuideView(AutoStyleGuideView): 95 | 96 | template_name = 'styleguide-inline-content.html' 97 | 98 | def get_context_data(self, **kwargs): 99 | context = super(InlineTemplateStyleGuideView, self).get_context_data(**kwargs) 100 | context.update({'inline_content': 'prototype/' + self.kwargs['html']}) 101 | context.update({'html': self.kwargs['html']}) 102 | return context 103 | 104 | class FullTemplateStyleGuideView(AutoStyleGuideView): 105 | 106 | template_name = 'styleguide-full-content.html' 107 | 108 | def get_context_data(self, **kwargs): 109 | context = super(FullTemplateStyleGuideView, self).get_context_data(**kwargs) 110 | context.update({'inline_content': 'prototype/' + self.kwargs['html']}) 111 | return context 112 | -------------------------------------------------------------------------------- /django_kss/templatetags/pykss.py: -------------------------------------------------------------------------------- 1 | from django.template.loader import render_to_string 2 | from django import template 3 | from pygments import highlight 4 | from pygments.lexers import get_lexer_by_name 5 | from pygments.formatters import HtmlFormatter 6 | 7 | from pygments.util import ClassNotFound 8 | 9 | 10 | register = template.Library() 11 | 12 | 13 | @register.filter(name='highlight_code') 14 | def highlight_code(code, lang): 15 | if code is not None: 16 | try: 17 | lexer = get_lexer_by_name(lang, encoding='utf-8', stripall=True, startinline=True) 18 | except ClassNotFound: 19 | lexer = get_lexer_by_name('text') 20 | formatter = HtmlFormatter(encoding='utf-8', style='colorful', cssclass='highlight', 21 | lineanchors="line") 22 | return highlight(code, lexer, formatter) 23 | else: 24 | return code 25 | 26 | 27 | class BaseStyleguideNode(template.Node): 28 | 29 | def __init__(self, styleguide, reference, template_name, nodelist): 30 | self.styleguide = styleguide 31 | self.reference = reference 32 | self.template_name = template_name 33 | self.nodelist = nodelist 34 | 35 | def __repr__(self): 36 | return '' 37 | 38 | @classmethod 39 | def as_tag(cls, parser, token): 40 | bits = token.split_contents()[1:] 41 | 42 | if len(bits) < 2: 43 | raise template.TemplateSyntaxError("styleguideblock expected at " 44 | "least two arguments") 45 | 46 | elif len(bits) == 2: 47 | styleguide, reference = bits 48 | template_name = '"pykss/styleguideblock.html"' 49 | 50 | elif len(bits) >= 3 and bits[2] != 'using': 51 | raise template.TemplateSyntaxError("styleguideblock expected using " 52 | "as the third argument") 53 | 54 | elif len(bits) == 3: 55 | raise template.TemplateSyntaxError("styleguideblock expects a " 56 | "template name after 'using'") 57 | 58 | else: 59 | styleguide, reference, _using, template_name = bits 60 | 61 | return cls.dispatch(parser, styleguide, reference, template_name) 62 | 63 | @classmethod 64 | def dispatch(cls, parser, styleguide, reference, template_name): 65 | raise NotImplementedError 66 | 67 | def render(self, context): 68 | styleguide = self.styleguide.resolve(context) 69 | reference = self.reference.resolve(context) 70 | template_name = self.template_name.resolve(context) 71 | 72 | sections = sorted([sec for ref, sec in styleguide.sections.iteritems() 73 | if ref.startswith(reference)], key=lambda s: s.section) 74 | 75 | if self.nodelist: 76 | example = self.nodelist.render(context).strip() 77 | for section in sections: 78 | section.add_example(example) 79 | 80 | output = [] 81 | 82 | for section in sections: 83 | context.update({'section': section}) 84 | html = render_to_string(template_name, context) 85 | output.append(html) 86 | context.pop() 87 | 88 | return ''.join(output) 89 | 90 | 91 | class StyleguideBlockNode(BaseStyleguideNode): 92 | 93 | @classmethod 94 | def dispatch(cls, parser, styleguide, reference, template_name): 95 | nodelist = parser.parse(('endstyleguideblock',)) 96 | parser.delete_first_token() 97 | return cls( 98 | styleguide=parser.compile_filter(styleguide), 99 | reference=parser.compile_filter(reference), 100 | template_name=parser.compile_filter(template_name), 101 | nodelist=nodelist, 102 | ) 103 | 104 | 105 | class RenderStyleguideNode(BaseStyleguideNode): 106 | 107 | @classmethod 108 | def dispatch(cls, parser, styleguide, reference, template_name): 109 | return cls( 110 | styleguide=parser.compile_filter(styleguide), 111 | reference=parser.compile_filter(reference), 112 | template_name=parser.compile_filter(template_name), 113 | nodelist=None, 114 | ) 115 | 116 | 117 | @register.tag 118 | def styleguideblock(parser, token): 119 | """ 120 | {% styleguideblock styleguide "1.1" %} 121 | 122 | {% endstyleguideblock %} 123 | 124 | {% styleguideblock styleguide "1.1" using "custom.html" %} 125 | 126 | {% endstyleguideblock %} 127 | 128 | """ 129 | return StyleguideBlockNode.as_tag(parser, token) 130 | 131 | 132 | @register.tag 133 | def renderstyleguide(parser, token): 134 | """ 135 | {% renderstyleguide styleguide "1.1" %} 136 | 137 | {% renderstyleguide styleguide "1.1" using "custom.html" %} 138 | """ 139 | return RenderStyleguideNode.as_tag(parser, token) 140 | -------------------------------------------------------------------------------- /django_kss/static/css/layout.css: -------------------------------------------------------------------------------- 1 | /*---------------------------------------------------------------------------- 2 | @group Global Reset 3 | ----------------------------------------------------------------------------*/ 4 | /* 5 | 6 | .access{ display:none; } /* For accessibility related elements */ 7 | 8 | .clear{ clear:both; height:0px; font-size:0px; line-height:0px; overflow:hidden; } 9 | 10 | 11 | @media (max-width: 767px) { 12 | .styleguide-hidden-xs { 13 | display: none !important; 14 | } 15 | } 16 | 17 | /* 18 | .styleguide-header{ 19 | padding:10px; 20 | 21 | font-size:16px; 22 | color:#666; 23 | 24 | background:#f1f1f1; 25 | border-bottom:1px solid #ddd; 26 | font-weight: bold; 27 | } 28 | */ 29 | 30 | #wrapper{ 31 | min-width:600px; 32 | padding-left:200px; 33 | } 34 | @media (max-width: 767px) { 35 | .styleguide-hidden-xs { 36 | display: none !important; 37 | } 38 | #wrapper{ 39 | width:100%; 40 | max-width: 100%; 41 | min-width: 0; 42 | padding-left: 0; 43 | } 44 | } 45 | 46 | [role=main]{ 47 | float:left; 48 | margin-left:-200px; 49 | width:160px; 50 | } 51 | 52 | 53 | /*---------------------------------------------------------------------------- 54 | @group Styleguide Styles 55 | ----------------------------------------------------------------------------*/ 56 | 57 | .styleguide { 58 | margin: 0 0 -5px 0; 59 | font-size: 24px; 60 | color: #000; } 61 | 62 | .styleguide-example { 63 | margin: 15px 0; 64 | background: rgba(255, 255, 255, 0.5); 65 | border: 1px solid #ddd; 66 | box-shadow: 0 0 5px rgba(0, 0, 0, 0.1); } 67 | .styleguide-example > h3 { 68 | margin: 0; 69 | padding: 5px; 70 | color: #fff; 71 | font-size: 12px; 72 | text-transform: uppercase; 73 | background: #333; 74 | border-top: 1px solid #000; } 75 | .styleguide-example > h3 em { 76 | float: right; 77 | text-transform: none; 78 | font-style: normal; 79 | font-weight: normal; 80 | color: #999; } 81 | .styleguide-example .styleguide-description { 82 | padding: 10px 5px; 83 | background: #f1f1f1; 84 | border-bottom: 1px solid #ddd; } 85 | .styleguide-example .styleguide-description p:first-child { 86 | margin-top: 0; } 87 | .styleguide-example .styleguide-description p:last-child { 88 | margin-bottom: 0; } 89 | .styleguide-example .styleguide-element { 90 | position: relative; 91 | padding: 20px; } 92 | .styleguide-example .styleguide-element + .styleguide-element { 93 | margin-top: -5px; 94 | padding-top: 15px; 95 | border-top: 1px solid #eee; } 96 | .styleguide-example .styleguide-element .styleguide-modifier-name { 97 | display: block; 98 | background-color: rgba(0,0,0,0.5); 99 | /* position: absolute; */ 100 | /* top: 0; */ 101 | /* right: 0; */ 102 | /* padding: 5px 8px; */ 103 | font-size: 14px; 104 | color: gray; 105 | background: rgba(0,0,0,0.1); 106 | border: 1px solid #eee; 107 | border-top: none; 108 | } 109 | .styleguide-example .styleguide-html { 110 | padding: 5px 10px; 111 | background: #edf6f8; 112 | border-top: 1px solid #dde7ea; 113 | overflow: auto; } 114 | .styleguide-example .styleguide-html .highlight { 115 | background: none; } 116 | .styleguide-example ul.styleguide-modifiers { 117 | margin: 0 0 0 10px; } 118 | .styleguide-example ul.styleguide-modifiers li { 119 | list-style-type: none; 120 | margin-left: 0; } 121 | .styleguide-example ul.styleguide-modifiers li strong { 122 | font-family: Monaco, monospace; 123 | font-size: 12px; 124 | font-weight: normal; 125 | color: #222; } 126 | .styleguide-example > .styleguide-code { 127 | font-family: Monaco, monospace; 128 | } 129 | 130 | .styleguide-modifier li{ 131 | } 132 | 133 | .styleguide-nav{ 134 | box-sizing: border-box; 135 | color: rgb(51, 51, 51); 136 | display: block; 137 | font-family: 'Helvetica Neue', Helvetica, Arial, sans-serif; 138 | font-size: 14px; 139 | height: 124px; 140 | line-height: 20px; 141 | list-style-image: none; 142 | list-style-position: outside; 143 | list-style-type: none; 144 | margin-bottom: 0px; 145 | margin-top: 0px; 146 | max-width: 300px; 147 | padding-left: 0px; 148 | } 149 | 150 | .styleguide-nav-list{ 151 | box-sizing: border-box; 152 | color: rgb(51, 51, 51); 153 | display: block; 154 | float: none; 155 | font-family: 'Helvetica Neue', Helvetica, Arial, sans-serif; 156 | font-size: 14px; 157 | min-height: 40px; 158 | line-height: 20px; 159 | list-style-image: none; 160 | list-style-position: outside; 161 | list-style-type: none; 162 | margin-left: 0px; 163 | position: relative; 164 | } 165 | 166 | .styleguid-nav-anchor{ 167 | background-color: rgb(238, 238, 238); 168 | border-bottom-left-radius: 0; 169 | border-bottom-right-radius: 0; 170 | border-top-left-radius: 0; 171 | border-top-right-radius: 0; 172 | box-sizing: border-box; 173 | color: rgb(42, 100, 150); 174 | display: block; 175 | font-family: 'Helvetica Neue', Helvetica, Arial, sans-serif; 176 | font-size: 14px; 177 | min-height: 40px; 178 | line-height: 20px; 179 | list-style-image: none; 180 | list-style-position: outside; 181 | list-style-type: none; 182 | outline-color: rgb(42, 100, 150); 183 | outline-style: none; 184 | outline-width: 0px; 185 | padding-bottom: 10px; 186 | padding-left: 15px; 187 | padding-right: 15px; 188 | padding-top: 10px; 189 | position: relative; 190 | text-decoration: none; 191 | } 192 | .active .styleguid-nav-anchor{ 193 | background-color: rgb(66, 139, 202); 194 | box-sizing: border-box; 195 | color: rgb(255, 255, 255); 196 | display: block; 197 | font-family: 'Helvetica Neue', Helvetica, Arial, sans-serif; 198 | font-size: 14px; 199 | height: 40px; 200 | line-height: 20px; 201 | list-style-image: none; 202 | list-style-position: outside; 203 | list-style-type: none; 204 | padding-bottom: 10px; 205 | padding-left: 15px; 206 | padding-right: 15px; 207 | padding-top: 10px; 208 | position: relative; 209 | text-decoration: none; 210 | } 211 | /* @end */ 212 | 213 | 214 | .styleguide-element{ 215 | background-image: url(); 216 | } 217 | 218 | .styleguide-header .brand{ 219 | background-color: #444; 220 | float: left; 221 | font-size: 1.2em; 222 | margin-left: 20px; 223 | color: #fff; 224 | } 225 | 226 | .styleguide-header .nav ul { 227 | list-style: none; 228 | background-color: #444; 229 | text-align: center; 230 | padding: 0; 231 | margin: 0; 232 | } 233 | .styleguide-header .nav li { 234 | font-family: 'Oswald', sans-serif; 235 | font-size: 1.2em; 236 | line-height: 40px; 237 | height: 40px; 238 | border-bottom: 1px solid #888; 239 | } 240 | 241 | .styleguide-header .nav a { 242 | text-decoration: none; 243 | color: #fff; 244 | display: block; 245 | transition: .3s background-color; 246 | padding-left: 10px; 247 | padding-right: 10px; 248 | } 249 | 250 | .styleguide-header .nav a:hover, .styleguide-header .nav .active { 251 | background-color: #005f5f; 252 | } 253 | 254 | .styleguide-header .nav a.active { 255 | background-color: #fff; 256 | color: #444; 257 | cursor: default; 258 | } 259 | 260 | @media screen and (min-width: 600px) { 261 | .styleguide-header .nav li , .styleguide-header .nav .brand{ 262 | min-width: 120px; 263 | border-bottom: none; 264 | height: 50px; 265 | line-height: 50px; 266 | font-size: 1.4em; 267 | } 268 | 269 | /* Option 1 - Display Inline */ 270 | .styleguide-header .nav li { 271 | display: inline-block; 272 | margin-right: -4px; 273 | margin-left: 10px; 274 | } 275 | 276 | /* Options 2 - Float 277 | .nav li { 278 | float: left; 279 | } 280 | .nav ul { 281 | overflow: auto; 282 | width: 600px; 283 | margin: 0 auto; 284 | } 285 | .nav { 286 | background-color: #444; 287 | } 288 | */ 289 | } 290 | --------------------------------------------------------------------------------