3
0
Fork 0

Compare commits

...

3 commits

Author SHA1 Message Date
Lance Edgar 8b23be7422 bump: version 0.20.3 → 0.20.4 2025-01-15 08:50:47 -06:00
Lance Edgar 9e0e36d536 fix: add WuttaDateWidget and associated logic 2025-01-15 08:40:56 -06:00
Lance Edgar e3c432aa37 fix: add serialize_object() method for ObjectRef schema node
so we can use this node type for non-wutta mapped class, with non-uuid
primary key
2025-01-15 08:21:15 -06:00
7 changed files with 127 additions and 11 deletions

View file

@ -5,6 +5,13 @@ 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.4 (2025-01-15)
### Fix
- add `WuttaDateWidget` and associated logic
- add `serialize_object()` method for `ObjectRef` schema node
## v0.20.3 (2025-01-14) ## v0.20.3 (2025-01-14)
### Fix ### Fix

View file

@ -6,7 +6,7 @@ build-backend = "hatchling.build"
[project] [project]
name = "WuttaWeb" name = "WuttaWeb"
version = "0.20.3" version = "0.20.4"
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

@ -546,10 +546,13 @@ class Form:
This is generally only possible if :attr:`model_class` is set This is generally only possible if :attr:`model_class` is set
to a valid SQLAlchemy mapped class. to a valid SQLAlchemy mapped class.
As of writing this only looks for This only checks for a couple of data types, with mapping as
:class:`sqlalchemy:sqlalchemy.types.DateTime` fields and if follows:
any are found, they are configured to use
:class:`~wuttaweb.forms.widgets.WuttaDateTimeWidget()`. * :class:`sqlalchemy:sqlalchemy.types.Date` ->
:class:`~wuttaweb.forms.widgets.WuttaDateWidget`
* :class:`sqlalchemy:sqlalchemy.types.DateTime` ->
:class:`~wuttaweb.forms.widgets.WuttaDateTimeWidget`
""" """
from wuttaweb.forms import widgets from wuttaweb.forms import widgets
@ -565,8 +568,9 @@ class Form:
prop = getattr(attr, 'prop', None) prop = getattr(attr, 'prop', None)
if prop and isinstance(prop, orm.ColumnProperty): if prop and isinstance(prop, orm.ColumnProperty):
column = prop.columns[0] column = prop.columns[0]
if isinstance(column.type, sa.DateTime): if isinstance(column.type, sa.Date):
# self.set_renderer(key, self.render_datetime) self.set_widget(key, widgets.WuttaDateWidget(self.request))
elif isinstance(column.type, sa.DateTime):
self.set_widget(key, widgets.WuttaDateTimeWidget(self.request)) self.set_widget(key, widgets.WuttaDateTimeWidget(self.request))
def set_grid(self, key, grid): def set_grid(self, key, grid):

View file

@ -334,8 +334,21 @@ class ObjectRef(colander.SchemaType):
# nb. keep a ref to this for later use # nb. keep a ref to this for later use
node.model_instance = appstruct node.model_instance = appstruct
# serialize to uuid # serialize to PK as string
return appstruct.uuid.hex return self.serialize_object(appstruct)
def serialize_object(self, obj):
"""
Serialize the given object to its primary key as string.
Default logic assumes the object has a UUID; subclass can
override as needed.
:param obj: Object reference for the node.
:returns: Object primary key as string.
"""
return obj.uuid.hex
def deserialize(self, node, cstruct): def deserialize(self, node, cstruct):
""" """ """ """
@ -417,7 +430,7 @@ class ObjectRef(colander.SchemaType):
if 'values' not in kwargs: if 'values' not in kwargs:
query = self.get_query() query = self.get_query()
objects = query.all() objects = query.all()
values = [(obj.uuid.hex, str(obj)) values = [(self.serialize_object(obj), str(obj))
for obj in objects] for obj in objects]
if self.empty_option: if self.empty_option:
values.insert(0, self.empty_option) values.insert(0, self.empty_option)

View file

