UNPKG

prettier-plugin-imports

Version:

A prettier plugins to sort imports in provided RegEx order

88 lines (81 loc) 3.56 kB
import { chunkTypeUnsortable, newLineNode, TYPES_SPECIAL_WORD, } from '../constants'; import { adjustCommentsOnSortedNodes } from './adjust-comments-on-sorted-nodes'; import { explodeTypeAndValueSpecifiers } from './explode-type-and-value-specifiers'; import { getChunkTypeOfNode } from './get-chunk-type-of-node'; import { getSortedNodesByImportOrder } from './get-sorted-nodes-by-import-order'; import { mergeNodesWithMatchingImportFlavors } from './merge-nodes-with-matching-flavors'; import type { GetSortedNodes, ImportChunk, ImportOrLine } from '../types'; /** * This function returns the given nodes, sorted in the order as indicated by * the importOrder array. The plugin considers these import nodes as local * import declarations * * In addition, this method preserves the relative order of side effect imports * and non side effect imports. A side effect import is an ImportDeclaration * without any import specifiers. It does this by splitting the import nodes at * each side effect node, then sorting only the non side effect import nodes * between the side effect nodes according to the given options. * * @param nodes All import nodes that should be sorted. * @param options Options to influence the behavior of the sorting algorithm. * @returns A sorted array of the remaining import nodes */ export const getSortedNodes: GetSortedNodes = (nodes, options) => { const { importOrder, importOrderCombineTypeAndValueImports, hasAnyCustomGroupSeparatorsInImportOrder, provideGapAfterTopOfFileComments, } = options; // Split nodes at each boundary between a side-effect node and a // non-side-effect node, keeping both types of nodes together. const splitBySideEffectNodes = nodes.reduce<ImportChunk[]>((chunks, node) => { const type = getChunkTypeOfNode(node); const last = chunks[chunks.length - 1]; if (last === undefined || last.type !== type) { chunks.push({ type, nodes: [node] }); } else { last.nodes.push(node); } return chunks; }, []); const finalNodes: ImportOrLine[] = []; // Sort each chunk of side-effect and non-side-effect nodes for (const chunk of splitBySideEffectNodes) { // do not sort side effect nodes if (chunk.type === chunkTypeUnsortable) { // If users use custom separators, add newlines around the side effect node if (hasAnyCustomGroupSeparatorsInImportOrder) { // Add newline before chunk if it has no leading comment #ConditionalNewLineAfterSideEffectWithSeparatorsGivenLeadingComment if (!chunk.nodes[0].leadingComments?.length) { finalNodes.push(newLineNode); } finalNodes.push(...chunk.nodes, newLineNode); } else { finalNodes.push(...chunk.nodes); } } else { let nodes = mergeNodesWithMatchingImportFlavors(chunk.nodes, { importOrderCombineTypeAndValueImports, }); // If type ordering is specified explicitly, we need to break apart type and value specifiers if (importOrder.some((group) => group.includes(TYPES_SPECIAL_WORD))) { nodes = explodeTypeAndValueSpecifiers(nodes); } // sort non-side effect nodes const sorted = getSortedNodesByImportOrder(nodes, options); finalNodes.push(...sorted); } } if (finalNodes.length > 0) { finalNodes.push(newLineNode); } // Adjust the comments on the sorted nodes to match the original comments return adjustCommentsOnSortedNodes(nodes, finalNodes, { leadingSeparator: provideGapAfterTopOfFileComments, }); };