Все ответы здесь не соответствуют одному или нескольким из следующих:
- Они переписывают что-то (плохо!), Которое находится в стандартной библиотеке шаблонов (ack, лучший ответ!)
- Они не используют
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:
# First item
s = item
elif i == (num_items - 1):
# Last item.
s += ', and ' + item
else:
# Items in the middle
s += ', ' + item
return mark_safe(s)
Вы можете использовать это в шаблоне django с:
{% load my_filters %}
{{ items|oxford_comma }}