'use strict';var _docsUrl = require('../docsUrl');var _docsUrl2 = _interopRequireDefault(_docsUrl);function _interopRequireDefault(obj) {return obj && obj.__esModule ? obj : { 'default': obj };} function isComma(token) { return token.type === 'Punctuator' && token.value === ','; } function removeSpecifiers(fixes, fixer, sourceCode, specifiers) {var _iteratorNormalCompletion = true;var _didIteratorError = false;var _iteratorError = undefined;try { for (var _iterator = specifiers[Symbol.iterator](), _step; !(_iteratorNormalCompletion = (_step = _iterator.next()).done); _iteratorNormalCompletion = true) {var specifier = _step.value; // remove the trailing comma var token = sourceCode.getTokenAfter(specifier); if (token && isComma(token)) { fixes.push(fixer.remove(token)); } fixes.push(fixer.remove(specifier)); }} catch (err) {_didIteratorError = true;_iteratorError = err;} finally {try {if (!_iteratorNormalCompletion && _iterator['return']) {_iterator['return']();}} finally {if (_didIteratorError) {throw _iteratorError;}}} } function getImportText( node, sourceCode, specifiers, kind) { var sourceString = sourceCode.getText(node.source); if (specifiers.length === 0) { return ''; } var names = specifiers.map(function (s) { if (s.imported.name === s.local.name) { return s.imported.name; } return String(s.imported.name) + ' as ' + String(s.local.name); }); // insert a fresh top-level import return 'import ' + String(kind) + ' {' + String(names.join(', ')) + '} from ' + String(sourceString) + ';'; } module.exports = { meta: { type: 'suggestion', docs: { category: 'Style guide', description: 'Enforce or ban the use of inline type-only markers for named imports.', url: (0, _docsUrl2['default'])('consistent-type-specifier-style') }, fixable: 'code', schema: [ { type: 'string', 'enum': ['prefer-inline', 'prefer-top-level'], 'default': 'prefer-inline' }] }, create: function () {function create(context) { var sourceCode = context.getSourceCode(); if (context.options[0] === 'prefer-inline') { return { ImportDeclaration: function () {function ImportDeclaration(node) { if (node.importKind === 'value' || node.importKind == null) { // top-level value / unknown is valid return; } if ( // no specifiers (import type {} from '') have no specifiers to mark as inline node.specifiers.length === 0 || node.specifiers.length === 1 // default imports are both "inline" and "top-level" && ( node.specifiers[0].type === 'ImportDefaultSpecifier' // namespace imports are both "inline" and "top-level" || node.specifiers[0].type === 'ImportNamespaceSpecifier')) { return; } context.report({ node: node, message: 'Prefer using inline {{kind}} specifiers instead of a top-level {{kind}}-only import.', data: { kind: node.importKind }, fix: function () {function fix(fixer) { var kindToken = sourceCode.getFirstToken(node, { skip: 1 }); return [].concat( kindToken ? fixer.remove(kindToken) : [], node.specifiers.map(function (specifier) {return fixer.insertTextBefore(specifier, String(node.importKind) + ' ');})); }return fix;}() }); }return ImportDeclaration;}() }; } // prefer-top-level return { ImportDeclaration: function () {function ImportDeclaration(node) { if ( // already top-level is valid node.importKind === 'type' || node.importKind === 'typeof' // no specifiers (import {} from '') cannot have inline - so is valid || node.specifiers.length === 0 || node.specifiers.length === 1 // default imports are both "inline" and "top-level" && ( node.specifiers[0].type === 'ImportDefaultSpecifier' // namespace imports are both "inline" and "top-level" || node.specifiers[0].type === 'ImportNamespaceSpecifier')) { return; } var typeSpecifiers = []; var typeofSpecifiers = []; var valueSpecifiers = []; var defaultSpecifier = null;var _iteratorNormalCompletion2 = true;var _didIteratorError2 = false;var _iteratorError2 = undefined;try { for (var _iterator2 = node.specifiers[Symbol.iterator](), _step2; !(_iteratorNormalCompletion2 = (_step2 = _iterator2.next()).done); _iteratorNormalCompletion2 = true) {var specifier = _step2.value; if (specifier.type === 'ImportDefaultSpecifier') { defaultSpecifier = specifier; continue; } if (specifier.importKind === 'type') { typeSpecifiers.push(specifier); } else if (specifier.importKind === 'typeof') { typeofSpecifiers.push(specifier); } else if (specifier.importKind === 'value' || specifier.importKind == null) { valueSpecifiers.push(specifier); } }} catch (err) {_didIteratorError2 = true;_iteratorError2 = err;} finally {try {if (!_iteratorNormalCompletion2 && _iterator2['return']) {_iterator2['return']();}} finally {if (_didIteratorError2) {throw _iteratorError2;}}} var typeImport = getImportText(node, sourceCode, typeSpecifiers, 'type'); var typeofImport = getImportText(node, sourceCode, typeofSpecifiers, 'typeof'); var newImports = (String(typeImport) + '\n' + String(typeofImport)).trim(); if (typeSpecifiers.length + typeofSpecifiers.length === node.specifiers.length) { // all specifiers have inline specifiers - so we replace the entire import var kind = [].concat( typeSpecifiers.length > 0 ? 'type' : [], typeofSpecifiers.length > 0 ? 'typeof' : []); context.report({ node: node, message: 'Prefer using a top-level {{kind}}-only import instead of inline {{kind}} specifiers.', data: { kind: kind.join('/') }, fix: function () {function fix(fixer) { return fixer.replaceText(node, newImports); }return fix;}() }); } else { // remove specific specifiers and insert new imports for them var _iteratorNormalCompletion3 = true;var _didIteratorError3 = false;var _iteratorError3 = undefined;try {for (var _iterator3 = typeSpecifiers.concat(typeofSpecifiers)[Symbol.iterator](), _step3; !(_iteratorNormalCompletion3 = (_step3 = _iterator3.next()).done); _iteratorNormalCompletion3 = true) {var _specifier = _step3.value; context.report({ node: _specifier, message: 'Prefer using a top-level {{kind}}-only import instead of inline {{kind}} specifiers.', data: { kind: _specifier.importKind }, fix: function () {function fix(fixer) { var fixes = []; // if there are no value specifiers, then the other report fixer will be called, not this one if (valueSpecifiers.length > 0) { // import { Value, type Type } from 'mod'; // we can just remove the type specifiers removeSpecifiers(fixes, fixer, sourceCode, typeSpecifiers); removeSpecifiers(fixes, fixer, sourceCode, typeofSpecifiers); // make the import nicely formatted by also removing the trailing comma after the last value import // eg // import { Value, type Type } from 'mod'; // to // import { Value } from 'mod'; // not // import { Value, } from 'mod'; var maybeComma = sourceCode.getTokenAfter(valueSpecifiers[valueSpecifiers.length - 1]); if (isComma(maybeComma)) { fixes.push(fixer.remove(maybeComma)); } } else if (defaultSpecifier) { // import Default, { type Type } from 'mod'; // remove the entire curly block so we don't leave an empty one behind // NOTE - the default specifier *must* be the first specifier always! // so a comma exists that we also have to clean up or else it's bad syntax var comma = sourceCode.getTokenAfter(defaultSpecifier, isComma); var closingBrace = sourceCode.getTokenAfter( node.specifiers[node.specifiers.length - 1], function (token) {return token.type === 'Punctuator' && token.value === '}';}); fixes.push(fixer.removeRange([ comma.range[0], closingBrace.range[1]])); } return fixes.concat( // insert the new imports after the old declaration fixer.insertTextAfter(node, '\n' + String(newImports))); }return fix;}() }); }} catch (err) {_didIteratorError3 = true;_iteratorError3 = err;} finally {try {if (!_iteratorNormalCompletion3 && _iterator3['return']) {_iterator3['return']();}} finally {if (_didIteratorError3) {throw _iteratorError3;}}} } }return ImportDeclaration;}() }; }return create;}() }; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/rules/consistent-type-specifier-style.js"],"names":["isComma","token","type","value","removeSpecifiers","fixes","fixer","sourceCode","specifiers","specifier","getTokenAfter","push","remove","getImportText","node","kind","sourceString","getText","source","length","names","map","s","imported","name","local","join","module","exports","meta","docs","category","description","url","fixable","schema","create","context","getSourceCode","options","ImportDeclaration","importKind","report","message","data","fix","kindToken","getFirstToken","skip","concat","insertTextBefore","typeSpecifiers","typeofSpecifiers","valueSpecifiers","defaultSpecifier","typeImport","typeofImport","newImports","trim","replaceText","maybeComma","comma","closingBrace","removeRange","range","insertTextAfter"],"mappings":"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","file":"consistent-type-specifier-style.js","sourcesContent":["import docsUrl from '../docsUrl';\n\nfunction isComma(token) {\n  return token.type === 'Punctuator' && token.value === ',';\n}\n\nfunction removeSpecifiers(fixes, fixer, sourceCode, specifiers) {\n  for (const specifier of specifiers) {\n    // remove the trailing comma\n    const token = sourceCode.getTokenAfter(specifier);\n    if (token && isComma(token)) {\n      fixes.push(fixer.remove(token));\n    }\n    fixes.push(fixer.remove(specifier));\n  }\n}\n\nfunction getImportText(\n  node,\n  sourceCode,\n  specifiers,\n  kind,\n) {\n  const sourceString = sourceCode.getText(node.source);\n  if (specifiers.length === 0) {\n    return '';\n  }\n\n  const names = specifiers.map((s) => {\n    if (s.imported.name === s.local.name) {\n      return s.imported.name;\n    }\n    return `${s.imported.name} as ${s.local.name}`;\n  });\n  // insert a fresh top-level import\n  return `import ${kind} {${names.join(', ')}} from ${sourceString};`;\n}\n\nmodule.exports = {\n  meta: {\n    type: 'suggestion',\n    docs: {\n      category: 'Style guide',\n      description: 'Enforce or ban the use of inline type-only markers for named imports.',\n      url: docsUrl('consistent-type-specifier-style'),\n    },\n    fixable: 'code',\n    schema: [\n      {\n        type: 'string',\n        enum: ['prefer-inline', 'prefer-top-level'],\n        default: 'prefer-inline',\n      },\n    ],\n  },\n\n  create(context) {\n    const sourceCode = context.getSourceCode();\n\n    if (context.options[0] === 'prefer-inline') {\n      return {\n        ImportDeclaration(node) {\n          if (node.importKind === 'value' || node.importKind == null) {\n            // top-level value / unknown is valid\n            return;\n          }\n\n          if (\n            // no specifiers (import type {} from '') have no specifiers to mark as inline\n            node.specifiers.length === 0\n            || node.specifiers.length === 1\n            // default imports are both \"inline\" and \"top-level\"\n            && (\n              node.specifiers[0].type === 'ImportDefaultSpecifier'\n              // namespace imports are both \"inline\" and \"top-level\"\n              || node.specifiers[0].type === 'ImportNamespaceSpecifier'\n            )\n          ) {\n            return;\n          }\n\n          context.report({\n            node,\n            message: 'Prefer using inline {{kind}} specifiers instead of a top-level {{kind}}-only import.',\n            data: {\n              kind: node.importKind,\n            },\n            fix(fixer) {\n              const kindToken = sourceCode.getFirstToken(node, { skip: 1 });\n\n              return [].concat(\n                kindToken ? fixer.remove(kindToken) : [],\n                node.specifiers.map((specifier) => fixer.insertTextBefore(specifier, `${node.importKind} `)),\n              );\n            },\n          });\n        },\n      };\n    }\n\n    // prefer-top-level\n    return {\n      ImportDeclaration(node) {\n        if (\n          // already top-level is valid\n          node.importKind === 'type'\n          || node.importKind === 'typeof'\n          // no specifiers (import {} from '') cannot have inline - so is valid\n          || node.specifiers.length === 0\n          || node.specifiers.length === 1\n          // default imports are both \"inline\" and \"top-level\"\n          && (\n            node.specifiers[0].type === 'ImportDefaultSpecifier'\n            // namespace imports are both \"inline\" and \"top-level\"\n            || node.specifiers[0].type === 'ImportNamespaceSpecifier'\n          )\n        ) {\n          return;\n        }\n\n        const typeSpecifiers = [];\n        const typeofSpecifiers = [];\n        const valueSpecifiers = [];\n        let defaultSpecifier = null;\n        for (const specifier of node.specifiers) {\n          if (specifier.type === 'ImportDefaultSpecifier') {\n            defaultSpecifier = specifier;\n            continue;\n          }\n\n          if (specifier.importKind === 'type') {\n            typeSpecifiers.push(specifier);\n          } else if (specifier.importKind === 'typeof') {\n            typeofSpecifiers.push(specifier);\n          } else if (specifier.importKind === 'value' || specifier.importKind == null) {\n            valueSpecifiers.push(specifier);\n          }\n        }\n\n        const typeImport = getImportText(node, sourceCode, typeSpecifiers, 'type');\n        const typeofImport = getImportText(node, sourceCode, typeofSpecifiers, 'typeof');\n        const newImports = `${typeImport}\\n${typeofImport}`.trim();\n\n        if (typeSpecifiers.length + typeofSpecifiers.length === node.specifiers.length) {\n          // all specifiers have inline specifiers - so we replace the entire import\n          const kind = [].concat(\n            typeSpecifiers.length > 0 ? 'type' : [],\n            typeofSpecifiers.length > 0 ? 'typeof' : [],\n          );\n\n          context.report({\n            node,\n            message: 'Prefer using a top-level {{kind}}-only import instead of inline {{kind}} specifiers.',\n            data: {\n              kind: kind.join('/'),\n            },\n            fix(fixer) {\n              return fixer.replaceText(node, newImports);\n            },\n          });\n        } else {\n          // remove specific specifiers and insert new imports for them\n          for (const specifier of typeSpecifiers.concat(typeofSpecifiers)) {\n            context.report({\n              node: specifier,\n              message: 'Prefer using a top-level {{kind}}-only import instead of inline {{kind}} specifiers.',\n              data: {\n                kind: specifier.importKind,\n              },\n              fix(fixer) {\n                const fixes = [];\n\n                // if there are no value specifiers, then the other report fixer will be called, not this one\n\n                if (valueSpecifiers.length > 0) {\n                  // import { Value, type Type } from 'mod';\n\n                  // we can just remove the type specifiers\n                  removeSpecifiers(fixes, fixer, sourceCode, typeSpecifiers);\n                  removeSpecifiers(fixes, fixer, sourceCode, typeofSpecifiers);\n\n                  // make the import nicely formatted by also removing the trailing comma after the last value import\n                  // eg\n                  // import { Value, type Type } from 'mod';\n                  // to\n                  // import { Value  } from 'mod';\n                  // not\n                  // import { Value,  } from 'mod';\n                  const maybeComma = sourceCode.getTokenAfter(valueSpecifiers[valueSpecifiers.length - 1]);\n                  if (isComma(maybeComma)) {\n                    fixes.push(fixer.remove(maybeComma));\n                  }\n                } else if (defaultSpecifier) {\n                  // import Default, { type Type } from 'mod';\n\n                  // remove the entire curly block so we don't leave an empty one behind\n                  // NOTE - the default specifier *must* be the first specifier always!\n                  //        so a comma exists that we also have to clean up or else it's bad syntax\n                  const comma = sourceCode.getTokenAfter(defaultSpecifier, isComma);\n                  const closingBrace = sourceCode.getTokenAfter(\n                    node.specifiers[node.specifiers.length - 1],\n                    (token) => token.type === 'Punctuator' && token.value === '}',\n                  );\n                  fixes.push(fixer.removeRange([\n                    comma.range[0],\n                    closingBrace.range[1],\n                  ]));\n                }\n\n                return fixes.concat(\n                  // insert the new imports after the old declaration\n                  fixer.insertTextAfter(node, `\\n${newImports}`),\n                );\n              },\n            });\n          }\n        }\n      },\n    };\n  },\n};\n"]}