All patches and comments are welcome. Please squash your changes to logical
commits before using git-format-patch and git-send-email to
patches@git.madduck.net.
If you'd read over the Git project's submission guidelines and adhered to them,
I'd be especially grateful.
   3 shared_examples_for "vim" do
 
   8     # Insert two blank lines.
 
   9     # The first line is a corner case in this plugin that would shadow the
 
  10     # correct behaviour of other tests. Thus we explicitly jump to the first
 
  11     # line when we require so.
 
  12     vim.feedkeys 'i\<CR>\<CR>\<ESC>'
 
  15   describe "when using the indent plugin" do
 
  16     it "sets the indentexpr and indentkeys options" do
 
  17       vim.command("set indentexpr?").should include "GetPythonPEPIndent("
 
  18       vim.command("set indentkeys?").should include "=elif"
 
  21     it "sets autoindent and expandtab" do
 
  22       vim.command("set autoindent?").should match(/\s*autoindent/)
 
  23       vim.command("set expandtab?").should match(/\s*expandtab/)
 
  27   describe "when entering the first line" do
 
  28     before { vim.feedkeys '0ggipass' }
 
  30     it "does not indent" do
 
  32       proposed_indent.should == 0
 
  35     it "does not indent when using '=='" do
 
  41   describe "when after a '(' that is at the end of its line" do
 
  42     before { vim.feedkeys 'itest(\<CR>' }
 
  44     it "indents by one level" do
 
  45       proposed_indent.should == shiftwidth
 
  46       vim.feedkeys 'something'
 
  47       indent.should == shiftwidth
 
  49       indent.should == shiftwidth
 
  52     it "puts the closing parenthesis at the same level" do
 
  54       indent.should == (hang_closing ? shiftwidth : 0)
 
  58   describe "when after an '(' that is followed by something" do
 
  59     before { vim.feedkeys 'itest(something,\<CR>' }
 
  61     it "lines up on following lines" do
 
  63       vim.feedkeys 'more,\<CR>'
 
  67     it "lines up the closing parenthesis" do
 
  72     it "does not touch the closing parenthesis if it is already indented further" do
 
  78   describe "when after an '{' that is followed by a comment" do
 
  79     before { vim.feedkeys 'imydict = {  # comment\<CR>' }
 
  81     it "indent by one level" do
 
  82       indent.should == shiftwidth
 
  83       vim.feedkeys '1: 1,\<CR>'
 
  84       indent.should == shiftwidth
 
  87     it "lines up the closing parenthesis" do
 
  89       indent.should == (hang_closing ? shiftwidth : 0)
 
  93   describe "when using gq to reindent a '(' that is" do
 
  94     before { vim.feedkeys 'itest(' }
 
  95     it "something and has a string without spaces at the end" do
 
  96       vim.feedkeys 'something_very_long_blaaaaaaaaa, "some_very_long_string_blaaaaaaaaaaaaaaaaaaaa"\<esc>gqq'
 
 101   describe "when after multiple parens of different types" do
 
 102     it "indents by one level" do
 
 103       vim.feedkeys 'if({\<CR>'
 
 104       indent.should == shiftwidth
 
 107     it "lines up with the last paren" do
 
 108       vim.feedkeys 'ifff({123: 456,\<CR>'
 
 113   describe "when '#' is contained in a string that is followed by a colon" do
 
 114     it "indents by one level" do
 
 115         vim.feedkeys 'iif "some#thing" == "test":#test\<CR>pass'
 
 116         indent.should == shiftwidth
 
 120   describe "when '#' is not contained in a string and is followed by a colon" do
 
 121     it "does not indent" do
 
 122         vim.feedkeys 'iif "some#thing" == "test"#:test\<CR>'
 
 127   describe "when inside an unfinished string" do
 
 128     it "does not indent" do
 
 129       vim.feedkeys 'i"test:\<ESC>'
 
 130       vim.echo('synIDattr(synID(line("."), col("."), 0), "name")'
 
 131               ).downcase.should include 'string'
 
 132       vim.feedkeys 'a\<CR>'
 
 133       proposed_indent.should == -1
 
 137     it "does not dedent" do
 
 138       vim.feedkeys 'iif True:\<CR>"test:\<ESC>'
 
 139       vim.echo('synIDattr(synID(line("."), col("."), 0), "name")'
 
 140               ).downcase.should include 'string'
 
 141       proposed_indent.should == shiftwidth
 
 142       indent.should == shiftwidth
 
 146   describe "when the previous line has a colon in a string" do
 
 147     before { vim.feedkeys 'itest(":".join(["1","2"]))\<CR>' }
 
 148     it "does not indent" do
 
 149       vim.feedkeys 'if True:'
 
 151       proposed_indent.should == 0
 
 155   describe "when the previous line has a list slice" do
 
 156     it "does not indent" do
 
 157       vim.feedkeys 'ib = a[2:]\<CR>'
 
 159       proposed_indent.should == 0
 
 163   describe "when line is empty inside a block" do
 
 164     it "is indented like the previous line" do
 
 165       vim.feedkeys 'idef a():\<CR>1\<CR>\<CR>2\<ESC>kcc'
 
 166       indent.should == shiftwidth
 
 170   describe "when an empty line is after empty line / before non-empty" do
 
 171     it "is indented like the next line" do
 
 172       vim.feedkeys 'idef a():\<CR>1\<CR>\<CR>\<CR>2\<ESC><<kcc'
 
 177   describe "when an empty line is after empty line / before non-empty (nested)" do
 
 178     it "is indented like the next line" do
 
 179       vim.feedkeys 'idef a():\<CR>1\<CR>\<CR>\<CR>\<ESC>0i\<TAB>2\<ESC>kcc'
 
 180       indent.should == shiftwidth
 
 184   describe "when line is empty inside a block following multi-line statement" do
 
 185     it "is indented like the previous line" do
 
 186       vim.feedkeys 'idef a():\<CR>x = (1 +\<CR>2)\<CR>\<CR>y\<ESC>kcc'
 
 187       indent.should == shiftwidth
 
 191   describe "when line is empty inside a block following stop statement" do
 
 192     it "is indented like the previous line minus shiftwidth" do
 
 193       vim.feedkeys 'iif x:\<CR>if y:\<CR>pass\<CR>\<CR>z\<ESC>kcc'
 
 194       indent.should == shiftwidth
 
 198   describe "when using simple control structures" do
 
 199       it "indents shiftwidth spaces" do
 
 200           vim.feedkeys 'iwhile True:\<CR>pass'
 
 201           indent.should == shiftwidth
 
 205   describe "when using a function definition" do
 
 206       it "indents shiftwidth spaces" do
 
 207           vim.feedkeys 'idef long_function_name(\<CR>arg'
 
 208           indent.should == shiftwidth
 
 212   describe "when using a class definition" do
 
 213       it "indents shiftwidth spaces" do
 
 214           vim.feedkeys 'iclass Foo(\<CR>'
 
 215           indent.should == shiftwidth
 
 219   describe "when writing an 'else' block" do
 
 220     it "aligns to the preceeding 'for' block" do
 
 221       vim.feedkeys 'ifor x in "abc":\<CR>pass\<CR>else:'
 
 225     it "aligns to the preceeding 'if' block" do
 
 226       vim.feedkeys 'ifor x in "abc":\<CR>if True:\<CR>pass\<CR>else:'
 
 227       indent.should == shiftwidth
 
 231   describe "when using parens and control statements" do
 
 232     it "avoids ambiguity by using extra indentation" do
 
 233       vim.feedkeys 'iif (111 and\<CR>'
 
 235         indent.should == shiftwidth * 2
 
 239       vim.feedkeys '222):\<CR>'
 
 240       indent.should == shiftwidth
 
 241       vim.feedkeys 'pass\<CR>'
 
 245     it "still aligns parens properly if not ambiguous" do
 
 246       vim.feedkeys 'iwhile (111 and\<CR>'
 
 248       vim.feedkeys '222):\<CR>'
 
 249       indent.should == shiftwidth
 
 250       vim.feedkeys 'pass\<CR>'
 
 254     it "handles nested expressions (Flake8's E127)" do
 
 255       vim.feedkeys 'i[\<CR>x for x in foo\<CR>if (\<CR>'
 
 256       indent.should == shiftwidth * 2
 
 259     it "still handles multiple parens correctly" do
 
 260       vim.feedkeys 'iif (111 and (222 and 333\<CR>'
 
 262       vim.feedkeys 'and 444\<CR>'
 
 264       vim.feedkeys ')\<CR>'
 
 266         indent.should == shiftwidth * 2
 
 270       vim.feedkeys 'and 555):\<CR>'
 
 271       indent.should == shiftwidth
 
 272       vim.feedkeys 'pass\<CR>'
 
 277   describe "when a line breaks with a manual '\\'" do
 
 278     it "indents shiftwidth spaces on normal line" do
 
 279         vim.feedkeys 'ivalue = test + \\\\\<CR>'
 
 280         indent.should == shiftwidth
 
 283     it "indents 2x shiftwidth spaces for control structures" do
 
 284         vim.feedkeys 'iif somevalue == xyz and \\\\\<CR>'
 
 285         indent.should == shiftwidth * 2
 
 288     it "indents relative to line above" do
 
 289         vim.feedkeys 'i\<TAB>value = test + \\\\\<CR>'
 
 290         indent.should == shiftwidth * 2
 
 294   describe "when current line is dedented compared to previous line" do
 
 295      before { vim.feedkeys 'i\<TAB>\<TAB>if x:\<CR>y = True\<CR>\<ESC>' }
 
 296      it "and current line has a valid indentation (Part 1)" do
 
 297         vim.feedkeys '0i\<TAB>if y:'
 
 298         proposed_indent.should == -1
 
 301      it "and current line has a valid indentation (Part 2)" do
 
 302         vim.feedkeys '0i\<TAB>\<TAB>if y:'
 
 303         proposed_indent.should == -1
 
 306      it "and current line has an invalid indentation" do
 
 307         vim.feedkeys 'i    while True:\<CR>'
 
 308         indent.should == previous_indent + shiftwidth
 
 312   describe "when current line is dedented compared to the last non-empty line" do
 
 313      before { vim.feedkeys 'i\<TAB>\<TAB>if x:\<CR>y = True\<CR>\<CR>\<ESC>' }
 
 314      it "and current line has a valid indentation" do
 
 315         vim.feedkeys '0i\<TAB>if y:'
 
 316         proposed_indent.should == -1
 
 320   describe "when an 'if' is followed by" do
 
 321      before { vim.feedkeys 'i\<TAB>\<TAB>if x:\<CR>' }
 
 322      it "an elif, it lines up with the 'if'" do
 
 323         vim.feedkeys 'elif y:'
 
 324         indent.should == shiftwidth * 2
 
 327      it "an 'else', it lines up with the 'if'" do
 
 329         indent.should == shiftwidth * 2
 
 333   describe "when an 'if' contains a try-except" do
 
 335        vim.feedkeys 'iif x:\<CR>try:\<CR>pass\<CR>except:\<CR>pass\<CR>'
 
 336        indent.should == shiftwidth
 
 338      it "an 'else' should be indented to the try" do
 
 340        indent.should == shiftwidth
 
 341        proposed_indent.should == shiftwidth
 
 343      it "an 'else' should keep the indent of the 'if'" do
 
 344        vim.feedkeys 'else:\<ESC><<'
 
 346        proposed_indent.should == 0
 
 350   describe "when a 'for' is followed by" do
 
 351      before { vim.feedkeys 'i\<TAB>\<TAB>for x in y:\<CR>' }
 
 352      it "an 'else', it lines up with the 'for'" do
 
 354         indent.should == shiftwidth * 2
 
 358   describe "when an 'else' is followed by" do
 
 359      before { vim.feedkeys 'i\<TAB>\<TAB>else:\<CR>XXX\<CR>' }
 
 360      it "a 'finally', it lines up with the 'else'" do
 
 361         vim.feedkeys 'finally:'
 
 362         indent.should == shiftwidth * 2
 
 367   describe "when a 'try' is followed by" do
 
 368      before { vim.feedkeys 'i\<TAB>\<TAB>try:\<CR>' }
 
 369      it "an 'except', it lines up with the 'try'" do
 
 370         vim.feedkeys 'except:'
 
 371         indent.should == shiftwidth * 2
 
 374      it "an 'else', it lines up with the 'try'" do
 
 376         indent.should == shiftwidth * 2
 
 379      it "a 'finally', it lines up with the 'try'" do
 
 380         vim.feedkeys 'finally:'
 
 381         indent.should == shiftwidth * 2
 
 385   describe "when an 'except' is followed by" do
 
 386      before { vim.feedkeys 'i\<TAB>\<TAB>except:\<CR>' }
 
 387      it "an 'else', it lines up with the 'except'" do
 
 389         indent.should == shiftwidth * 2
 
 392      it "another 'except', it lines up with the previous 'except'" do
 
 393         vim.feedkeys 'except:'
 
 394         indent.should == shiftwidth * 2
 
 397      it "a 'finally', it lines up with the 'except'" do
 
 398         vim.feedkeys 'finally:'
 
 399         indent.should == shiftwidth * 2
 
 403   describe "when an else is used inside of a nested if" do
 
 404     before { vim.feedkeys 'iif foo:\<CR>if bar:\<CR>pass\<CR>' }
 
 405     it "indents the else to the inner if" do
 
 407       indent.should == shiftwidth
 
 411   describe "when an else is used outside of a nested if" do
 
 412     before { vim.feedkeys 'iif True:\<CR>if True:\<CR>pass\<CR>\<Esc>0' }
 
 413     it "indents the else to the outer if" do
 
 415       proposed_indent.should == shiftwidth
 
 417       vim.feedkeys 'ielse:'
 
 419       proposed_indent.should == 0
 
 423   describe "when jedi-vim call signatures are used" do
 
 424     before { vim.command 'syn match jediFunction "JEDI_CALL_SIGNATURE" keepend extend' }
 
 426     it "ignores the call signature after a colon" do
 
 427       vim.feedkeys 'iif True:  JEDI_CALL_SIGNATURE\<CR>'
 
 428       indent.should == shiftwidth
 
 431     it "ignores the call signature after a function" do
 
 432       vim.feedkeys 'idef f(  JEDI_CALL_SIGNATURE\<CR>'
 
 433       indent.should == shiftwidth
 
 438 shared_examples_for "multiline strings" do
 
 443     # Insert two blank lines.
 
 444     # The first line is a corner case in this plugin that would shadow the
 
 445     # correct behaviour of other tests. Thus we explicitly jump to the first
 
 446     # line when we require so.
 
 447     vim.feedkeys 'i\<CR>\<CR>\<ESC>'
 
 450   describe "when after an '(' that is followed by an unfinished string" do
 
 451     before { vim.feedkeys 'itest("""' }
 
 453     it "it indents the next line" do
 
 455       expected_proposed, expected_indent = multiline_indent(0, shiftwidth)
 
 456       proposed_indent.should == expected_proposed
 
 457       indent.should == expected_indent
 
 460     it "with contents it indents the second line to the parenthesis" do
 
 461       vim.feedkeys 'second line\<CR>'
 
 462       expected_proposed, expected_indent = multiline_indent(0, 5)
 
 463       proposed_indent.should == expected_proposed
 
 464       indent.should == expected_indent
 
 468   describe "when after assigning an unfinished string" do
 
 469     before { vim.feedkeys 'itest = """' }
 
 471     it "it indents the next line" do
 
 473       expected_proposed, expected_indent = multiline_indent(0, shiftwidth)
 
 474       proposed_indent.should == expected_proposed
 
 475       indent.should == expected_indent
 
 479   describe "when after assigning an indented unfinished string" do
 
 480     before { vim.feedkeys 'i    test = """' }
 
 482     it "it indents the next line" do
 
 484       expected_proposed, expected_indent = multiline_indent(4, shiftwidth + 4)
 
 485       proposed_indent.should == expected_proposed
 
 486       indent.should == expected_indent
 
 490   describe "when after assigning an indented finished string" do
 
 491     before { vim.feedkeys 'i    test = ""' }
 
 493     it "it does indent the next line" do
 
 498     it "and writing a new string, it does indent the next line" do
 
 499       vim.feedkeys '\<CR>""'
 
 504   describe "when after a docstring" do
 
 505     it "it does indent the next line to the docstring" do
 
 506       vim.feedkeys 'i    """\<CR>'
 
 508       proposed_indent.should == 4
 
 511     it "indents the closing docstring quotes" do
 
 512       vim.feedkeys 'i    """\<CR>\<CR>"""'
 
 514       proposed_indent.should == 4
 
 515       vim.echo('getline(3)').should == '    """'
 
 518     it "indents non-matching docstring quotes" do
 
 519       vim.feedkeys 'i    """\<CR>\<Esc>'
 
 521       vim.echo('line(".")').should == "4"
 
 522       vim.echo('getline(".")').should == "'''"
 
 524       proposed_indent.should == -1
 
 528   describe "when after a docstring with contents" do
 
 529     before { vim.feedkeys 'i    """First line' }
 
 530     it "it does indent the next line to the docstring" do
 
 533       proposed_indent.should == 4
 
 537   describe "when breaking a string after opening parenthesis" do
 
 538     before { vim.feedkeys 'i    foo("""bar\<Left>\<Left>\<Left>' }
 
 539     it "it does indent the next line as after an opening multistring" do
 
 541       _, expected_indent = multiline_indent(4, 4 + shiftwidth)
 
 542       indent.should == expected_indent
 
 543       proposed_indent.should == -1
 
 545       # it keeps the indent after an empty line
 
 547       proposed_indent, expected_indent = multiline_indent(4, 4 + shiftwidth)
 
 548       indent.should == expected_indent
 
 549       proposed_indent.should == proposed_indent
 
 551       # it keeps the indent of nonblank above
 
 552       vim.feedkeys '\<End>\<CR>'
 
 553       proposed_indent, expected_indent = multiline_indent(4, 4 + shiftwidth)
 
 554       indent.should == expected_indent
 
 555       proposed_indent.should == proposed_indent
 
 557       # it keeps the indent of nonblank above before an empty line
 
 559       proposed_indent, expected_indent = multiline_indent(4, 4 + shiftwidth)
 
 560       indent.should == expected_indent
 
 561       proposed_indent.should == proposed_indent
 
 566 SUITE_SHIFTWIDTHS = [4, 3]
 
 567 SUITE_HANG_CLOSINGS = [false, true]
 
 569 SUITE_SHIFTWIDTHS.each do |sw|
 
 570   describe "vim when using width of #{sw}" do
 
 572       vim.command("set sw=#{sw} ts=#{sw} sts=#{sw} et")
 
 574     it "sets shiftwidth to #{sw}" do
 
 575       shiftwidth.should == sw
 
 578     SUITE_HANG_CLOSINGS.each do |hc|
 
 579       describe "vim when hang_closing is set to #{hc}" do
 
 583         it "sets hang_closing to #{hc}" do
 
 584           hang_closing.should == !!hc
 
 587         it_behaves_like "vim"
 
 593 describe "vim when not using python_pep8_indent_multiline_string" do
 
 595     vim.command("set sw=4 ts=4 sts=4 et")
 
 596     vim.command("unlet! g:python_pep8_indent_multiline_string")
 
 598   it_behaves_like "multiline strings"
 
 601 describe "vim when using python_pep8_indent_multiline_first=0" do
 
 603     vim.command("set sw=4 ts=4 sts=4 et")
 
 604     vim.command("let g:python_pep8_indent_multiline_string=0")
 
 606   it_behaves_like "multiline strings"
 
 609 describe "vim when using python_pep8_indent_multiline_string=-1" do
 
 611     vim.command("set sw=4 ts=4 sts=4 et")
 
 612     vim.command("let g:python_pep8_indent_multiline_string=-1")
 
 614   it_behaves_like "multiline strings"
 
 617 describe "vim when using python_pep8_indent_multiline_string=-2" do
 
 619     vim.command("set sw=4 ts=4 sts=4 et")
 
 620     vim.command("let g:python_pep8_indent_multiline_string=-2")
 
 622   it_behaves_like "multiline strings"
 
 625 describe "Handles far away opening parens" do
 
 626   before { vim.feedkeys '\<ESC>ggdGifrom foo import (' }
 
 628   it "indents by one level" do
 
 630     proposed_indent.should == shiftwidth
 
 633   it "indents by one level for 10 lines" do
 
 634     vim.command('set paste | exe "norm 9o" | set nopaste')
 
 635     vim.feedkeys '\<Esc>o'
 
 636     indent.should == shiftwidth
 
 639   it "indents by one level for 50 lines" do
 
 640     vim.command('set paste | exe "norm 49o" | set nopaste')
 
 641     vim.feedkeys '\<Esc>o'
 
 642     indent.should == shiftwidth
 
 646 describe "Handles far away opening square brackets" do
 
 647   before { vim.feedkeys '\<ESC>ggdGibar = [' }
 
 649   it "indents by one level" do
 
 651     proposed_indent.should == shiftwidth
 
 654   it "indents by one level for 10 lines" do
 
 655     vim.command('set paste | exe "norm 9o" | set nopaste')
 
 656     vim.feedkeys '\<Esc>o'
 
 657     indent.should == shiftwidth
 
 660   it "indents by one level for 100 lines" do
 
 661     vim.command('set paste | exe "norm 99o" | set nopaste')
 
 662     vim.feedkeys '\<Esc>o'
 
 663     indent.should == shiftwidth
 
 667 describe "Handles far away opening curly brackets" do
 
 668   before { vim.feedkeys '\<ESC>ggdGijson = {' }
 
 670   it "indents by one level" do
 
 672     vim.feedkeys '\<Esc>o'
 
 673     proposed_indent.should == shiftwidth
 
 676   it "indents by one level for 10 lines" do
 
 677     vim.command('set paste | exe "norm 9o" | set nopaste')
 
 678     vim.feedkeys '\<Esc>o'
 
 679     indent.should == shiftwidth
 
 682   it "indents by one level for 1000 lines" do
 
 683     vim.command('set paste | exe "norm 999o" | set nopaste')
 
 684     vim.feedkeys '\<Esc>o'
 
 685     indent.should == shiftwidth
 
 689 describe "Compact multiline dict" do
 
 690   before { vim.feedkeys '\<ESC>ggdGid = {"one": 1,' }
 
 692   it "gets indented correctly" do
 
 694     proposed_indent.should == 5
 
 696     vim.feedkeys '"two": 2}'
 
 697     proposed_indent.should == 5
 
 700     proposed_indent.should == 0
 
 704 describe "Using O" do
 
 706     vim.feedkeys '\<ESC>ggdG'
 
 707     vim.feedkeys 'iif foo:\<CR>'
 
 710   it "respects autoindent" do
 
 711     vim.feedkeys '1\<CR>\<CR>'
 
 712     indent.should == shiftwidth
 
 713     vim.feedkeys '\<Esc>ko'
 
 714     indent.should == shiftwidth
 
 715     vim.feedkeys '\<Esc>kO'
 
 716     indent.should == shiftwidth
 
 717     # Uses/keeps indent from line above
 
 718     vim.feedkeys '\<Esc>i2\<Esc>O'
 
 719     indent.should == shiftwidth
 
 720     # Uses/keeps indent from line above
 
 721     vim.feedkeys '\<Esc>j\<Esc>O'
 
 726 describe "searchpairpos" do
 
 727   before { vim.feedkeys '\<ESC>ggdG' }
 
 728   it "handles nested parenthesis" do
 
 729     vim.feedkeys 'iif foo.startswith("("):\<CR>'
 
 730     indent.should == shiftwidth
 
 734 describe "o within TODO" do
 
 736     vim.feedkeys '\<ESC>ggdG'
 
 737     vim.feedkeys 'iif 1:  # TODO\<Esc>'
 
 738     # Assertion that we have a pythonTodo here.
 
 739     vim.echo('synIDattr(synID(line("."), col("."), 0), "name")').should match 'pythonTodo'
 
 742   it "respects autoindent" do
 
 744     indent.should == shiftwidth
 
 748 describe "elif after else" do
 
 750     vim.feedkeys '\<ESC>ggdG'
 
 753   it "is indented to the outer if" do
 
 754     vim.feedkeys 'iif 1:\<CR>if 2:\<CR>pass\<CR>else:\<CR>pass\<CR>elif 3:\<Esc>'
 
 757     vim.feedkeys '\<ESC>ggdG'
 
 758     vim.feedkeys 'i    if 1:\<CR>if 2:\<CR>pass\<CR>else:\<CR>pass\<CR>elif 3:\<Esc>'