import { parse } from '../src'
import { baseParse, baseCompile } from '@vue/compiler-core'
import { SourceMapConsumer } from 'source-map'
describe('compiler:sfc', () => {
describe('source map', () => {
test('style block', () => {
// Padding determines how many blank lines will there be before the style block
const padding = Math.round(Math.random() * 10)
const style = parse(
`${'\n'.repeat(padding)}\n`
).descriptor.styles[0]
expect(style.map).not.toBeUndefined()
const consumer = new SourceMapConsumer(style.map!)
consumer.eachMapping(mapping => {
expect(mapping.originalLine - mapping.generatedLine).toBe(padding)
})
})
test('script block', () => {
// Padding determines how many blank lines will there be before the style block
const padding = Math.round(Math.random() * 10)
const script = parse(
`${'\n'.repeat(padding)}\n`
).descriptor.script
expect(script!.map).not.toBeUndefined()
const consumer = new SourceMapConsumer(script!.map!)
consumer.eachMapping(mapping => {
expect(mapping.originalLine - mapping.generatedLine).toBe(padding)
})
})
test('custom block', () => {
const padding = Math.round(Math.random() * 10)
const custom = parse(
`${'\n'.repeat(padding)}\n{\n "greeting": "hello"\n}\n\n`
).descriptor.customBlocks[0]
expect(custom!.map).not.toBeUndefined()
const consumer = new SourceMapConsumer(custom!.map!)
consumer.eachMapping(mapping => {
expect(mapping.originalLine - mapping.generatedLine).toBe(padding)
})
})
})
test('pad content', () => {
const content = `
{ "greeting": "hello" }
`
const padFalse = parse(content.trim(), { pad: false }).descriptor
expect(padFalse.template!.content).toBe('\n
\n')
expect(padFalse.script!.content).toBe('\nexport default {}\n')
expect(padFalse.styles[0].content).toBe('\nh1 { color: red }\n')
expect(padFalse.customBlocks[0].content).toBe('\n{ "greeting": "hello" }\n')
const padTrue = parse(content.trim(), { pad: true }).descriptor
expect(padTrue.script!.content).toBe(
Array(3 + 1).join('//\n') + '\nexport default {}\n'
)
expect(padTrue.styles[0].content).toBe(
Array(6 + 1).join('\n') + '\nh1 { color: red }\n'
)
expect(padTrue.customBlocks[0].content).toBe(
Array(9 + 1).join('\n') + '\n{ "greeting": "hello" }\n'
)
const padLine = parse(content.trim(), { pad: 'line' }).descriptor
expect(padLine.script!.content).toBe(
Array(3 + 1).join('//\n') + '\nexport default {}\n'
)
expect(padLine.styles[0].content).toBe(
Array(6 + 1).join('\n') + '\nh1 { color: red }\n'
)
expect(padLine.customBlocks[0].content).toBe(
Array(9 + 1).join('\n') + '\n{ "greeting": "hello" }\n'
)
const padSpace = parse(content.trim(), { pad: 'space' }).descriptor
expect(padSpace.script!.content).toBe(
`\n\n\n\n\n`.replace(
/./g,
' '
) + '\n{ "greeting": "hello" }\n'
)
})
test('should parse correct range for root level self closing tag', () => {
const content = `\n \n`
const { descriptor } = parse(`${content}`)
expect(descriptor.template).toBeTruthy()
expect(descriptor.template!.content).toBe(content)
expect(descriptor.template!.loc).toMatchObject({
start: { line: 1, column: 11, offset: 10 },
end: {
line: 3,
column: 1,
offset: 10 + content.length
},
source: content
})
})
test('should parse correct range for blocks with no content (self closing)', () => {
const { descriptor } = parse(``)
expect(descriptor.template).toBeTruthy()
expect(descriptor.template!.content).toBeFalsy()
expect(descriptor.template!.loc).toMatchObject({
start: { line: 1, column: 1, offset: 0 },
end: { line: 1, column: 1, offset: 0 },
source: ''
})
})
test('should parse correct range for blocks with no content (explicit)', () => {
const { descriptor } = parse(``)
expect(descriptor.template).toBeTruthy()
expect(descriptor.template!.content).toBeFalsy()
expect(descriptor.template!.loc).toMatchObject({
start: { line: 1, column: 11, offset: 10 },
end: { line: 1, column: 11, offset: 10 },
source: ''
})
})
test('should ignore other nodes with no content', () => {
expect(parse(``).descriptor.script).toBe(null)
expect(parse(``).descriptor.script).toBe(null)
expect(parse(``).descriptor.styles.length).toBe(0)
expect(parse(``).descriptor.styles.length).toBe(0)
expect(parse(``).descriptor.customBlocks.length).toBe(0)
expect(
parse(` \n\t `).descriptor.customBlocks.length
).toBe(0)
})
test('handle empty nodes with src attribute', () => {
const { descriptor } = parse(``)
expect(descriptor.script).toBeTruthy()
expect(descriptor.script!.content).toBeFalsy()
expect(descriptor.script!.attrs['src']).toBe('com')
})
test('ignoreEmpty: false', () => {
const { descriptor } = parse(
`\n`,
{
ignoreEmpty: false
}
)
expect(descriptor.script).toBeTruthy()
expect(descriptor.script!.loc).toMatchObject({
source: '',
start: { line: 1, column: 9, offset: 8 },
end: { line: 1, column: 9, offset: 8 }
})
expect(descriptor.scriptSetup).toBeTruthy()
expect(descriptor.scriptSetup!.loc).toMatchObject({
source: '\n',
start: { line: 2, column: 15, offset: 32 },
end: { line: 3, column: 1, offset: 33 }
})
})
test('nested templates', () => {
const content = `
ok
`
const { descriptor } = parse(`${content}`)
expect(descriptor.template!.content).toBe(content)
})
test('treat empty lang attribute as the html', () => {
const content = `ok
`
const { descriptor, errors } = parse(
`${content}`
)
expect(descriptor.template!.content).toBe(content)
expect(errors.length).toBe(0)
})
// #1120
test('alternative template lang should be treated as plain text', () => {
const content = `p(v-if="1 < 2") test`
const { descriptor, errors } = parse(
`` + content + ``
)
expect(errors.length).toBe(0)
expect(descriptor.template!.content).toBe(content)
})
//#2566
test('div lang should not be treated as plain text', () => {
const { errors } = parse(`
`)
expect(errors.length).toBe(0)
})
test('slotted detection', async () => {
expect(parse(`hi`).descriptor.slotted).toBe(false)
expect(
parse(`hi`).descriptor
.slotted
).toBe(false)
expect(
parse(
`hi`
).descriptor.slotted
).toBe(true)
expect(
parse(
`hi`
).descriptor.slotted
).toBe(true)
})
test('error tolerance', () => {
const { errors } = parse(``)
expect(errors.length).toBe(1)
})
test('should parse as DOM by default', () => {
const { errors } = parse(``)
expect(errors.length).toBe(0)
})
test('custom compiler', () => {
const { errors } = parse(``, {
compiler: {
parse: baseParse,
compile: baseCompile
}
})
expect(errors.length).toBe(1)
})
test('treat custom blocks as raw text', () => {
const { errors, descriptor } = parse(` <-& `)
expect(errors.length).toBe(0)
expect(descriptor.customBlocks[0].content).toBe(` <-& `)
})
describe('warnings', () => {
function assertWarning(errors: Error[], msg: string) {
expect(errors.some(e => e.message.match(msg))).toBe(true)
}
test('should only allow single template element', () => {
assertWarning(
parse(``).errors,
`Single file component can contain only one element`
)
})
test('should only allow single script element', () => {
assertWarning(
parse(``)
.errors,
`Single file component can contain only one `
).errors,
`Single file component can contain only one `
).errors.length
).toBe(0)
})
})
})