Alle Antworten hier schlagen eine oder mehrere der folgenden fehl:
- Sie schreiben etwas (schlecht!) Um, das in der Standardvorlagenbibliothek enthalten ist (ack, top answer!)
- Sie werden nicht
and
für den letzten Artikel verwendet.
- Ihnen fehlt ein serielles (Oxford) Komma.
- Sie verwenden eine negative Indizierung, die für Django-Abfragesätze nicht funktioniert.
- Sie handhaben normalerweise die Saitenhygiene nicht richtig.
Hier ist mein Eintrag in diesen Kanon. Zunächst die Tests:
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')
Und jetzt der Code. Ja, es wird ein bisschen chaotisch, aber Sie werden sehen, dass keine negative Indizierung verwendet wird:
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)
Sie können dies in einer Django-Vorlage verwenden mit:
{% load my_filters %}
{{ items|oxford_comma }}