astx
Version:
super powerful structural search and replace for JavaScript and TypeScript to automate your refactoring
113 lines (104 loc) • 14.1 kB
JavaScript
import ensureArray from '../util/ensureArray.mjs'
import forEachNode from '../util/forEachNode.mjs'
function parse0(backend, strings, ...quasis) {
try {
const result = backend.template.statements(strings, ...quasis)
if (result.length === 0)
return backend.template.expression(strings, ...quasis)
if (result.length > 1) return result
const node = result[0]
return node.type === 'ExpressionStatement' ? node.expression : node
} catch (error) {
// fallthrough
}
return backend.template.expression(strings, ...quasis)
}
export function parsePattern(strings, ...quasis) {
const ast = parse0(this, strings, ...quasis)
let result = ensureArray(ast).map((n) => new this.t.NodePath(n))
const allComments = []
forEachNode(this.t, result, ['Node'], (path) => {
if (allComments.length >= 2) return
const { node } = path
for (const comment of this.comments(node)) {
allComments.push(comment)
}
})
let done = false
if (allComments.length >= 2) {
let from = Infinity,
to = -Infinity
for (const comment of allComments) {
const { start, end } = this.location(comment)
if (start != null && start > to) to = start
if (end != null && end < from) from = end
}
if (from != null && to != null && from < to) {
const pathInRange = (path) => {
const { start, end } = this.location(path.node)
return start != null && end != null && start >= from && end <= to
}
forEachNode(this.t, result, ['Node'], (path) => {
if (done) return
if (pathInRange(path)) {
while (path.parentPath != null && pathInRange(path.parentPath))
path = path.parentPath
result = path
done = true
}
})
if (done) return result
}
}
let extractNext = false
forEachNode(this.t, result, ['Node'], (path) => {
if (done) return
if (extractNext) {
result = path
done = true
return
}
const { node } = path
const { comments, leadingComments, innerComments } = node
if (comments) {
for (let i = 0; i < comments.length; i++) {
const c = comments[i]
if (!c.value && c.leading) {
comments.splice(i, 1)
result = path
done = true
return
} else if (!c.value && c.inner) {
extractNext = true
return
}
}
}
if (leadingComments) {
for (let i = 0; i < leadingComments.length; i++) {
const c = leadingComments[i]
if (!c.value) {
leadingComments.splice(i, 1)
result = path
done = true
return
}
}
}
if (innerComments) {
for (let i = 0; i < innerComments.length; i++) {
const c = innerComments[i]
if (!c.value) {
innerComments.splice(i, 1)
extractNext = true
return
}
}
}
})
return result
}
export function parsePatternToNodes(strings, ...quasis) {
const paths = this.parsePattern(strings, ...quasis)
return Array.isArray(paths) ? paths.map((p) => p.node) : paths.node
} //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["ensureArray","forEachNode","parse0","backend","strings","quasis","result","template","statements","length","expression","node","type","error","parsePattern","ast","map","n","t","NodePath","allComments","path","comment","comments","push","done","from","Infinity","to","start","end","location","pathInRange","parentPath","extractNext","leadingComments","innerComments","i","c","value","leading","splice","inner","parsePatternToNodes","paths","Array","isArray","p"],"sources":["../../src/backend/parse.ts"],"sourcesContent":["import { Backend } from './Backend'\nimport { NodePath, Expression, Statement, Node, Comment } from '../types'\nimport ensureArray from '../util/ensureArray'\nimport forEachNode from '../util/forEachNode'\n\nfunction parse0(\n  backend: Backend,\n  strings: TemplateStringsArray | string | string[],\n  ...quasis: any[]\n): Expression | Statement | Statement[] {\n  try {\n    const result = backend.template.statements(strings, ...quasis)\n    if (result.length === 0)\n      return backend.template.expression(strings, ...quasis)\n    if (result.length > 1) return result\n    const node = result[0]\n    return node.type === 'ExpressionStatement' ? node.expression : node\n  } catch (error) {\n    // fallthrough\n  }\n  return backend.template.expression(strings, ...quasis)\n}\n\nexport function parsePattern(\n  this: Backend,\n  strings: TemplateStringsArray | string | string[],\n  ...quasis: any[]\n): NodePath | NodePath[] {\n  const ast = parse0(this, strings, ...quasis)\n  let result: NodePath | NodePath[] = ensureArray(ast).map(\n    (n) => new this.t.NodePath(n)\n  )\n  const allComments: Comment[] = []\n  forEachNode(this.t, result, ['Node'], (path: NodePath) => {\n    if (allComments.length >= 2) return\n    const { node } = path\n    for (const comment of this.comments(node)) {\n      allComments.push(comment)\n    }\n  })\n  let done = false\n  if (allComments.length >= 2) {\n    let from = Infinity,\n      to = -Infinity\n    for (const comment of allComments) {\n      const { start, end } = this.location(comment)\n      if (start != null && start > to) to = start\n      if (end != null && end < from) from = end\n    }\n    if (from != null && to != null && from < to) {\n      const pathInRange = (path: NodePath) => {\n        const { start, end } = this.location(path.node)\n        return start != null && end != null && start >= from && end <= to\n      }\n      forEachNode(this.t, result, ['Node'], (path: NodePath) => {\n        if (done) return\n        if (pathInRange(path)) {\n          while (path.parentPath != null && pathInRange(path.parentPath))\n            path = path.parentPath\n          result = path\n          done = true\n        }\n      })\n      if (done) return result\n    }\n  }\n  let extractNext = false\n  forEachNode(this.t, result, ['Node'], (path: NodePath) => {\n    if (done) return\n    if (extractNext) {\n      result = path\n      done = true\n      return\n    }\n    const { node } = path\n    const { comments, leadingComments, innerComments } = node as any\n    if (comments) {\n      for (let i = 0; i < comments.length; i++) {\n        const c = comments[i]\n        if (!c.value && c.leading) {\n          comments.splice(i, 1)\n          result = path\n          done = true\n          return\n        } else if (!c.value && c.inner) {\n          extractNext = true\n          return\n        }\n      }\n    }\n    if (leadingComments) {\n      for (let i = 0; i < leadingComments.length; i++) {\n        const c = leadingComments[i]\n        if (!c.value) {\n          leadingComments.splice(i, 1)\n          result = path\n          done = true\n          return\n        }\n      }\n    }\n    if (innerComments) {\n      for (let i = 0; i < innerComments.length; i++) {\n        const c = innerComments[i]\n        if (!c.value) {\n          innerComments.splice(i, 1)\n          extractNext = true\n          return\n        }\n      }\n    }\n  })\n  return result\n}\n\nexport function parsePatternToNodes(\n  this: Backend,\n  strings: TemplateStringsArray | string | string[],\n  ...quasis: any[]\n): Node | Node[] {\n  const paths = this.parsePattern(strings, ...quasis)\n  return Array.isArray(paths) ? paths.map((p) => p.node) : paths.node\n}\n"],"mappings":"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"}