Усі відповіді тут не відповідають одному чи декільком із наступного:
- Вони переписують щось (погано!), Що є у стандартній бібліотеці шаблонів (так, відповідь зверху!)
- Вони не використовують
and
для останнього предмета.
- Їм бракує серійної (оксфордської) коми.
- Вони використовують негативну індексацію, яка не працюватиме для наборів запитів django.
- Зазвичай вони не справляються належним чином з санітарною обробкою струн.
Ось мій вступ до цього канону. Спочатку тести:
class TestTextFilters(TestCase):
def test_oxford_zero_items(self):
self.assertEqual(oxford_comma([]), '')
def test_oxford_one_item(self):
self.assertEqual(oxford_comma(['a']), 'a')
def test_oxford_two_items(self):
self.assertEqual(oxford_comma(['a', 'b']), 'a and b')
def test_oxford_three_items(self):
self.assertEqual(oxford_comma(['a', 'b', 'c']), 'a, b, and c')
А тепер код. Так, стає трохи брудно, але ви побачите, що в ньому не використовується негативне індексування:
from django.utils.encoding import force_text
from django.utils.html import conditional_escape
from django.utils.safestring import mark_safe
@register.filter(is_safe=True, needs_autoescape=True)
def oxford_comma(l, autoescape=True):
"""Join together items in a list, separating them with commas or ', and'"""
l = map(force_text, l)
if autoescape:
l = map(conditional_escape, l)
num_items = len(l)
if num_items == 0:
s = ''
elif num_items == 1:
s = l[0]
elif num_items == 2:
s = l[0] + ' and ' + l[1]
elif num_items > 2:
for i, item in enumerate(l):
if i == 0:
s = item
elif i == (num_items - 1):
s += ', and ' + item
else:
s += ', ' + item
return mark_safe(s)
Ви можете використовувати це в шаблоні django з:
{% load my_filters %}
{{ items|oxford_comma }}