Added SpecPartialsTests and fixed issues
This commit is contained in:
@@ -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))
|
||||||
}
|
}
|
||||||
|
if newLine && hasLineFinished(&parser) {
|
||||||
|
setNewLine = true
|
||||||
|
if parser.current() == "\n" {
|
||||||
parser.unsafeAdvance()
|
parser.unsafeAdvance()
|
||||||
let (name, _) = try parseName(&parser)
|
}
|
||||||
tokens.append(.partial(name))
|
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 {
|
||||||
|
|||||||
@@ -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)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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]
|
||||||
|
|||||||
@@ -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")
|
||||||
|
|||||||
@@ -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)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -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
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user