import MagicString, { SourceMap } from 'magic-string'
import { beforeEach, describe, expect, it } from 'vitest'
import { preprocess } from 'svelte/compiler'
import { MagicSFC as MagicSvelteSFC, magicSvelteSfcOptions } from '../src/svelte/sfc'
import { completeSvelteComponent, svelteScript, svelteStyle, svelteTemplate, svelteTypescriptScript } from './utils'
describe('Magic Svelte SFC', () => {
beforeEach(() => {
// Set default parser for MagicSvelteSFC
magicSvelteSfcOptions.parser = preprocess
})
it('Can create the class', async () => {
const sfc = await new MagicSvelteSFC(svelteScript).parse()
expect(sfc.toString()).toBe(svelteScript)
})
it('Cannot create a Magic Svelte SFC without a parser function', async () => {
magicSvelteSfcOptions.parser = undefined
try {
await new MagicSvelteSFC(svelteScript).parse()
throw new Error('Code should not reach this point!')
}
catch (e) {
expect(e.message).toBe('You must provide a `parser` function (usually preprocess from svelte/compiler) in options when using MagicSvelteSFC.')
}
})
it('Can create the class from a MagicString', async () => {
const ms = new MagicString(svelteScript)
const sfc = await new MagicSvelteSFC(ms).parse()
const appended = '\nlet secondTest: string'
sfc.scripts[0].append(appended)
expect(sfc.toString()).toBe(``)
})
it('Can get a sourcemap', async () => {
const sfc = await new MagicSvelteSFC(svelteScript).parse()
expect(sfc.getSourcemap()).toBeInstanceOf(SourceMap)
})
it('Can parse a
Variants component
`
const sfc = await new MagicSvelteSFC(component).parse()
expect(sfc.scripts[0]).toBeInstanceOf(Object)
expect(sfc.scripts[0].attrs.lang).toBe('ts')
expect(sfc.scripts[1]).toBeInstanceOf(Object)
expect(sfc.scripts[1].attrs).toStrictEqual({})
})
it('Can parse a HTML content', async () => {
const sfc = await new MagicSvelteSFC(svelteTemplate).parse()
expect(sfc.templates[0]).toBeInstanceOf(Object)
})
it('Can parse a '
const expectedStyle = ''
const sfc = await new MagicSvelteSFC(originalStyle).parse()
sfc.styles[0].overwrite(18, 21, 'blue')
expect(sfc.toString()).toBe(expectedStyle)
})
it('Can manipulate nested elements in a block', async () => {
const originalNestedTemplate = 'Hello, world!
'
const expectedNestedTemplate = 'Hello, Mars!
'
const sfc = await new MagicSvelteSFC(originalNestedTemplate).parse()
sfc.templates[0].overwrite(18, 23, 'Mars')
expect(sfc.toString()).toBe(expectedNestedTemplate)
})
it('Can handle empty blocks', async () => {
const emptyScript = ''
const emptyTemplate = '\n\n'
const emptyStyle = ''
const sfc = await new MagicSvelteSFC(`${emptyScript}\n${emptyTemplate}\n${emptyStyle}`).parse()
// Svelte SFC parser does detect empty script blocks
expect(sfc.scripts.length).toBeFalsy()
// Svelte SFC parser does detect empty style blocks
expect(sfc.styles.length).toBeFalsy()
// Svelte SFC parser does not detect empty HTML
expect(sfc.templates.length).toBeTruthy()
})
it('Can append content to a `
const expectedScript = ``
const sfc = await new MagicSvelteSFC(originalScript).parse()
sfc.scripts[0].append(appended)
expect(sfc.toString()).toBe(expectedScript)
})
it('Can appendLeft content to a `
const expectedScript = ``
const sfc = await new MagicSvelteSFC(originalScript).parse()
sfc.scripts[0].appendLeft(0, appended)
expect(sfc.toString()).toBe(expectedScript)
})
it('Can appendRight content to a `
const expectedScript = ``
const sfc = await new MagicSvelteSFC(originalScript).parse()
sfc.scripts[0].appendRight(baseContent.length, appended)
expect(sfc.toString()).toBe(expectedScript)
})
it('Can prepend content to a `
const expectedScript = ``
const sfc = await new MagicSvelteSFC(originalScript).parse()
sfc.scripts[0].prepend(prepended)
expect(sfc.toString()).toBe(expectedScript)
})
it('Can prependLeft content to a `
const expectedScript = ``
const sfc = await new MagicSvelteSFC(originalScript).parse()
sfc.scripts[0].prependLeft(0, prepended)
expect(sfc.toString()).toBe(expectedScript)
})
it('Can prependRight content to a `
const expectedScript = ``
const sfc = await new MagicSvelteSFC(originalScript).parse()
sfc.scripts[0].prependRight(baseContent.length, prepended)
expect(sfc.toString()).toBe(expectedScript)
})
it('Can manipulate every block of an SFC', async () => {
const originalSFC = `{{ msg }}
`
const expectedSFC = `{{ updatedMsg }}
`
const sfc = await new MagicSvelteSFC(originalSFC).parse()
sfc.templates[0].overwrite(8, 11, 'updatedMsg')
sfc.scripts[0].overwrite(61, 66, 'Mars')
sfc.styles[0].overwrite(18, 21, 'blue')
expect(sfc.toString()).toBe(expectedSFC)
})
it('Uses all methods on different blocks', async () => {
const originalSFC = `Hello, world!
`
const expectedSFC = `Hi, Mars!
Hello, world!
`
const sfc = await new MagicSvelteSFC(originalSFC).parse()
// Manipulate block
sfc.templates[0].prepend('Hi, Mars!\n')
// Manipulate