UNPKG

nuclide-format-js

Version:

A collection of codemods to help auto format javascript

203 lines (162 loc) 20.4 kB
'use strict'; var _extends = Object.assign || function (target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i]; for (var key in source) { if (Object.prototype.hasOwnProperty.call(source, key)) { target[key] = source[key]; } } } return target; }; var formatCode = function () { var _ref = _asyncToGenerator(regeneratorRuntime.mark(function _callee(sourceOptions) { var parameters = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {}; var editor, options, buffer, inputSource, _transformCodeOrShowE, outputSource, error, position; return regeneratorRuntime.wrap(function _callee$(_context) { while (1) { switch (_context.prev = _context.next) { case 0: editor = parameters.editor || atom.workspace.getActiveTextEditor(); if (editor) { _context.next = 4; break; } // eslint-disable-next-line no-console console.log('- format-js: No active text editor'); return _context.abrupt('return'); case 4: options = dontAddRequiresIfUsedAsService(sourceOptions, parameters); // Save things buffer = editor.getBuffer(); inputSource = buffer.getText(); // Auto-require transform. _transformCodeOrShowE = transformCodeOrShowError(inputSource, options, parameters), outputSource = _transformCodeOrShowE.outputSource, error = _transformCodeOrShowE.error; // Update position if source has a syntax error if (error && atom.config.get('nuclide-format-js.moveCursorToSyntaxError')) { position = syntaxErrorPosition(error); if (position) { editor.setCursorBufferPosition(position); } } // Update the source and position after all transforms are done. Do nothing // if the source did not change at all. if (!(outputSource === inputSource)) { _context.next = 11; break; } return _context.abrupt('return'); case 11: buffer.setTextViaDiff(outputSource); // Save the file if that option is specified. if (atom.config.get('nuclide-format-js.saveAfterRun')) { editor.save(); } case 13: case 'end': return _context.stop(); } } }, _callee, this); })); return function formatCode(_x) { return _ref.apply(this, arguments); }; }(); /* * Copyright (c) 2015-present, Facebook, Inc. * All rights reserved. * * This source code is licensed under the license found in the LICENSE file in * the root directory of this source tree. * * */ /* globals atom */ function _asyncToGenerator(fn) { return function () { var gen = fn.apply(this, arguments); return new Promise(function (resolve, reject) { function step(key, arg) { try { var info = gen[key](arg); var value = info.value; } catch (error) { reject(error); return; } if (info.done) { resolve(value); } else { return Promise.resolve(value).then(function (value) { step("next", value); }, function (err) { step("throw", err); }); } } return step("next"); }); }; } function transformCodeOrShowError(inputSource, options, parameters) { var _require = require('../common'), transform = _require.transform; // TODO: Add a limit so the transform is not run on files over a certain size. var outputSource = void 0; try { outputSource = transform(inputSource, options); } catch (error) { showErrorNotification(error, parameters); return { outputSource: inputSource, error: error }; } dismissNotification(ERROR_TITLE(parameters)); dismissNotification(INFO_TITLE(parameters)); if (outputSource === inputSource && // Do not confirm success if user opted out atom.config.get('nuclide-format-js.confirmNoChangeSuccess')) { if (parameters.missingExports) { showMissingExportsNotification(parameters); } else { showSuccessNotification(parameters); } } return { outputSource: outputSource }; } var ERROR_TITLE = function ERROR_TITLE(parameters) { return notificationTitle(parameters, 'failed'); }; function showErrorNotification(error, parameters) { var title = ERROR_TITLE(parameters); dismissNotification(title); atom.notifications.addError(title, { detail: error.toString(), stack: error.stack, dismissable: true }); } var SUCCESS_TITLE = function SUCCESS_TITLE(parameters) { return notificationTitle(parameters, 'succeeded'); }; var notificationTimeouts = {}; function showSuccessNotification(parameters) { var title = SUCCESS_TITLE(parameters); dismissExistingNotification(title); atom.notifications.addSuccess(title, { detail: 'No changes were needed.', dismissable: true }); timeoutNotification(title); } function timeoutNotification(title) { notificationTimeouts[title] = setTimeout(function () { dismissExistingNotification(title); }, 2000); } function dismissExistingNotification(title) { dismissNotification(title); clearTimeout(notificationTimeouts[title]); } var INFO_TITLE = function INFO_TITLE(parameters) { return notificationTitle(parameters, 'couldn\'t fix all problems'); }; function showMissingExportsNotification(parameters) { var title = INFO_TITLE(parameters); dismissNotification(title); atom.notifications.addInfo(title, { detail: 'Exports for these references couldn\'t be determined. ' + 'Either there are multiple possible export candidates, ' + 'or none exist, or the Language Server or Flow are still ' + 'initializing.', dismissable: true }); } function dismissNotification(title) { atom.notifications.getNotifications().filter(function (notification) { return notification.getMessage() === title; }).forEach(function (notification) { return notification.dismiss(); }); } function notificationTitle(parameters, message) { return (parameters.addedRequires != null ? 'Nuclide JS Imports: Auto Require ' : 'Nuclide Format JS: Fix Requires') + message; } function syntaxErrorPosition(error) { var _ref2 = error.loc || {}, line = _ref2.line, column = _ref2.column; return Number.isInteger(line) && Number.isInteger(column) ? [line - 1, column] : null; } function dontAddRequiresIfUsedAsService(sourceOptions, parameters) { var blacklist = new Set(sourceOptions.blacklist); if (parameters.addedRequires != null) { blacklist.add('requires.addMissingRequires').add('requires.addMissingTypes'); } return _extends({}, sourceOptions, { blacklist: blacklist }); } module.exports = formatCode; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/atom/formatCode.js"],"names":["sourceOptions","parameters","editor","atom","workspace","getActiveTextEditor","console","log","options","dontAddRequiresIfUsedAsService","buffer","getBuffer","inputSource","getText","transformCodeOrShowError","outputSource","error","config","get","position","syntaxErrorPosition","setCursorBufferPosition","setTextViaDiff","save","formatCode","require","transform","showErrorNotification","dismissNotification","ERROR_TITLE","INFO_TITLE","missingExports","showMissingExportsNotification","showSuccessNotification","notificationTitle","title","notifications","addError","detail","toString","stack","dismissable","SUCCESS_TITLE","notificationTimeouts","dismissExistingNotification","addSuccess","timeoutNotification","setTimeout","clearTimeout","addInfo","getNotifications","filter","notification","getMessage","forEach","dismiss","message","addedRequires","loc","line","column","Number","isInteger","blacklist","Set","add","module","exports"],"mappings":"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","file":"formatCode.js","sourcesContent":["/*\n * Copyright (c) 2015-present, Facebook, Inc.\n * All rights reserved.\n *\n * This source code is licensed under the license found in the LICENSE file in\n * the root directory of this source tree.\n *\n * @flow\n */\n\n/* globals atom */\n\nimport type {SourceOptions} from '../common/options/SourceOptions';\n\ntype ErrorWithLocation = {loc?: {line: number, column: number}};\n\ntype ServiceParams = {\n  addedRequires?: boolean,\n  missingExports?: boolean,\n};\n\nasync function formatCode(\n  sourceOptions: SourceOptions,\n  parameters: {\n    addedRequires?: boolean,\n    missingExports?: boolean,\n    editor?: TextEditor,\n  } = {},\n): Promise<void> {\n  const editor = parameters.editor || atom.workspace.getActiveTextEditor();\n  if (!editor) {\n    // eslint-disable-next-line no-console\n    console.log('- format-js: No active text editor');\n    return;\n  }\n\n  const options = dontAddRequiresIfUsedAsService(sourceOptions, parameters);\n\n  // Save things\n  const buffer = editor.getBuffer();\n  const inputSource = buffer.getText();\n\n  // Auto-require transform.\n  const {outputSource, error} = transformCodeOrShowError(\n    inputSource,\n    options,\n    parameters,\n  );\n\n  // Update position if source has a syntax error\n  if (error && atom.config.get('nuclide-format-js.moveCursorToSyntaxError')) {\n    const position = syntaxErrorPosition(error);\n    if (position) {\n      editor.setCursorBufferPosition(position);\n    }\n  }\n\n  // Update the source and position after all transforms are done. Do nothing\n  // if the source did not change at all.\n  if (outputSource === inputSource) {\n    return;\n  }\n\n  buffer.setTextViaDiff(outputSource);\n\n  // Save the file if that option is specified.\n  if (atom.config.get('nuclide-format-js.saveAfterRun')) {\n    editor.save();\n  }\n}\n\n\nfunction transformCodeOrShowError(\n  inputSource: string,\n  options: SourceOptions,\n  parameters: ServiceParams,\n): {outputSource: string, error?: ErrorWithLocation} {\n  const {transform} = require('../common');\n  // TODO: Add a limit so the transform is not run on files over a certain size.\n  let outputSource;\n  try {\n    outputSource = transform(inputSource, options);\n  } catch (error) {\n    showErrorNotification(error, parameters);\n    return {outputSource: inputSource, error};\n  }\n  dismissNotification(ERROR_TITLE(parameters));\n  dismissNotification(INFO_TITLE(parameters));\n  if (\n    outputSource === inputSource &&\n    // Do not confirm success if user opted out\n    atom.config.get('nuclide-format-js.confirmNoChangeSuccess')\n  ) {\n    if (parameters.missingExports) {\n      showMissingExportsNotification(parameters);\n    } else {\n      showSuccessNotification(parameters);\n    }\n  }\n  return {outputSource};\n}\n\nconst ERROR_TITLE = parameters => notificationTitle(parameters, 'failed');\n\nfunction showErrorNotification(error: Error, parameters: ServiceParams): void {\n  const title = ERROR_TITLE(parameters);\n  dismissNotification(title);\n  atom.notifications.addError(title, {\n    detail: error.toString(),\n    stack: error.stack,\n    dismissable: true,\n  });\n}\n\nconst SUCCESS_TITLE = parameters => notificationTitle(parameters, 'succeeded');\n\nconst notificationTimeouts = {};\nfunction showSuccessNotification(parameters: ServiceParams): void {\n  const title = SUCCESS_TITLE(parameters);\n  dismissExistingNotification(title);\n  atom.notifications.addSuccess(title, {\n    detail: 'No changes were needed.',\n    dismissable: true,\n  });\n  timeoutNotification(title);\n}\n\nfunction timeoutNotification(title: string) {\n  notificationTimeouts[title] = setTimeout(() => {\n    dismissExistingNotification(title);\n  }, 2000);\n}\n\nfunction dismissExistingNotification(title: string): void {\n  dismissNotification(title);\n  clearTimeout(notificationTimeouts[title]);\n}\n\nconst INFO_TITLE = parameters =>\n  notificationTitle(parameters, 'couldn\\'t fix all problems');\n\nfunction showMissingExportsNotification(parameters: ServiceParams): void {\n  const title = INFO_TITLE(parameters);\n  dismissNotification(title);\n  atom.notifications.addInfo(title, {\n    detail: 'Exports for these references couldn\\'t be determined. ' +\n      'Either there are multiple possible export candidates, ' +\n      'or none exist, or the Language Server or Flow are still ' +\n      'initializing.',\n    dismissable: true,\n  });\n}\n\nfunction dismissNotification(title: string): void {\n  atom.notifications.getNotifications()\n    .filter(notification => notification.getMessage() === title)\n    .forEach(notification => notification.dismiss());\n}\n\nfunction notificationTitle(parameters: ServiceParams, message: string): string {\n  return (\n    (parameters.addedRequires != null\n      ? 'Nuclide JS Imports: Auto Require '\n      : 'Nuclide Format JS: Fix Requires') +\n    message\n  );\n}\n\nfunction syntaxErrorPosition(error: ErrorWithLocation): ?[number, number] {\n  const {line, column} = error.loc || {};\n  return Number.isInteger(line) && Number.isInteger(column)\n    ? [line - 1, column]\n    : null;\n}\n\nfunction dontAddRequiresIfUsedAsService(\n  sourceOptions: SourceOptions,\n  parameters: {addedRequires?: boolean},\n): SourceOptions {\n  const blacklist = new Set(sourceOptions.blacklist);\n  if (parameters.addedRequires != null) {\n    blacklist\n      .add('requires.addMissingRequires')\n      .add('requires.addMissingTypes');\n  }\n  return {\n    ...sourceOptions,\n    blacklist,\n  };\n}\n\nmodule.exports = formatCode;\n"]}