3
0
Fork 0

Compare commits

...

4 commits

Author SHA1 Message Date
Lance Edgar 51c456eb38 bump: version 0.20.1 → 0.20.2 2025-01-14 13:19:27 -06:00
Lance Edgar ecb1dce590 fix: improve support for composite model_key in MasterView
in particular, had a table (Catapult) with composite primary key,
where both prop keys are named differently than columns.

this also splits out the route kwargs logic for action urls, because
of another situation where i wanted to use non-primary field as model
key, but it also needed to be stripped of whitespace.  this allows for
such an override but in the end i did not pursue that method and just
wound up using default model key anyway..
2025-01-14 11:51:03 -06:00
Lance Edgar 72a663a80b fix: let content header text be a bit longer 2025-01-14 11:28:38 -06:00
Lance Edgar 59fe324872 fix: add optional target attr for GridAction 2025-01-14 11:28:12 -06:00
7 changed files with 74 additions and 12 deletions

View file

@ -5,6 +5,15 @@ All notable changes to wuttaweb will be documented in this file.
The format is based on [Keep a Changelog](http://keepachangelog.com/en/1.0.0/) The format is based on [Keep a Changelog](http://keepachangelog.com/en/1.0.0/)
and this project adheres to [Semantic Versioning](http://semver.org/spec/v2.0.0.html). and this project adheres to [Semantic Versioning](http://semver.org/spec/v2.0.0.html).
## v0.20.2 (2025-01-14)
### Fix
- improve support for composite `model_key` in MasterView
- let content header text be a bit longer
- add optional `target` attr for GridAction
- add `render_date()` method for grids
## v0.20.1 (2025-01-13) ## v0.20.1 (2025-01-13)
### Fix ### Fix

View file

@ -6,7 +6,7 @@ build-backend = "hatchling.build"
[project] [project]
name = "WuttaWeb" name = "WuttaWeb"
version = "0.20.1" version = "0.20.2"
description = "Web App for Wutta Framework" description = "Web App for Wutta Framework"
readme = "README.md" readme = "README.md"
authors = [{name = "Lance Edgar", email = "lance@wuttaproject.org"}] authors = [{name = "Lance Edgar", email = "lance@wuttaproject.org"}]

View file

@ -2280,6 +2280,10 @@ class GridAction:
See also :meth:`get_url()`. See also :meth:`get_url()`.
.. attribute:: target
Optional ``target`` attribute for the ``<a>`` tag.
.. attribute:: icon .. attribute:: icon
Name of icon to be shown for the action link. Name of icon to be shown for the action link.
@ -2297,6 +2301,7 @@ class GridAction:
key, key,
label=None, label=None,
url=None, url=None,
target=None,
icon=None, icon=None,
link_class=None, link_class=None,
): ):
@ -2305,6 +2310,7 @@ class GridAction:
self.app = self.config.get_app() self.app = self.config.get_app()
self.key = key self.key = key
self.url = url self.url = url
self.target = target
self.label = label or self.app.make_title(key) self.label = label or self.app.make_title(key)
self.icon = icon or key self.icon = icon or key
self.link_class = link_class or '' self.link_class = link_class or ''

View file

@ -155,7 +155,7 @@
} }
#content-title h1 { #content-title h1 {
max-width: 80%; max-width: 85%;
overflow: hidden; overflow: hidden;
padding-left: 0.5rem; padding-left: 0.5rem;
text-overflow: ellipsis; text-overflow: ellipsis;

View file

@ -180,6 +180,9 @@
% for action in grid.actions: % for action in grid.actions:
<a v-if="props.row._action_url_${action.key}" <a v-if="props.row._action_url_${action.key}"
:href="props.row._action_url_${action.key}" :href="props.row._action_url_${action.key}"
% if action.target:
target="${action.target}"
% endif
class="${action.link_class}"> class="${action.link_class}">
${action.render_icon_and_label()} ${action.render_icon_and_label()}
</a> </a>

View file

