Added SpecPartialsTests and fixed issues

This commit is contained in:
Adam Fowler
2021-03-17 18:30:55 +00:00
parent b4d6a518c7
commit 169a7bbbf4
6 changed files with 164 additions and 14 deletions

View File

@@ -32,10 +32,12 @@ extension HBMustacheTemplate {
continue continue
} else if parser.current() == "{" { } else if parser.current() == "{" {
parser.unsafeAdvance() parser.unsafeAdvance()
// if next character is not "{" then is normal text
if parser.current() != "{" { if parser.current() != "{" {
if text.count > 0 { if text.count > 0 {
tokens.append(.text(whiteSpaceBefore + text.string + "{")) tokens.append(.text(whiteSpaceBefore + text.string + "{"))
whiteSpaceBefore = "" whiteSpaceBefore = ""
newLine = false
} }
continue continue
} else { } else {
@@ -43,26 +45,30 @@ extension HBMustacheTemplate {
} }
} }
// whatever text we found before the "{{" should be added
if text.count > 0 { if text.count > 0 {
tokens.append(.text(whiteSpaceBefore + text.string)) tokens.append(.text(whiteSpaceBefore + text.string))
whiteSpaceBefore = "" whiteSpaceBefore = ""
newLine = false newLine = false
} }
// have we reached the end of the text
if parser.reachedEnd() { if parser.reachedEnd() {
break break
} }
var setNewLine = false
switch parser.current() { switch parser.current() {
case "#": case "#":
// section // section
parser.unsafeAdvance() parser.unsafeAdvance()
let (name, method) = try parseName(&parser) let (name, method) = try parseName(&parser)
if newLine && hasLineFinished(&parser) { if newLine && hasLineFinished(&parser) {
newLine = true setNewLine = true
if parser.current() == "\n" { if parser.current() == "\n" {
parser.unsafeAdvance() parser.unsafeAdvance()
} }
} else if whiteSpaceBefore.count > 0 { } else if whiteSpaceBefore.count > 0 {
tokens.append(.text(whiteSpaceBefore)) tokens.append(.text(whiteSpaceBefore))
whiteSpaceBefore = ""
} }
let sectionTokens = try parse(&parser, sectionName: name, newLine: newLine) let sectionTokens = try parse(&parser, sectionName: name, newLine: newLine)
tokens.append(.section(name: name, method: method, template: HBMustacheTemplate(sectionTokens))) tokens.append(.section(name: name, method: method, template: HBMustacheTemplate(sectionTokens)))
@@ -72,12 +78,13 @@ extension HBMustacheTemplate {
parser.unsafeAdvance() parser.unsafeAdvance()
let (name, method) = try parseName(&parser) let (name, method) = try parseName(&parser)
if newLine && hasLineFinished(&parser) { if newLine && hasLineFinished(&parser) {
newLine = true setNewLine = true
if parser.current() == "\n" { if parser.current() == "\n" {
parser.unsafeAdvance() parser.unsafeAdvance()
} }
} else if whiteSpaceBefore.count > 0 { } else if whiteSpaceBefore.count > 0 {
tokens.append(.text(whiteSpaceBefore)) tokens.append(.text(whiteSpaceBefore))
whiteSpaceBefore = ""
} }
let sectionTokens = try parse(&parser, sectionName: name, newLine: newLine) let sectionTokens = try parse(&parser, sectionName: name, newLine: newLine)
tokens.append(.invertedSection(name: name, method: method, template: HBMustacheTemplate(sectionTokens))) tokens.append(.invertedSection(name: name, method: method, template: HBMustacheTemplate(sectionTokens)))
@@ -90,12 +97,13 @@ extension HBMustacheTemplate {
throw Error.sectionCloseNameIncorrect throw Error.sectionCloseNameIncorrect
} }
if newLine && hasLineFinished(&parser) { if newLine && hasLineFinished(&parser) {
newLine = true setNewLine = true
if parser.current() == "\n" { if parser.current() == "\n" {
parser.unsafeAdvance() parser.unsafeAdvance()
} }
} else if whiteSpaceBefore.count > 0 { } else if whiteSpaceBefore.count > 0 {
tokens.append(.text(whiteSpaceBefore)) tokens.append(.text(whiteSpaceBefore))
whiteSpaceBefore = ""
} }
return tokens return tokens
@@ -104,7 +112,7 @@ extension HBMustacheTemplate {
parser.unsafeAdvance() parser.unsafeAdvance()
_ = try parseComment(&parser) _ = try parseComment(&parser)
if newLine && hasLineFinished(&parser) { if newLine && hasLineFinished(&parser) {
newLine = true setNewLine = true
if !parser.reachedEnd() { if !parser.reachedEnd() {
parser.unsafeAdvance() parser.unsafeAdvance()
} }
@@ -114,6 +122,7 @@ extension HBMustacheTemplate {
// unescaped variable // unescaped variable
if whiteSpaceBefore.count > 0 { if whiteSpaceBefore.count > 0 {
tokens.append(.text(whiteSpaceBefore)) tokens.append(.text(whiteSpaceBefore))
whiteSpaceBefore = ""
} }
parser.unsafeAdvance() parser.unsafeAdvance()
let (name, method) = try parseName(&parser) let (name, method) = try parseName(&parser)
@@ -124,6 +133,7 @@ extension HBMustacheTemplate {
// unescaped variable // unescaped variable
if whiteSpaceBefore.count > 0 { if whiteSpaceBefore.count > 0 {
tokens.append(.text(whiteSpaceBefore)) tokens.append(.text(whiteSpaceBefore))
whiteSpaceBefore = ""
} }
parser.unsafeAdvance() parser.unsafeAdvance()
let (name, method) = try parseName(&parser) let (name, method) = try parseName(&parser)
@@ -131,21 +141,38 @@ extension HBMustacheTemplate {
case ">": case ">":
// partial // partial
parser.unsafeAdvance()
let (name, _) = try parseName(&parser)
/*if newLine && hasLineFinished(&parser) {
setNewLine = true
if parser.current() == "\n" {
parser.unsafeAdvance()
}
}*/
if whiteSpaceBefore.count > 0 { if whiteSpaceBefore.count > 0 {
tokens.append(.text(whiteSpaceBefore)) tokens.append(.text(whiteSpaceBefore))
} }
parser.unsafeAdvance() if newLine && hasLineFinished(&parser) {
let (name, _) = try parseName(&parser) setNewLine = true
tokens.append(.partial(name)) if parser.current() == "\n" {
parser.unsafeAdvance()
}
tokens.append(.partial(name, indentation: whiteSpaceBefore))
} else {
tokens.append(.partial(name, indentation: nil))
}
whiteSpaceBefore = ""
default: default:
// variable // variable
if whiteSpaceBefore.count > 0 { if whiteSpaceBefore.count > 0 {
tokens.append(.text(whiteSpaceBefore)) tokens.append(.text(whiteSpaceBefore))
whiteSpaceBefore = ""
} }
let (name, method) = try parseName(&parser) let (name, method) = try parseName(&parser)
tokens.append(.variable(name: name, method: method)) tokens.append(.variable(name: name, method: method))
} }
newLine = setNewLine
} }
// should never get here if reading section // should never get here if reading section
guard sectionName == nil else { guard sectionName == nil else {

View File

@@ -5,9 +5,12 @@ extension HBMustacheTemplate {
/// - object: Object /// - object: Object
/// - context: Context that render is occurring in. Contains information about position in sequence /// - context: Context that render is occurring in. Contains information about position in sequence
/// - Returns: Rendered text /// - Returns: Rendered text
func render(_ object: Any, context: HBMustacheContext? = nil) -> String { func render(_ object: Any, context: HBMustacheContext? = nil, indentation: String? = nil) -> String {
var string = "" var string = ""
for token in tokens { for token in tokens {
if let indentation = indentation, string.last == "\n" {
string += indentation
}
switch token { switch token {
case let .text(text): case let .text(text):
string += text string += text
@@ -31,9 +34,9 @@ extension HBMustacheTemplate {
let child = getChild(named: variable, from: object, method: method, context: context) let child = getChild(named: variable, from: object, method: method, context: context)
string += renderInvertedSection(child, parent: object, with: template) string += renderInvertedSection(child, parent: object, with: template)
case let .partial(name): case let .partial(name, indentation):
if let text = library?.render(object, withTemplate: name) { if let template = library?.getTemplate(named: name) {
string += text string += template.render(object, indentation: indentation)
} }
} }
} }

View File

@@ -36,7 +36,7 @@ public final class HBMustacheTemplate {
case unescapedVariable(name: String, method: String? = nil) case unescapedVariable(name: String, method: String? = nil)
case section(name: String, method: String? = nil, template: HBMustacheTemplate) case section(name: String, method: String? = nil, template: HBMustacheTemplate)
case invertedSection(name: String, method: String? = nil, template: HBMustacheTemplate) case invertedSection(name: String, method: String? = nil, template: HBMustacheTemplate)
case partial(String) case partial(String, indentation: String?)
} }
let tokens: [Token] let tokens: [Token]

View File

@@ -13,6 +13,7 @@ final class PartialTests: XCTestCase {
""") """)
let template2 = try HBMustacheTemplate(string: """ let template2 = try HBMustacheTemplate(string: """
<strong>{{.}}</strong> <strong>{{.}}</strong>
""") """)
library.register(template, named: "base") library.register(template, named: "base")
library.register(template2, named: "user") library.register(template2, named: "user")

View File

@@ -525,3 +525,122 @@ final class SpecInvertedTests: XCTestCase {
try test(object, template, expected) try test(object, template, expected)
} }
} }
// MARK: Partials
final class SpecPartialsTests: XCTestCase {
func testPartial(_ object: Any, _ template: String, _ partials: [String: String], _ expected: String) throws {
let library = HBMustacheLibrary()
let template = try HBMustacheTemplate(string: template)
library.register(template, named: "template")
for (key, value) in partials {
let template = try HBMustacheTemplate(string: value)
library.register(template, named: key)
}
let result = library.render(object, withTemplate: "template")
XCTAssertEqual(result, expected)
}
func testBasic() throws {
let object = {}
let template = #""{{>text}}""#
let partial = "from partial"
let expected = #""from partial""#
try testPartial(object, template, ["text": partial], expected)
}
func testFailedLookup() throws {
let object = {}
let template = #""{{>text}}""#
let expected = "\"\""
try testPartial(object, template, [:], expected)
}
func testContext() throws {
let object = ["text": "content"]
let template = #""{{>partial}}""#
let partial = "*{{text}}*"
let expected = #""*content*""#
try testPartial(object, template, ["partial": partial], expected)
}
func testRecursion() throws {
let object: [String: Any] = ["content": "X", "nodes": [["content": "Y", "nodes": []]]]
let template = #"{{>node}}"#
let partial = "{{content}}<{{#nodes}}{{>node}}{{/nodes}}>"
let expected = #"X<Y<>>"#
try testPartial(object, template, ["node": partial], expected)
}
func testSurroundingWhitespace() throws {
let object = {}
let template = "| {{>partial}} |"
let partial = "\t|\t"
let expected = "| \t|\t |"
try testPartial(object, template, ["partial": partial], expected)
}
func testInlineIdention() throws {
let object = ["data": "|"]
let template = " {{data}} {{> partial}}\n"
let partial = ">\n>"
let expected = " | >\n>\n"
try testPartial(object, template, ["partial": partial], expected)
}
func testStandaloneLineEndings() throws {
let object = {}
let template = "|\r\n{{>partial}}\r\n|"
let partial = ">"
let expected = "|\r\n>|"
try testPartial(object, template, ["partial": partial], expected)
}
func testStandaloneWithoutPreviousLine() throws {
let object = {}
let template = " {{>partial}}\n>"
let partial = ">\n>"
let expected = " >\n >>"
try testPartial(object, template, ["partial": partial], expected)
}
func testStandaloneWithoutNewLine() throws {
let object = {}
let template = ">\n {{>partial}}"
let partial = ">\n>"
let expected = ">\n >\n >"
try testPartial(object, template, ["partial": partial], expected)
}
func testStandaloneIndentation() throws {
let object = ["content": "<\n->"]
let template = """
\
{{>partial}}
/
"""
let partial = """
|
{{{content}}}
|
"""
let expected = """
\
|
<
->
|
/
"""
try testPartial(object, template, ["partial": partial], expected)
}
func testPaddingWhitespace() throws {
let object = ["boolean": true ]
let template = "|{{> partial }}|"
let partial = "[]"
let expected = "|[]|"
try testPartial(object, template, ["partial": partial], expected)
}
}

View File

@@ -83,8 +83,8 @@ extension HBMustacheTemplate.Token: Equatable {
return lhs1 == rhs1 && lhs2 == rhs2 && lhs3 == rhs3 return lhs1 == rhs1 && lhs2 == rhs2 && lhs3 == rhs3
case let (.invertedSection(lhs1, lhs2, lhs3), .invertedSection(rhs1, rhs2, rhs3)): case let (.invertedSection(lhs1, lhs2, lhs3), .invertedSection(rhs1, rhs2, rhs3)):
return lhs1 == rhs1 && lhs2 == rhs2 && lhs3 == rhs3 return lhs1 == rhs1 && lhs2 == rhs2 && lhs3 == rhs3
case let (.partial(name1), .partial(name2)): case let (.partial(name1, indent1), .partial(name2, indent2)):
return name1 == name2 return name1 == name2 && indent1 == indent2
default: default:
return false return false
} }