|
|
@ -25,6 +25,51 @@ interface Computation {
|
|
|
|
deps: string[]
|
|
|
|
deps: string[]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function detectIndentation(str: string) {
|
|
|
|
|
|
|
|
const pattern = /^[\t\s]{1,4}/gm;
|
|
|
|
|
|
|
|
let match;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
while (match = pattern.exec(str)) {
|
|
|
|
|
|
|
|
if (match[0][0] === '\t') return '\t';
|
|
|
|
|
|
|
|
if (match[0].length === 2) return ' ';
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return ' ';
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function getIndentationLevel(str: string, b: number) {
|
|
|
|
|
|
|
|
let a = b;
|
|
|
|
|
|
|
|
while (a > 0 && str[a - 1] !== '\n') a -= 1;
|
|
|
|
|
|
|
|
return /^\s*/.exec(str.slice(a, b))[0];
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function getIndentExclusionRanges(node: Node) {
|
|
|
|
|
|
|
|
const ranges: Node[] = [];
|
|
|
|
|
|
|
|
walk(node, {
|
|
|
|
|
|
|
|
enter(node: Node) {
|
|
|
|
|
|
|
|
if (node.type === 'TemplateElement') ranges.push(node);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
return ranges;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
function removeIndentation(
|
|
|
|
|
|
|
|
code: MagicString,
|
|
|
|
|
|
|
|
start: number,
|
|
|
|
|
|
|
|
end: number,
|
|
|
|
|
|
|
|
indentationLevel: string,
|
|
|
|
|
|
|
|
ranges: Node[]
|
|
|
|
|
|
|
|
) {
|
|
|
|
|
|
|
|
const str = code.original.slice(start, end);
|
|
|
|
|
|
|
|
const pattern = new RegExp(`^${indentationLevel}`, 'gm');
|
|
|
|
|
|
|
|
let match;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
while (match = pattern.exec(str)) {
|
|
|
|
|
|
|
|
// TODO bail if we're inside an exclusion range
|
|
|
|
|
|
|
|
code.remove(start + match.index, start + match.index + indentationLevel.length);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
export default class Generator {
|
|
|
|
export default class Generator {
|
|
|
|
ast: Parsed;
|
|
|
|
ast: Parsed;
|
|
|
|
parsed: Parsed;
|
|
|
|
parsed: Parsed;
|
|
|
@ -45,7 +90,6 @@ export default class Generator {
|
|
|
|
importedComponents: Map<string, string>;
|
|
|
|
importedComponents: Map<string, string>;
|
|
|
|
namespace: string;
|
|
|
|
namespace: string;
|
|
|
|
hasComponents: boolean;
|
|
|
|
hasComponents: boolean;
|
|
|
|
hasJs: boolean;
|
|
|
|
|
|
|
|
computations: Computation[];
|
|
|
|
computations: Computation[];
|
|
|
|
templateProperties: Record<string, Node>;
|
|
|
|
templateProperties: Record<string, Node>;
|
|
|
|
slots: Set<string>;
|
|
|
|
slots: Set<string>;
|
|
|
@ -409,7 +453,7 @@ export default class Generator {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
parseJs() {
|
|
|
|
parseJs() {
|
|
|
|
const { source } = this;
|
|
|
|
const { code, source } = this;
|
|
|
|
const { js } = this.parsed;
|
|
|
|
const { js } = this.parsed;
|
|
|
|
|
|
|
|
|
|
|
|
const imports = this.imports;
|
|
|
|
const imports = this.imports;
|
|
|
@ -418,11 +462,14 @@ export default class Generator {
|
|
|
|
const componentDefinition = new CodeBuilder();
|
|
|
|
const componentDefinition = new CodeBuilder();
|
|
|
|
|
|
|
|
|
|
|
|
let namespace = null;
|
|
|
|
let namespace = null;
|
|
|
|
let hasJs = !!js;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (js) {
|
|
|
|
if (js) {
|
|
|
|
this.addSourcemapLocations(js.content);
|
|
|
|
this.addSourcemapLocations(js.content);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
const indentation = detectIndentation(source.slice(js.start, js.end));
|
|
|
|
|
|
|
|
const indentationLevel = getIndentationLevel(source, js.content.body[0].start);
|
|
|
|
|
|
|
|
const indentExclusionRanges = getIndentExclusionRanges(js.content);
|
|
|
|
|
|
|
|
|
|
|
|
const scope = annotateWithScopes(js.content);
|
|
|
|
const scope = annotateWithScopes(js.content);
|
|
|
|
scope.declarations.forEach(name => {
|
|
|
|
scope.declarations.forEach(name => {
|
|
|
|
this.userVars.add(name);
|
|
|
|
this.userVars.add(name);
|
|
|
@ -500,6 +547,12 @@ export default class Generator {
|
|
|
|
let name = this.getUniqueName(key);
|
|
|
|
let name = this.getUniqueName(key);
|
|
|
|
this.templateVars.set(disambiguator ? `${disambiguator}-${key}` : key, name);
|
|
|
|
this.templateVars.set(disambiguator ? `${disambiguator}-${key}` : key, name);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// deindent
|
|
|
|
|
|
|
|
const indentationLevel = getIndentationLevel(source, node.start);
|
|
|
|
|
|
|
|
if (indentationLevel) {
|
|
|
|
|
|
|
|
removeIndentation(code, node.start, node.end, indentationLevel, indentExclusionRanges);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// TODO disambiguate between different categories, and ensure
|
|
|
|
// TODO disambiguate between different categories, and ensure
|
|
|
|
// no conflicts with existing aliases
|
|
|
|
// no conflicts with existing aliases
|
|
|
|
if (node.type === 'ArrowFunctionExpression') {
|
|
|
|
if (node.type === 'ArrowFunctionExpression') {
|
|
|
@ -624,35 +677,46 @@ export default class Generator {
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// now that we've analysed the default export, we can determine whether or not we need to keep it
|
|
|
|
|
|
|
|
let hasDefaultExport = !!defaultExport;
|
|
|
|
|
|
|
|
if (defaultExport && defaultExport.declaration.properties.length === 0) {
|
|
|
|
|
|
|
|
hasDefaultExport = false;
|
|
|
|
|
|
|
|
removeNode(this.code, js.content, defaultExport);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// if we do need to keep it, then we need to replace `export default`
|
|
|
|
// if we do need to keep it, then we need to replace `export default`
|
|
|
|
if (hasDefaultExport) {
|
|
|
|
// if (defaultExport) {
|
|
|
|
this.code.overwrite(
|
|
|
|
// this.code.overwrite(
|
|
|
|
defaultExport.start,
|
|
|
|
// defaultExport.start,
|
|
|
|
defaultExport.declaration.start,
|
|
|
|
// defaultExport.declaration.start,
|
|
|
|
`var ${this.alias('template')} = `
|
|
|
|
// `var ${this.alias('template')} = `
|
|
|
|
);
|
|
|
|
// );
|
|
|
|
|
|
|
|
// }
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (indentationLevel) {
|
|
|
|
|
|
|
|
if (defaultExport) {
|
|
|
|
|
|
|
|
removeIndentation(code, js.content.start, defaultExport.start, indentationLevel, indentExclusionRanges);
|
|
|
|
|
|
|
|
removeIndentation(code, defaultExport.end, js.content.end, indentationLevel, indentExclusionRanges);
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
removeIndentation(code, js.content.start, js.content.end, indentationLevel, indentExclusionRanges);
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (js.content.body.length === 0) {
|
|
|
|
if (js.content.body.length === 0) {
|
|
|
|
// if there's no need to include user code, remove it altogether
|
|
|
|
// if there's no need to include user code, remove it altogether
|
|
|
|
this.code.remove(js.content.start, js.content.end);
|
|
|
|
this.code.remove(js.content.start, js.content.end);
|
|
|
|
hasJs = false;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
this.javascript = hasDefaultExport ?
|
|
|
|
let a = js.content.start;
|
|
|
|
`[✂${js.content.start}-${defaultExport.start}✂]${componentDefinition}[✂${defaultExport.end}-${js.content.end}✂]` :
|
|
|
|
while (/\s/.test(source[a])) a += 1;
|
|
|
|
`[✂${js.content.start}-${js.content.end}✂]`;
|
|
|
|
|
|
|
|
|
|
|
|
let b = js.content.end;
|
|
|
|
|
|
|
|
while (/\s/.test(source[b - 1])) b -= 1;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (defaultExport) {
|
|
|
|
|
|
|
|
this.javascript = '';
|
|
|
|
|
|
|
|
if (a !== defaultExport.start) this.javascript += `[✂${a}-${defaultExport.start}✂]`;
|
|
|
|
|
|
|
|
if (!componentDefinition.isEmpty()) this.javascript += componentDefinition;
|
|
|
|
|
|
|
|
if (defaultExport.end !== b) this.javascript += `[✂${defaultExport.end}-${b}✂]`;
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
this.javascript = a === b ? null : `[✂${a}-${b}✂]`;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
this.computations = computations;
|
|
|
|
this.computations = computations;
|
|
|
|
this.hasJs = hasJs;
|
|
|
|
|
|
|
|
this.namespace = namespace;
|
|
|
|
this.namespace = namespace;
|
|
|
|
this.templateProperties = templateProperties;
|
|
|
|
this.templateProperties = templateProperties;
|
|
|
|
}
|
|
|
|
}
|
|
|
|