Django-components

Latest version: v0.135

Safety actively analyzes 723882 Python packages for vulnerabilities to keep your Python projects secure.

Scan your dependencies

Page 5 of 9

0.111

⚠️ Attention ⚠️ - Please update to v0.117 to fix known bugs. See [791](https://github.com/django-components/django-components/issues/791) and [#789](https://github.com/django-components/django-components/issues/789) and [#818](https://github.com/django-components/django-components/issues/818).

Fix

- Prevent rendering Component tags during fill discovery stage to fix a case when a component inside the default slot
tried to access provided data too early.

0.110

⚠️ Attention ⚠️ - Please update to v0.117 to fix known bugs. See [791](https://github.com/django-components/django-components/issues/791) and [#789](https://github.com/django-components/django-components/issues/789) and [#818](https://github.com/django-components/django-components/issues/818).

General

🚨📢 BREAKING CHANGES

- Installation changes:

- If your components include JS or CSS, you now must use the middleware and add django-components' URLs to your `urlpatterns`
(See "[Adding support for JS and CSS](https://github.com/django-components/django-components#adding-support-for-js-and-css)")

- Component typing signature changed from

py
Component[Args, Kwargs, Data, Slots]


to

py
Component[Args, Kwargs, Slots, Data, JsData, CssData]


- If you rendered a component A with `Component.render()` and then inserted that into another component B, now you must pass `render_dependencies=False` to component A:

py
prerendered_a = CompA.render(
args=[...],
kwargs={...},
render_dependencies=False,
)

html = CompB.render(
kwargs={
content=prerendered_a,
},
)


Feat

- Intellisense and mypy validation for settings:

Instead of defining the `COMPONENTS` settings as a plain dict, you can use `ComponentsSettings`:

py
settings.py
from django_components import ComponentsSettings

COMPONENTS = ComponentsSettings(
autodiscover=True,
...
)


- Use `get_component_dirs()` and `get_component_files()` to get the same list of dirs / files that would be imported by `autodiscover()`, but without actually
importing them.

Refactor

- For advanced use cases, use can omit the middleware and instead manage component JS and CSS dependencies yourself with [`render_dependencies`](https://github.com/django-components/django-components#render_dependencies-and-deep-dive-into-rendering-js--css-without-the-middleware)

- The [`ComponentRegistry`](../apidjango_components.ComponentRegistry) settings [`RegistrySettings`](../apidjango_components.RegistrySettings)
were lowercased to align with the global settings:
- `RegistrySettings.CONTEXT_BEHAVIOR` -> `RegistrySettings.context_behavior`
- `RegistrySettings.TAG_FORMATTER` -> `RegistrySettings.tag_formatter`

The old uppercase settings `CONTEXT_BEHAVIOR` and `TAG_FORMATTER` are deprecated and will be removed in v1.

- The setting `reload_on_template_change` was renamed to
[`reload_on_file_change`](../settingsdjango_components.app_settings.ComponentsSettingsreload_on_file_change).
And now it properly triggers server reload when any file in the component dirs change. The old name `reload_on_template_change`
is deprecated and will be removed in v1.

- The setting `forbidden_static_files` was renamed to
[`static_files_forbidden`](../settingsdjango_components.app_settings.ComponentsSettingsstatic_files_forbidden)
to align with [`static_files_allowed`](../settingsdjango_components.app_settings.ComponentsSettingsstatic_files_allowed)
The old name `forbidden_static_files` is deprecated and will be removed in v1.

Tags

🚨📢 BREAKING CHANGES

- `{% component_dependencies %}` tag was removed. Instead, use `{% component_js_dependencies %}` and `{% component_css_dependencies %}`

- The combined tag was removed to encourage the best practice of putting JS scripts at the end of `<body>`, and CSS styles inside `<head>`.

On the other hand, co-locating JS script and CSS styles can lead to
a [flash of unstyled content](https://en.wikipedia.org/wiki/Flash_of_unstyled_content),
as either JS scripts will block the rendering, or CSS will load too late.

- The undocumented keyword arg `preload` of `{% component_js_dependencies %}` and `{% component_css_dependencies %}` tags was removed.
This will be replaced with HTML fragment support.

Fix

- Allow using forward slash (`/`) when defining custom TagFormatter,
e.g. `{% MyComp %}..{% /MyComp %}`.

Refactor

- `{% component_dependencies %}` tags are now OPTIONAL - If your components use JS and CSS, but you don't use `{% component_dependencies %}` tags, the JS and CSS will now be, by default, inserted at the end of `<body>` and at the end of `<head>` respectively.

Slots

Feat

- Fills can now be defined within loops (`{% for %}`) or other tags (like `{% with %}`),
or even other templates using `{% include %}`.

Following is now possible

django
{% component "table" %}
{% for slot_name in slots %}
{% fill name=slot_name %}
{% endfill %}
{% endfor %}
{% endcomponent %}


- If you need to access the data or the default content of a default fill, you can
set the `name` kwarg to `"default"`.

Previously, a default fill would be defined simply by omitting the `{% fill %}` tags:

django
{% component "child" %}
Hello world
{% endcomponent %}


But in that case you could not access the slot data or the default content, like it's possible
for named fills:

django
{% component "child" %}
{% fill name="header" data="data" %}
Hello {{ data.user.name }}
{% endfill %}
{% endcomponent %}


Now, you can specify default tag by using `name="default"`:

django
{% component "child" %}
{% fill name="default" data="data" %}
Hello {{ data.user.name }}
{% endfill %}
{% endcomponent %}


- When inside `get_context_data()` or other component methods, the default fill
can now be accessed as `Component.input.slots["default"]`, e.g.:

py
class MyTable(Component):
def get_context_data(self, *args, **kwargs):
default_slot = self.input.slots["default"]
...


- You can now dynamically pass all slots to a child component. This is similar to
[passing all slots in Vue](https://vue-land.github.io/faq/forwarding-slots#passing-all-slots):

py
class MyTable(Component):
def get_context_data(self, *args, **kwargs):
return {
"slots": self.input.slots,
}

template: """
<div>
{% component "child" %}
{% for slot_name in slots %}
{% fill name=slot_name data="data" %}
{% slot name=slot_name ...data / %}
{% endfill %}
{% endfor %}
{% endcomponent %}
</div>
"""


Fix

- Slots defined with `{% fill %}` tags are now properly accessible via `self.input.slots` in `get_context_data()`

- Do not raise error if multiple slots with same name are flagged as default

- Slots can now be defined within loops (`{% for %}`) or other tags (like `{% with %}`),
or even other templates using `{% include %}`.

Previously, following would cause the kwarg `name` to be an empty string:

django
{% for slot_name in slots %}
{% slot name=slot_name %}
{% endfor %}


Refactor

- When you define multiple slots with the same name inside a template,
you now have to set the `default` and `required` flags individually.

htmldjango
<div class="calendar-component">
<div class="header">
{% slot "image" default required %}Image here{% endslot %}
</div>
<div class="body">
{% slot "image" default required %}Image here{% endslot %}
</div>
</div>


This means you can also have multiple slots with the same name but
different conditions.

E.g. in this example, we have a component that renders a user avatar
- a small circular image with a profile picture of name initials.

If the component is given `image_src` or `name_initials` variables,
the `image` slot is optional. But if neither of those are provided,
you MUST fill the `image` slot.

htmldjango
<div class="avatar">
{% if image_src %}
{% slot "image" default %}
<img src="{{ image_src }}" />
{% endslot %}
{% elif name_initials %}
{% slot "image" default required %}
<div style="
border-radius: 25px;
width: 50px;
height: 50px;
background: blue;
">
{{ name_initials }}
</div>
{% endslot %}
{% else %}
{% slot "image" default required / %}
{% endif %}
</div>


- The slot fills that were passed to a component and which can be accessed as `Component.input.slots`
can now be passed through the Django template, e.g. as inputs to other tags.

Internally, django-components handles slot fills as functions.

Previously, if you tried to pass a slot fill within a template, Django would try to call it as a function.

Now, something like this is possible:

py
class MyTable(Component):
def get_context_data(self, *args, **kwargs):
return {
"child_slot": self.input.slots["child_slot"],
}

template: """
<div>
{% component "child" content=child_slot / %}
</div>
"""


NOTE: Using `{% slot %}` and `{% fill %}` tags is still the preferred method, but the approach above
may be necessary in some complex or edge cases.

- The `is_filled` variable (and the `{{ component_vars.is_filled }}` context variable) now returns
`False` when you try to access a slot name which has not been defined:

Before:

django
{{ component_vars.is_filled.header }} -> True
{{ component_vars.is_filled.footer }} -> False
{{ component_vars.is_filled.nonexist }} -> "" (empty string)


After:
django
{{ component_vars.is_filled.header }} -> True
{{ component_vars.is_filled.footer }} -> False
{{ component_vars.is_filled.nonexist }} -> False


- Components no longer raise an error if there are extra slot fills

- Components will raise error when a slot is doubly-filled.

E.g. if we have a component with a default slot:

django
{% slot name="content" default / %}


Now there is two ways how we can target this slot: Either using `name="default"`
or `name="content"`.

In case you specify BOTH, the component will raise an error:

django
{% component "child" %}
{% fill slot="default" %}
Hello from default slot
{% endfill %}
{% fill slot="content" data="data" %}
Hello from content slot
{% endfill %}
{% endcomponent %}

0.100

BREAKING CHANGES

- `django_components.safer_staticfiles` app was removed. It is no longer needed.

- Installation changes:

- Instead of defining component directories in `STATICFILES_DIRS`, set them to [`COMPONENTS.dirs`](https://github.com/django-components/django-components#dirs).
- You now must define `STATICFILES_FINDERS`

- [See here how to migrate your settings.py](https://github.com/django-components/django-components/blob/master/docs/migrating_from_safer_staticfiles.md)

Feat

- Beside the top-level `/components` directory, you can now define also app-level components dirs, e.g. `[app]/components`
(See [`COMPONENTS.app_dirs`](https://github.com/django-components/django-components#app_dirs)).

Refactor

- When you call `as_view()` on a component instance, that instance will be passed to `View.as_view()`

0.97

Fix

- Fixed template caching. You can now also manually create cached templates with [`cached_template()`](https://github.com/django-components/django-components#template_cache_size---tune-the-template-cache)

Refactor

- The previously undocumented `get_template` was made private.

- In it's place, there's a new `get_template`, which supersedes `get_template_string` (will be removed in v1). The new `get_template` is the same as `get_template_string`, except
it allows to return either a string or a Template instance.

- You now must use only one of `template`, `get_template`, `template_name`, or `get_template_name`.

0.96

Feat

- Run-time type validation for Python 3.11+ - If the `Component` class is typed, e.g. `Component[Args, Kwargs, ...]`, the args, kwargs, slots, and data are validated against the given types. (See [Runtime input validation with types](https://github.com/django-components/django-components#runtime-input-validation-with-types))

- Render hooks - Set `on_render_before` and `on_render_after` methods on `Component` to intercept or modify the template or context before rendering, or the rendered result afterwards. (See [Component hooks](https://github.com/django-components/django-components#component-hooks))

- `component_vars.is_filled` context variable can be accessed from within `on_render_before` and `on_render_after` hooks as `self.is_filled.my_slot`

0.95

Feat

- Added support for dynamic components, where the component name is passed as a variable. (See [Dynamic components](https://github.com/django-components/django-components#dynamic-components))

Refactor

- Changed `Component.input` to raise `RuntimeError` if accessed outside of render context. Previously it returned `None` if unset.

Page 5 of 9

© 2025 Safety CLI Cybersecurity Inc. All Rights Reserved.