Skip to content

Commit 89efdab

Browse files
committed
Fixed #5363 -- HTML5 datetime-local valid format HTMLFormRenderer
Co-authored-by: Peter Thomassen
1 parent 577bb3c commit 89efdab

File tree

2 files changed

+98
-4
lines changed

2 files changed

+98
-4
lines changed

rest_framework/renderers.py

Lines changed: 20 additions & 4 deletions
Original file line numberDiff line numberDiff line change
@@ -22,7 +22,7 @@
2222
from django.utils.http import parse_header_parameters
2323
from django.utils.safestring import SafeString
2424

25-
from rest_framework import VERSION, exceptions, serializers, status
25+
from rest_framework import ISO_8601, VERSION, exceptions, serializers, status
2626
from rest_framework.compat import (
2727
INDENT_SEPARATORS, LONG_SEPARATORS, SHORT_SEPARATORS, coreapi, coreschema,
2828
pygments_css, yaml
@@ -339,11 +339,27 @@ def render_field(self, field, parent_style):
339339
style['template_pack'] = parent_style.get('template_pack', self.template_pack)
340340
style['renderer'] = self
341341

342-
# Get a clone of the field with text-only value representation.
342+
# Get a clone of the field with text-only value representation ('' if None or False).
343343
field = field.as_form_field()
344344

345-
if style.get('input_type') == 'datetime-local' and isinstance(field.value, str):
346-
field.value = field.value.rstrip('Z')
345+
if style.get('input_type') == 'datetime-local':
346+
try:
347+
format_ = field._field.format
348+
except AttributeError:
349+
format_ = api_settings.DATETIME_FORMAT
350+
351+
if format_ is not None:
352+
# field.value is expected to be a string
353+
# https://www.django-rest-framework.org/api-guide/fields/#datetimefield
354+
field.value = (
355+
datetime.datetime.fromisoformat(field.value.rstrip('Z')) if format_ == ISO_8601
356+
else datetime.datetime.strptime(field.value, format_)
357+
)
358+
359+
# The format of an input type="datetime-local" is "yyyy-MM-ddThh:mm"
360+
# followed by optional ":ss" or ":ss.SSS", so keep only the first three
361+
# digits of milliseconds to avoid browser console error.
362+
field.value = field.value.replace(tzinfo=None).isoformat(timespec="milliseconds")
347363

348364
if 'template' in style:
349365
template_name = style['template']

tests/test_renderers.py

Lines changed: 78 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
import re
22
from collections.abc import MutableMapping
3+
from datetime import datetime
34

45
import pytest
56
from django.core.cache import cache
@@ -488,6 +489,83 @@ class TestSerializer(serializers.Serializer):
488489
assert rendered == ''
489490

490491

492+
class TestDateTimeFieldHTMLFormRender(TestCase):
493+
"""
494+
Default USE_TZ is True.
495+
Default TIME_ZONE is 'America/Chicago'.
496+
"""
497+
498+
def _assert_datetime_rendering(self, appointment, expected, datetimefield_kwargs=None):
499+
datetimefield_kwargs = datetimefield_kwargs or {}
500+
501+
class TestSerializer(serializers.Serializer):
502+
appointment = serializers.DateTimeField(**datetimefield_kwargs)
503+
504+
serializer = TestSerializer(data={"appointment": appointment})
505+
serializer.is_valid()
506+
renderer = HTMLFormRenderer()
507+
field = serializer['appointment']
508+
rendered = renderer.render_field(field, {})
509+
expected_html = (
510+
'<input name="appointment" class="form-control" '
511+
f'type="datetime-local" value="{expected}">'
512+
)
513+
514+
self.assertInHTML(expected_html, rendered)
515+
516+
def test_datetime_field_rendering_milliseconds(self):
517+
self._assert_datetime_rendering(
518+
datetime(2024, 12, 24, 0, 55, 30, 345678), "2024-12-24T00:55:30.345"
519+
)
520+
521+
def test_datetime_field_rendering_no_seconds_and_no_milliseconds(self):
522+
self._assert_datetime_rendering(
523+
datetime(2024, 12, 24, 0, 55, 0, 0), "2024-12-24T00:55:00.000"
524+
)
525+
526+
def test_datetime_field_rendering_with_format_as_none(self):
527+
self._assert_datetime_rendering(
528+
datetime(2024, 12, 24, 0, 55, 30, 345678),
529+
"2024-12-24T00:55:30.345",
530+
{"format": None}
531+
)
532+
533+
def test_datetime_field_rendering_with_format(self):
534+
self._assert_datetime_rendering(
535+
datetime(2024, 12, 24, 0, 55, 30, 345678),
536+
"2024-12-24T00:55:00.000",
537+
{"format": "%a %d %b %Y, %I:%M%p"}
538+
)
539+
540+
@override_settings(TIME_ZONE='UTC')
541+
def test_datetime_field_rendering_utc(self):
542+
self._assert_datetime_rendering(
543+
datetime(2024, 12, 24, 0, 55, 30, 345678),
544+
"2024-12-24T00:55:30.345"
545+
)
546+
547+
@override_settings(TIME_ZONE='Asia/Tokyo') # +09:00
548+
def test_datetime_field_rendering_non_zero_timezone(self):
549+
self._assert_datetime_rendering(
550+
datetime(2024, 12, 24, 0, 55, 30, 345678),
551+
"2024-12-24T00:55:30.345"
552+
)
553+
554+
@override_settings(REST_FRAMEWORK={'DATETIME_FORMAT': '%a %d %b %Y, %I:%M%p'})
555+
def test_datetime_field_rendering_with_custom_datetime_format(self):
556+
self._assert_datetime_rendering(
557+
datetime(2024, 12, 24, 0, 55, 30, 345678),
558+
"2024-12-24T00:55:00.000"
559+
)
560+
561+
@override_settings(REST_FRAMEWORK={'DATETIME_FORMAT': None})
562+
def test_datetime_field_rendering_datetime_format_is_none(self):
563+
self._assert_datetime_rendering(
564+
datetime(2024, 12, 24, 0, 55, 30, 345678),
565+
"2024-12-24T00:55:30.345"
566+
)
567+
568+
491569
class TestHTMLFormRenderer(TestCase):
492570
def setUp(self):
493571
class TestSerializer(serializers.Serializer):

0 commit comments

Comments
 (0)