mirror of
https://github.com/JonnyBro/JaBa.git
synced 2025-01-20 09:23:51 +05:00
380 lines
13 KiB
JavaScript
380 lines
13 KiB
JavaScript
/* eslint-env mocha */
|
|
/* eslint-disable quotes */
|
|
|
|
mocha.setup({
|
|
ui: 'bdd'
|
|
})
|
|
|
|
describe('unit tests', () => {
|
|
const assert = chai.assert
|
|
|
|
const add = html => {
|
|
const tpl = document.createElement('template')
|
|
tpl.innerHTML = html
|
|
return document.body.appendChild(tpl.content.firstElementChild)
|
|
}
|
|
|
|
const sleep = t => new Promise(resolve => setTimeout(resolve, t))
|
|
|
|
const tick = () => new Promise(resolve => requestAnimationFrame(resolve))
|
|
|
|
describe('constructor()', () => {
|
|
it('should not load marked if marked already loaded', async () => {
|
|
window.marked = true
|
|
const fixture = add(`<zero-md manual-render></zero-md>`)
|
|
await fixture.waitForReady()
|
|
const nodes = document.head.querySelectorAll('script')
|
|
for (let a = 0; a < nodes.length; a++) {
|
|
assert(!nodes[a].src.endsWith('marked.min.js'))
|
|
}
|
|
fixture.remove()
|
|
})
|
|
|
|
it('should not load prism if prism already loaded', async () => {
|
|
window.marked = false
|
|
let nodes = document.head.querySelectorAll('script')
|
|
for (let a = 0; a < nodes.length; a++) {
|
|
if (nodes[a].src.includes('prism')) {
|
|
nodes[a].remove()
|
|
}
|
|
}
|
|
const f = add(`<zero-md manual-render></zero-md>`)
|
|
await f.loadScript(f.config.markedUrl)
|
|
await f.waitForReady()
|
|
nodes = document.head.querySelectorAll('script')
|
|
for (let a = 0; a < nodes.length; a++) {
|
|
assert(!nodes[a].src.includes('prism'))
|
|
}
|
|
f.remove()
|
|
})
|
|
|
|
it('should merge ZeroMdConfig opts into config', async () => {
|
|
const f = add(`<zero-md manual-render></zero-md>`)
|
|
await f.waitForReady()
|
|
assert(f.config.foo === 'bar')
|
|
f.remove()
|
|
})
|
|
})
|
|
|
|
describe('getters and setters', () => {
|
|
let f
|
|
before(() => { f = add(`<zero-md src="dummy.md" manual-render></zero-md>`) })
|
|
after(() => f.remove())
|
|
|
|
it('src reflects', () => {
|
|
assert(f.src === 'dummy.md')
|
|
f.src = 'dummy2.md'
|
|
assert(f.getAttribute('src') === 'dummy2.md')
|
|
})
|
|
|
|
it('boolean equates to true in class prop', () => {
|
|
assert(f.manualRender === true)
|
|
})
|
|
|
|
it('boolean reflects', () => {
|
|
f.manualRender = false
|
|
assert(!f.hasAttribute('manual-render'))
|
|
})
|
|
})
|
|
|
|
describe('buildStyles()', () => {
|
|
let f
|
|
afterEach(() => f.remove())
|
|
|
|
it('uses default styles if no template declared', () => {
|
|
f = add(`<zero-md manual-render></zero-md>`)
|
|
const s = f.makeNode(f.buildStyles()).outerHTML
|
|
assert(s.includes('/github-markdown.min.css'))
|
|
})
|
|
|
|
it('uses template styles', () => {
|
|
f = add(`<zero-md manual-render><template><link rel="stylesheet" href="example.css"></template></zero-md>`)
|
|
const s = f.makeNode(f.buildStyles()).outerHTML
|
|
assert(!s.includes('/github-markdown.min.css'))
|
|
assert(s.includes('example.css'))
|
|
})
|
|
|
|
it('prepends correctly', () => {
|
|
f = add(`<zero-md manual-render><template data-merge="prepend"><style>p{color:red;}</style></template></zero-md>`)
|
|
const s = f.makeNode(f.buildStyles()).outerHTML
|
|
assert(s.indexOf('p{color:red;}') < s.indexOf('markdown.min'))
|
|
})
|
|
|
|
it('appends correctly', () => {
|
|
f = add(`<zero-md manual-render><template data-merge="append"><style>p{color:red;}</style></template></zero-md>`)
|
|
const s = f.makeNode(f.buildStyles()).outerHTML
|
|
assert(s.indexOf('p{color:red;}') > s.indexOf('markdown.min'))
|
|
})
|
|
|
|
it('allows passing an empty template to override default template', () => {
|
|
f = add(`<zero-md manual-render><template></template></zero-md>`)
|
|
const s = f.makeNode(f.buildStyles())
|
|
assert(s.querySelectorAll('link').length === 0)
|
|
})
|
|
})
|
|
|
|
describe('buildMd()', () => {
|
|
let f
|
|
beforeEach(() => { f = add(`<zero-md manual-render></zero-md>`) })
|
|
afterEach(() => f.remove())
|
|
|
|
it('converts src to md', async () => {
|
|
f.src = 'fixture.md'
|
|
await f.render()
|
|
assert(f.shadowRoot.querySelector('.markdown-body>h1').innerHTML === 'markdown-fixture')
|
|
})
|
|
|
|
it('falls back to script when src is falsy', async () => {
|
|
const el = document.createElement('script')
|
|
el.setAttribute('type', 'text/markdown')
|
|
el.text = `# fallback`
|
|
f.appendChild(el)
|
|
await f.render()
|
|
assert(f.shadowRoot.querySelector('.markdown-body>h1').innerHTML === 'fallback')
|
|
})
|
|
|
|
it('highlights java code too', async () => {
|
|
f.src = 'fixture.md'
|
|
await f.render()
|
|
await sleep(200) // freaking ugly but blame prism
|
|
const el = f.shadowRoot.querySelector('.markdown-body pre>code.language-java :first-child')
|
|
assert(el.classList.contains('token'))
|
|
})
|
|
|
|
it('language-detects unhinted code blocks', async () => {
|
|
f.src = 'fixture.md'
|
|
await f.render()
|
|
const nodes = [...f.shadowRoot.querySelectorAll('p')].filter(i => i.textContent === 'Unhinted:')
|
|
assert(nodes[0].nextElementSibling.className.includes('language-'))
|
|
})
|
|
|
|
it('dedents when script data-dedent set', async () => {
|
|
const el = document.createElement('script')
|
|
el.setAttribute('type', 'text/markdown')
|
|
el.setAttribute('data-dedent', '')
|
|
el.text = `
|
|
# fallback`
|
|
f.appendChild(el)
|
|
await f.render()
|
|
assert(f.shadowRoot.querySelector('.markdown-body>h1').innerHTML === 'fallback')
|
|
})
|
|
|
|
it('resolves md base urls relative to src', async () => {
|
|
f.src = 'test1/fixture.md'
|
|
await f.render()
|
|
const a = document.createElement('a')
|
|
a.href = f.shadowRoot.querySelector('img').src
|
|
assert(a.pathname === '/test1/cat.jpg')
|
|
})
|
|
})
|
|
|
|
describe('stampBody()', () => {
|
|
let f
|
|
beforeEach(() => { f = add(`<zero-md manual-render></zero-md>`) })
|
|
afterEach(() => f.remove())
|
|
|
|
it('stamps html body into shadow dom', () => {
|
|
f.stampBody('<div class="test">hello</div>')
|
|
assert(f.shadowRoot.querySelector('.test').innerHTML === 'hello')
|
|
})
|
|
|
|
it('stamps html body into light dom if no-shadow set', () => {
|
|
f.remove()
|
|
f = add(`<zero-md manual-render no-shadow></zero-md>`)
|
|
f.stampBody('<div class="test">hello</div>')
|
|
assert(f.querySelector('.test').innerHTML === 'hello')
|
|
})
|
|
})
|
|
|
|
describe('stampStyles()', () => {
|
|
let f
|
|
beforeEach(() => { f = add(`<zero-md manual-render></zero-md>`) })
|
|
afterEach(() => f.remove())
|
|
|
|
it('stamps html styles and wait for stylesheet links to resolve', async () => {
|
|
const html = '<div><link rel="stylesheet" href="fixture.css"></div>'
|
|
let loaded = false
|
|
f.shadowRoot.addEventListener('load', () => {
|
|
loaded = true
|
|
}, {
|
|
once: true,
|
|
capture: true
|
|
})
|
|
await f.stampStyles(html)
|
|
assert(loaded)
|
|
})
|
|
|
|
it('still stamps html styles if a link errors', async () => {
|
|
const html = '<div><link rel="stylesheet" href="error.css"><link rel="stylesheet" href="fixture.css"></div>'
|
|
await f.stampStyles(html)
|
|
assert(f.shadowRoot.querySelector('link[href="fixture.css"]'))
|
|
})
|
|
})
|
|
|
|
describe('render()', () => {
|
|
let f
|
|
afterEach(() => f.remove())
|
|
|
|
it('auto re-renders when src change', done => {
|
|
f = add(`<zero-md src="fixture.md"></zero-md>`)
|
|
f.addEventListener('zero-md-rendered', () => {
|
|
if (f.src === 'fixture.md') {
|
|
assert(f.shadowRoot.querySelector('h1').innerHTML === 'markdown-fixture')
|
|
f.src = 'test1/fixture.md'
|
|
} else if (f.src === 'test1/fixture.md') {
|
|
assert(f.shadowRoot.querySelector('h1').innerHTML === 'relative-link-test')
|
|
done()
|
|
}
|
|
})
|
|
})
|
|
|
|
it('prevents FOUC by ensuring styles are stamped and resolved first, before stamping md', async () => {
|
|
f = add(`<zero-md manual-render>
|
|
<template>
|
|
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bootstrap@4.5.2/dist/css/bootstrap.css">
|
|
</template>
|
|
<script type="text/markdown"># fixture</script></zero-md>`)
|
|
const job = f.render()
|
|
await tick()
|
|
assert(f.shadowRoot.querySelector('link'))
|
|
assert(!f.shadowRoot.querySelector('h1'))
|
|
await job
|
|
assert(f.shadowRoot.querySelector('h1'))
|
|
})
|
|
|
|
it('renders markdown-body with optional classes', async () => {
|
|
f = add(`<zero-md manual-render><script type="text/markdown"># test</script></zero-md>`)
|
|
await f.render({ classes: 'test-class' })
|
|
assert(f.shadowRoot.querySelector('.markdown-body').classList.contains('test-class'))
|
|
await f.render({ classes: ['test2', 'test3'] })
|
|
assert(f.shadowRoot.querySelector('.markdown-body').classList.contains('test3'))
|
|
})
|
|
|
|
it('renders partially if body changes but styles do not', async () => {
|
|
f = add(`<zero-md manual-render><template><style>h1{color:red;}</style></template><script type="text/markdown"># test</script></zero-md>`)
|
|
await f.render()
|
|
let detail = {}
|
|
f.addEventListener('zero-md-rendered', e => {
|
|
detail = e.detail
|
|
})
|
|
f.querySelector('script').innerText = '# test2'
|
|
await f.render()
|
|
await tick()
|
|
assert(detail.stamped && detail.stamped.body === true)
|
|
assert(detail.stamped && !detail.stamped.styles)
|
|
const h1 = f.shadowRoot.querySelector('h1')
|
|
assert(window.getComputedStyle(h1).getPropertyValue('color') === 'rgb(255, 0, 0)')
|
|
})
|
|
|
|
it('renders partially if styles change but body does not', async () => {
|
|
f = add(`<zero-md manual-render><template><style>h1{color:red;}</style></template><script type="text/markdown"># test</script></zero-md>`)
|
|
await f.render()
|
|
let detail = {}
|
|
f.addEventListener('zero-md-rendered', e => {
|
|
detail = e.detail
|
|
})
|
|
const tpl = f.querySelector('template')
|
|
tpl.content.firstElementChild.innerText = 'h1{color:blue}'
|
|
await f.render()
|
|
await tick()
|
|
assert(detail.stamped && detail.stamped.styles === true)
|
|
assert(detail.stamped && !detail.stamped.body)
|
|
const h1 = f.shadowRoot.querySelector('h1')
|
|
assert(window.getComputedStyle(h1).getPropertyValue('color') === 'rgb(0, 0, 255)')
|
|
})
|
|
})
|
|
|
|
describe('hash-link scrolls', () => {
|
|
let f
|
|
afterEach(() => {
|
|
location.hash = ''
|
|
f.remove()
|
|
})
|
|
|
|
it('scrolls to element if location.hash set on first render', async () => {
|
|
location.hash = 'tamen-et-veri'
|
|
f = add(`<div style="height:200px;overflow:hidden;"><zero-md src="fixture.md"></zero-md></div>`)
|
|
await sleep(500)
|
|
assert(f.scrollTop > 0)
|
|
})
|
|
|
|
it('hijacks same-doc hash links and scrolls id into view', async () => {
|
|
f = add(`<div style="height:200px;overflow:hidden;"><zero-md src="fixture.md" manual-render></zero-md></div>`)
|
|
const el = f.querySelector('zero-md')
|
|
await el.render()
|
|
const a = el.shadowRoot.querySelector('a[href="#tamen-et-veri"]')
|
|
a.click()
|
|
await sleep(50)
|
|
assert(f.scrollTop > 0)
|
|
assert(location.hash === '#tamen-et-veri')
|
|
})
|
|
})
|
|
|
|
describe('Mutation Observer tests', () => {
|
|
let f
|
|
afterEach(() => f.remove())
|
|
|
|
it('auto re-renders content when inline markdown script changes', done => {
|
|
let isInitialRender = true
|
|
f = add(`<zero-md><script type="text/markdown"># markdown-fixture</script></zero-md>`)
|
|
f.addEventListener('zero-md-rendered', () => {
|
|
if (isInitialRender) {
|
|
assert(f.shadowRoot.querySelector('h1').innerHTML === 'markdown-fixture')
|
|
isInitialRender = false
|
|
f.querySelector('script').innerHTML = '# updated markdown-fixture'
|
|
} else {
|
|
assert(f.shadowRoot.querySelector('h1').innerHTML === 'updated markdown-fixture')
|
|
done()
|
|
}
|
|
})
|
|
})
|
|
|
|
it('auto re-renders styles when styles template changes', done => {
|
|
let isInitialRender = true
|
|
f = add(`<zero-md>
|
|
<template>
|
|
<style>h1 { color: rgb(255, 0, 0); }</style>
|
|
</template>
|
|
<script type="text/markdown"># fixture</script></zero-md>`)
|
|
f.addEventListener('zero-md-rendered', () => {
|
|
const h1 = f.shadowRoot.querySelector('h1')
|
|
const computedStyle = window.getComputedStyle(h1)
|
|
if (isInitialRender) {
|
|
assert(computedStyle.color === 'rgb(255, 0, 0)')
|
|
isInitialRender = false
|
|
f.querySelector('template').content.firstElementChild.innerHTML = 'h1 { color: rgb(0, 255, 0); }'
|
|
} else {
|
|
assert(computedStyle.color === 'rgb(0, 255, 0)')
|
|
done()
|
|
}
|
|
})
|
|
})
|
|
})
|
|
|
|
describe('running console tests - please ensure no error messages generated in console', () => {
|
|
it('element should reconnect properly', async () => {
|
|
console.log('Running element reconnection test... (this should not generate any errors)')
|
|
let count = 0
|
|
const handler = () => count++
|
|
window.addEventListener('zero-md-ready', handler)
|
|
const el = document.createElement('zero-md')
|
|
el.setAttribute('manual-render', '')
|
|
let node = document.body.appendChild(el)
|
|
await tick()
|
|
node.remove()
|
|
await tick()
|
|
node = document.body.appendChild(el)
|
|
await tick()
|
|
node.remove()
|
|
window.removeEventListener('zero-md-ready', handler)
|
|
assert(count === 2)
|
|
console.log('Complete')
|
|
})
|
|
})
|
|
|
|
describe('other cool features', () => {
|
|
|
|
})
|
|
})
|
|
|
|
mocha.run()
|