Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/smileychris/django-includecontents
https://github.com/smileychris/django-includecontents
django templating
Last synced: 2 months ago
JSON representation
- Host: GitHub
- URL: https://github.com/smileychris/django-includecontents
- Owner: SmileyChris
- Created: 2024-04-08T05:01:03.000Z (9 months ago)
- Default Branch: main
- Last Pushed: 2024-07-25T23:21:12.000Z (6 months ago)
- Last Synced: 2024-10-31T15:47:53.019Z (2 months ago)
- Topics: django, templating
- Language: Python
- Homepage: https://pypi.org/project/django-includecontents/
- Size: 114 KB
- Stars: 14
- Watchers: 2
- Forks: 1
- Open Issues: 2
-
Metadata Files:
- Readme: README.md
- Changelog: changes/.gitignore
Awesome Lists containing this project
README
# Django IncludeContents tag
Provides a component-like `{% includecontents %}` tag to Django.
For example:
```jinja
{% load includecontents %}
{% includecontents "hello.html" %}
World
{% endincludecontents %}
```It also provides an optional Django template engine that extends this tag to
work like an HTML component.In this example, it will include and render `components/pretty-card.html`:
```html
World
```
This engine also allows for multi-line template tags. For example:
```jinja
{% if
user.is_authenticated
and user.is_staff
%}
...
{% endif %}
```## Installation
```bash
pip install django-includecontents
```Either install the custom template engine or just add this app to your `INSTALLED_APPS`.
### Install with template engine
Replace the default `django.template.backends.django.DjangoTemplates` backend in your settings:
```python
TEMPLATES = [
{
'BACKEND': 'includecontents.django.DjangoTemplates',
...
},
]
```This engine also adds `includecontents` to the built-in tag libraries so there is no need to load it in your templates.
### Install without template engine
Alternatively, add this app to your `INSTALLED_APPS` and use `{% load includecontents %}` in your templates:
```python
INSTALLED_APPS = [
...
'includecontents',
]
```## Template tag usage
The `includecontents` tag works like the `include` tag but the contents is rendered and passed to the included template as a `contents` variable.
```jinja
{% includecontents "hello.html" %}
World
{% endincludecontents %}
```### Named contents blocks
You can also have named contents blocks within the component content.
For example:
```jinja
{% includecontents "hello.html" %}
World
{% contents footer %}Footer{% endcontents %}
{% endincludecontents %}
```Where `hello.html` template could look something like:
```jinja
{{ contents }}
{% if contents.footer %}
{% endif %}
```## HTML Components Usage
This requires the custom template engine to be installed.
Create a `components` directory in your templates directory. This is where you will put your component templates that are used via the HTML component format.
These components are normal Django templates that will be rendered with an isolated context. The context is passed to the component via component's attributes.Components must not match any standard HTML tags. Actually, it's best practice to name them as HTML custom elements (1 or more ASCII letters; a hyphen; 1 or more more ASCII letters, digits or hyphens).
For example, a `components/my-card.html` template could look like:
```html
{{ title }}
{{ contents }}
```Components are always rendered isolated from the parent template, so they should not rely on any context variables from the parent template.
Which will allow you to use it like this (without the need to load any template library):
```html
World
```
You can use directories within `components` to group your components. For example, `components/forms/field.html`:
```html
```
You can use named [`{% contents %}` blocks](#named-contents-blocks), just like with the `includecontents` tag.
Some HTML formatters (like prettier) insist on quoting HTML attribute values, you can avoid this by optionally wrapping template values in `{}`:
```html
```
You can also use short-hand syntax for HTML attributes when the attribute name matches the variable name:
```html
```
### Component Props
You can define the required or default props of the component in a comment at the top of its template that begins with `props ` (or `def ` to match what JinjaX uses). An exception will be raised if a required prop is not provided.
Any other attributes passed to the component that are not listed in this definition will be added to an `attrs` context variable that can render them as HTML attributes.
```jinja
{# props #}
{{ contents }}
```You can also provide default values for these attributes via the `{% attrs %}` template tag.
```jinja
{# props title, large=False #}
...
```This example component above would require a `title` attribute and allow an optional `large` attribute. Any other attributes will be rendered on the div, with a default class of `card` if you don't specify a class attribute.
If you want to provide multiple groups of undefined attributes, you can use `group.name` as the format.
Then render them with `{{ attrs.group }}` (or `{% attrs.group %}` if you want fallback values).For example to call a component like this:
```html
```
It could be defined like this:
```jinja
{# props value, label="" #}
{% if label %}{{ ''|safe }}{% endif %}
{{ label }}
{% if label %}{{ ''|safe }}{% endif %}
```#### Kebab-cased props
Non-defined props can be "kebab-cased", for example:
```html
```
To access a kebab-cased prop to a via the ``attrs`` variable, use its CamelCase equivalent. The ``{% attrs %}`` tag works with kebab-case fine though::
```jinja
{# props #}{% attrs my-prop="fallback" %}
{% if attrs.myProp %}
my-prop is explicitly set to {{ attrs.myProp }}
{% endif %}
```#### Extended / conditional classes
Prepend your class list with `"& "` to have it extended rather than replaced:
```jinja
{% attrs class="lg" %} {# sets default class attribute to "lg" but can be overridden #}
{% attrs class="& lg p-3" %} {# always add 'lg p-3' classes #}
```You can provide conditional classes for the `class` attribute using the svelte class directive format:
```jinja
{# props large=False #}{% attrs class:lg=large %} {# adds 'lg' class if large prop is truthy #}
{% attrs class:lg %} {# always adds 'lg' class #}
```You can use this same conditional format on the component attributes directly:
```html
World
```
#### Conditional classes in tailwindcss
Add this transform in your `tailwind.config.js` so that Tailwind picks up the
Svelte-like class directives:```js
/** @type {import('tailwindcss').Config} */
module.exports = {
content: {
files: ["**/*.{html,js}", "!node_modules"],
transform: {
// Also handle Svelte-like class:class-name directives
html: (content) => content.replace(/(\w):([-\w]+)/g, '$1 "$2" '),
},
},
...
```## Prettier formatting
While not part of this package, `django-includecontents` plays very well with the [`prettier-plugin-jinja-template` plugin](https://www.npmjs.com/package/prettier-plugin-jinja-template) for Prettier.
This plugin can provide consistent formatting for your Django (and Jinja, obviously) templates.First install it with `npm install --save-dev prettier-plugin-jinja-template`.
Then create a `.prettierrc` file in your project root with the following content:
```json
{
"plugins": ["prettier-plugin-jinja-template"],
"overrides": [
{
"files": ["**/{templates,jinja2}/**/*.html"],
"options": {
"parser": "jinja-template",
"quoteAttributes": false
}
}
]
}
````quoteAttributes` is set to `false` to avoid quoting HTML attribute values that are template variables (e.g. `title={mytitle}`).
The workaround to keep the quotes when using the alpineJS `x-data` directive is to simply put a space between the quote and the curly brace: `x-data=" {...} "`.### VScode formatting
To use this Prettier formatting within VScode, use the following two extensions:
* [Django](https://marketplace.visualstudio.com/items?itemName=batisteo.vscode-django)
* [Prettier - Code formatter](https://marketplace.visualstudio.com/items?itemName=esbenp.prettier-vscode)Add this to your settings (`Ctrl-P`, paste `>Preferences: Open Workspace Settings (JSON)`):
```json
{
"[django-html]": {
"editor.formatOnSave": true,
"editor.defaultFormatter": "esbenp.prettier-vscode"
},
}
```