@ -2176,6 +2176,28 @@ class MasterView(View):
""" """
return str(instance) or "(no title)" return str(instance) or "(no title)"
def get_action_route_kwargs(self, obj):
"""
Get a dict of route kwargs for the given object.
This is called from :meth:`get_action_url()` and must return
kwargs suitable for use with ``request.route_url()``.
In practice this should return a dict which has keys for each
field from :meth:`get_model_key()` and values which come from
the object.
:param obj: Model instance object.
:returns: The dict of route kwargs for the object.
"""
try:
return dict([(key, obj[key])
for key in self.get_model_key()])
except TypeError:
return dict([(key, getattr(obj, key))
for key in self.get_model_key()])
def get_action_url(self, action, obj, **kwargs): def get_action_url(self, action, obj, **kwargs):
""" """
Generate an "action" URL for the given model instance. Generate an "action" URL for the given model instance.
@ -2183,22 +2205,21 @@ class MasterView(View):
This is a shortcut which generates a route name based on This is a shortcut which generates a route name based on
:meth:`get_route_prefix()` and the ``action`` param. :meth:`get_route_prefix()` and the ``action`` param.
It returns the URL based on generated route name and object's It calls :meth:`get_action_route_kwargs()` and then passes
model key values. those along with route name to ``request.route_url()``, and
returns the result.
:param action: String name for the action, which corresponds :param action: String name for the action, which corresponds
to part of some named route, e.g. ``'view'`` or ``'edit'``. to part of some named route, e.g. ``'view'`` or ``'edit'``.
:param obj: Model instance object. :param obj: Model instance object.
:param \**kwargs: Additional kwargs to be passed to
``request.route_url()``, if needed.
""" """
route_prefix = self.get_route_prefix() kw = self.get_action_route_kwargs(obj)
try:
kw = dict([(key, obj[key])
for key in self.get_model_key()])
except TypeError:
kw = dict([(key, getattr(obj, key))
for key in self.get_model_key()])
kw.update(kwargs) kw.update(kwargs)
route_prefix = self.get_route_prefix()
return self.request.route_url(f'{route_prefix}.{action}', **kw) return self.request.route_url(f'{route_prefix}.{action}', **kw)
def get_action_url_view(self, obj, i): def get_action_url_view(self, obj, i):
@ -2729,7 +2750,7 @@ class MasterView(View):
inspector = sa.inspect(model_class) inspector = sa.inspect(model_class)
keys = [col.name for col in inspector.primary_key] keys = [col.name for col in inspector.primary_key]
return tuple([prop.key for prop in inspector.column_attrs return tuple([prop.key for prop in inspector.column_attrs
if [col.name for col in prop.columns] == keys]) if all([col.name in keys for col in prop.columns])])
raise AttributeError(f"you must define model_key for view class: {cls}") raise AttributeError(f"you must define model_key for view class: {cls}")

View file

@ -750,6 +750,29 @@ class TestMasterView(WebTestCase):
self.request.matchdict = {'name': 'blarg'} self.request.matchdict = {'name': 'blarg'}
self.assertRaises(HTTPNotFound, view.get_instance, session=self.session) self.assertRaises(HTTPNotFound, view.get_instance, session=self.session)
def test_get_action_route_kwargs(self):
model = self.app.model
with patch.object(mod.MasterView, 'model_class', new=model.Setting, create=True):
view = self.make_view()
# dict object
setting = {'name': 'foo', 'value': 'bar'}
kw = view.get_action_route_kwargs(setting)
self.assertEqual(kw, {'name': 'foo'})
# mapped object
setting = model.Setting(name='foo', value='bar')
kw = view.get_action_route_kwargs(setting)
self.assertEqual(kw, {'name': 'foo'})
# non-standard object
class MySetting:
def __init__(self, **kw):
self.__dict__.update(kw)
setting = MySetting(name='foo', value='bar')
kw = view.get_action_route_kwargs(setting)
self.assertEqual(kw, {'name': 'foo'})
def test_get_action_url_for_dict(self): def test_get_action_url_for_dict(self):
model = self.app.model model = self.app.model
setting = {'name': 'foo', 'value': 'bar'} setting = {'name': 'foo', 'value': 'bar'}