Compare commits
4 commits
a405b19217
...
51c456eb38
Author | SHA1 | Date | |
---|---|---|---|
|
51c456eb38 | ||
|
ecb1dce590 | ||
|
72a663a80b | ||
|
59fe324872 |
|
@ -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
|
||||||
|
|
|
@ -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"}]
|
||||||
|
|
|
@ -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 ''
|
||||||
|
|
|
@ -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;
|
||||||
|
|
|
@ -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>
|
||||||
|
|
|
@ -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}")
|
||||||
|
|
||||||
|
|
|
@ -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'}
|
||||||
|
|
Loading…
Reference in a new issue