mirror of https://github.com/django/django.git
Fixed #29654 -- Made text truncation an ellipsis character instead of three dots.
Thanks Sudhanshu Mishra for the initial patch and Tim Graham for the review.
This commit is contained in:
parent
939dcff24f
commit
201017df30
|
@ -193,7 +193,7 @@ class ForeignKeyRawIdWidget(forms.TextInput):
|
||||||
except NoReverseMatch:
|
except NoReverseMatch:
|
||||||
url = '' # Admin not registered for target model.
|
url = '' # Admin not registered for target model.
|
||||||
|
|
||||||
return Truncator(obj).words(14, truncate='...'), url
|
return Truncator(obj).words(14), url
|
||||||
|
|
||||||
|
|
||||||
class ManyToManyRawIdWidget(ForeignKeyRawIdWidget):
|
class ManyToManyRawIdWidget(ForeignKeyRawIdWidget):
|
||||||
|
|
|
@ -280,7 +280,7 @@ def truncatewords(value, arg):
|
||||||
length = int(arg)
|
length = int(arg)
|
||||||
except ValueError: # Invalid literal for int().
|
except ValueError: # Invalid literal for int().
|
||||||
return value # Fail silently.
|
return value # Fail silently.
|
||||||
return Truncator(value).words(length, truncate=' ...')
|
return Truncator(value).words(length, truncate=' …')
|
||||||
|
|
||||||
|
|
||||||
@register.filter(is_safe=True)
|
@register.filter(is_safe=True)
|
||||||
|
@ -294,7 +294,7 @@ def truncatewords_html(value, arg):
|
||||||
length = int(arg)
|
length = int(arg)
|
||||||
except ValueError: # invalid literal for int()
|
except ValueError: # invalid literal for int()
|
||||||
return value # Fail silently.
|
return value # Fail silently.
|
||||||
return Truncator(value).words(length, html=True, truncate=' ...')
|
return Truncator(value).words(length, html=True, truncate=' …')
|
||||||
|
|
||||||
|
|
||||||
@register.filter(is_safe=False)
|
@register.filter(is_safe=False)
|
||||||
|
|
|
@ -245,7 +245,7 @@ def urlize(text, trim_url_limit=None, nofollow=False, autoescape=False):
|
||||||
leading punctuation (opening parens) and it'll still do the right thing.
|
leading punctuation (opening parens) and it'll still do the right thing.
|
||||||
|
|
||||||
If trim_url_limit is not None, truncate the URLs in the link text longer
|
If trim_url_limit is not None, truncate the URLs in the link text longer
|
||||||
than this limit to trim_url_limit-3 characters and append an ellipsis.
|
than this limit to trim_url_limit - 1 characters and append an ellipsis.
|
||||||
|
|
||||||
If nofollow is True, give the links a rel="nofollow" attribute.
|
If nofollow is True, give the links a rel="nofollow" attribute.
|
||||||
|
|
||||||
|
@ -256,7 +256,7 @@ def urlize(text, trim_url_limit=None, nofollow=False, autoescape=False):
|
||||||
def trim_url(x, limit=trim_url_limit):
|
def trim_url(x, limit=trim_url_limit):
|
||||||
if limit is None or len(x) <= limit:
|
if limit is None or len(x) <= limit:
|
||||||
return x
|
return x
|
||||||
return '%s...' % x[:max(0, limit - 3)]
|
return '%s…' % x[:max(0, limit - 1)]
|
||||||
|
|
||||||
def unescape(text, trail):
|
def unescape(text, trail):
|
||||||
"""
|
"""
|
||||||
|
|
|
@ -64,7 +64,7 @@ class Truncator(SimpleLazyObject):
|
||||||
if truncate is None:
|
if truncate is None:
|
||||||
truncate = pgettext(
|
truncate = pgettext(
|
||||||
'String to return when truncating text',
|
'String to return when truncating text',
|
||||||
'%(truncated_text)s...')
|
'%(truncated_text)s…')
|
||||||
if '%(truncated_text)s' in truncate:
|
if '%(truncated_text)s' in truncate:
|
||||||
return truncate % {'truncated_text': text}
|
return truncate % {'truncated_text': text}
|
||||||
# The truncation text didn't contain the %(truncated_text)s string
|
# The truncation text didn't contain the %(truncated_text)s string
|
||||||
|
@ -81,8 +81,7 @@ class Truncator(SimpleLazyObject):
|
||||||
of characters.
|
of characters.
|
||||||
|
|
||||||
`truncate` specifies what should be used to notify that the string has
|
`truncate` specifies what should be used to notify that the string has
|
||||||
been truncated, defaulting to a translatable string of an ellipsis
|
been truncated, defaulting to a translatable string of an ellipsis.
|
||||||
(...).
|
|
||||||
"""
|
"""
|
||||||
self._setup()
|
self._setup()
|
||||||
length = int(num)
|
length = int(num)
|
||||||
|
@ -123,7 +122,7 @@ class Truncator(SimpleLazyObject):
|
||||||
"""
|
"""
|
||||||
Truncate a string after a certain number of words. `truncate` specifies
|
Truncate a string after a certain number of words. `truncate` specifies
|
||||||
what should be used to notify that the string has been truncated,
|
what should be used to notify that the string has been truncated,
|
||||||
defaulting to ellipsis (...).
|
defaulting to ellipsis.
|
||||||
"""
|
"""
|
||||||
self._setup()
|
self._setup()
|
||||||
length = int(num)
|
length = int(num)
|
||||||
|
|
|
@ -2265,15 +2265,15 @@ If ``value`` is ``"my FIRST post"``, the output will be ``"My First Post"``.
|
||||||
-----------------
|
-----------------
|
||||||
|
|
||||||
Truncates a string if it is longer than the specified number of characters.
|
Truncates a string if it is longer than the specified number of characters.
|
||||||
Truncated strings will end with a translatable ellipsis sequence ("...").
|
Truncated strings will end with a translatable ellipsis character ("…").
|
||||||
|
|
||||||
**Argument:** Number of characters to truncate to
|
**Argument:** Number of characters to truncate to
|
||||||
|
|
||||||
For example::
|
For example::
|
||||||
|
|
||||||
{{ value|truncatechars:9 }}
|
{{ value|truncatechars:7 }}
|
||||||
|
|
||||||
If ``value`` is ``"Joel is a slug"``, the output will be ``"Joel i..."``.
|
If ``value`` is ``"Joel is a slug"``, the output will be ``"Joel i…"``.
|
||||||
|
|
||||||
.. templatefilter:: truncatechars_html
|
.. templatefilter:: truncatechars_html
|
||||||
|
|
||||||
|
@ -2286,10 +2286,10 @@ are closed immediately after the truncation.
|
||||||
|
|
||||||
For example::
|
For example::
|
||||||
|
|
||||||
{{ value|truncatechars_html:9 }}
|
{{ value|truncatechars_html:7 }}
|
||||||
|
|
||||||
If ``value`` is ``"<p>Joel is a slug</p>"``, the output will be
|
If ``value`` is ``"<p>Joel is a slug</p>"``, the output will be
|
||||||
``"<p>Joel i...</p>"``.
|
``"<p>Joel i…</p>"``.
|
||||||
|
|
||||||
Newlines in the HTML content will be preserved.
|
Newlines in the HTML content will be preserved.
|
||||||
|
|
||||||
|
@ -2306,7 +2306,7 @@ For example::
|
||||||
|
|
||||||
{{ value|truncatewords:2 }}
|
{{ value|truncatewords:2 }}
|
||||||
|
|
||||||
If ``value`` is ``"Joel is a slug"``, the output will be ``"Joel is ..."``.
|
If ``value`` is ``"Joel is a slug"``, the output will be ``"Joel is …"``.
|
||||||
|
|
||||||
Newlines within the string will be removed.
|
Newlines within the string will be removed.
|
||||||
|
|
||||||
|
@ -2327,7 +2327,7 @@ For example::
|
||||||
{{ value|truncatewords_html:2 }}
|
{{ value|truncatewords_html:2 }}
|
||||||
|
|
||||||
If ``value`` is ``"<p>Joel is a slug</p>"``, the output will be
|
If ``value`` is ``"<p>Joel is a slug</p>"``, the output will be
|
||||||
``"<p>Joel is ...</p>"``.
|
``"<p>Joel is …</p>"``.
|
||||||
|
|
||||||
Newlines in the HTML content will be preserved.
|
Newlines in the HTML content will be preserved.
|
||||||
|
|
||||||
|
@ -2454,7 +2454,7 @@ For example::
|
||||||
|
|
||||||
If ``value`` is ``"Check out www.djangoproject.com"``, the output would be
|
If ``value`` is ``"Check out www.djangoproject.com"``, the output would be
|
||||||
``'Check out <a href="http://www.djangoproject.com"
|
``'Check out <a href="http://www.djangoproject.com"
|
||||||
rel="nofollow">www.djangopr...</a>'``.
|
rel="nofollow">www.djangoproj…</a>'``.
|
||||||
|
|
||||||
As with urlize_, this filter should only be applied to plain text.
|
As with urlize_, this filter should only be applied to plain text.
|
||||||
|
|
||||||
|
|
|
@ -273,6 +273,12 @@ Miscellaneous
|
||||||
* The return value of :func:`django.utils.text.slugify` is no longer marked as
|
* The return value of :func:`django.utils.text.slugify` is no longer marked as
|
||||||
HTML safe.
|
HTML safe.
|
||||||
|
|
||||||
|
* The default truncation character used by the :tfilter:`urlizetrunc`,
|
||||||
|
:tfilter:`truncatechars`, :tfilter:`truncatechars_html`,
|
||||||
|
:tfilter:`truncatewords`, and :tfilter:`truncatewords_html` template filters
|
||||||
|
is now the real ellipsis character (``…``) instead of 3 dots. You may have to
|
||||||
|
adapt some test output comparisons.
|
||||||
|
|
||||||
.. _deprecated-features-2.2:
|
.. _deprecated-features-2.2:
|
||||||
|
|
||||||
Features deprecated in 2.2
|
Features deprecated in 2.2
|
||||||
|
|
|
@ -346,7 +346,7 @@ class MigrateTests(MigrationTestBase):
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
'Planned operations:\n'
|
'Planned operations:\n'
|
||||||
'migrations.0004_fourth\n'
|
'migrations.0004_fourth\n'
|
||||||
' Raw SQL operation -> SELECT * FROM migrations_author W...\n',
|
' Raw SQL operation -> SELECT * FROM migrations_author WHE…\n',
|
||||||
out.getvalue()
|
out.getvalue()
|
||||||
)
|
)
|
||||||
# Migrate to the fourth migration.
|
# Migrate to the fourth migration.
|
||||||
|
|
|
@ -5,10 +5,10 @@ from ..utils import setup
|
||||||
|
|
||||||
class TruncatecharsTests(SimpleTestCase):
|
class TruncatecharsTests(SimpleTestCase):
|
||||||
|
|
||||||
@setup({'truncatechars01': '{{ a|truncatechars:5 }}'})
|
@setup({'truncatechars01': '{{ a|truncatechars:3 }}'})
|
||||||
def test_truncatechars01(self):
|
def test_truncatechars01(self):
|
||||||
output = self.engine.render_to_string('truncatechars01', {'a': 'Testing, testing'})
|
output = self.engine.render_to_string('truncatechars01', {'a': 'Testing, testing'})
|
||||||
self.assertEqual(output, 'Te...')
|
self.assertEqual(output, 'Te…')
|
||||||
|
|
||||||
@setup({'truncatechars02': '{{ a|truncatechars:7 }}'})
|
@setup({'truncatechars02': '{{ a|truncatechars:7 }}'})
|
||||||
def test_truncatechars02(self):
|
def test_truncatechars02(self):
|
||||||
|
|
|
@ -5,18 +5,18 @@ from django.test import SimpleTestCase
|
||||||
class FunctionTests(SimpleTestCase):
|
class FunctionTests(SimpleTestCase):
|
||||||
|
|
||||||
def test_truncate_zero(self):
|
def test_truncate_zero(self):
|
||||||
self.assertEqual(truncatechars_html('<p>one <a href="#">two - three <br>four</a> five</p>', 0), '...')
|
self.assertEqual(truncatechars_html('<p>one <a href="#">two - three <br>four</a> five</p>', 0), '…')
|
||||||
|
|
||||||
def test_truncate(self):
|
def test_truncate(self):
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
truncatechars_html('<p>one <a href="#">two - three <br>four</a> five</p>', 6),
|
truncatechars_html('<p>one <a href="#">two - three <br>four</a> five</p>', 4),
|
||||||
'<p>one...</p>',
|
'<p>one…</p>',
|
||||||
)
|
)
|
||||||
|
|
||||||
def test_truncate2(self):
|
def test_truncate2(self):
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
truncatechars_html('<p>one <a href="#">two - three <br>four</a> five</p>', 11),
|
truncatechars_html('<p>one <a href="#">two - three <br>four</a> five</p>', 9),
|
||||||
'<p>one <a href="#">two ...</a></p>',
|
'<p>one <a href="#">two …</a></p>',
|
||||||
)
|
)
|
||||||
|
|
||||||
def test_truncate3(self):
|
def test_truncate3(self):
|
||||||
|
@ -26,7 +26,7 @@ class FunctionTests(SimpleTestCase):
|
||||||
)
|
)
|
||||||
|
|
||||||
def test_truncate_unicode(self):
|
def test_truncate_unicode(self):
|
||||||
self.assertEqual(truncatechars_html('<b>\xc5ngstr\xf6m</b> was here', 5), '<b>\xc5n...</b>')
|
self.assertEqual(truncatechars_html('<b>\xc5ngstr\xf6m</b> was here', 3), '<b>\xc5n…</b>')
|
||||||
|
|
||||||
def test_truncate_something(self):
|
def test_truncate_something(self):
|
||||||
self.assertEqual(truncatechars_html('a<b>b</b>c', 3), 'a<b>b</b>c')
|
self.assertEqual(truncatechars_html('a<b>b</b>c', 3), 'a<b>b</b>c')
|
||||||
|
|
|
@ -14,25 +14,25 @@ class TruncatewordsTests(SimpleTestCase):
|
||||||
output = self.engine.render_to_string(
|
output = self.engine.render_to_string(
|
||||||
'truncatewords01', {'a': 'alpha & bravo', 'b': mark_safe('alpha & bravo')}
|
'truncatewords01', {'a': 'alpha & bravo', 'b': mark_safe('alpha & bravo')}
|
||||||
)
|
)
|
||||||
self.assertEqual(output, 'alpha & ... alpha & ...')
|
self.assertEqual(output, 'alpha & … alpha & …')
|
||||||
|
|
||||||
@setup({'truncatewords02': '{{ a|truncatewords:"2" }} {{ b|truncatewords:"2"}}'})
|
@setup({'truncatewords02': '{{ a|truncatewords:"2" }} {{ b|truncatewords:"2"}}'})
|
||||||
def test_truncatewords02(self):
|
def test_truncatewords02(self):
|
||||||
output = self.engine.render_to_string(
|
output = self.engine.render_to_string(
|
||||||
'truncatewords02', {'a': 'alpha & bravo', 'b': mark_safe('alpha & bravo')}
|
'truncatewords02', {'a': 'alpha & bravo', 'b': mark_safe('alpha & bravo')}
|
||||||
)
|
)
|
||||||
self.assertEqual(output, 'alpha & ... alpha & ...')
|
self.assertEqual(output, 'alpha & … alpha & …')
|
||||||
|
|
||||||
|
|
||||||
class FunctionTests(SimpleTestCase):
|
class FunctionTests(SimpleTestCase):
|
||||||
|
|
||||||
def test_truncate(self):
|
def test_truncate(self):
|
||||||
self.assertEqual(truncatewords('A sentence with a few words in it', 1), 'A ...')
|
self.assertEqual(truncatewords('A sentence with a few words in it', 1), 'A …')
|
||||||
|
|
||||||
def test_truncate2(self):
|
def test_truncate2(self):
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
truncatewords('A sentence with a few words in it', 5),
|
truncatewords('A sentence with a few words in it', 5),
|
||||||
'A sentence with a few ...',
|
'A sentence with a few …',
|
||||||
)
|
)
|
||||||
|
|
||||||
def test_overtruncate(self):
|
def test_overtruncate(self):
|
||||||
|
|
|
@ -10,13 +10,13 @@ class FunctionTests(SimpleTestCase):
|
||||||
def test_truncate(self):
|
def test_truncate(self):
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
truncatewords_html('<p>one <a href="#">two - three <br>four</a> five</p>', 2),
|
truncatewords_html('<p>one <a href="#">two - three <br>four</a> five</p>', 2),
|
||||||
'<p>one <a href="#">two ...</a></p>',
|
'<p>one <a href="#">two …</a></p>',
|
||||||
)
|
)
|
||||||
|
|
||||||
def test_truncate2(self):
|
def test_truncate2(self):
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
truncatewords_html('<p>one <a href="#">two - three <br>four</a> five</p>', 4),
|
truncatewords_html('<p>one <a href="#">two - three <br>four</a> five</p>', 4),
|
||||||
'<p>one <a href="#">two - three <br>four ...</a></p>',
|
'<p>one <a href="#">two - three <br>four …</a></p>',
|
||||||
)
|
)
|
||||||
|
|
||||||
def test_truncate3(self):
|
def test_truncate3(self):
|
||||||
|
@ -32,12 +32,12 @@ class FunctionTests(SimpleTestCase):
|
||||||
)
|
)
|
||||||
|
|
||||||
def test_truncate_unicode(self):
|
def test_truncate_unicode(self):
|
||||||
self.assertEqual(truncatewords_html('\xc5ngstr\xf6m was here', 1), '\xc5ngstr\xf6m ...')
|
self.assertEqual(truncatewords_html('\xc5ngstr\xf6m was here', 1), '\xc5ngstr\xf6m …')
|
||||||
|
|
||||||
def test_truncate_complex(self):
|
def test_truncate_complex(self):
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
truncatewords_html('<i>Buenos días! ¿Cómo está?</i>', 3),
|
truncatewords_html('<i>Buenos días! ¿Cómo está?</i>', 3),
|
||||||
'<i>Buenos días! ¿Cómo ...</i>',
|
'<i>Buenos días! ¿Cómo …</i>',
|
||||||
)
|
)
|
||||||
|
|
||||||
def test_invalid_arg(self):
|
def test_invalid_arg(self):
|
||||||
|
|
|
@ -20,8 +20,8 @@ class UrlizetruncTests(SimpleTestCase):
|
||||||
)
|
)
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
output,
|
output,
|
||||||
'"Unsafe" <a href="http://example.com/x=&y=" rel="nofollow">http:...</a> '
|
'"Unsafe" <a href="http://example.com/x=&y=" rel="nofollow">http://…</a> '
|
||||||
'"Safe" <a href="http://example.com?x=&y=" rel="nofollow">http:...</a>'
|
'"Safe" <a href="http://example.com?x=&y=" rel="nofollow">http://…</a>'
|
||||||
)
|
)
|
||||||
|
|
||||||
@setup({'urlizetrunc02': '{{ a|urlizetrunc:"8" }} {{ b|urlizetrunc:"8" }}'})
|
@setup({'urlizetrunc02': '{{ a|urlizetrunc:"8" }} {{ b|urlizetrunc:"8" }}'})
|
||||||
|
@ -35,8 +35,8 @@ class UrlizetruncTests(SimpleTestCase):
|
||||||
)
|
)
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
output,
|
output,
|
||||||
'"Unsafe" <a href="http://example.com/x=&y=" rel="nofollow">http:...</a> '
|
'"Unsafe" <a href="http://example.com/x=&y=" rel="nofollow">http://…</a> '
|
||||||
'"Safe" <a href="http://example.com?x=&y=" rel="nofollow">http:...</a>'
|
'"Safe" <a href="http://example.com?x=&y=" rel="nofollow">http://…</a>'
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@ -55,13 +55,13 @@ class FunctionTests(SimpleTestCase):
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
urlizetrunc(uri, 30),
|
urlizetrunc(uri, 30),
|
||||||
'<a href="http://31characteruri.com/test/" rel="nofollow">'
|
'<a href="http://31characteruri.com/test/" rel="nofollow">'
|
||||||
'http://31characteruri.com/t...</a>',
|
'http://31characteruri.com/tes…</a>',
|
||||||
)
|
)
|
||||||
|
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
urlizetrunc(uri, 2),
|
urlizetrunc(uri, 1),
|
||||||
'<a href="http://31characteruri.com/test/"'
|
'<a href="http://31characteruri.com/test/"'
|
||||||
' rel="nofollow">...</a>',
|
' rel="nofollow">…</a>',
|
||||||
)
|
)
|
||||||
|
|
||||||
def test_overtruncate(self):
|
def test_overtruncate(self):
|
||||||
|
@ -74,7 +74,7 @@ class FunctionTests(SimpleTestCase):
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
urlizetrunc('http://www.google.co.uk/search?hl=en&q=some+long+url&btnG=Search&meta=', 20),
|
urlizetrunc('http://www.google.co.uk/search?hl=en&q=some+long+url&btnG=Search&meta=', 20),
|
||||||
'<a href="http://www.google.co.uk/search?hl=en&q=some+long+url&btnG=Search&'
|
'<a href="http://www.google.co.uk/search?hl=en&q=some+long+url&btnG=Search&'
|
||||||
'meta=" rel="nofollow">http://www.google...</a>',
|
'meta=" rel="nofollow">http://www.google.c…</a>',
|
||||||
)
|
)
|
||||||
|
|
||||||
def test_non_string_input(self):
|
def test_non_string_input(self):
|
||||||
|
@ -89,5 +89,5 @@ class FunctionTests(SimpleTestCase):
|
||||||
def test_autoescape_off(self):
|
def test_autoescape_off(self):
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
urlizetrunc('foo<a href=" google.com ">bar</a>buz', 9, autoescape=False),
|
urlizetrunc('foo<a href=" google.com ">bar</a>buz', 9, autoescape=False),
|
||||||
'foo<a href=" <a href="http://google.com" rel="nofollow">google...</a> ">bar</a>buz',
|
'foo<a href=" <a href="http://google.com" rel="nofollow">google.c…</a> ">bar</a>buz',
|
||||||
)
|
)
|
||||||
|
|
|
@ -168,7 +168,7 @@ class FilterSyntaxTests(SimpleTestCase):
|
||||||
Numbers as filter arguments should work
|
Numbers as filter arguments should work
|
||||||
"""
|
"""
|
||||||
output = self.engine.render_to_string('filter-syntax19', {"var": "hello world"})
|
output = self.engine.render_to_string('filter-syntax19', {"var": "hello world"})
|
||||||
self.assertEqual(output, "hello ...")
|
self.assertEqual(output, "hello …")
|
||||||
|
|
||||||
@setup({'filter-syntax20': '{{ ""|default_if_none:"was none" }}'})
|
@setup({'filter-syntax20': '{{ ""|default_if_none:"was none" }}'})
|
||||||
def test_filter_syntax20(self):
|
def test_filter_syntax20(self):
|
||||||
|
|
|
@ -56,22 +56,22 @@ class TestUtilsText(SimpleTestCase):
|
||||||
def test_truncate_chars(self):
|
def test_truncate_chars(self):
|
||||||
truncator = text.Truncator('The quick brown fox jumped over the lazy dog.')
|
truncator = text.Truncator('The quick brown fox jumped over the lazy dog.')
|
||||||
self.assertEqual('The quick brown fox jumped over the lazy dog.', truncator.chars(100)),
|
self.assertEqual('The quick brown fox jumped over the lazy dog.', truncator.chars(100)),
|
||||||
self.assertEqual('The quick brown fox ...', truncator.chars(23)),
|
self.assertEqual('The quick brown fox …', truncator.chars(21)),
|
||||||
self.assertEqual('The quick brown fo.....', truncator.chars(23, '.....')),
|
self.assertEqual('The quick brown fo.....', truncator.chars(23, '.....')),
|
||||||
|
|
||||||
nfc = text.Truncator('o\xfco\xfco\xfco\xfc')
|
nfc = text.Truncator('o\xfco\xfco\xfco\xfc')
|
||||||
nfd = text.Truncator('ou\u0308ou\u0308ou\u0308ou\u0308')
|
nfd = text.Truncator('ou\u0308ou\u0308ou\u0308ou\u0308')
|
||||||
self.assertEqual('oüoüoüoü', nfc.chars(8))
|
self.assertEqual('oüoüoüoü', nfc.chars(8))
|
||||||
self.assertEqual('oüoüoüoü', nfd.chars(8))
|
self.assertEqual('oüoüoüoü', nfd.chars(8))
|
||||||
self.assertEqual('oü...', nfc.chars(5))
|
self.assertEqual('oü…', nfc.chars(3))
|
||||||
self.assertEqual('oü...', nfd.chars(5))
|
self.assertEqual('oü…', nfd.chars(3))
|
||||||
|
|
||||||
# Ensure the final length is calculated correctly when there are
|
# Ensure the final length is calculated correctly when there are
|
||||||
# combining characters with no precomposed form, and that combining
|
# combining characters with no precomposed form, and that combining
|
||||||
# characters are not split up.
|
# characters are not split up.
|
||||||
truncator = text.Truncator('-B\u030AB\u030A----8')
|
truncator = text.Truncator('-B\u030AB\u030A----8')
|
||||||
self.assertEqual('-B\u030A...', truncator.chars(5))
|
self.assertEqual('-B\u030A…', truncator.chars(3))
|
||||||
self.assertEqual('-B\u030AB\u030A-...', truncator.chars(7))
|
self.assertEqual('-B\u030AB\u030A-…', truncator.chars(5))
|
||||||
self.assertEqual('-B\u030AB\u030A----8', truncator.chars(8))
|
self.assertEqual('-B\u030AB\u030A----8', truncator.chars(8))
|
||||||
|
|
||||||
# Ensure the length of the end text is correctly calculated when it
|
# Ensure the length of the end text is correctly calculated when it
|
||||||
|
@ -82,18 +82,18 @@ class TestUtilsText(SimpleTestCase):
|
||||||
|
|
||||||
# Make a best effort to shorten to the desired length, but requesting
|
# Make a best effort to shorten to the desired length, but requesting
|
||||||
# a length shorter than the ellipsis shouldn't break
|
# a length shorter than the ellipsis shouldn't break
|
||||||
self.assertEqual('...', text.Truncator('asdf').chars(1))
|
self.assertEqual('…', text.Truncator('asdf').chars(0))
|
||||||
# lazy strings are handled correctly
|
# lazy strings are handled correctly
|
||||||
self.assertEqual(text.Truncator(lazystr('The quick brown fox')).chars(12), 'The quick...')
|
self.assertEqual(text.Truncator(lazystr('The quick brown fox')).chars(10), 'The quick…')
|
||||||
|
|
||||||
def test_truncate_words(self):
|
def test_truncate_words(self):
|
||||||
truncator = text.Truncator('The quick brown fox jumped over the lazy dog.')
|
truncator = text.Truncator('The quick brown fox jumped over the lazy dog.')
|
||||||
self.assertEqual('The quick brown fox jumped over the lazy dog.', truncator.words(10))
|
self.assertEqual('The quick brown fox jumped over the lazy dog.', truncator.words(10))
|
||||||
self.assertEqual('The quick brown fox...', truncator.words(4))
|
self.assertEqual('The quick brown fox…', truncator.words(4))
|
||||||
self.assertEqual('The quick brown fox[snip]', truncator.words(4, '[snip]'))
|
self.assertEqual('The quick brown fox[snip]', truncator.words(4, '[snip]'))
|
||||||
# lazy strings are handled correctly
|
# lazy strings are handled correctly
|
||||||
truncator = text.Truncator(lazystr('The quick brown fox jumped over the lazy dog.'))
|
truncator = text.Truncator(lazystr('The quick brown fox jumped over the lazy dog.'))
|
||||||
self.assertEqual('The quick brown fox...', truncator.words(4))
|
self.assertEqual('The quick brown fox…', truncator.words(4))
|
||||||
|
|
||||||
def test_truncate_html_words(self):
|
def test_truncate_html_words(self):
|
||||||
truncator = text.Truncator(
|
truncator = text.Truncator(
|
||||||
|
@ -104,7 +104,7 @@ class TestUtilsText(SimpleTestCase):
|
||||||
truncator.words(10, html=True)
|
truncator.words(10, html=True)
|
||||||
)
|
)
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
'<p id="par"><strong><em>The quick brown fox...</em></strong></p>',
|
'<p id="par"><strong><em>The quick brown fox…</em></strong></p>',
|
||||||
truncator.words(4, html=True)
|
truncator.words(4, html=True)
|
||||||
)
|
)
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
|
@ -121,21 +121,21 @@ class TestUtilsText(SimpleTestCase):
|
||||||
'<p>The quick <a href="xyz.html"\n id="mylink">brown fox</a> jumped over the lazy dog.</p>'
|
'<p>The quick <a href="xyz.html"\n id="mylink">brown fox</a> jumped over the lazy dog.</p>'
|
||||||
)
|
)
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
'<p>The quick <a href="xyz.html"\n id="mylink">brown...</a></p>',
|
'<p>The quick <a href="xyz.html"\n id="mylink">brown…</a></p>',
|
||||||
truncator.words(3, '...', html=True)
|
truncator.words(3, html=True)
|
||||||
)
|
)
|
||||||
|
|
||||||
# Test self-closing tags
|
# Test self-closing tags
|
||||||
truncator = text.Truncator('<br/>The <hr />quick brown fox jumped over the lazy dog.')
|
truncator = text.Truncator('<br/>The <hr />quick brown fox jumped over the lazy dog.')
|
||||||
self.assertEqual('<br/>The <hr />quick brown...', truncator.words(3, '...', html=True))
|
self.assertEqual('<br/>The <hr />quick brown…', truncator.words(3, html=True))
|
||||||
truncator = text.Truncator('<br>The <hr/>quick <em>brown fox</em> jumped over the lazy dog.')
|
truncator = text.Truncator('<br>The <hr/>quick <em>brown fox</em> jumped over the lazy dog.')
|
||||||
self.assertEqual('<br>The <hr/>quick <em>brown...</em>', truncator.words(3, '...', html=True))
|
self.assertEqual('<br>The <hr/>quick <em>brown…</em>', truncator.words(3, html=True))
|
||||||
|
|
||||||
# Test html entities
|
# Test html entities
|
||||||
truncator = text.Truncator('<i>Buenos días! ¿Cómo está?</i>')
|
truncator = text.Truncator('<i>Buenos días! ¿Cómo está?</i>')
|
||||||
self.assertEqual('<i>Buenos días! ¿Cómo...</i>', truncator.words(3, '...', html=True))
|
self.assertEqual('<i>Buenos días! ¿Cómo…</i>', truncator.words(3, html=True))
|
||||||
truncator = text.Truncator('<p>I <3 python, what about you?</p>')
|
truncator = text.Truncator('<p>I <3 python, what about you?</p>')
|
||||||
self.assertEqual('<p>I <3 python...</p>', truncator.words(3, '...', html=True))
|
self.assertEqual('<p>I <3 python…</p>', truncator.words(3, html=True))
|
||||||
|
|
||||||
re_tag_catastrophic_test = ('</a' + '\t' * 50000) + '//>'
|
re_tag_catastrophic_test = ('</a' + '\t' * 50000) + '//>'
|
||||||
truncator = text.Truncator(re_tag_catastrophic_test)
|
truncator = text.Truncator(re_tag_catastrophic_test)
|
||||||
|
|
Loading…
Reference in New Issue