prompts.py 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128
  1. """Terminal input and output prompts."""
  2. from pygments.token import Token
  3. import sys
  4. from IPython.core.displayhook import DisplayHook
  5. from prompt_toolkit.formatted_text import fragment_list_width, PygmentsTokens
  6. from prompt_toolkit.shortcuts import print_formatted_text
  7. from prompt_toolkit.enums import EditingMode
  8. class Prompts(object):
  9. def __init__(self, shell):
  10. self.shell = shell
  11. def vi_mode(self):
  12. if (getattr(self.shell.pt_app, 'editing_mode', None) == EditingMode.VI
  13. and self.shell.prompt_includes_vi_mode):
  14. mode = str(self.shell.pt_app.app.vi_state.input_mode)
  15. if mode.startswith('InputMode.'):
  16. mode = mode[10:13].lower()
  17. elif mode.startswith('vi-'):
  18. mode = mode[3:6]
  19. return '['+mode+'] '
  20. return ''
  21. def current_line(self) -> int:
  22. if self.shell.pt_app is not None:
  23. return self.shell.pt_app.default_buffer.document.cursor_position_row or 0
  24. return 0
  25. def in_prompt_tokens(self):
  26. return [
  27. (Token.Prompt, self.vi_mode()),
  28. (
  29. Token.Prompt,
  30. self.shell.prompt_line_number_format.format(
  31. line=1, rel_line=-self.current_line()
  32. ),
  33. ),
  34. (Token.Prompt, "In ["),
  35. (Token.PromptNum, str(self.shell.execution_count)),
  36. (Token.Prompt, ']: '),
  37. ]
  38. def _width(self):
  39. return fragment_list_width(self.in_prompt_tokens())
  40. def continuation_prompt_tokens(self, width=None, *, lineno=None):
  41. if width is None:
  42. width = self._width()
  43. line = lineno + 1 if lineno is not None else 0
  44. prefix = " " * len(
  45. self.vi_mode()
  46. ) + self.shell.prompt_line_number_format.format(
  47. line=line, rel_line=line - self.current_line() - 1
  48. )
  49. return [
  50. (
  51. Token.Prompt,
  52. prefix + (" " * (width - len(prefix) - 5)) + "...: ",
  53. ),
  54. ]
  55. def rewrite_prompt_tokens(self):
  56. width = self._width()
  57. return [
  58. (Token.Prompt, ('-' * (width - 2)) + '> '),
  59. ]
  60. def out_prompt_tokens(self):
  61. return [
  62. (Token.OutPrompt, 'Out['),
  63. (Token.OutPromptNum, str(self.shell.execution_count)),
  64. (Token.OutPrompt, ']: '),
  65. ]
  66. class ClassicPrompts(Prompts):
  67. def in_prompt_tokens(self):
  68. return [
  69. (Token.Prompt, '>>> '),
  70. ]
  71. def continuation_prompt_tokens(self, width=None):
  72. return [
  73. (Token.Prompt, '... ')
  74. ]
  75. def rewrite_prompt_tokens(self):
  76. return []
  77. def out_prompt_tokens(self):
  78. return []
  79. class RichPromptDisplayHook(DisplayHook):
  80. """Subclass of base display hook using coloured prompt"""
  81. def write_output_prompt(self):
  82. sys.stdout.write(self.shell.separate_out)
  83. # If we're not displaying a prompt, it effectively ends with a newline,
  84. # because the output will be left-aligned.
  85. self.prompt_end_newline = True
  86. if self.do_full_cache:
  87. tokens = self.shell.prompts.out_prompt_tokens()
  88. prompt_txt = "".join(s for _, s in tokens)
  89. if prompt_txt and not prompt_txt.endswith("\n"):
  90. # Ask for a newline before multiline output
  91. self.prompt_end_newline = False
  92. if self.shell.pt_app:
  93. print_formatted_text(PygmentsTokens(tokens),
  94. style=self.shell.pt_app.app.style, end='',
  95. )
  96. else:
  97. sys.stdout.write(prompt_txt)
  98. def write_format_data(self, format_dict, md_dict=None) -> None:
  99. assert self.shell is not None
  100. if self.shell.mime_renderers:
  101. for mime, handler in self.shell.mime_renderers.items():
  102. if mime in format_dict:
  103. handler(format_dict[mime], None)
  104. return
  105. super().write_format_data(format_dict, md_dict)