Insert arbitrary parts in Table. There is a namespace
outeron table where you can insert things.
Filter form fields failed to respect include directive from filter
Fix for live edit when used with iommi path decoding
Fixed a crash in the debug tree that sometimes happened if you had an non-editable
Live edit internal API is now a bit more flexible, making it possible to adapt it to your own use cases better
Late evaluation of
Field.editable, so you can now give a lambda for
Rare crash when Field.on_bind tries to access form.all_fields, that is removed
Admin debug message was shown in non-debug
Add group name of form group specifier to lambda args when rendering the corresponding tags
Admin: respect django permissions system
Compatibility with Python 3.11
Enable usage of
redirect_toto be an instance of
django.utils.functional.Promisethat is a proxy for a string (which is what
Admin delete page should present 404 page on non-existing object
Menus should attempt to respect query parameters when doing set_active
Dependencies tri.declarative and tri.struct are merged into iommi and are thus no longer required. You should remove them from your project and fix your imports to only import from iommi if you had imports from
Fixes for how style definitions are merged. This fixes an issue where select2 wasn’t used properly in table filters.
Refurbished priority for settings on refinable objects. E.g style settings can now overwrite things declared by shortcuts.
Fixed paginator and select2/choice/radio control when django is configured to use thousands separator
Drop Field parameters
assets. Assets parameters now strictly go in
Silenced deprecation warning from Django for app configs
Improved error messages
Workaround for crash deep in pytest internals when using synthetic traceback
Introduced new decorator
@with_defaultsfor giving parameters in shortcuts in a more natural way.
Introduced new decorator
@superinvoking_classmethodto enable a classmethod to call a base class implementation with the same name. Useful when declaring more specific shortcuts when customizing using inheritance.
Improved exceptions: you now get an exception frame that points to the declaration if you use the declarative style. This makes it much easier to know what code is responsible for a crash.
Fixed CSV rendering endpoint and added an example for it
EditTableis no longer experimental!
Fix rendering of choices in filter select widgets from CharField with choices
Bulk editing: fields should be default None, even if the model has a default value
Fixed rendering of disabled checkbox
Base style no longer includes the select2 assets
h_tag should be possible to set to None and to set include=False
Fixed default handling for get_sql_debug
Fixed issue where you couldn’t set non-editable input tag to ‘input’
Fix auto rowspan. It was broken if you did multiple bind() on the same object
Live edit improvements: Handle slower pages MUCH better, and editing of classes works
Fixed styling for non-editable select2
Minor fix for SQL tracing when function names could not be retrieved
Path decoding feature added. See https://docs.iommi.rocks/en/latest/path.html
bootstrap5. Thanks to Nigel Metheringham!
perform_dispatchfor legacy FBVs. See https://docs.iommi.rocks/en/latest/legacy_fbv.html for how to use it.
Forms: Some asserts to make
redirect_tomore fool proof
Forms: A little check to make setting
Fieldmore fool proof
Send full state of current form when doing ajax select2 callback, enabling more advanced filtering
Fixed bug where
Fixed bug where style application on an object only had an effect on the child objects
Form.deletewill now show a nice error message if the object can’t be deleted due to restricted rows
EditTable: Add row feature added
EditTable: Delete row feature added
EditColumnnow match the behavior of
fieldfor a normal
EditTable: Fixed auto__rows
EditTable: if instance.pk < 0, treat this as a create
EditTablecan now be nested inside a
Form, similar to how forms can be nested in forms.
Make it possible to turn off freetext_search query form field by passing
Optimized writing for
Several fixes for
EditTable. Basically it works now :)
Due to a mistake in packaging the experimental feature EditTable was not included in previous release. This is now fixed.
Freetext field could be shown when it shouldn’t, causing a crash on search
iommi.experimental.edit_table.EditTable. This API can change at any time and also move away from
Logout redirect for
Nicer error message for missing
Fixed error for when you specify conf in a way that isn’t supported. This would silently ignore the configuration before.
Field groups in forms. There is a new string
Fieldthat is used to group, a
Formyou can use to set attrs, tag, etc. Global styling for form groups is done via the
FieldGroupclass. The bootstrap style has been updated to support this feature out of the box.
Validation could be bypassed for forms if they have been saved via
form.refine_done(). This became the default behavior for
.as_view()in iommi 4.1 so that release is broken.
refine_done, giving you a nice little performance win for free
@iommi_renderview decorator. Use this to get correct behavior when using transactions by default in views. The iommi middleware will now produce an error if you try to use it incorrectly.
iommi_init_all_select2to re-initialize iommi select2 components
Break out the select2 enhancement from the base style into a separate
select2_enhanced_formsstyle, and added it to all the built in styles. If you have a custom style that extended
baseyou must now also add
select2_enhanced_formsto that style to get the same behavior as before.
should_ignore_frame() is more robust against acrobatic frames. This is a rather obscure bug that won’t affect normal iommi usage.
Dropped support for
__in names of declared columns/fields/filters (deprecated since 3.2.0)
Big internal refactoring. You should see some performance improvements accross the board.
Ability to customize the
Cellsclasses used by
Improved ability to customize
Table.tbody. You can now add html after or before the rows from the table itself
Template-based rendering should get iommi_evaluate_parameters as context. This was the case in some cases but not all, most notably when rendering a
Added snakeviz profiling (use it by passing
_iommi_prof=snakeas a url parameter)
Fixed stack traces in SQL tracing
Fixed jump to code for several scenarios
German translation fixes and updates
Improved error message for invalid admin config
write_nested_form_to_instancenow takes keyword arguments
Fix bug causing any endpoint invocation of table fields to force a bind of the paginator (Which should be lazy)
Fix enforcement on
Field.multi_choiceand others where value is a list.
Names with underscore are deprecated and will be removed in the next major version. This means you can no longer write this:
class MyTable(Table): foo__bar = Column()
You must now instead write:
class MyTable(Table): some_name = Column(attr='foo__bar')
foo__bar had some weird consequences like you not being able to later target that name without getting ambiguities in what
Expand ajax reload on filter change of tables to also include the bulk form. If not done, the bulk options are not in sync with the filtering.
Remove reference to non-existant
errors.htmlin bootstrap style
Table.visible_rowsnon-lazy and not a property
Table.rowsis no longer a property
Form: Evaluate parameters now contain instance
Use the same redirect logic for delete as create/edit. This means you can now use
extra__redirect_tofor delete too
When stopping the live editing, a full runserver restart is now triggered so you get the new code you just edited
Styles have a new
sub_stylesparameter. This change greatly simplifies how you set up a custom style for your project if you want to customize the query form.
IOMMI_DEFAULT_STYLEcan now be a
Breaking change: The horizontal styles are removed and replaced with the substyle feature. If you use for example
'bootstrap_horizontal', you need to replace it with
Mixed case filter fields didn’t work
Respect browsers preferred dark/light mode for profiler and sql tracer
Major bug: tables based on querysets would implicitly use the django result cache. This resulted in the contents of the table not changing until after process restart
Fixed nesting tables inside forms. This would previously crash with a strange error message.
Avoid infinite loop in sort_after on too large indicies
Read defaults from model for initial of fields
Increased log level of SQL logging from 11 to 21 (DEBUG+1 -> INFO+1)
Added null factory for
Fixed live editing code to use the same logic as ‘jump to code’ to find the code
Fixed one case where live edit broke
Prettier debug menu for live editing
Prettier query help text (thanks new contributor flying_sausages!)
Fixed bad html escape in SQL trace magnitude graph (this is not a security problem, as it’s a developer tool with very restricted access)
Renamed freetext to freetext_search. It was too easy to collide with a user defined model with a field called freetext
Automatically generating a Query from a model with a foreign key was broken in cases where the name field wasn’t the same as name field of the parent model
Make it possible to pass a lambda to title of Page/Form/Table
Improved error when trying to register an already registered style
Revert to the old (pre 2.8.2) way of using
search_fieldsto compose queries.
The new approach failed for cases when there was a custom
value_to_qdefinition. A proper fix needs to have a unified approach also when using
Render title of
Pageobjects. To turn off the rendering of the title pass
Removed the register_search_fields warning, it was 90% annoying and 10% useful
Form: support passing instance as a lambda, even in combination with
Removed bad assert that prevented passing instance as a lambda for auto__model of Form
SQL trace was broken for postgres
query_from_indexes should automatically generate filters for foreign keys. This especially affected the admin.
search_fieldswhen composing queries from model filter values. Always using the
.pkfallback approach is more stable when the search field values might not be unique. This will remove a bunch of warnings that weren’t very helpful too.
Fixed crash when setting
capitalize()now handles safe strings properly. This will enable you to pass safe strings to
Translation of Yes/No
Fixed error message for
Updated to fontawesome 4.7
Renamed live edit asset to not conflict with the name ‘custom’ which might be fairly common
Nicer title in the admin for apps
Auto generated tables had “ID” as the column name for foreign keys, instead of the name of the remote model.
Profiler fixed: the bind and render of iommi objects that were handled by the middleware weren’t profiled
Fixed live edit to work for views with URL arguments
Handle settings.BASE_DIR as Path objects
fix bulk__include = False on table
Make DebugMenu created on demand to avoid setting of breakpoints when debugging your own code
Models in admin are now in alphabetical order
Fieldis not a
Tag, so you can render a
Formas a div if you want.
The root menu item for the iommi admin was broken if you inherited from Admin
Force the live edit view to be bootstrap. This avoids the live edit feature looking a big broken for your own custom styles.
Minor bootstrap styling fix for non-editable fields
The paginator is now lazy. This means we can avoid a potentially expensive
.count()database hit in many situations
Table.post_bulk_edittakes evaluate parameters now
Column.include=False implies that the column shouldn’t get anything in the bulk form. If you want bulk editing without a visible column use Column.render_column=False
Fix reinvoke/reinvoke_new_defaults when shortcut is changed
Date/datetime parsing bugs fixed after mutation testing
Do not do form post_validation if we are in initial display mode
Forms now don’t create a submit button by default. If you have a post handler you will get a submit button though.
SQL trace bugfixes
Custom raw_data callback should have same semantics as constant value (and parsed_data callback)
Improved error message on disallowed unbound object access
Documentation improvements, for example new pages for dev tools, and styles
Live editing on
.as_view()style views work in the case of an explicitly declared class
Fixed bug where the ajax enhanced table didn’t work if you used
Table.divor otherwise changed the
Fixed auto__model column/filter for
Formcan now contain non-
Fieldparts. Iterate over everything to render with
form.partsand all the fields to be validated with
form.fields. Fields that are not direct children are also collected, so you can easily add extra structure by wrapping a bunch of fields in a
You can now customize the name shown in the advanced search via
Form submit buttons (
Actions.submit) are now rendered as
Added SQL trace feature
You can now apply styles on the root object. Example:
Edit button only present in debug menu when the edit middleware is installed
Added profile button to debug menu
Make collected assets more accessible when rendering iommi in your own templating environment: you can now access them on the iommi objects:
Removed broken validation of sort columns. This validation prevented sorting on annotations which was very confusing as it worked in debug mode
Make it possible to target the live edit page with styles (via
The live edit view can be flipped between horizontal and vertical layouts
The debug tree view is slimmed down (by not including endpoints and assets on lots of things)
Field.raw_data_listis removed. You can know if it’s a list or not by checking
raw_datacovers the uses cases.
Include decorators in live edit
The debug jump to code feature should work for some more scenarios, and it will not display if it has no good guess.
Field.choice_to_option. This is replaced by
Fixed live editing to work when distributing iommi
Live editing of function based views in DEBUG. Works for both iommi views and normal django views.
Added ajax enhanced table filtering
You can now turn off the advanced mode on queries:
Queryhas two new refinables:
post_process. These are hook points if you need to further customize what query is generated.
Enable profiling when DEBUG mode is on, even if you’re not staff
Fixed multiselect on empty list
get_errors()member function on
Fixed select2 widget when the base url do not end with
Styling fixes. Primarily for bulma.
include=False on a Column should imply not generating the query filter and bulk field. If you want to not render a column but still want the filters, use the render_column=False feature
Added callbacks for saving a form:
Form.createfor custom object creation
The errors list has been changed. You should always use
add_error()to add an error on a
It is now possible to call
get_errors()and get what you expect from
Query forms can now have additional fields, that are ignored by the filter handling code (when you want to do additional filtering outside of the query logic)
Bug fixes with state leaking between binds
Fixed jump to code
Improved error message for
Added a nice error message if you try to shoot in
classas raw strings
Fixed empty table message, and invalid query form messages
rowsqueryset and filtering were not respected for the “Select all rows” bulk feature. This could produce some pretty bad bugs!
Support custom bulk post_handlers on lists and not just querysets
Tablehas a few new members:
initial_rows: the rows you pass (or that gets created by
auto__model) is stored unchanged here
initial_rows+ sorting applied
sorted_rows+ filtering applied
sorted_and_filtered_rows+ pagination applied
rows: this is now a property and will map to the old behavior which is the “most applied” member that exists
Fixed passing dunder paths to
auto__include. You got a weird crash if the target of the path was a foreign key. There are still issues to be resolved adjacent to this, but the base case now works.
Fixed the “select all” feature for pages with multiple tables.
Only include select2 assets when needed (possible because of the point above)
Filtering on booleans was very broken. It always returned empty querysets and didn’t produce errors when you tried to do stuff like
It’s now possible to configure stuff on the freetext field of a query
iommi will now grab the root page title from the text from
Headerinstances in addition to
Render date fields as such
Fixed date and time formatting
Support for optgroups in forms
Make it possible to insert fields into the form of a query, and filters into a query
Differentiate between primary and other actions. This should make iommi pages look more in line with the majority of design systems. If you have a custom style you probably want to add a style definition for
Fixed a case of a silent overwrite that could be surprising. This was found during reading the code and has never happened to us in practice.
Style fixes for bulma
Fix so that style application does not alter definitions destructively. This could lead to some strange behavior if you tried to switch between styles, and it could leak over definitions between things you would not expect.
The title of
Nonewhen there is no model
Assets as first class concept. You can now insert asset definitions into your style with
assets__js=...instead of defining a
base_template. This change also removes the base templates for all the built in styles as they are now obsolete.
Made it easy to hide the label of a Field by setting
Internationalization! iommi now has i18n support and ships with English, German and Swedish languages out of the box. We welcome more translations.
Out of the box support for the Bulma CSS framework
auto__includespecifications allow foreign key paths
By default we now grab display_name from the model fields verbose_name (if applicable)
Sometimes you got reordering of parts when doing a post to a form for example, this is now fixed
traversableargument to lambdas is now the leaf and not the root. This was a bug.
reverse_lazyas url argument to MenuItem
Two id attributes were rendered on the input tags in forms (thanks Benedikt Grundmann for reporting!)
instance.idwhich might be valid. It should have accessed
instance.pkwhich is always valid.
Stylemust now take a
base_templateargument. This replaces the setting
IOMMI_CONTENT_BLOCKis removed. Replaced by the
Allow table rows to be provided from a generator. (Disabling paginator)
Added blocks (
iommi_bottom) as useful hook points to add custom data in the templates if you don’t need a totally new template but want to just customize a little bit.
The default sort_key on a Column.foreign_key now looks at the searchable field of the remote field (‘name’ by default). This means by default sorting will mostly be more what you expect.
Changed the error from get_search_field() for non-unique name to a warning.
Removed <table> for layout in query advanced/simple stuff.
Don’t warn for missing register_search_fields when attr=None
Set admin to bootstrap by default.
Added form for changing password. Used by the admin but also usable from your code.
Added form for login. Used by the admin but also usable from your code.
Fixed foundation styling for query form.
Field.help. This is the fragment that renders the help text for a
Field. This means you can now style and customize this part of forms more easily. For example set a CSS class:
Use django default date and time formatting in tables.
New shortcut for
Table.divfor when you want to render a
Tableas a bunch of divs. This is useful because a
Tableis really a view model of a sequence of stuff, not just a
Possibility to set
Noneto not get a wrapping html tag.
Table.outeras a tag you can style that encompasses the entire table part.
Form.h_tagrendering inside the form tag to make it stylable as a coherent whole.
Grab html title from first part if no title is given explicitly. This means you’ll get the
<title>tag filled more often by what you expect automatically.
Templateinstances are now collected properly by
Read admin config from modules.
The Admin is now opt in, not opt out.
The admin is now MUCH prettier and better.
Tableare now rendered above the table by default. Set
Trueto render them the old way.
Many misc improvements
Table.bulk. The old name was a mistake as the name was always
bulk. This meant that styling didn’t work like you expected and the pick feature also lead you down the wrong path.
Support user inputted relative dates/datetimes
Support more time formats automatically
Introduced Filter.parse() which is a hook point for handling special parsing in the query language. The query language will no longer try to convert to integers, floats and dates for you. You have to specify a parse() method.
traversablekey to evaluate parameters. Think of it like something similar to
cell__formatnow gets all evaluate parameters like you’d expect
Nonebut you’ve specified
value_to_qthen your filter is now included
Various bug fixes
Use select2 as the default for multi_choice
Improved usability: Make icon column behavior on falsy values more guessable
Accidentally changed default style to foundation, change back to bootstrap
Improved usability: Don’t fall back to default template name if the user specified an explicit template name: fail on TemplateNotFound
Style on root uses correct base template
Allow model fields called
register_name_field. This new system is a list of field names and not just a single field. There is also new searching and filtering behavior based on this that means you will get better search results
field_nameas used by model factories is replaced with
model_field_name. If you used
register_factoryyou will need to change this. The field names on
Filterare also renamed.
Support fields named
itemson Django models
Added basic styling support for CSS frameworks Water and Foundation
Fix include to make None mean False
Change Filter.text to search using icontains instead of iexact by default in the basic search mode
Change post_validation callback to receive standard evaluate parameters
Improved help text for queries
Field.radio was broken in the bootstrap style: it specified the input template as the template for the entire field, so the label got erased
Fixed default text argument to Fragment
Fixed issue where endpoint dispatch parameter was left over in the pagination and sorting links
Parts that are None should not be collected. This affected the admin where it printed “None” below the “Admin” link.
Added header for bulk edit form in tables
Fixed textarea readonly when field is not editable
Fixed is_paginated function on Paginator
Add request to evaluate parameters
Make evaluate and evaluate_recursive match even the
**_case by default
No dispatch command on a POST is invalid and will now produce an error
Lazy bind() on members. This is a performance fix.
Fixed bug where display_name could not be overridden with a lambda due to incorrect evaluate handling
Removed Table.rendered_columns container. You have to look at the columns and check if they have render_column=False
Fixed data-endpoint attribute on table
Fixed tbody endpoint and added a div to make the endpoint easier to use
Fixed an issue where fragments couldn’t be customized later if built with the
Fragment. This should be mostly transparent.
You can now pass multiple argument to
html.div('foo', 'bar')is now valid and creates two child nodes
auto__only suggests a default. This avoids some confusion one could get if mixing
rowsin some ways.
Fixed setting active on nested submenus where the parent had url None
Include iommi/base_bootstrap.html and iommi/base_semantic_ui.html in package, and use them if no base.html is present. This improves the out of the box experience for new projects a lot
Support mixing of
auto__rowbased columns and declarative columns
Support attrs__class and attrs__style as callables
Added support for context namespace on Page, which is passed to the template when rendering (for now only available on the root page)
Fixed how we set title of bulk edit and delete buttons to make configuration more obvious
Fixed rendering of grouped actions for bootstrap
Respect auto__include order
boolean_tristate should be the default for the Field of a Column.boolean
New class Header that is used to automatically get h1/h2/etc tags according to nesting of headers
Table.rows should be able to be evaluated
Added feature that you can type ‘now’ into date/datetime/time fields
Feature to be able to force rendering of paginator for single page tables
Paginator fixes: it’s now no longer possible to use the Django paginator, but the iommi paginator is more full features in trade.
Removed jQuery dependency for JS parts
Big improvements to the Menu component
filters that have freetext mode now hide their field by default
Added “pick” in the debug toolbar. This is a feature to quickly find the part of the document you want to configure
Changed so that Query defaults to having the Field included by default
Renamed BoundRow/bound_row to Cells/cells
Major improvements to the admin
Lots and lots of cleanup and bug fixes