75 lines
		
	
	
		
			2.1 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			75 lines
		
	
	
		
			2.1 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
import LinesAndColumns from "lines-and-columns";
 | 
						|
 | 
						|
 | 
						|
import {formatTokenType} from "../parser/tokenizer/types";
 | 
						|
 | 
						|
export default function formatTokens(code, tokens) {
 | 
						|
  if (tokens.length === 0) {
 | 
						|
    return "";
 | 
						|
  }
 | 
						|
 | 
						|
  const tokenKeys = Object.keys(tokens[0]).filter(
 | 
						|
    (k) => k !== "type" && k !== "value" && k !== "start" && k !== "end" && k !== "loc",
 | 
						|
  );
 | 
						|
  const typeKeys = Object.keys(tokens[0].type).filter((k) => k !== "label" && k !== "keyword");
 | 
						|
 | 
						|
  const headings = ["Location", "Label", "Raw", ...tokenKeys, ...typeKeys];
 | 
						|
 | 
						|
  const lines = new LinesAndColumns(code);
 | 
						|
  const rows = [headings, ...tokens.map(getTokenComponents)];
 | 
						|
  const padding = headings.map(() => 0);
 | 
						|
  for (const components of rows) {
 | 
						|
    for (let i = 0; i < components.length; i++) {
 | 
						|
      padding[i] = Math.max(padding[i], components[i].length);
 | 
						|
    }
 | 
						|
  }
 | 
						|
  return rows
 | 
						|
    .map((components) => components.map((component, i) => component.padEnd(padding[i])).join(" "))
 | 
						|
    .join("\n");
 | 
						|
 | 
						|
  function getTokenComponents(token) {
 | 
						|
    const raw = code.slice(token.start, token.end);
 | 
						|
    return [
 | 
						|
      formatRange(token.start, token.end),
 | 
						|
      formatTokenType(token.type),
 | 
						|
      truncate(String(raw), 14),
 | 
						|
      // @ts-ignore: Intentional dynamic access by key.
 | 
						|
      ...tokenKeys.map((key) => formatValue(token[key], key)),
 | 
						|
      // @ts-ignore: Intentional dynamic access by key.
 | 
						|
      ...typeKeys.map((key) => formatValue(token.type[key], key)),
 | 
						|
    ];
 | 
						|
  }
 | 
						|
 | 
						|
  // eslint-disable-next-line @typescript-eslint/no-explicit-any
 | 
						|
  function formatValue(value, key) {
 | 
						|
    if (value === true) {
 | 
						|
      return key;
 | 
						|
    } else if (value === false || value === null) {
 | 
						|
      return "";
 | 
						|
    } else {
 | 
						|
      return String(value);
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  function formatRange(start, end) {
 | 
						|
    return `${formatPos(start)}-${formatPos(end)}`;
 | 
						|
  }
 | 
						|
 | 
						|
  function formatPos(pos) {
 | 
						|
    const location = lines.locationForIndex(pos);
 | 
						|
    if (!location) {
 | 
						|
      return "Unknown";
 | 
						|
    } else {
 | 
						|
      return `${location.line + 1}:${location.column + 1}`;
 | 
						|
    }
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
function truncate(s, length) {
 | 
						|
  if (s.length > length) {
 | 
						|
    return `${s.slice(0, length - 3)}...`;
 | 
						|
  } else {
 | 
						|
    return s;
 | 
						|
  }
 | 
						|
}
 |