From 8a669e166486c450572383cce8e9c611b6dcac50 Mon Sep 17 00:00:00 2001 From: Odin H B Date: Fri, 8 Mar 2024 11:22:00 +0100 Subject: [PATCH] Render literals like vanilla js template strings fixes https://github.com/developit/vhtml/issues/40 and https://github.com/developit/vhtml/issues/37 inspired by https://github.com/developit/vhtml/pull/41 The only place we don't have parity right now is arrays (because that's how children are passed) and bigint (because I'm not gonna do dependency wrangling rn) --- src/stringjsx.js | 10 +++-- test/stringjsx.js | 105 ++++++++++++++++++++++++++++++++++++++++++++++ 2 files changed, 112 insertions(+), 3 deletions(-) diff --git a/src/stringjsx.js b/src/stringjsx.js index 7ae48ed..b309671 100644 --- a/src/stringjsx.js +++ b/src/stringjsx.js @@ -87,8 +87,9 @@ function render(tagName, attributes) { const children = extractChildren(arguments); let result = ''; + const isFragment = tagName !== null && tagName !== undefined; - if (tagName) { // null is passed when rendering a fragment + if (isFragment) { result += '<'; result += tagName; @@ -104,7 +105,10 @@ function render(tagName, attributes) { } else while(children.length) { const child = children.shift(); - if (!child) continue; + if (child === undefined || child === null) { + result += child; + continue; + } if (isCollection(child)) { for (let i = child.length - 1; i >= 0; i--) { @@ -116,7 +120,7 @@ function render(tagName, attributes) { } } - if (tagName) result += ``; + if (isFragment) result += ``; } // Read about the fun world of javascript strings diff --git a/test/stringjsx.js b/test/stringjsx.js index 33c5f44..3643089 100644 --- a/test/stringjsx.js +++ b/test/stringjsx.js @@ -269,6 +269,14 @@ describe('stringjsx', () => { ); }); + it('also supports fragments w/ undefined', () => { + expect( + h(h.Fragment, null,

foo

, bar,
baz
).toString() + ).to.equal( + '

foo

bar
baz
' + ); + }); + // regression test for https://github.com/developit/vhtml/issues/34 it('does not allow cache-based html injection anymore', () => { const injectable = '

test

'; @@ -302,4 +310,101 @@ describe('stringjsx', () => { '' ) }); + + describe('literal rendering', () => { + it('renders the empty string', () => { + expect( +
{''}
.toString() + ).to.equal('
') + }); + + it('renders blank strings', () => { + expect( +
{' '} and also {'\n\n\n \t\t\t'}
.toString() + ).to.equal('
and also \n\n\n \t\t\t
') + }); + + it('renders a goofy string', () => { + expect( +
{'haha'}
.toString() + ).to.equal('
haha
') + }); + + it('renders numbers', () => { + expect( +
+ {63452} + num: {12385} +

{-882}, {942}

+
.toString() + ).to.equal('
63452num: 12385

-882, 942

') + }); + + it('renders infinities', () => { + // impressive + expect( +
+ {Infinity}+{-Infinity} +
.toString() + ).to.equal('
Infinity+-Infinity
') + }); + + it('renders a "very big" number', () => { + expect( +
{5463454363452342352665745632523423423}
.toString() + ).to.equal('
5.463454363452342e+36
') + }); + + // TODO: hmm maybe isn't supported by any node tooling yet? + // it('renders a bigint', () => { + // expect( + //
{5463454363452342352665745632523423423n}
.toString() + // ).to.equal('
5463454363452342352665745632523423423
') + // }); + + // regression test for https://github.com/developit/vhtml/issues/40 + it('renders zero', () => { + expect( +
{0} elephants carrying {0} trees
.toString() + ).to.equal('
0 elephants carrying 0 trees
') + }); + + it('renders booleans', () => { + expect( +
+

{true} love

+ {false} promises +
.toString() + ).to.equal('

true love

false promises
') + }); + + it('renders undefined', () => { + expect(
{undefined} behaviour
.toString()) + .to.equal('
undefined behaviour
') + }); + + it('renders null', () => { + expect(
{null} and void
.toString()) + .to.equal('
null and void
') + }); + + it('"renders" an object', () => { + expect(
object? {{}} object
.toString()) + .to.equal('
object? [object Object] object
') + }); + + it('renders an array', () => { + expect(
little bit of {['a', 'b']}
.toString()) + .to.equal('
little bit of ab
') + }); + + it('renders toString', () => { + const instrument = { + toString: function() { return 'ukulele' }, + } + + expect(
instrument: {instrument}
.toString()) + .to.equal('
instrument: ukulele
') + }); + }); });