@ -36,6 +36,7 @@ in the namespace:
* :class:`deform:deform.widget.CheckboxWidget` * :class:`deform:deform.widget.CheckboxWidget`
* :class:`deform:deform.widget.SelectWidget` * :class:`deform:deform.widget.SelectWidget`
* :class:`deform:deform.widget.CheckboxChoiceWidget` * :class:`deform:deform.widget.CheckboxChoiceWidget`
* :class:`deform:deform.widget.DateInputWidget`
* :class:`deform:deform.widget.DateTimeInputWidget` * :class:`deform:deform.widget.DateTimeInputWidget`
* :class:`deform:deform.widget.MoneyInputWidget` * :class:`deform:deform.widget.MoneyInputWidget`
""" """
@ -49,7 +50,7 @@ import humanize
from deform.widget import (Widget, TextInputWidget, TextAreaWidget, from deform.widget import (Widget, TextInputWidget, TextAreaWidget,
PasswordWidget, CheckedPasswordWidget, PasswordWidget, CheckedPasswordWidget,
CheckboxWidget, SelectWidget, CheckboxChoiceWidget, CheckboxWidget, SelectWidget, CheckboxChoiceWidget,
DateTimeInputWidget, MoneyInputWidget) DateInputWidget, DateTimeInputWidget, MoneyInputWidget)
from webhelpers2.html import HTML from webhelpers2.html import HTML
from wuttjamaican.conf import parse_list from wuttjamaican.conf import parse_list
@ -153,6 +154,43 @@ class WuttaCheckboxChoiceWidget(CheckboxChoiceWidget):
self.app = self.config.get_app() self.app = self.config.get_app()
class WuttaDateWidget(DateInputWidget):
"""
Custom widget for :class:`python:datetime.date` fields.
The main purpose of this widget is to leverage
:meth:`~wuttjamaican:wuttjamaican.app.AppHandler.render_date()`
for the readonly display.
It is automatically used for SQLAlchemy mapped classes where the
field maps to a :class:`sqlalchemy:sqlalchemy.types.Date` column.
For other (non-mapped) date fields, or mapped datetime fields for
which a date widget is preferred, use
:meth:`~wuttaweb.forms.base.Form.set_widget()`.
This is a subclass of
:class:`deform:deform.widget.DateInputWidget` and uses these
Deform templates:
* ``dateinput``
"""
def __init__(self, request, *args, **kwargs):
super().__init__(*args, **kwargs)
self.request = request
self.config = self.request.wutta_config
self.app = self.config.get_app()
def serialize(self, field, cstruct, **kw):
""" """
readonly = kw.get('readonly', self.readonly)
if readonly and cstruct:
dt = datetime.datetime.fromisoformat(cstruct)
return self.app.render_date(dt)
return super().serialize(field, cstruct, **kw)
class WuttaDateTimeWidget(DateTimeInputWidget): class WuttaDateTimeWidget(DateTimeInputWidget):
""" """
Custom widget for :class:`python:datetime.datetime` fields. Custom widget for :class:`python:datetime.datetime` fields.

View file

@ -3,6 +3,8 @@
from unittest import TestCase from unittest import TestCase
from unittest.mock import MagicMock, patch from unittest.mock import MagicMock, patch
import sqlalchemy as sa
import colander import colander
import deform import deform
from pyramid import testing from pyramid import testing
@ -188,6 +190,18 @@ class TestForm(TestCase):
self.assertIn('created', form.widgets) self.assertIn('created', form.widgets)
self.assertIsInstance(form.widgets['created'], MyWidget) self.assertIsInstance(form.widgets['created'], MyWidget)
# mock up a table with all relevant column types
class Whatever(model.Base):
__tablename__ = 'whatever'
id = sa.Column(sa.Integer(), primary_key=True)
date = sa.Column(sa.Date())
date_time = sa.Column(sa.DateTime())
# widget set for all known types
form = self.make_form(model_class=Whatever)
self.assertIsInstance(form.widgets['date'], widgets.WuttaDateWidget)
self.assertIsInstance(form.widgets['date_time'], widgets.WuttaDateTimeWidget)
def test_set_grid(self): def test_set_grid(self):
form = self.make_form(fields=['foo', 'bar']) form = self.make_form(fields=['foo', 'bar'])
self.assertNotIn('foo', form.widgets) self.assertNotIn('foo', form.widgets)

View file

@ -87,6 +87,46 @@ class TestObjectRefWidget(WebTestCase):
self.assertNotIn('url', values) self.assertNotIn('url', values)
class TestWuttaDateWidget(WebTestCase):
def make_field(self, node, **kwargs):
# TODO: not sure why default renderer is in use even though
# pyramid_deform was included in setup? but this works..
kwargs.setdefault('renderer', deform.Form.default_renderer)
return deform.Field(node, **kwargs)
def make_widget(self, **kwargs):
return mod.WuttaDateWidget(self.request, **kwargs)
def test_serialize(self):
node = colander.SchemaNode(colander.Date())
field = self.make_field(node)
# first try normal date
widget = self.make_widget()
dt = datetime.date(2025, 1, 15)
# editable widget has normal picker html
result = widget.serialize(field, str(dt))
self.assertIn('<wutta-datepicker', result)
# readonly is rendered per app convention
result = widget.serialize(field, str(dt), readonly=True)
self.assertEqual(result, '2025-01-15')
# now try again with datetime
widget = self.make_widget()
dt = datetime.datetime(2025, 1, 15, 8, 35)
# editable widget has normal picker html
result = widget.serialize(field, str(dt))
self.assertIn('<wutta-datepicker', result)
# readonly is rendered per app convention
result = widget.serialize(field, str(dt), readonly=True)
self.assertEqual(result, '2025-01-15')
class TestWuttaDateTimeWidget(WebTestCase): class TestWuttaDateTimeWidget(WebTestCase):
def make_field(self, node, **kwargs): def make_field(self, node, **kwargs):