#5025 closed New feature (fixed)
Add a "truncate" template filter
Reported by: | Chris Beaven | Owned by: | nobody |
---|---|---|---|
Component: | Template system | Version: | dev |
Severity: | Normal | Keywords: | |
Cc: | semente@…, paulschreiber@…, Gabriel Hurley, mikexstudios, floguy@…, timdumol | Triage Stage: | Ready for checkin |
Has patch: | yes | Needs documentation: | no |
Needs tests: | no | Patch needs improvement: | no |
Easy pickings: | no | UI/UX: | no |
Description
A truncate filter seems to tie in nicely with the truncatewords and truncatewords_html filters (and it's something I needed recently).
The patch adds a truncate
text util which is shared for the new truncate
and the existing urlizetrunc
filter.
Attachments (5)
Change History (49)
by , 17 years ago
Attachment: | truncate.patch added |
---|
comment:1 by , 17 years ago
Summary: | truncate filter → Add a "truncate" template filter |
---|
comment:2 by , 17 years ago
Triage Stage: | Unreviewed → Design decision needed |
---|
comment:3 by , 17 years ago
Resolution: | → fixed |
---|---|
Status: | new → closed |
Triage Stage: | Design decision needed → Accepted |
comment:5 by , 17 years ago
Resolution: | fixed |
---|---|
Status: | closed → reopened |
comment:7 by , 16 years ago
Nope, but we do have http://docs.djangoproject.com/en/dev/ref/templates/builtins/#slice
comment:8 by , 16 years ago
Triage Stage: | Accepted → Design decision needed |
---|
But you're right, it has sat in Accepted for with a patch for long enough - it needs a design decision.
comment:9 by , 16 years ago
I tried slice, but it doesn't do exactly the same thing. The ... in the end give it a quite different meaning.
At the moment I just implemented this filter as a custom filter on my application.
I couldn't find any other built-in feature that would do the same task.
I will be glad to drop the custom filter if this patch gets in though.
Thanks for the reply.
comment:10 by , 15 years ago
There are more differences between slice and truncate: slice doesn't work on things that are not strings, while truncate do. So trying to do {{ article|slice:":20" }} (hoping to limit the unicode_ representation of the article), will not work. {{ article|truncate:20 }} does work like intended.
The patch works fine and fills a real need. I'm voting +1 on adding this.
comment:11 by , 15 years ago
PS: Bringing it up on the django dev list makes more sense than adding a +1 here.
comment:12 by , 15 years ago
Cc: | added |
---|
I made a similar and simple filter, but your patch is better to Django.
For reference: http://www.djangosnippets.org/snippets/1715/.
comment:13 by , 15 years ago
The proposal in #12200 uses a unicode ellipsis and provides for head, tail and middle truncation. It also avoids breaking in the middle of words.
comment:14 by , 15 years ago
Cc: | added |
---|
comment:15 by , 15 years ago
Cc: | added |
---|
comment:16 by , 15 years ago
Cc: | added |
---|
comment:17 by , 15 years ago
One problem with this filter is how to correctly handle unicode, where grapheme != code point. Perhaps some library can do it, I don't think that Python bundles such a library. I don't think this is as much of a problem with truncatewords
because you rarely have a combining character preceding a space (is that ever valid?).
comment:18 by , 14 years ago
Triage Stage: | Design decision needed → Accepted |
---|
Design decision (based on in-person discussion between some maintainers and "some" mailing list dicussion): we will add such a filter. It will be written like utils.text.truncatewords so that the truncate filter can be replaced with a user's own, but with different ellipsis sequence. The default ellipsis sequence will be marked as translatable. It must operate on unicode strings, not treat them as bytes. In fact, it must use Python's unicodedata library to evalute combining characters correctly w.r.t. length.
comment:19 by , 14 years ago
Needs documentation: | set |
---|
marking as needs improvement, as per mtredinnick's comment, above
comment:20 by , 14 years ago
Needs documentation: | unset |
---|---|
Patch needs improvement: | set |
comment:21 by , 14 years ago
Patch needs improvement: | unset |
---|
comment:22 by , 14 years ago
Patch needs improvement: | set |
---|
This implementation does not correctly deal with the situation where you have combining characters that do not have precomposed forms:
>>> print truncate_chars(u"__B\u030A____", 6) __B... >>> print truncate_chars(u"__B\u030A____", 7) __B̊...
(Contrast u"__A\u03A____"
, which is dealt with correctly because there is a precomposed Å character).
This kind of thing does matter - http://gizmodo.com/382026/a-cellphones-missing-dot-kills-two-people-puts-three-more-in-jail
I don't know how easy it is to fix - standing on the sidelines and cheering/booing is more my style today :-)
There are also bugs where length - len_end_text < 0
.
comment:23 by , 14 years ago
Patch needs improvement: | unset |
---|
Thanks for the good catches Luke, I've updated the implementation to handle your two issues, as well as added tests to ensure it doesn't regress.
comment:24 by , 14 years ago
I'm not sure that _(end_text)
makes sense: the default ellipsis should just use ugettext_lazy
instead.
comment:28 by , 14 years ago
Cc: | added |
---|
comment:29 by , 14 years ago
floguy: I understand the suggestion as: "translate "..." directly instead of the variable". Makes sense?
comment:30 by , 14 years ago
Severity: | → Normal |
---|---|
Type: | → New feature |
by , 14 years ago
comment:31 by , 14 years ago
Easy pickings: | unset |
---|
The length of the truncated string in floguy's previous patch wasn't taking into account the length of the string when it had combining characters with no precomposed form.
I'm pretty happy with the patch I uploaded - I wonder though that if instead of having ugettext_lazy('...')
we should use: ugettext_lazy('%(truncated_text)s...')
comment:32 by , 14 years ago
Patch needs improvement: | set |
---|
A few things:
- Use
pgettext_lazy
instead ofugettext_lazy
to give the rather ambiguous string'...'
some context. (e.g.end_text=pgettext_lazy('truncatechars end text', '...')
. This will explain the translators better what this is about. - The naming of
truncate_chars
andtruncatechars
seems weird. How doestruncate_chars
relate totruncate_words
? Can they be combined? Why isn't thetruncate
filter extended in the first place? - Don't use inline imports in the template filter.
comment:33 by , 14 years ago
Ah, and using pgettext_lazy('%(truncated_text)s...')
seems like a good idea, indeed.
follow-up: 35 comment:34 by , 14 years ago
Thanks for the review, Jannis.
- do we still need pgettext if it's
'%(truncated_text)s...'
?
- the naming seems to reflect the current naming of
truncate_words
/truncatewords
, why is that weird?
- i'm not sure what you mean by extending the
truncate
filter - whattruncate
filter?
- just about every template filter in defaultfilters has an inline import - should this patch refactor them all or are we setting a new precident?
comment:35 by , 14 years ago
Replying to SmileyChris:
Thanks for the review, Jannis.
- do we still need pgettext if it's
'%(truncated_text)s...'
?
Yes, pgettext is still useful in the sense that it gives a hint to the translators where it's going to be used.
- the naming seems to reflect the current naming of
truncate_words
/truncatewords
, why is that weird?
Adding two new functions with almost the same name for a feature that is almost exactly implemented in a different tag and function doesn't make it an awful naming to you?
- i'm not sure what you mean by extending the
truncate
filter - whattruncate
filter?
Apologies, I meant the truncatewords
filter. IMO, both serve the same purpose: truncate a string. Couldn't we combine the utility function?
- just about every template filter in defaultfilters has an inline import - should this patch refactor them all or are we setting a new precident?
Nope, fixing that is a matter of another ticket, I assume this was once done because of import dependencies, but there is no reason to keep following that path, as far as I can see it.
by , 14 years ago
Attachment: | 5025.2.diff added |
---|
comment:36 by , 14 years ago
Patch needs improvement: | unset |
---|
New patch uses pgettext and takes on Jannis' suggestion of using a common utility for truncation of strings (I've called the class Truncator
, but Truncatorminator
would probably have been cooler :P).
For now, I've left the behavior of the truncate end-text to be backwards compatible for the truncatewords
and truncatewords_html
filters (that is, hardcoded strings of ' ...'
). There's a slight behaviour change otherwise: we're dropping the space between the last truncated word and the ellipsis.
Also, truncatewords_html
doesn't really make sense to allow for the translated ellipsis anywhere but the end, so it will always be appended to the end, regardless of where it contains '%(truncated_text)s'
comment:37 by , 14 years ago
Triage Stage: | Accepted → Ready for checkin |
---|---|
UI/UX: | unset |
This looks great!
comment:39 by , 13 years ago
Cc: | added |
---|
It seems the contributor forgot to register the new tag. Should I open a new ticket, or reopen this?
by , 13 years ago
Attachment: | 5025.3.diff added |
---|
To be applied on top of 5025.2.diff. Fixes the issues mentioned below.
comment:40 by , 13 years ago
Also, L257 has an extra argument to Truncator.chars
(.chars(value, length)
should just be .chars(length)
) which causes a ValueError
and weird behavior. The patch I just attached fixes both.
comment:41 by , 13 years ago
A closed ticket is unlikely to get any attention. Could you please open a new ticket?
comment:44 by , 13 years ago
Replying to hanson2010:
Will this be checked in for 1.4?
It is, see https://docs.djangoproject.com/en/1.4/ref/templates/builtins/#truncatechars
I'll push to DD since it's a complete patch, but inclusion needs approval.