halfcab 14.0.5 → 15.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/eventEmitter/test.js +1 -1
- package/halfcab.mjs +44 -113
- package/package.json +3 -4
- package/test.js +7 -9
- package/test_ssr_switch.mjs +33 -0
package/eventEmitter/test.js
CHANGED
|
@@ -2,7 +2,7 @@ import chai from 'chai'
|
|
|
2
2
|
import dirtyChai from 'dirty-chai'
|
|
3
3
|
import sinon from 'sinon'
|
|
4
4
|
import sinonChai from 'sinon-chai'
|
|
5
|
-
import eventEmitter from './index
|
|
5
|
+
import eventEmitter from './index'
|
|
6
6
|
import jsdomGlobal from 'jsdom-global'
|
|
7
7
|
|
|
8
8
|
const {expect} = chai
|
package/halfcab.mjs
CHANGED
|
@@ -2,11 +2,8 @@ import shiftyRouterModule from 'shifty-router'
|
|
|
2
2
|
import hrefModule from 'shifty-router/href.js'
|
|
3
3
|
import historyModule from 'shifty-router/history.js'
|
|
4
4
|
import createLocation from 'shifty-router/create-location.js'
|
|
5
|
-
import
|
|
6
|
-
import
|
|
7
|
-
import { repeat } from 'lit-html/directives/repeat.js'
|
|
8
|
-
import { classMap } from 'lit-html/directives/class-map.js'
|
|
9
|
-
import { styleMap } from 'lit-html/directives/style-map.js'
|
|
5
|
+
import bel from 'nanohtml'
|
|
6
|
+
import update from 'nanomorph'
|
|
10
7
|
import axios from 'axios'
|
|
11
8
|
import cssInject from 'csjs-inject'
|
|
12
9
|
import merge from 'deepmerge'
|
|
@@ -14,9 +11,12 @@ import marked from 'marked'
|
|
|
14
11
|
import { decode } from 'html-entities'
|
|
15
12
|
import eventEmitter from './eventEmitter/index.mjs'
|
|
16
13
|
import qs from 'qs'
|
|
14
|
+
import LRU from 'nanolru'
|
|
15
|
+
import Component from 'nanocomponent'
|
|
17
16
|
import * as deepDiff from 'deep-object-diff'
|
|
18
17
|
import clone from 'fast-clone'
|
|
19
|
-
|
|
18
|
+
|
|
19
|
+
const cache = LRU(5000)
|
|
20
20
|
|
|
21
21
|
let cssTag = cssInject
|
|
22
22
|
let componentCSSString = ''
|
|
@@ -28,7 +28,6 @@ let rootEl
|
|
|
28
28
|
let components
|
|
29
29
|
let dataInitial
|
|
30
30
|
let el
|
|
31
|
-
let componentIndex = 0
|
|
32
31
|
|
|
33
32
|
marked.setOptions({
|
|
34
33
|
breaks: true
|
|
@@ -70,96 +69,19 @@ if (typeof window !== 'undefined') {
|
|
|
70
69
|
let geb = new eventEmitter({state})
|
|
71
70
|
|
|
72
71
|
let html = (strings, ...values) => {
|
|
73
|
-
// fix for allowing csjs to coexist with
|
|
72
|
+
// fix for allowing csjs to coexist with nanohtml SSR
|
|
74
73
|
values = values.map(value => {
|
|
75
|
-
|
|
76
|
-
// TemplateResult usually has 'strings' and 'values' or '_$litType$'
|
|
77
|
-
// DirectiveResult (unsafeHTML) uses default toString, so we shouldn't call it.
|
|
78
|
-
if (value && typeof value !== 'function' && !Array.isArray(value) && typeof value.toString === 'function' && value.toString !== Object.prototype.toString && !value.strings && !value._$litType$) {
|
|
74
|
+
if (value && value.hasOwnProperty('toString')) {
|
|
79
75
|
return value.toString()
|
|
80
76
|
}
|
|
81
77
|
return value
|
|
82
78
|
})
|
|
83
79
|
|
|
84
|
-
return
|
|
85
|
-
}
|
|
86
|
-
|
|
87
|
-
// Capture directive classes for SSR identification
|
|
88
|
-
const RepeatDirective = repeat([], () => {})['_$litDirective$']
|
|
89
|
-
const ClassMapDirective = classMap({})['_$litDirective$']
|
|
90
|
-
const StyleMapDirective = styleMap({})['_$litDirective$']
|
|
91
|
-
|
|
92
|
-
function resolveTemplate (value) {
|
|
93
|
-
if (value === nothing || value === noChange) {
|
|
94
|
-
return ''
|
|
95
|
-
}
|
|
96
|
-
|
|
97
|
-
if (Array.isArray(value)) {
|
|
98
|
-
return value.map(resolveTemplate).join('')
|
|
99
|
-
}
|
|
100
|
-
if (value && typeof value === 'object') {
|
|
101
|
-
if (value.strings) {
|
|
102
|
-
let result = ''
|
|
103
|
-
const { strings, values } = value
|
|
104
|
-
for (let i = 0; i < strings.length; i++) {
|
|
105
|
-
result += strings[i]
|
|
106
|
-
if (i < values.length) {
|
|
107
|
-
result += resolveTemplate(values[i])
|
|
108
|
-
}
|
|
109
|
-
}
|
|
110
|
-
return result
|
|
111
|
-
}
|
|
112
|
-
|
|
113
|
-
if (value['_$litDirective$']) {
|
|
114
|
-
const directiveClass = value['_$litDirective$']
|
|
115
|
-
if (directiveClass.directiveName === 'unsafeHTML' && value.values && value.values.length > 0) {
|
|
116
|
-
return String(value.values[0])
|
|
117
|
-
}
|
|
118
|
-
|
|
119
|
-
if (directiveClass === RepeatDirective) {
|
|
120
|
-
const items = value.values[0]
|
|
121
|
-
const templateFn = value.values[value.values.length - 1]
|
|
122
|
-
if (items && typeof templateFn === 'function') {
|
|
123
|
-
return Array.from(items).map((item, index) => resolveTemplate(templateFn(item, index))).join('')
|
|
124
|
-
}
|
|
125
|
-
return ''
|
|
126
|
-
}
|
|
127
|
-
|
|
128
|
-
if (directiveClass === ClassMapDirective) {
|
|
129
|
-
const classObj = value.values[0]
|
|
130
|
-
if (typeof classObj === 'object') {
|
|
131
|
-
return Object.keys(classObj)
|
|
132
|
-
.filter(key => classObj[key])
|
|
133
|
-
.join(' ')
|
|
134
|
-
}
|
|
135
|
-
return ''
|
|
136
|
-
}
|
|
137
|
-
|
|
138
|
-
if (directiveClass === StyleMapDirective) {
|
|
139
|
-
const styleObj = value.values[0]
|
|
140
|
-
if (typeof styleObj === 'object') {
|
|
141
|
-
return Object.keys(styleObj)
|
|
142
|
-
.map(key => `${key}:${styleObj[key]}`)
|
|
143
|
-
.join(';')
|
|
144
|
-
}
|
|
145
|
-
return ''
|
|
146
|
-
}
|
|
147
|
-
}
|
|
148
|
-
}
|
|
149
|
-
|
|
150
|
-
if (typeof value === 'function') {
|
|
151
|
-
return ''
|
|
152
|
-
}
|
|
153
|
-
|
|
154
|
-
return value === undefined || value === null ? '' : String(value)
|
|
80
|
+
return bel(strings, ...values)
|
|
155
81
|
}
|
|
156
82
|
|
|
157
83
|
function ssr (rootComponent) {
|
|
158
|
-
|
|
159
|
-
let componentsString = ''
|
|
160
|
-
try {
|
|
161
|
-
componentsString = resolveTemplate(rootComponent)
|
|
162
|
-
} catch (e) {}
|
|
84
|
+
let componentsString = `${rootComponent}`
|
|
163
85
|
return {componentsString, stylesString: componentCSSString}
|
|
164
86
|
}
|
|
165
87
|
|
|
@@ -320,12 +242,11 @@ function nextTick (func) {
|
|
|
320
242
|
|
|
321
243
|
function stateUpdated () {
|
|
322
244
|
if (rootEl) {
|
|
323
|
-
componentIndex = 0
|
|
324
245
|
let startTime = Date.now()
|
|
325
246
|
let newEl = components(state)
|
|
326
247
|
console.log(`Component render: ${Date.now() - startTime}`)
|
|
327
248
|
startTime = Date.now()
|
|
328
|
-
|
|
249
|
+
update(rootEl, newEl)
|
|
329
250
|
console.log(`DOM morph: ${Date.now() - startTime}`)
|
|
330
251
|
}
|
|
331
252
|
}
|
|
@@ -367,7 +288,6 @@ function updateState (updateObject, options) {
|
|
|
367
288
|
function emptySSRVideos (c) {
|
|
368
289
|
//SSR videos with source tags don't like morphing and you get double audio,
|
|
369
290
|
// so remove src from the new one so it never starts
|
|
370
|
-
if (!c || !c.querySelectorAll) return
|
|
371
291
|
let autoplayTrue = c.querySelectorAll('video[autoplay="true"]')
|
|
372
292
|
let autoplayAutoplay = c.querySelectorAll('video[autoplay="autoplay"]')
|
|
373
293
|
let autoplayOn = c.querySelectorAll('video[autoplay="on"]')
|
|
@@ -384,13 +304,13 @@ function emptySSRVideos (c) {
|
|
|
384
304
|
|
|
385
305
|
function injectHTML (htmlString, options) {
|
|
386
306
|
if (options && options.wrapper === false) {
|
|
387
|
-
return
|
|
307
|
+
return html([htmlString])
|
|
388
308
|
}
|
|
389
|
-
return html`<div>${
|
|
309
|
+
return html([`<div>${htmlString}</div>`]) // using html as a regular function instead of a tag function, and prevent double encoding of ampersands while we're at it
|
|
390
310
|
}
|
|
391
311
|
|
|
392
312
|
function injectMarkdown (mdString, options) {
|
|
393
|
-
return injectHTML(decode(marked(mdString)), options)
|
|
313
|
+
return injectHTML(decode(marked(mdString)), options) //using html as a regular function instead of a tag function, and prevent double encoding of ampersands while we're at it
|
|
394
314
|
}
|
|
395
315
|
|
|
396
316
|
function gotoRoute (route) {
|
|
@@ -502,24 +422,16 @@ export default (config, {shiftyRouter = shiftyRouterModule, href = hrefModule, h
|
|
|
502
422
|
gotoRoute(location.href)
|
|
503
423
|
})
|
|
504
424
|
|
|
505
|
-
componentIndex = 0
|
|
506
425
|
let c = components(state)//root element generated by components
|
|
507
426
|
if (el) {
|
|
508
427
|
|
|
509
|
-
|
|
428
|
+
emptySSRVideos(c)
|
|
510
429
|
|
|
511
430
|
let r = document.querySelector(el)
|
|
512
|
-
|
|
513
|
-
// Fallback if element not found
|
|
514
|
-
rootEl = document.createElement('div')
|
|
515
|
-
} else {
|
|
516
|
-
rootEl = r
|
|
517
|
-
}
|
|
518
|
-
render(c, rootEl)
|
|
431
|
+
rootEl = update(r, c)
|
|
519
432
|
return resolve({rootEl, state})
|
|
520
433
|
}
|
|
521
|
-
rootEl =
|
|
522
|
-
render(c, rootEl)
|
|
434
|
+
rootEl = c
|
|
523
435
|
resolve({rootEl, state})//if no root element provided, just return the root
|
|
524
436
|
// component and the state
|
|
525
437
|
})
|
|
@@ -529,20 +441,39 @@ function rerender () {
|
|
|
529
441
|
debounce(stateUpdated)
|
|
530
442
|
}
|
|
531
443
|
|
|
532
|
-
class Component {
|
|
533
|
-
|
|
534
|
-
|
|
535
|
-
return html``
|
|
444
|
+
class PureComponent extends Component {
|
|
445
|
+
createElement (args) {
|
|
446
|
+
this.args = clone(args)
|
|
536
447
|
}
|
|
537
|
-
}
|
|
538
448
|
|
|
539
|
-
|
|
449
|
+
update (args) {
|
|
450
|
+
let diff = deepDiff.diff(this.args, args)
|
|
451
|
+
Object.keys(diff).forEach(key => {
|
|
452
|
+
if (typeof diff[key] === 'function') {
|
|
453
|
+
this[key] = args[key]
|
|
454
|
+
}
|
|
455
|
+
})
|
|
456
|
+
return !!Object.keys(diff).find(key => typeof diff[key] !== 'function')
|
|
457
|
+
}
|
|
458
|
+
}
|
|
540
459
|
|
|
541
460
|
function cachedComponent (Class, args, id) {
|
|
542
|
-
|
|
461
|
+
let instance
|
|
462
|
+
if (id) {
|
|
463
|
+
let found = cache.get(id)
|
|
464
|
+
if (found) {
|
|
465
|
+
instance = found
|
|
466
|
+
} else {
|
|
467
|
+
instance = new Class()
|
|
468
|
+
cache.set(id, instance)
|
|
469
|
+
}
|
|
470
|
+
return instance.render(args)
|
|
471
|
+
} else {
|
|
472
|
+
instance = new Class()
|
|
473
|
+
return instance.createElement(args)
|
|
474
|
+
}
|
|
543
475
|
}
|
|
544
476
|
|
|
545
|
-
|
|
546
477
|
export {
|
|
547
478
|
getRouteComponent,
|
|
548
479
|
rerender,
|
package/package.json
CHANGED
|
@@ -1,17 +1,17 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "halfcab",
|
|
3
|
-
"version": "
|
|
3
|
+
"version": "15.0.0",
|
|
4
4
|
"type": "module",
|
|
5
5
|
"description": "A simple universal JavaScript framework focused on making use of es2015 template strings to build components.",
|
|
6
6
|
"main": "halfcab.mjs",
|
|
7
7
|
"module": "halfcab.mjs",
|
|
8
8
|
"jsnext:main": "halfcab.mjs",
|
|
9
9
|
"scripts": {
|
|
10
|
-
"test": "mocha './{,!(node_modules)/**}/test.js'",
|
|
10
|
+
"test": "mocha --experimental-modules --es-module-specifier-resolution=node --experimental-json-modules './{,!(node_modules)/**}/test.js'",
|
|
11
11
|
"test:coverage": "c8 --reporter=html --check-coverage --lines 75 --functions 75 --branches 75 npm test",
|
|
12
12
|
"test:coveralls": "c8 npm test && c8 report --reporter=text-lcov | coveralls",
|
|
13
13
|
"versionbump:fix": "npm version patch --no-git-tag-version",
|
|
14
|
-
"versionbump:feature": "npm version
|
|
14
|
+
"versionbump:feature": "npm version major --no-git-tag-version",
|
|
15
15
|
"versionbump:breakingchanges": "npm version major --no-git-tag-version",
|
|
16
16
|
"npm-publish": "npm publish"
|
|
17
17
|
},
|
|
@@ -54,7 +54,6 @@
|
|
|
54
54
|
"event-emitter": "^0.3.5",
|
|
55
55
|
"fast-clone": "^1.5.13",
|
|
56
56
|
"html-entities": "^2.3.2",
|
|
57
|
-
"lit-html": "^3.3.1",
|
|
58
57
|
"marked": "^0.7.0",
|
|
59
58
|
"nanocomponent": "^6.5.2",
|
|
60
59
|
"nanohtml": "^1.6.3",
|
package/test.js
CHANGED
|
@@ -3,7 +3,7 @@ import dirtyChai from 'dirty-chai'
|
|
|
3
3
|
import sinon from 'sinon'
|
|
4
4
|
import sinonChai from 'sinon-chai'
|
|
5
5
|
import jsdomGlobal from 'jsdom-global'
|
|
6
|
-
import server from 'nanohtml/lib/server
|
|
6
|
+
import server from 'nanohtml/lib/server'
|
|
7
7
|
|
|
8
8
|
const {expect} = chai
|
|
9
9
|
chai.use(dirtyChai)
|
|
@@ -37,7 +37,7 @@ describe('halfcab', () => {
|
|
|
37
37
|
before(async () => {
|
|
38
38
|
jsdomGlobal()
|
|
39
39
|
intialData('{"contactForm":{"sendDisabled":false,"showThanks":false},"login":{"disabled":false},"loading":false,"showContact":true,"products":[{"name":"SeeMonster","productType":"Digital Signage","description":"Digital signage built for ski areas.","price":"$3,200 USD per winter","logo":{"url":"//images.contentful.com/6p8ohxfikak1/6gZcP64mTm22m02YaoMGOQ/d7fef90eeaf292d0470c3b69a82b2c65/seemonster.svg","name":"SeeMonster Logo"},"detailSections":[{"name":"What is SeeMonster","description":"### SeeMonster enables you to display dynamic messages on screens around your ski area and on hotel room TVs, all managed over the Internet.","list":"","media":{"file":"//images.contentful.com/6p8ohxfikak1/bfmANPtMfQCQuQ4QkaWUE/249642578bfa0ec173be38ca9324aa13/seemonster-1-2-3.png","width":866},"link":""},{"name":"SeeMonster Includes","description":"","list":["Plays on any computer & TV combo","Bandwidth, storage & support included","Easy drag and drop admin.","Built in animation.","Upload images and video in any format","Use with both digital signs and in-room TV systems","Use live data to populate text fields & swap out images","Remotely control your screens","Schedule content","Social media feeds","Display videos and photos","Maps, lift & trail status"],"media":{"file":"//images.contentful.com/6p8ohxfikak1/3y0AaPPMzCGKiuKcMuea2Q/f52b579dc9a2080e4849ee87226124c2/QT-weatherforecast-screen.jpg","width":667},"link":""}]},{"name":"vicoMap","productType":"Interactive Trail Map","description":"*The* interactive map for ski areas.","price":"$2,800 USD per winter","logo":{"url":"//images.contentful.com/6p8ohxfikak1/1sC9N1WHW82AWkoeIcKQY2/688cbb96e7be602dd1f9bfbaff24e290/vicomap.svg","name":"vicoMap Logo"},"detailSections":[{"name":"Demo","description":"*The best way to check out vicoMap, is to just start using it! Have a go with this one and let us know what you think.*","list":"","media":"","link":"https://vicomap-cdn.resorts-interactive.com/map/12"},{"name":"vicoMap includes","description":"","list":["Responsive (works on desktop and mobile devices)","Live lift and trail data (free basic Report Pal connector included)","Cloud hosted (AWS)","Bandwidth, storage & support included","Easy website embed","No setup cost - built from your existing Illustrator file"],"media":"","link":""}]},{"name":"Report Pal","productType":"Snow Reporting","description":"Easy snow reporting in the cloud.","price":"$3,000 USD per winter","logo":{"url":"//images.contentful.com/6p8ohxfikak1/6JppGdAWzKc8mM8smE8w8m/ae5a5d43928027869f324aaa46b6f968/reportpal.svg","name":"Report Pal Logo"},"detailSections":[{"name":"Report Pal Summary","description":"The idea behind Report pal is a simple one - move snow reporting away from your website CMS and into the cloud.\n\nInstead of controlling your snow report from your website (which can change every couple of years, resulting in more costs to build, time re-training staff and having to re-integrate with 3rd parties), move it into the cloud and have a consistent platform, season after season.\n\nReport Pal makes use of the latest technologies on offer from Amazon Web Services, giving you a rock solid, always on snow reporting platform that's lightning fast.\n\nEvery ski resort is different, so Report Pal has been built from the ground up so that we can customise it to suit you. No more working your way around limitations of your website CMS, you can input and output what you like.\n\nWe'll setup your output in [MTN.XML](http://mtnxml.org \"MTN.XML\") format - the industry standard. Your data will be easily shared with 3rd parties.","list":"","media":{"file":"//images.contentful.com/6p8ohxfikak1/1ZPyCTXc08w6U8gC2k6Sgm/6b5b2a77395ec1965e0246bf8be8e74a/Screen-Shot-2016-12-11-at-4.18.34-pm.png","width":385},"link":""},{"name":"Report Pal Includes","description":"","list":["Unlimited users","Intelligent report merging","Scheduled reports","Social media integration","MTN.XML output","Report history and rollback","Automated lift & trail open counts, acreage and length","Rock solid - cloud hosted on Amazon Web Services"],"media":"","link":""}]}],"company":{"name":"Resorts Interactive","description":"Resorts Interactive has been making cloud software for the ski industry for since 2004. The way our software is built allows us to evolve over time and embrace changing technologies – a must have quality of any cloud based partner. Our current suite of Report Pal, vicoMap and SeeMonster covers a broad range of ski area marketing and operations and allows you to get back to what’s important (like taking a few runs on your lunch break!) Get in touch today, we’d love to talk.","logo":{"url":"//images.contentful.com/6p8ohxfikak1/6rv2tBpY5yiEUoyICu0A4W/4ab915134368789c1343c3444005e3aa/resortsinteractive.svg","name":"Resorts Interactive Logo"},"companyDetailSections":[{"image":"//images.contentful.com/6p8ohxfikak1/6ImVzlBXSEiKSeeqgEQksg/c748af57491f7804845038b4c6a177a8/powder_shot.jpg","title":"Resorts Interactive website","body":""},{"image":"//images.contentful.com/6p8ohxfikak1/3y0AaPPMzCGKiuKcMuea2Q/f52b579dc9a2080e4849ee87226124c2/QT-weatherforecast-screen.jpg","title":"What sets us apart?","body":"Resorts Interactive has been making cloud software for the ski industry since 2005 so we know a thing or two about the subject! The way our software is built allows us to evolve over time and embrace changing technologies – a must have quality of any cloud based partner. Our current suite of Report Pal, vicoMap and SeeMonster covers a broad range of ski area marketing and operations and allows you to get back to what’s important (like taking a few runs on your lunch break!) [Get in touch today, we’d love to talk.](mailto:info@resorts-interactive.com)"},{"image":"//images.contentful.com/6p8ohxfikak1/4eRwd92gowomMKAYOggcw0/9f83c2f2b2a7eaeb132a9dfe60509bca/Devices.jpg","title":"Clever Software Making Life Easy","body":"- You work in the ski industry because you love it, so we build software to save you time and get you back on the hill\n- Cloud based software is designed to be used anywhere – desktop, mobile, Colorado, Timbuktu. If you can get an Internet connection, we’ve got you covered\n- Simplicity and stability form a core part of our software architecture. Our products are hosted on Amazon Web Services and make use of their best and latest technologies like CloudFront, Aurora, and Elastic Beanstalk to make sure we’re always fast and available\n- Include live data and images from social media, weather, news and other websites\n- Upload video in any format and it’ll be converted for you"}],"icon":{"sys":{"space":{"sys":{"type":"Link","linkType":"Space","id":"6p8ohxfikak1"}},"id":"kmG0X3fFQcEseimceUkIs","type":"Asset","createdAt":"2017-06-18T10:24:54.580Z","updatedAt":"2017-06-18T10:24:54.580Z","revision":1,"locale":"en-NZ"},"fields":{"title":"Semi-flake-png","description":"PNG version of semi-flake","file":{"url":"//images.contentful.com/6p8ohxfikak1/kmG0X3fFQcEseimceUkIs/943c04791a2d442f1f50802a3868e29e/semiflake.png","details":{"size":24190,"image":{"width":512,"height":512}},"fileName":"semiflake.png","contentType":"image/png"}}}}}')
|
|
40
|
-
let halfcabModule = await import('./halfcab
|
|
40
|
+
let halfcabModule = await import('./halfcab')
|
|
41
41
|
;({
|
|
42
42
|
ssr,
|
|
43
43
|
html,
|
|
@@ -73,7 +73,7 @@ describe('halfcab', () => {
|
|
|
73
73
|
before(async () => {
|
|
74
74
|
jsdomGlobal()
|
|
75
75
|
intialData('{"contactForm":{"sendDisabled":false,"showThanks":false},"login":{"disabled":false},"loading":false,"showContact":true,"products":[{"name":"SeeMonster","productType":"Digital Signage","description":"Digital signage built for ski areas.","price":"$3,200 USD per winter","logo":{"url":"//images.contentful.com/6p8ohxfikak1/6gZcP64mTm22m02YaoMGOQ/d7fef90eeaf292d0470c3b69a82b2c65/seemonster.svg","name":"SeeMonster Logo"},"detailSections":[{"name":"What is SeeMonster","description":"### SeeMonster enables you to display dynamic messages on screens around your ski area and on hotel room TVs, all managed over the Internet.","list":"","media":{"file":"//images.contentful.com/6p8ohxfikak1/bfmANPtMfQCQuQ4QkaWUE/249642578bfa0ec173be38ca9324aa13/seemonster-1-2-3.png","width":866},"link":""},{"name":"SeeMonster Includes","description":"","list":["Plays on any computer & TV combo","Bandwidth, storage & support included","Easy drag and drop admin.","Built in animation.","Upload images and video in any format","Use with both digital signs and in-room TV systems","Use live data to populate text fields & swap out images","Remotely control your screens","Schedule content","Social media feeds","Display videos and photos","Maps, lift & trail status"],"media":{"file":"//images.contentful.com/6p8ohxfikak1/3y0AaPPMzCGKiuKcMuea2Q/f52b579dc9a2080e4849ee87226124c2/QT-weatherforecast-screen.jpg","width":667},"link":""}]},{"name":"vicoMap","productType":"Interactive Trail Map","description":"*The* interactive map for ski areas.","price":"$2,800 USD per winter","logo":{"url":"//images.contentful.com/6p8ohxfikak1/1sC9N1WHW82AWkoeIcKQY2/688cbb96e7be602dd1f9bfbaff24e290/vicomap.svg","name":"vicoMap Logo"},"detailSections":[{"name":"Demo","description":"*The best way to check out vicoMap, is to just start using it! Have a go with this one and let us know what you think.*","list":"","media":"","link":"https://vicomap-cdn.resorts-interactive.com/map/12"},{"name":"vicoMap includes","description":"","list":["Responsive (works on desktop and mobile devices)","Live lift and trail data (free basic Report Pal connector included)","Cloud hosted (AWS)","Bandwidth, storage & support included","Easy website embed","No setup cost - built from your existing Illustrator file"],"media":"","link":""}]},{"name":"Report Pal","productType":"Snow Reporting","description":"Easy snow reporting in the cloud.","price":"$3,000 USD per winter","logo":{"url":"//images.contentful.com/6p8ohxfikak1/6JppGdAWzKc8mM8smE8w8m/ae5a5d43928027869f324aaa46b6f968/reportpal.svg","name":"Report Pal Logo"},"detailSections":[{"name":"Report Pal Summary","description":"The idea behind Report pal is a simple one - move snow reporting away from your website CMS and into the cloud.\n\nInstead of controlling your snow report from your website (which can change every couple of years, resulting in more costs to build, time re-training staff and having to re-integrate with 3rd parties), move it into the cloud and have a consistent platform, season after season.\n\nReport Pal makes use of the latest technologies on offer from Amazon Web Services, giving you a rock solid, always on snow reporting platform that's lightning fast.\n\nEvery ski resort is different, so Report Pal has been built from the ground up so that we can customise it to suit you. No more working your way around limitations of your website CMS, you can input and output what you like.\n\nWe'll setup your output in [MTN.XML](http://mtnxml.org \"MTN.XML\") format - the industry standard. Your data will be easily shared with 3rd parties.","list":"","media":{"file":"//images.contentful.com/6p8ohxfikak1/1ZPyCTXc08w6U8gC2k6Sgm/6b5b2a77395ec1965e0246bf8be8e74a/Screen-Shot-2016-12-11-at-4.18.34-pm.png","width":385},"link":""},{"name":"Report Pal Includes","description":"","list":["Unlimited users","Intelligent report merging","Scheduled reports","Social media integration","MTN.XML output","Report history and rollback","Automated lift & trail open counts, acreage and length","Rock solid - cloud hosted on Amazon Web Services"],"media":"","link":""}]}],"company":{"name":"Resorts Interactive","description":"Resorts Interactive has been making cloud software for the ski industry for since 2004. The way our software is built allows us to evolve over time and embrace changing technologies – a must have quality of any cloud based partner. Our current suite of Report Pal, vicoMap and SeeMonster covers a broad range of ski area marketing and operations and allows you to get back to what’s important (like taking a few runs on your lunch break!) Get in touch today, we’d love to talk.","logo":{"url":"//images.contentful.com/6p8ohxfikak1/6rv2tBpY5yiEUoyICu0A4W/4ab915134368789c1343c3444005e3aa/resortsinteractive.svg","name":"Resorts Interactive Logo"},"companyDetailSections":[{"image":"//images.contentful.com/6p8ohxfikak1/6ImVzlBXSEiKSeeqgEQksg/c748af57491f7804845038b4c6a177a8/powder_shot.jpg","title":"Resorts Interactive website","body":""},{"image":"//images.contentful.com/6p8ohxfikak1/3y0AaPPMzCGKiuKcMuea2Q/f52b579dc9a2080e4849ee87226124c2/QT-weatherforecast-screen.jpg","title":"What sets us apart?","body":"Resorts Interactive has been making cloud software for the ski industry since 2005 so we know a thing or two about the subject! The way our software is built allows us to evolve over time and embrace changing technologies – a must have quality of any cloud based partner. Our current suite of Report Pal, vicoMap and SeeMonster covers a broad range of ski area marketing and operations and allows you to get back to what’s important (like taking a few runs on your lunch break!) [Get in touch today, we’d love to talk.](mailto:info@resorts-interactive.com)"},{"image":"//images.contentful.com/6p8ohxfikak1/4eRwd92gowomMKAYOggcw0/9f83c2f2b2a7eaeb132a9dfe60509bca/Devices.jpg","title":"Clever Software Making Life Easy","body":"- You work in the ski industry because you love it, so we build software to save you time and get you back on the hill\n- Cloud based software is designed to be used anywhere – desktop, mobile, Colorado, Timbuktu. If you can get an Internet connection, we’ve got you covered\n- Simplicity and stability form a core part of our software architecture. Our products are hosted on Amazon Web Services and make use of their best and latest technologies like CloudFront, Aurora, and Elastic Beanstalk to make sure we’re always fast and available\n- Include live data and images from social media, weather, news and other websites\n- Upload video in any format and it’ll be converted for you"}],"icon":{"sys":{"space":{"sys":{"type":"Link","linkType":"Space","id":"6p8ohxfikak1"}},"id":"kmG0X3fFQcEseimceUkIs","type":"Asset","createdAt":"2017-06-18T10:24:54.580Z","updatedAt":"2017-06-18T10:24:54.580Z","revision":1,"locale":"en-NZ"},"fields":{"title":"Semi-flake-png","description":"PNG version of semi-flake","file":{"url":"//images.contentful.com/6p8ohxfikak1/kmG0X3fFQcEseimceUkIs/943c04791a2d442f1f50802a3868e29e/semiflake.png","details":{"size":24190,"image":{"width":512,"height":512}},"fileName":"semiflake.png","contentType":"image/png"}}}}}')
|
|
76
|
-
let halfcabModule = await import('./halfcab
|
|
76
|
+
let halfcabModule = await import('./halfcab')
|
|
77
77
|
;({
|
|
78
78
|
ssr,
|
|
79
79
|
html,
|
|
@@ -91,22 +91,20 @@ describe('halfcab', () => {
|
|
|
91
91
|
halfcab = halfcabModule.default
|
|
92
92
|
})
|
|
93
93
|
|
|
94
|
-
it('Produces
|
|
94
|
+
it('Produces an HTML element when rendering', () => {
|
|
95
95
|
let el = html`
|
|
96
96
|
<div oninput=${() => {
|
|
97
97
|
}}></div>
|
|
98
98
|
`
|
|
99
|
-
expect(
|
|
100
|
-
expect(el).to.have.property('strings')
|
|
99
|
+
expect(el instanceof HTMLDivElement).to.be.true()
|
|
101
100
|
})
|
|
102
101
|
|
|
103
|
-
it('Produces
|
|
102
|
+
it('Produces an HTML element wrapping as a reusable component', () => {
|
|
104
103
|
let el = args => html`
|
|
105
104
|
<div oninput=${() => {
|
|
106
105
|
}}></div>
|
|
107
106
|
`
|
|
108
|
-
expect(
|
|
109
|
-
expect(el({})).to.have.property('strings')
|
|
107
|
+
expect(el({}) instanceof HTMLDivElement).to.be.true()
|
|
110
108
|
})
|
|
111
109
|
|
|
112
110
|
it('Runs halfcab function without error', () => {
|
|
@@ -0,0 +1,33 @@
|
|
|
1
|
+
import { ssr, html } from './halfcab.mjs';
|
|
2
|
+
import { unsafeHTML } from 'lit-html/directives/unsafe-html.js';
|
|
3
|
+
import { repeat } from 'lit-html/directives/repeat.js';
|
|
4
|
+
import { classMap } from 'lit-html/directives/class-map.js';
|
|
5
|
+
import { styleMap } from 'lit-html/directives/style-map.js';
|
|
6
|
+
|
|
7
|
+
console.log('--- Testing SSR ---');
|
|
8
|
+
|
|
9
|
+
const items = ['A', 'B', 'C'];
|
|
10
|
+
const classes = { active: true, disabled: false };
|
|
11
|
+
const styles = { color: 'red', 'font-size': '20px' };
|
|
12
|
+
const fn = () => console.log('clicked');
|
|
13
|
+
|
|
14
|
+
const template = html`
|
|
15
|
+
<div class="container">
|
|
16
|
+
<h1>Hello SSR</h1>
|
|
17
|
+
<div class=${classMap(classes)} style=${styleMap(styles)}>Styled</div>
|
|
18
|
+
<ul>
|
|
19
|
+
${repeat(items, (item) => html`<li>${item}</li>`)}
|
|
20
|
+
</ul>
|
|
21
|
+
<div>${unsafeHTML('<span>Unsafe</span>')}</div>
|
|
22
|
+
<button onclick=${fn}>Click</button>
|
|
23
|
+
<button disabled=${null}>Disabled Null</button>
|
|
24
|
+
</div>
|
|
25
|
+
`;
|
|
26
|
+
|
|
27
|
+
try {
|
|
28
|
+
const result = ssr(template);
|
|
29
|
+
console.log('Result componentsString:');
|
|
30
|
+
console.log(result.componentsString);
|
|
31
|
+
} catch (e) {
|
|
32
|
+
console.error('SSR Error:', e);
|
|
33
|
+
}
|