mirror of
https://github.com/python/cpython.git
synced 2025-10-09 16:34:44 +00:00
gh-95191: IDLE: Include prompts when saving Shell #95554
This commit is contained in:
parent
d2c1a9c76c
commit
b85411fc5e
5 changed files with 43 additions and 24 deletions
|
@ -4,6 +4,8 @@ Released on 2022-10-03
|
||||||
=========================
|
=========================
|
||||||
|
|
||||||
|
|
||||||
|
gh-95191: Include prompts when saving Shell (interactive input/output).
|
||||||
|
|
||||||
gh-95511: Fix the Shell context menu copy-with-prompts bug of copying
|
gh-95511: Fix the Shell context menu copy-with-prompts bug of copying
|
||||||
an extra line when one selects whole lines.
|
an extra line when one selects whole lines.
|
||||||
|
|
||||||
|
|
|
@ -1,10 +1,12 @@
|
||||||
"Test , coverage 17%."
|
"Test , coverage 17%."
|
||||||
|
|
||||||
from idlelib import iomenu, util
|
from idlelib import iomenu
|
||||||
import unittest
|
import unittest
|
||||||
from test.support import requires
|
from test.support import requires
|
||||||
from tkinter import Tk
|
from tkinter import Tk
|
||||||
from idlelib.editor import EditorWindow
|
from idlelib.editor import EditorWindow
|
||||||
|
from idlelib import util
|
||||||
|
from idlelib.idle_test.mock_idle import Func
|
||||||
|
|
||||||
|
|
||||||
class IOBindingTest(unittest.TestCase):
|
class IOBindingTest(unittest.TestCase):
|
||||||
|
@ -36,9 +38,14 @@ class IOBindingTest(unittest.TestCase):
|
||||||
io = self.io
|
io = self.io
|
||||||
fix = io.fixnewlines
|
fix = io.fixnewlines
|
||||||
text = io.editwin.text
|
text = io.editwin.text
|
||||||
|
|
||||||
|
# Make the editor temporarily look like Shell.
|
||||||
self.editwin.interp = None
|
self.editwin.interp = None
|
||||||
eq(fix(), '')
|
shelltext = '>>> if 1'
|
||||||
del self.editwin.interp
|
self.editwin.get_prompt_text = Func(result=shelltext)
|
||||||
|
eq(fix(), shelltext) # Get... call and '\n' not added.
|
||||||
|
del self.editwin.interp, self.editwin.get_prompt_text
|
||||||
|
|
||||||
text.insert(1.0, 'a')
|
text.insert(1.0, 'a')
|
||||||
eq(fix(), 'a'+io.eol_convention)
|
eq(fix(), 'a'+io.eol_convention)
|
||||||
eq(text.get('1.0', 'end-1c'), 'a\n')
|
eq(text.get('1.0', 'end-1c'), 'a\n')
|
||||||
|
|
|
@ -251,11 +251,17 @@ class IOBinding:
|
||||||
return False
|
return False
|
||||||
|
|
||||||
def fixnewlines(self):
|
def fixnewlines(self):
|
||||||
"Return text with final \n if needed and os eols."
|
"""Return text with os eols.
|
||||||
if (self.text.get("end-2c") != '\n'
|
|
||||||
and not hasattr(self.editwin, "interp")): # Not shell.
|
Add prompts if shell else final \n if missing.
|
||||||
self.text.insert("end-1c", "\n")
|
"""
|
||||||
text = self.text.get("1.0", "end-1c")
|
|
||||||
|
if hasattr(self.editwin, "interp"): # Saving shell.
|
||||||
|
text = self.editwin.get_prompt_text('1.0', self.text.index('end-1c'))
|
||||||
|
else:
|
||||||
|
if self.text.get("end-2c") != '\n':
|
||||||
|
self.text.insert("end-1c", "\n") # Changes 'end-1c' value.
|
||||||
|
text = self.text.get('1.0', "end-1c")
|
||||||
if self.eol_convention != "\n":
|
if self.eol_convention != "\n":
|
||||||
text = text.replace("\n", self.eol_convention)
|
text = text.replace("\n", self.eol_convention)
|
||||||
return text
|
return text
|
||||||
|
|
|
@ -986,6 +986,23 @@ class PyShell(OutputWindow):
|
||||||
def get_standard_extension_names(self):
|
def get_standard_extension_names(self):
|
||||||
return idleConf.GetExtensions(shell_only=True)
|
return idleConf.GetExtensions(shell_only=True)
|
||||||
|
|
||||||
|
def get_prompt_text(self, first, last):
|
||||||
|
"""Return text between first and last with prompts added."""
|
||||||
|
text = self.text.get(first, last)
|
||||||
|
lineno_range = range(
|
||||||
|
int(float(first)),
|
||||||
|
int(float(last))
|
||||||
|
)
|
||||||
|
prompts = [
|
||||||
|
self.shell_sidebar.line_prompts.get(lineno)
|
||||||
|
for lineno in lineno_range
|
||||||
|
]
|
||||||
|
return "\n".join(
|
||||||
|
line if prompt is None else f"{prompt} {line}"
|
||||||
|
for prompt, line in zip(prompts, text.splitlines())
|
||||||
|
) + "\n"
|
||||||
|
|
||||||
|
|
||||||
def copy_with_prompts_callback(self, event=None):
|
def copy_with_prompts_callback(self, event=None):
|
||||||
"""Copy selected lines to the clipboard, with prompts.
|
"""Copy selected lines to the clipboard, with prompts.
|
||||||
|
|
||||||
|
@ -1002,23 +1019,9 @@ class PyShell(OutputWindow):
|
||||||
sellast = text.index('sel.last')
|
sellast = text.index('sel.last')
|
||||||
if sellast[-1] != '0':
|
if sellast[-1] != '0':
|
||||||
sellast = text.index("sel.last+1line linestart")
|
sellast = text.index("sel.last+1line linestart")
|
||||||
|
|
||||||
selected_text = self.text.get(selfirst, sellast)
|
|
||||||
selection_lineno_range = range(
|
|
||||||
int(float(selfirst)),
|
|
||||||
int(float(sellast))
|
|
||||||
)
|
|
||||||
prompts = [
|
|
||||||
self.shell_sidebar.line_prompts.get(lineno)
|
|
||||||
for lineno in selection_lineno_range
|
|
||||||
]
|
|
||||||
selected_text_with_prompts = "\n".join(
|
|
||||||
line if prompt is None else f"{prompt} {line}"
|
|
||||||
for prompt, line in zip(prompts, selected_text.splitlines())
|
|
||||||
) + "\n"
|
|
||||||
|
|
||||||
text.clipboard_clear()
|
text.clipboard_clear()
|
||||||
text.clipboard_append(selected_text_with_prompts)
|
prompt_text = self.get_prompt_text(selfirst, sellast)
|
||||||
|
text.clipboard_append(prompt_text)
|
||||||
|
|
||||||
reading = False
|
reading = False
|
||||||
executing = False
|
executing = False
|
||||||
|
|
|
@ -0,0 +1 @@
|
||||||
|
Include prompts when saving Shell (interactive input and output).
|
Loading…
Add table
Add a link
Reference in a new issue