UNPKG

astx

Version:

super powerful structural search and replace for JavaScript and TypeScript to automate your refactoring

113 lines (104 loc) 14.1 kB
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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"}