2019-07-17 10:44:44 -03:00
|
|
|
"Test format, coverage 99%."
|
2018-06-18 05:47:59 -03:00
|
|
|
|
2019-07-17 10:44:44 -03:00
|
|
|
from idlelib import format as ft
|
2013-08-10 17:56:28 -03:00
|
|
|
import unittest
|
2019-07-17 10:44:44 -03:00
|
|
|
from unittest import mock
|
2013-08-10 17:56:28 -03:00
|
|
|
from test.support import requires
|
2018-06-18 05:47:59 -03:00
|
|
|
from tkinter import Tk, Text
|
|
|
|
from idlelib.editor import EditorWindow
|
2019-07-17 21:48:36 -03:00
|
|
|
from idlelib.idle_test.mock_idle import Editor as MockEditor
|
2013-08-10 17:56:28 -03:00
|
|
|
|
|
|
|
|
|
|
|
class Is_Get_Test(unittest.TestCase):
|
|
|
|
"""Test the is_ and get_ functions"""
|
|
|
|
test_comment = '# This is a comment'
|
|
|
|
test_nocomment = 'This is not a comment'
|
|
|
|
trailingws_comment = '# This is a comment '
|
|
|
|
leadingws_comment = ' # This is a comment'
|
|
|
|
leadingws_nocomment = ' This is not a comment'
|
|
|
|
|
|
|
|
def test_is_all_white(self):
|
2019-07-17 10:44:44 -03:00
|
|
|
self.assertTrue(ft.is_all_white(''))
|
|
|
|
self.assertTrue(ft.is_all_white('\t\n\r\f\v'))
|
|
|
|
self.assertFalse(ft.is_all_white(self.test_comment))
|
2013-08-10 17:56:28 -03:00
|
|
|
|
|
|
|
def test_get_indent(self):
|
|
|
|
Equal = self.assertEqual
|
2019-07-17 10:44:44 -03:00
|
|
|
Equal(ft.get_indent(self.test_comment), '')
|
|
|
|
Equal(ft.get_indent(self.trailingws_comment), '')
|
|
|
|
Equal(ft.get_indent(self.leadingws_comment), ' ')
|
|
|
|
Equal(ft.get_indent(self.leadingws_nocomment), ' ')
|
2013-08-10 17:56:28 -03:00
|
|
|
|
|
|
|
def test_get_comment_header(self):
|
|
|
|
Equal = self.assertEqual
|
|
|
|
# Test comment strings
|
2019-07-17 10:44:44 -03:00
|
|
|
Equal(ft.get_comment_header(self.test_comment), '#')
|
|
|
|
Equal(ft.get_comment_header(self.trailingws_comment), '#')
|
|
|
|
Equal(ft.get_comment_header(self.leadingws_comment), ' #')
|
2013-08-10 17:56:28 -03:00
|
|
|
# Test non-comment strings
|
2019-07-17 10:44:44 -03:00
|
|
|
Equal(ft.get_comment_header(self.leadingws_nocomment), ' ')
|
|
|
|
Equal(ft.get_comment_header(self.test_nocomment), '')
|
2013-08-10 17:56:28 -03:00
|
|
|
|
|
|
|
|
|
|
|
class FindTest(unittest.TestCase):
|
2016-05-28 14:22:31 -03:00
|
|
|
"""Test the find_paragraph function in paragraph module.
|
2013-08-10 17:56:28 -03:00
|
|
|
|
|
|
|
Using the runcase() function, find_paragraph() is called with 'mark' set at
|
|
|
|
multiple indexes before and inside the test paragraph.
|
|
|
|
|
|
|
|
It appears that code with the same indentation as a quoted string is grouped
|
|
|
|
as part of the same paragraph, which is probably incorrect behavior.
|
|
|
|
"""
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def setUpClass(cls):
|
|
|
|
from idlelib.idle_test.mock_tk import Text
|
|
|
|
cls.text = Text()
|
|
|
|
|
|
|
|
def runcase(self, inserttext, stopline, expected):
|
|
|
|
# Check that find_paragraph returns the expected paragraph when
|
|
|
|
# the mark index is set to beginning, middle, end of each line
|
|
|
|
# up to but not including the stop line
|
|
|
|
text = self.text
|
|
|
|
text.insert('1.0', inserttext)
|
|
|
|
for line in range(1, stopline):
|
|
|
|
linelength = int(text.index("%d.end" % line).split('.')[1])
|
|
|
|
for col in (0, linelength//2, linelength):
|
|
|
|
tempindex = "%d.%d" % (line, col)
|
2019-07-17 10:44:44 -03:00
|
|
|
self.assertEqual(ft.find_paragraph(text, tempindex), expected)
|
2013-08-10 17:56:28 -03:00
|
|
|
text.delete('1.0', 'end')
|
|
|
|
|
|
|
|
def test_find_comment(self):
|
|
|
|
comment = (
|
|
|
|
"# Comment block with no blank lines before\n"
|
|
|
|
"# Comment line\n"
|
|
|
|
"\n")
|
|
|
|
self.runcase(comment, 3, ('1.0', '3.0', '#', comment[0:58]))
|
|
|
|
|
|
|
|
comment = (
|
|
|
|
"\n"
|
|
|
|
"# Comment block with whitespace line before and after\n"
|
|
|
|
"# Comment line\n"
|
|
|
|
"\n")
|
|
|
|
self.runcase(comment, 4, ('2.0', '4.0', '#', comment[1:70]))
|
|
|
|
|
|
|
|
comment = (
|
|
|
|
"\n"
|
|
|
|
" # Indented comment block with whitespace before and after\n"
|
|
|
|
" # Comment line\n"
|
|
|
|
"\n")
|
|
|
|
self.runcase(comment, 4, ('2.0', '4.0', ' #', comment[1:82]))
|
|
|
|
|
|
|
|
comment = (
|
|
|
|
"\n"
|
|
|
|
"# Single line comment\n"
|
|
|
|
"\n")
|
|
|
|
self.runcase(comment, 3, ('2.0', '3.0', '#', comment[1:23]))
|
|
|
|
|
|
|
|
comment = (
|
|
|
|
"\n"
|
|
|
|
" # Single line comment with leading whitespace\n"
|
|
|
|
"\n")
|
|
|
|
self.runcase(comment, 3, ('2.0', '3.0', ' #', comment[1:51]))
|
|
|
|
|
|
|
|
comment = (
|
|
|
|
"\n"
|
|
|
|
"# Comment immediately followed by code\n"
|
|
|
|
"x = 42\n"
|
|
|
|
"\n")
|
|
|
|
self.runcase(comment, 3, ('2.0', '3.0', '#', comment[1:40]))
|
|
|
|
|
|
|
|
comment = (
|
|
|
|
"\n"
|
|
|
|
" # Indented comment immediately followed by code\n"
|
|
|
|
"x = 42\n"
|
|
|
|
"\n")
|
|
|
|
self.runcase(comment, 3, ('2.0', '3.0', ' #', comment[1:53]))
|
|
|
|
|
|
|
|
comment = (
|
|
|
|
"\n"
|
|
|
|
"# Comment immediately followed by indented code\n"
|
|
|
|
" x = 42\n"
|
|
|
|
"\n")
|
|
|
|
self.runcase(comment, 3, ('2.0', '3.0', '#', comment[1:49]))
|
|
|
|
|
|
|
|
def test_find_paragraph(self):
|
|
|
|
teststring = (
|
|
|
|
'"""String with no blank lines before\n'
|
|
|
|
'String line\n'
|
|
|
|
'"""\n'
|
|
|
|
'\n')
|
|
|
|
self.runcase(teststring, 4, ('1.0', '4.0', '', teststring[0:53]))
|
|
|
|
|
|
|
|
teststring = (
|
|
|
|
"\n"
|
|
|
|
'"""String with whitespace line before and after\n'
|
|
|
|
'String line.\n'
|
|
|
|
'"""\n'
|
|
|
|
'\n')
|
|
|
|
self.runcase(teststring, 5, ('2.0', '5.0', '', teststring[1:66]))
|
|
|
|
|
|
|
|
teststring = (
|
|
|
|
'\n'
|
|
|
|
' """Indented string with whitespace before and after\n'
|
|
|
|
' Comment string.\n'
|
|
|
|
' """\n'
|
|
|
|
'\n')
|
|
|
|
self.runcase(teststring, 5, ('2.0', '5.0', ' ', teststring[1:85]))
|
|
|
|
|
|
|
|
teststring = (
|
|
|
|
'\n'
|
|
|
|
'"""Single line string."""\n'
|
|
|
|
'\n')
|
|
|
|
self.runcase(teststring, 3, ('2.0', '3.0', '', teststring[1:27]))
|
|
|
|
|
|
|
|
teststring = (
|
|
|
|
'\n'
|
|
|
|
' """Single line string with leading whitespace."""\n'
|
|
|
|
'\n')
|
|
|
|
self.runcase(teststring, 3, ('2.0', '3.0', ' ', teststring[1:55]))
|
|
|
|
|
|
|
|
|
|
|
|
class ReformatFunctionTest(unittest.TestCase):
|
|
|
|
"""Test the reformat_paragraph function without the editor window."""
|
|
|
|
|
2016-08-30 14:47:49 -03:00
|
|
|
def test_reformat_paragraph(self):
|
2013-08-10 17:56:28 -03:00
|
|
|
Equal = self.assertEqual
|
2019-07-17 10:44:44 -03:00
|
|
|
reform = ft.reformat_paragraph
|
2013-08-10 17:56:28 -03:00
|
|
|
hw = "O hello world"
|
|
|
|
Equal(reform(' ', 1), ' ')
|
|
|
|
Equal(reform("Hello world", 20), "Hello world")
|
|
|
|
|
|
|
|
# Test without leading newline
|
|
|
|
Equal(reform(hw, 1), "O\nhello\nworld")
|
|
|
|
Equal(reform(hw, 6), "O\nhello\nworld")
|
|
|
|
Equal(reform(hw, 7), "O hello\nworld")
|
|
|
|
Equal(reform(hw, 12), "O hello\nworld")
|
|
|
|
Equal(reform(hw, 13), "O hello world")
|
|
|
|
|
|
|
|
# Test with leading newline
|
|
|
|
hw = "\nO hello world"
|
|
|
|
Equal(reform(hw, 1), "\nO\nhello\nworld")
|
|
|
|
Equal(reform(hw, 6), "\nO\nhello\nworld")
|
|
|
|
Equal(reform(hw, 7), "\nO hello\nworld")
|
|
|
|
Equal(reform(hw, 12), "\nO hello\nworld")
|
|
|
|
Equal(reform(hw, 13), "\nO hello world")
|
|
|
|
|
|
|
|
|
|
|
|
class ReformatCommentTest(unittest.TestCase):
|
|
|
|
"""Test the reformat_comment function without the editor window."""
|
|
|
|
|
|
|
|
def test_reformat_comment(self):
|
|
|
|
Equal = self.assertEqual
|
|
|
|
|
|
|
|
# reformat_comment formats to a minimum of 20 characters
|
|
|
|
test_string = (
|
|
|
|
" \"\"\"this is a test of a reformat for a triple quoted string"
|
|
|
|
" will it reformat to less than 70 characters for me?\"\"\"")
|
2019-07-17 10:44:44 -03:00
|
|
|
result = ft.reformat_comment(test_string, 70, " ")
|
2013-08-10 17:56:28 -03:00
|
|
|
expected = (
|
|
|
|
" \"\"\"this is a test of a reformat for a triple quoted string will it\n"
|
|
|
|
" reformat to less than 70 characters for me?\"\"\"")
|
|
|
|
Equal(result, expected)
|
|
|
|
|
|
|
|
test_comment = (
|
|
|
|
"# this is a test of a reformat for a triple quoted string will "
|
|
|
|
"it reformat to less than 70 characters for me?")
|
2019-07-17 10:44:44 -03:00
|
|
|
result = ft.reformat_comment(test_comment, 70, "#")
|
2013-08-10 17:56:28 -03:00
|
|
|
expected = (
|
|
|
|
"# this is a test of a reformat for a triple quoted string will it\n"
|
|
|
|
"# reformat to less than 70 characters for me?")
|
|
|
|
Equal(result, expected)
|
|
|
|
|
|
|
|
|
|
|
|
class FormatClassTest(unittest.TestCase):
|
|
|
|
def test_init_close(self):
|
2019-07-17 10:44:44 -03:00
|
|
|
instance = ft.FormatParagraph('editor')
|
2013-08-10 17:56:28 -03:00
|
|
|
self.assertEqual(instance.editwin, 'editor')
|
|
|
|
instance.close()
|
|
|
|
self.assertEqual(instance.editwin, None)
|
|
|
|
|
|
|
|
|
|
|
|
# For testing format_paragraph_event, Initialize FormatParagraph with
|
|
|
|
# a mock Editor with .text and .get_selection_indices. The text must
|
|
|
|
# be a Text wrapper that adds two methods
|
|
|
|
|
|
|
|
# A real EditorWindow creates unneeded, time-consuming baggage and
|
|
|
|
# sometimes emits shutdown warnings like this:
|
|
|
|
# "warning: callback failed in WindowList <class '_tkinter.TclError'>
|
|
|
|
# : invalid command name ".55131368.windows".
|
|
|
|
# Calling EditorWindow._close in tearDownClass prevents this but causes
|
|
|
|
# other problems (windows left open).
|
|
|
|
|
|
|
|
class TextWrapper:
|
|
|
|
def __init__(self, master):
|
|
|
|
self.text = Text(master=master)
|
|
|
|
def __getattr__(self, name):
|
|
|
|
return getattr(self.text, name)
|
|
|
|
def undo_block_start(self): pass
|
|
|
|
def undo_block_stop(self): pass
|
|
|
|
|
|
|
|
class Editor:
|
|
|
|
def __init__(self, root):
|
|
|
|
self.text = TextWrapper(root)
|
|
|
|
get_selection_indices = EditorWindow. get_selection_indices
|
|
|
|
|
|
|
|
class FormatEventTest(unittest.TestCase):
|
|
|
|
"""Test the formatting of text inside a Text widget.
|
|
|
|
|
|
|
|
This is done with FormatParagraph.format.paragraph_event,
|
2013-08-17 10:11:40 -03:00
|
|
|
which calls functions in the module as appropriate.
|
2013-08-10 17:56:28 -03:00
|
|
|
"""
|
|
|
|
test_string = (
|
|
|
|
" '''this is a test of a reformat for a triple "
|
|
|
|
"quoted string will it reformat to less than 70 "
|
|
|
|
"characters for me?'''\n")
|
|
|
|
multiline_test_string = (
|
|
|
|
" '''The first line is under the max width.\n"
|
|
|
|
" The second line's length is way over the max width. It goes "
|
|
|
|
"on and on until it is over 100 characters long.\n"
|
|
|
|
" Same thing with the third line. It is also way over the max "
|
|
|
|
"width, but FormatParagraph will fix it.\n"
|
|
|
|
" '''\n")
|
|
|
|
multiline_test_comment = (
|
|
|
|
"# The first line is under the max width.\n"
|
|
|
|
"# The second line's length is way over the max width. It goes on "
|
|
|
|
"and on until it is over 100 characters long.\n"
|
|
|
|
"# Same thing with the third line. It is also way over the max "
|
|
|
|
"width, but FormatParagraph will fix it.\n"
|
|
|
|
"# The fourth line is short like the first line.")
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def setUpClass(cls):
|
|
|
|
requires('gui')
|
|
|
|
cls.root = Tk()
|
2018-06-18 05:47:59 -03:00
|
|
|
cls.root.withdraw()
|
2013-08-10 17:56:28 -03:00
|
|
|
editor = Editor(root=cls.root)
|
|
|
|
cls.text = editor.text.text # Test code does not need the wrapper.
|
2019-07-17 10:44:44 -03:00
|
|
|
cls.formatter = ft.FormatParagraph(editor).format_paragraph_event
|
2013-08-10 17:56:28 -03:00
|
|
|
# Sets the insert mark just after the re-wrapped and inserted text.
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def tearDownClass(cls):
|
2016-06-03 23:19:17 -03:00
|
|
|
del cls.text, cls.formatter
|
2018-06-18 05:47:59 -03:00
|
|
|
cls.root.update_idletasks()
|
2013-08-10 17:56:28 -03:00
|
|
|
cls.root.destroy()
|
2014-02-27 19:47:49 -04:00
|
|
|
del cls.root
|
2013-08-10 17:56:28 -03:00
|
|
|
|
|
|
|
def test_short_line(self):
|
|
|
|
self.text.insert('1.0', "Short line\n")
|
|
|
|
self.formatter("Dummy")
|
|
|
|
self.assertEqual(self.text.get('1.0', 'insert'), "Short line\n" )
|
|
|
|
self.text.delete('1.0', 'end')
|
|
|
|
|
|
|
|
def test_long_line(self):
|
|
|
|
text = self.text
|
|
|
|
|
|
|
|
# Set cursor ('insert' mark) to '1.0', within text.
|
|
|
|
text.insert('1.0', self.test_string)
|
|
|
|
text.mark_set('insert', '1.0')
|
2014-04-22 02:11:03 -03:00
|
|
|
self.formatter('ParameterDoesNothing', limit=70)
|
2013-08-10 17:56:28 -03:00
|
|
|
result = text.get('1.0', 'insert')
|
|
|
|
# find function includes \n
|
|
|
|
expected = (
|
|
|
|
" '''this is a test of a reformat for a triple quoted string will it\n"
|
|
|
|
" reformat to less than 70 characters for me?'''\n") # yes
|
|
|
|
self.assertEqual(result, expected)
|
|
|
|
text.delete('1.0', 'end')
|
|
|
|
|
|
|
|
# Select from 1.11 to line end.
|
|
|
|
text.insert('1.0', self.test_string)
|
|
|
|
text.tag_add('sel', '1.11', '1.end')
|
2014-04-22 02:11:03 -03:00
|
|
|
self.formatter('ParameterDoesNothing', limit=70)
|
2013-08-10 17:56:28 -03:00
|
|
|
result = text.get('1.0', 'insert')
|
|
|
|
# selection excludes \n
|
|
|
|
expected = (
|
|
|
|
" '''this is a test of a reformat for a triple quoted string will it reformat\n"
|
|
|
|
" to less than 70 characters for me?'''") # no
|
|
|
|
self.assertEqual(result, expected)
|
|
|
|
text.delete('1.0', 'end')
|
|
|
|
|
|
|
|
def test_multiple_lines(self):
|
|
|
|
text = self.text
|
|
|
|
# Select 2 long lines.
|
|
|
|
text.insert('1.0', self.multiline_test_string)
|
|
|
|
text.tag_add('sel', '2.0', '4.0')
|
2014-04-22 02:11:03 -03:00
|
|
|
self.formatter('ParameterDoesNothing', limit=70)
|
2013-08-10 17:56:28 -03:00
|
|
|
result = text.get('2.0', 'insert')
|
|
|
|
expected = (
|
|
|
|
" The second line's length is way over the max width. It goes on and\n"
|
|
|
|
" on until it is over 100 characters long. Same thing with the third\n"
|
|
|
|
" line. It is also way over the max width, but FormatParagraph will\n"
|
|
|
|
" fix it.\n")
|
|
|
|
self.assertEqual(result, expected)
|
|
|
|
text.delete('1.0', 'end')
|
|
|
|
|
|
|
|
def test_comment_block(self):
|
|
|
|
text = self.text
|
|
|
|
|
|
|
|
# Set cursor ('insert') to '1.0', within block.
|
|
|
|
text.insert('1.0', self.multiline_test_comment)
|
2014-04-22 02:11:03 -03:00
|
|
|
self.formatter('ParameterDoesNothing', limit=70)
|
2013-08-10 17:56:28 -03:00
|
|
|
result = text.get('1.0', 'insert')
|
|
|
|
expected = (
|
|
|
|
"# The first line is under the max width. The second line's length is\n"
|
|
|
|
"# way over the max width. It goes on and on until it is over 100\n"
|
|
|
|
"# characters long. Same thing with the third line. It is also way over\n"
|
|
|
|
"# the max width, but FormatParagraph will fix it. The fourth line is\n"
|
|
|
|
"# short like the first line.\n")
|
|
|
|
self.assertEqual(result, expected)
|
|
|
|
text.delete('1.0', 'end')
|
|
|
|
|
|
|
|
# Select line 2, verify line 1 unaffected.
|
|
|
|
text.insert('1.0', self.multiline_test_comment)
|
|
|
|
text.tag_add('sel', '2.0', '3.0')
|
2014-04-22 02:11:03 -03:00
|
|
|
self.formatter('ParameterDoesNothing', limit=70)
|
2013-08-10 17:56:28 -03:00
|
|
|
result = text.get('1.0', 'insert')
|
|
|
|
expected = (
|
|
|
|
"# The first line is under the max width.\n"
|
|
|
|
"# The second line's length is way over the max width. It goes on and\n"
|
|
|
|
"# on until it is over 100 characters long.\n")
|
|
|
|
self.assertEqual(result, expected)
|
|
|
|
text.delete('1.0', 'end')
|
|
|
|
|
|
|
|
# The following block worked with EditorWindow but fails with the mock.
|
|
|
|
# Lines 2 and 3 get pasted together even though the previous block left
|
|
|
|
# the previous line alone. More investigation is needed.
|
|
|
|
## # Select lines 3 and 4
|
|
|
|
## text.insert('1.0', self.multiline_test_comment)
|
|
|
|
## text.tag_add('sel', '3.0', '5.0')
|
|
|
|
## self.formatter('ParameterDoesNothing')
|
|
|
|
## result = text.get('3.0', 'insert')
|
|
|
|
## expected = (
|
|
|
|
##"# Same thing with the third line. It is also way over the max width,\n"
|
|
|
|
##"# but FormatParagraph will fix it. The fourth line is short like the\n"
|
|
|
|
##"# first line.\n")
|
|
|
|
## self.assertEqual(result, expected)
|
|
|
|
## text.delete('1.0', 'end')
|
|
|
|
|
|
|
|
|
2019-07-17 10:44:44 -03:00
|
|
|
class DummyEditwin:
|
|
|
|
def __init__(self, root, text):
|
|
|
|
self.root = root
|
|
|
|
self.text = text
|
|
|
|
self.indentwidth = 4
|
|
|
|
self.tabwidth = 4
|
|
|
|
self.usetabs = False
|
|
|
|
self.context_use_ps1 = True
|
|
|
|
|
|
|
|
_make_blanks = EditorWindow._make_blanks
|
|
|
|
get_selection_indices = EditorWindow.get_selection_indices
|
|
|
|
|
|
|
|
|
|
|
|
class FormatRegionTest(unittest.TestCase):
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def setUpClass(cls):
|
|
|
|
requires('gui')
|
|
|
|
cls.root = Tk()
|
|
|
|
cls.root.withdraw()
|
|
|
|
cls.text = Text(cls.root)
|
|
|
|
cls.text.undo_block_start = mock.Mock()
|
|
|
|
cls.text.undo_block_stop = mock.Mock()
|
|
|
|
cls.editor = DummyEditwin(cls.root, cls.text)
|
|
|
|
cls.formatter = ft.FormatRegion(cls.editor)
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def tearDownClass(cls):
|
|
|
|
del cls.text, cls.formatter, cls.editor
|
|
|
|
cls.root.update_idletasks()
|
|
|
|
cls.root.destroy()
|
|
|
|
del cls.root
|
|
|
|
|
|
|
|
def setUp(self):
|
|
|
|
self.text.insert('1.0', self.code_sample)
|
|
|
|
|
|
|
|
def tearDown(self):
|
|
|
|
self.text.delete('1.0', 'end')
|
|
|
|
|
|
|
|
code_sample = """\
|
2019-11-20 02:18:39 -04:00
|
|
|
# WS line needed for test.
|
2019-07-17 10:44:44 -03:00
|
|
|
class C1():
|
|
|
|
# Class comment.
|
|
|
|
def __init__(self, a, b):
|
|
|
|
self.a = a
|
|
|
|
self.b = b
|
|
|
|
|
|
|
|
def compare(self):
|
|
|
|
if a > b:
|
|
|
|
return a
|
|
|
|
elif a < b:
|
|
|
|
return b
|
|
|
|
else:
|
|
|
|
return None
|
|
|
|
"""
|
|
|
|
|
|
|
|
def test_get_region(self):
|
|
|
|
get = self.formatter.get_region
|
|
|
|
text = self.text
|
|
|
|
eq = self.assertEqual
|
|
|
|
|
|
|
|
# Add selection.
|
|
|
|
text.tag_add('sel', '7.0', '10.0')
|
|
|
|
expected_lines = ['',
|
|
|
|
' def compare(self):',
|
|
|
|
' if a > b:',
|
|
|
|
'']
|
|
|
|
eq(get(), ('7.0', '10.0', '\n'.join(expected_lines), expected_lines))
|
|
|
|
|
|
|
|
# Remove selection.
|
|
|
|
text.tag_remove('sel', '1.0', 'end')
|
|
|
|
eq(get(), ('15.0', '16.0', '\n', ['', '']))
|
|
|
|
|
|
|
|
def test_set_region(self):
|
|
|
|
set_ = self.formatter.set_region
|
|
|
|
text = self.text
|
|
|
|
eq = self.assertEqual
|
|
|
|
|
|
|
|
save_bell = text.bell
|
|
|
|
text.bell = mock.Mock()
|
|
|
|
line6 = self.code_sample.splitlines()[5]
|
|
|
|
line10 = self.code_sample.splitlines()[9]
|
|
|
|
|
|
|
|
text.tag_add('sel', '6.0', '11.0')
|
|
|
|
head, tail, chars, lines = self.formatter.get_region()
|
|
|
|
|
|
|
|
# No changes.
|
|
|
|
set_(head, tail, chars, lines)
|
|
|
|
text.bell.assert_called_once()
|
|
|
|
eq(text.get('6.0', '11.0'), chars)
|
|
|
|
eq(text.get('sel.first', 'sel.last'), chars)
|
|
|
|
text.tag_remove('sel', '1.0', 'end')
|
|
|
|
|
|
|
|
# Alter selected lines by changing lines and adding a newline.
|
|
|
|
newstring = 'added line 1\n\n\n\n'
|
|
|
|
newlines = newstring.split('\n')
|
|
|
|
set_('7.0', '10.0', chars, newlines)
|
|
|
|
# Selection changed.
|
|
|
|
eq(text.get('sel.first', 'sel.last'), newstring)
|
|
|
|
# Additional line added, so last index is changed.
|
|
|
|
eq(text.get('7.0', '11.0'), newstring)
|
|
|
|
# Before and after lines unchanged.
|
|
|
|
eq(text.get('6.0', '7.0-1c'), line6)
|
|
|
|
eq(text.get('11.0', '12.0-1c'), line10)
|
|
|
|
text.tag_remove('sel', '1.0', 'end')
|
|
|
|
|
|
|
|
text.bell = save_bell
|
|
|
|
|
|
|
|
def test_indent_region_event(self):
|
|
|
|
indent = self.formatter.indent_region_event
|
|
|
|
text = self.text
|
|
|
|
eq = self.assertEqual
|
|
|
|
|
|
|
|
text.tag_add('sel', '7.0', '10.0')
|
|
|
|
indent()
|
|
|
|
# Blank lines aren't affected by indent.
|
|
|
|
eq(text.get('7.0', '10.0'), ('\n def compare(self):\n if a > b:\n'))
|
|
|
|
|
|
|
|
def test_dedent_region_event(self):
|
|
|
|
dedent = self.formatter.dedent_region_event
|
|
|
|
text = self.text
|
|
|
|
eq = self.assertEqual
|
|
|
|
|
|
|
|
text.tag_add('sel', '7.0', '10.0')
|
|
|
|
dedent()
|
|
|
|
# Blank lines aren't affected by dedent.
|
|
|
|
eq(text.get('7.0', '10.0'), ('\ndef compare(self):\n if a > b:\n'))
|
|
|
|
|
|
|
|
def test_comment_region_event(self):
|
|
|
|
comment = self.formatter.comment_region_event
|
|
|
|
text = self.text
|
|
|
|
eq = self.assertEqual
|
|
|
|
|
|
|
|
text.tag_add('sel', '7.0', '10.0')
|
|
|
|
comment()
|
|
|
|
eq(text.get('7.0', '10.0'), ('##\n## def compare(self):\n## if a > b:\n'))
|
|
|
|
|
|
|
|
def test_uncomment_region_event(self):
|
|
|
|
comment = self.formatter.comment_region_event
|
|
|
|
uncomment = self.formatter.uncomment_region_event
|
|
|
|
text = self.text
|
|
|
|
eq = self.assertEqual
|
|
|
|
|
|
|
|
text.tag_add('sel', '7.0', '10.0')
|
|
|
|
comment()
|
|
|
|
uncomment()
|
|
|
|
eq(text.get('7.0', '10.0'), ('\n def compare(self):\n if a > b:\n'))
|
|
|
|
|
|
|
|
# Only remove comments at the beginning of a line.
|
|
|
|
text.tag_remove('sel', '1.0', 'end')
|
|
|
|
text.tag_add('sel', '3.0', '4.0')
|
|
|
|
uncomment()
|
|
|
|
eq(text.get('3.0', '3.end'), (' # Class comment.'))
|
|
|
|
|
|
|
|
self.formatter.set_region('3.0', '4.0', '', ['# Class comment.', ''])
|
|
|
|
uncomment()
|
|
|
|
eq(text.get('3.0', '3.end'), (' Class comment.'))
|
|
|
|
|
|
|
|
@mock.patch.object(ft.FormatRegion, "_asktabwidth")
|
|
|
|
def test_tabify_region_event(self, _asktabwidth):
|
|
|
|
tabify = self.formatter.tabify_region_event
|
|
|
|
text = self.text
|
|
|
|
eq = self.assertEqual
|
|
|
|
|
|
|
|
text.tag_add('sel', '7.0', '10.0')
|
|
|
|
# No tabwidth selected.
|
|
|
|
_asktabwidth.return_value = None
|
|
|
|
self.assertIsNone(tabify())
|
|
|
|
|
|
|
|
_asktabwidth.return_value = 3
|
|
|
|
self.assertIsNotNone(tabify())
|
|
|
|
eq(text.get('7.0', '10.0'), ('\n\t def compare(self):\n\t\t if a > b:\n'))
|
|
|
|
|
|
|
|
@mock.patch.object(ft.FormatRegion, "_asktabwidth")
|
|
|
|
def test_untabify_region_event(self, _asktabwidth):
|
|
|
|
untabify = self.formatter.untabify_region_event
|
|
|
|
text = self.text
|
|
|
|
eq = self.assertEqual
|
|
|
|
|
|
|
|
text.tag_add('sel', '7.0', '10.0')
|
|
|
|
# No tabwidth selected.
|
|
|
|
_asktabwidth.return_value = None
|
|
|
|
self.assertIsNone(untabify())
|
|
|
|
|
|
|
|
_asktabwidth.return_value = 2
|
|
|
|
self.formatter.tabify_region_event()
|
|
|
|
_asktabwidth.return_value = 3
|
|
|
|
self.assertIsNotNone(untabify())
|
|
|
|
eq(text.get('7.0', '10.0'), ('\n def compare(self):\n if a > b:\n'))
|
|
|
|
|
|
|
|
@mock.patch.object(ft, "askinteger")
|
|
|
|
def test_ask_tabwidth(self, askinteger):
|
|
|
|
ask = self.formatter._asktabwidth
|
|
|
|
askinteger.return_value = 10
|
|
|
|
self.assertEqual(ask(), 10)
|
|
|
|
|
|
|
|
|
2019-11-20 02:18:39 -04:00
|
|
|
class IndentsTest(unittest.TestCase):
|
|
|
|
|
|
|
|
@mock.patch.object(ft, "askyesno")
|
|
|
|
def test_toggle_tabs(self, askyesno):
|
|
|
|
editor = DummyEditwin(None, None) # usetabs == False.
|
|
|
|
indents = ft.Indents(editor)
|
|
|
|
askyesno.return_value = True
|
|
|
|
|
|
|
|
indents.toggle_tabs_event(None)
|
|
|
|
self.assertEqual(editor.usetabs, True)
|
|
|
|
self.assertEqual(editor.indentwidth, 8)
|
|
|
|
|
|
|
|
indents.toggle_tabs_event(None)
|
|
|
|
self.assertEqual(editor.usetabs, False)
|
|
|
|
self.assertEqual(editor.indentwidth, 8)
|
|
|
|
|
|
|
|
@mock.patch.object(ft, "askinteger")
|
|
|
|
def test_change_indentwidth(self, askinteger):
|
|
|
|
editor = DummyEditwin(None, None) # indentwidth == 4.
|
|
|
|
indents = ft.Indents(editor)
|
|
|
|
|
|
|
|
askinteger.return_value = None
|
|
|
|
indents.change_indentwidth_event(None)
|
|
|
|
self.assertEqual(editor.indentwidth, 4)
|
|
|
|
|
|
|
|
askinteger.return_value = 3
|
|
|
|
indents.change_indentwidth_event(None)
|
|
|
|
self.assertEqual(editor.indentwidth, 3)
|
|
|
|
|
|
|
|
askinteger.return_value = 5
|
|
|
|
editor.usetabs = True
|
|
|
|
indents.change_indentwidth_event(None)
|
|
|
|
self.assertEqual(editor.indentwidth, 3)
|
|
|
|
|
|
|
|
|
|
|
|
class RstripTest(unittest.TestCase):
|
2019-07-17 21:48:36 -03:00
|
|
|
|
2019-11-24 17:29:29 -04:00
|
|
|
@classmethod
|
|
|
|
def setUpClass(cls):
|
|
|
|
requires('gui')
|
|
|
|
cls.root = Tk()
|
|
|
|
cls.root.withdraw()
|
|
|
|
cls.text = Text(cls.root)
|
|
|
|
cls.editor = MockEditor(text=cls.text)
|
|
|
|
cls.do_rstrip = ft.Rstrip(cls.editor).do_rstrip
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def tearDownClass(cls):
|
|
|
|
del cls.text, cls.do_rstrip, cls.editor
|
|
|
|
cls.root.update_idletasks()
|
|
|
|
cls.root.destroy()
|
|
|
|
del cls.root
|
2019-07-17 21:48:36 -03:00
|
|
|
|
2019-11-24 17:29:29 -04:00
|
|
|
def tearDown(self):
|
|
|
|
self.text.delete('1.0', 'end-1c')
|
2019-07-17 21:48:36 -03:00
|
|
|
|
2019-11-24 17:29:29 -04:00
|
|
|
def test_rstrip_lines(self):
|
2019-07-17 21:48:36 -03:00
|
|
|
original = (
|
|
|
|
"Line with an ending tab \n"
|
|
|
|
"Line ending in 5 spaces \n"
|
|
|
|
"Linewithnospaces\n"
|
|
|
|
" indented line\n"
|
|
|
|
" indented line with trailing space \n"
|
2019-11-24 17:29:29 -04:00
|
|
|
" \n")
|
2019-07-17 21:48:36 -03:00
|
|
|
stripped = (
|
|
|
|
"Line with an ending tab\n"
|
|
|
|
"Line ending in 5 spaces\n"
|
|
|
|
"Linewithnospaces\n"
|
|
|
|
" indented line\n"
|
|
|
|
" indented line with trailing space\n")
|
|
|
|
|
2019-11-24 17:29:29 -04:00
|
|
|
self.text.insert('1.0', original)
|
|
|
|
self.do_rstrip()
|
|
|
|
self.assertEqual(self.text.get('1.0', 'insert'), stripped)
|
|
|
|
|
|
|
|
def test_rstrip_end(self):
|
|
|
|
text = self.text
|
|
|
|
for code in ('', '\n', '\n\n\n'):
|
|
|
|
with self.subTest(code=code):
|
|
|
|
text.insert('1.0', code)
|
|
|
|
self.do_rstrip()
|
|
|
|
self.assertEqual(text.get('1.0','end-1c'), '')
|
|
|
|
for code in ('a\n', 'a\n\n', 'a\n\n\n'):
|
|
|
|
with self.subTest(code=code):
|
|
|
|
text.delete('1.0', 'end-1c')
|
|
|
|
text.insert('1.0', code)
|
|
|
|
self.do_rstrip()
|
|
|
|
self.assertEqual(text.get('1.0','end-1c'), 'a\n')
|
2019-07-17 21:48:36 -03:00
|
|
|
|
|
|
|
|
2013-08-10 17:56:28 -03:00
|
|
|
if __name__ == '__main__':
|
|
|
|
unittest.main(verbosity=2, exit=2)
|