truncatesentences_html-filter added.
This commit is contained in:
parent
b8e14c6183
commit
b28be5695d
@ -23,7 +23,7 @@ All additional filters/tags will be registered automatically.
|
|||||||
- Regulars
|
- Regulars
|
||||||
- **[filesizeformat](https://docs.djangoproject.com/en/dev/ref/templates/builtins/#filesizeformat)** (human-readable filesize; takes bytes as input)
|
- **[filesizeformat](https://docs.djangoproject.com/en/dev/ref/templates/builtins/#filesizeformat)** (human-readable filesize; takes bytes as input)
|
||||||
- **[slugify](https://docs.djangoproject.com/en/dev/ref/templates/builtins/#slugify)** (creates a slug for a given input)
|
- **[slugify](https://docs.djangoproject.com/en/dev/ref/templates/builtins/#slugify)** (creates a slug for a given input)
|
||||||
- **truncatesentences** (returns the first X sentences [like truncatechars/truncatewords]; please provide X as a parameter)
|
- **truncatesentences** / **truncatesentences_html** (returns the first X sentences [like truncatechars/truncatewords]; please provide X as a parameter)
|
||||||
|
|
||||||
- Markup
|
- Markup
|
||||||
- **markdown** (parses markdown text and outputs HTML; **hint**: use the **safe**-filter to make the output not being escaped)
|
- **markdown** (parses markdown text and outputs HTML; **hint**: use the **safe**-filter to make the output not being escaped)
|
||||||
|
166
filters.go
166
filters.go
@ -1,10 +1,13 @@
|
|||||||
package pongo2addons
|
package pongo2addons
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"bytes"
|
||||||
"errors"
|
"errors"
|
||||||
"time"
|
"fmt"
|
||||||
"regexp"
|
"regexp"
|
||||||
"strings"
|
"strings"
|
||||||
|
"time"
|
||||||
|
"unicode/utf8"
|
||||||
|
|
||||||
"github.com/flosch/pongo2"
|
"github.com/flosch/pongo2"
|
||||||
|
|
||||||
@ -18,6 +21,7 @@ func init() {
|
|||||||
pongo2.RegisterFilter("slugify", filterSlugify)
|
pongo2.RegisterFilter("slugify", filterSlugify)
|
||||||
pongo2.RegisterFilter("filesizeformat", filterFilesizeformat)
|
pongo2.RegisterFilter("filesizeformat", filterFilesizeformat)
|
||||||
pongo2.RegisterFilter("truncatesentences", filterTruncatesentences)
|
pongo2.RegisterFilter("truncatesentences", filterTruncatesentences)
|
||||||
|
pongo2.RegisterFilter("truncatesentences_html", filterTruncatesentencesHtml)
|
||||||
|
|
||||||
// Markup
|
// Markup
|
||||||
pongo2.RegisterFilter("markdown", filterMarkdown)
|
pongo2.RegisterFilter("markdown", filterMarkdown)
|
||||||
@ -54,6 +58,166 @@ func filterTruncatesentences(in *pongo2.Value, param *pongo2.Value) (*pongo2.Val
|
|||||||
return pongo2.AsValue(strings.TrimSpace(strings.Join(sentencens[:min(count, len(sentencens))], ""))), nil
|
return pongo2.AsValue(strings.TrimSpace(strings.Join(sentencens[:min(count, len(sentencens))], ""))), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Taken from pongo2/filters_builtin.go
|
||||||
|
func filterTruncateHtmlHelper(value string, new_output *bytes.Buffer, cond func() bool, fn func(c rune, s int, idx int) int, finalize func()) {
|
||||||
|
vLen := len(value)
|
||||||
|
tag_stack := make([]string, 0)
|
||||||
|
idx := 0
|
||||||
|
|
||||||
|
for idx < vLen && !cond() {
|
||||||
|
c, s := utf8.DecodeRuneInString(value[idx:])
|
||||||
|
if c == utf8.RuneError {
|
||||||
|
idx += s
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if c == '<' {
|
||||||
|
new_output.WriteRune(c)
|
||||||
|
idx += s // consume "<"
|
||||||
|
|
||||||
|
if idx+1 < vLen {
|
||||||
|
if value[idx] == '/' {
|
||||||
|
// Close tag
|
||||||
|
|
||||||
|
new_output.WriteString("/")
|
||||||
|
|
||||||
|
tag := ""
|
||||||
|
idx += 1 // consume "/"
|
||||||
|
|
||||||
|
for idx < vLen {
|
||||||
|
c2, size2 := utf8.DecodeRuneInString(value[idx:])
|
||||||
|
if c2 == utf8.RuneError {
|
||||||
|
idx += size2
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
// End of tag found
|
||||||
|
if c2 == '>' {
|
||||||
|
idx++ // consume ">"
|
||||||
|
break
|
||||||
|
}
|
||||||
|
tag += string(c2)
|
||||||
|
idx += size2
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(tag_stack) > 0 {
|
||||||
|
// Ideally, the close tag is TOP of tag stack
|
||||||
|
// In malformed HTML, it must not be, so iterate through the stack and remove the tag
|
||||||
|
for i := len(tag_stack) - 1; i >= 0; i-- {
|
||||||
|
if tag_stack[i] == tag {
|
||||||
|
// Found the tag
|
||||||
|
tag_stack[i] = tag_stack[len(tag_stack)-1]
|
||||||
|
tag_stack = tag_stack[:len(tag_stack)-1]
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
new_output.WriteString(tag)
|
||||||
|
new_output.WriteString(">")
|
||||||
|
} else {
|
||||||
|
// Open tag
|
||||||
|
|
||||||
|
tag := ""
|
||||||
|
|
||||||
|
params := false
|
||||||
|
for idx < vLen {
|
||||||
|
c2, size2 := utf8.DecodeRuneInString(value[idx:])
|
||||||
|
if c2 == utf8.RuneError {
|
||||||
|
idx += size2
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
new_output.WriteRune(c2)
|
||||||
|
|
||||||
|
// End of tag found
|
||||||
|
if c2 == '>' {
|
||||||
|
idx++ // consume ">"
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
if !params {
|
||||||
|
if c2 == ' ' {
|
||||||
|
params = true
|
||||||
|
} else {
|
||||||
|
tag += string(c2)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
idx += size2
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add tag to stack
|
||||||
|
tag_stack = append(tag_stack, tag)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
idx = fn(c, s, idx)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
finalize()
|
||||||
|
|
||||||
|
for i := len(tag_stack) - 1; i >= 0; i-- {
|
||||||
|
tag := tag_stack[i]
|
||||||
|
// Close everything from the regular tag stack
|
||||||
|
new_output.WriteString(fmt.Sprintf("</%s>", tag))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func filterTruncatesentencesHtml(in *pongo2.Value, param *pongo2.Value) (*pongo2.Value, error) {
|
||||||
|
count := param.Integer()
|
||||||
|
if count <= 0 {
|
||||||
|
return pongo2.AsValue(""), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
value := in.String()
|
||||||
|
newLen := max(param.Integer(), 0)
|
||||||
|
|
||||||
|
new_output := bytes.NewBuffer(nil)
|
||||||
|
|
||||||
|
sentencefilter := 0
|
||||||
|
|
||||||
|
filterTruncateHtmlHelper(value, new_output, func() bool {
|
||||||
|
return sentencefilter >= newLen
|
||||||
|
}, func(_ rune, _ int, idx int) int {
|
||||||
|
// Get next word
|
||||||
|
word_found := false
|
||||||
|
|
||||||
|
for idx < len(value) {
|
||||||
|
c2, size2 := utf8.DecodeRuneInString(value[idx:])
|
||||||
|
if c2 == utf8.RuneError {
|
||||||
|
idx += size2
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if c2 == '<' {
|
||||||
|
// HTML tag start, don't consume it
|
||||||
|
return idx
|
||||||
|
}
|
||||||
|
|
||||||
|
new_output.WriteRune(c2)
|
||||||
|
idx += size2
|
||||||
|
|
||||||
|
if (c2 == '.' && !(idx+1 < len(value) && value[idx+1] >= '0' && value[idx+1] <= '9')) ||
|
||||||
|
c2 == '!' || c2 == '?' || c2 == '\n' {
|
||||||
|
// Sentence ends here, stop capturing it now
|
||||||
|
break
|
||||||
|
} else {
|
||||||
|
word_found = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if word_found {
|
||||||
|
sentencefilter++
|
||||||
|
}
|
||||||
|
|
||||||
|
return idx
|
||||||
|
}, func() {})
|
||||||
|
|
||||||
|
return pongo2.AsValue(new_output.String()), nil
|
||||||
|
}
|
||||||
|
|
||||||
func filterTimeuntilTimesince(in *pongo2.Value, param *pongo2.Value) (*pongo2.Value, error) {
|
func filterTimeuntilTimesince(in *pongo2.Value, param *pongo2.Value) (*pongo2.Value, error) {
|
||||||
basetime, is_time := in.Interface().(time.Time)
|
basetime, is_time := in.Interface().(time.Time)
|
||||||
if !is_time {
|
if !is_time {
|
||||||
|
@ -18,9 +18,6 @@ type TestSuite1 struct{}
|
|||||||
|
|
||||||
var _ = Suite(&TestSuite1{})
|
var _ = Suite(&TestSuite1{})
|
||||||
|
|
||||||
// Taken from http://www.florian-schlachter.de/post/pongo2-10-rc1/
|
|
||||||
const demoText = `This is a first sentencen with a 4.50 number. The second one is even more fun! Isn't it? Last sentence, okay.`
|
|
||||||
|
|
||||||
func (s *TestSuite1) TestFilters(c *C) {
|
func (s *TestSuite1) TestFilters(c *C) {
|
||||||
// Markdown
|
// Markdown
|
||||||
c.Assert(pongo2.RenderTemplateString("{{ \"**test**\"|markdown|safe }}", nil), Equals, "<p><strong>test</strong></p>\n")
|
c.Assert(pongo2.RenderTemplateString("{{ \"**test**\"|markdown|safe }}", nil), Equals, "<p><strong>test</strong></p>\n")
|
||||||
@ -75,6 +72,15 @@ func (s *TestSuite1) TestFilters(c *C) {
|
|||||||
Equals, "1st 2nd 3rd 18241st")
|
Equals, "1st 2nd 3rd 18241st")
|
||||||
|
|
||||||
// Truncatesentences
|
// Truncatesentences
|
||||||
c.Assert(pongo2.RenderTemplateString("{{ text|truncatesentences:3|safe }}", pongo2.Context{"text": demoText}),
|
c.Assert(pongo2.RenderTemplateString("{{ text|truncatesentences:3|safe }}", pongo2.Context{
|
||||||
Equals, "This is a first sentencen with a 4.50 number. The second one is even more fun! Isn't it?")
|
"text": `This is a first sentence with a 4.50 number. The second one is even more fun! Isn't it? Last sentence, okay.`}),
|
||||||
|
Equals, "This is a first sentence with a 4.50 number. The second one is even more fun! Isn't it?")
|
||||||
|
|
||||||
|
// Truncatesentences_html
|
||||||
|
c.Assert(pongo2.RenderTemplateString("{{ text|truncatesentences_html:2|safe }}", pongo2.Context{
|
||||||
|
"text": `<div class="test"><ul><li>This is a first sentence with a 4.50 number.</li><li>The second one is even more fun! Isn't it?</li><li>Last sentence, okay.</li></ul></div>`}),
|
||||||
|
Equals, `<div class="test"><ul><li>This is a first sentence with a 4.50 number.</li><li>The second one is even more fun!</li></ul></div>`)
|
||||||
|
c.Assert(pongo2.RenderTemplateString("{{ text|truncatesentences_html:3|safe }}", pongo2.Context{
|
||||||
|
"text": `<div class="test"><ul><li>This is a first sentence with a 4.50 number.</li><li>The second one is even more fun! Isn't it?</li><li>Last sentence, okay.</li></ul></div>`}),
|
||||||
|
Equals, `<div class="test"><ul><li>This is a first sentence with a 4.50 number.</li><li>The second one is even more fun! Isn't it?</li></ul></div>`)
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user