index.js 2.9 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677
  1. "use strict";
  2. Object.defineProperty(exports, "__esModule", { value: true });
  3. const node_async_hooks_1 = require("node:async_hooks");
  4. const external_editor_1 = require("external-editor");
  5. const core_1 = require("@inquirer/core");
  6. const editorTheme = {
  7. validationFailureMode: 'keep',
  8. };
  9. exports.default = (0, core_1.createPrompt)((config, done) => {
  10. const { waitForUseInput = true, file: { postfix = config.postfix ?? '.txt', ...fileProps } = {}, validate = () => true, } = config;
  11. const theme = (0, core_1.makeTheme)(editorTheme, config.theme);
  12. const [status, setStatus] = (0, core_1.useState)('idle');
  13. const [value = '', setValue] = (0, core_1.useState)(config.default);
  14. const [errorMsg, setError] = (0, core_1.useState)();
  15. const prefix = (0, core_1.usePrefix)({ status, theme });
  16. function startEditor(rl) {
  17. rl.pause();
  18. // Note: The bind call isn't strictly required. But we need it for our mocks to work as expected.
  19. const editCallback = node_async_hooks_1.AsyncResource.bind(async (error, answer) => {
  20. rl.resume();
  21. if (error) {
  22. setError(error.toString());
  23. }
  24. else {
  25. setStatus('loading');
  26. const isValid = await validate(answer);
  27. if (isValid === true) {
  28. setError(undefined);
  29. setStatus('done');
  30. done(answer);
  31. }
  32. else {
  33. if (theme.validationFailureMode === 'clear') {
  34. setValue(config.default);
  35. }
  36. else {
  37. setValue(answer);
  38. }
  39. setError(isValid || 'You must provide a valid value');
  40. setStatus('idle');
  41. }
  42. }
  43. });
  44. (0, external_editor_1.editAsync)(value, (error, answer) => void editCallback(error, answer), {
  45. postfix,
  46. ...fileProps,
  47. });
  48. }
  49. (0, core_1.useEffect)((rl) => {
  50. if (!waitForUseInput) {
  51. startEditor(rl);
  52. }
  53. }, []);
  54. (0, core_1.useKeypress)((key, rl) => {
  55. // Ignore keypress while our prompt is doing other processing.
  56. if (status !== 'idle') {
  57. return;
  58. }
  59. if ((0, core_1.isEnterKey)(key)) {
  60. startEditor(rl);
  61. }
  62. });
  63. const message = theme.style.message(config.message, status);
  64. let helpTip = '';
  65. if (status === 'loading') {
  66. helpTip = theme.style.help('Received');
  67. }
  68. else if (status === 'idle') {
  69. const enterKey = theme.style.key('enter');
  70. helpTip = theme.style.help(`Press ${enterKey} to launch your preferred editor.`);
  71. }
  72. let error = '';
  73. if (errorMsg) {
  74. error = theme.style.error(errorMsg);
  75. }
  76. return [[prefix, message, helpTip].filter(Boolean).join(' '), error];
  77. });