summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorBram Moolenaar <Bram@vim.org>2018-05-04 20:15:38 +0200
committerBram Moolenaar <Bram@vim.org>2018-05-04 20:15:38 +0200
commitc136af29c0b1939076fbae7d36afd90dce740315 (patch)
tree8513e2fab04112190824d6cd2f308a01acbe58f6
parent606cb8b08ed510962fcdc8ef1abcc1fe35fbffef (diff)
downloadvim-git-c136af29c0b1939076fbae7d36afd90dce740315.tar.gz
patch 8.0.1790: 'winfixwidth' is not always respected by :closev8.0.1790
Problem: 'winfixwidth' is not always respected by :close. Solution: Prefer a frame without 'winfixwidth' or 'winfixheight'. (Jason Franklin)
-rw-r--r--src/testdir/test_winbuf_close.vim36
-rw-r--r--src/version.c2
-rw-r--r--src/window.c48
3 files changed, 73 insertions, 13 deletions
diff --git a/src/testdir/test_winbuf_close.vim b/src/testdir/test_winbuf_close.vim
index ed64dd79b..e4618610c 100644
--- a/src/testdir/test_winbuf_close.vim
+++ b/src/testdir/test_winbuf_close.vim
@@ -122,3 +122,39 @@ func Test_winbuf_close()
call delete('Xtest2')
call delete('Xtest3')
endfunc
+
+" Test that ":close" will respect 'winfixheight' when possible.
+func Test_winfixheight_on_close()
+ set nosplitbelow nosplitright
+
+ split | split | vsplit
+
+ $wincmd w
+ setlocal winfixheight
+ let l:height = winheight(0)
+
+ 3close
+
+ call assert_equal(l:height, winheight(0))
+
+ %bwipeout!
+ setlocal nowinfixheight splitbelow& splitright&
+endfunc
+
+" Test that ":close" will respect 'winfixwidth' when possible.
+func Test_winfixwidth_on_close()
+ set nosplitbelow nosplitright
+
+ vsplit | vsplit | split
+
+ $wincmd w
+ setlocal winfixwidth
+ let l:width = winwidth(0)
+
+ 3close
+
+ call assert_equal(l:width, winwidth(0))
+
+ %bwipeout!
+ setlocal nowinfixwidth splitbelow& splitright&
+endfunction
diff --git a/src/version.c b/src/version.c
index 9d00e46c8..467b14e4f 100644
--- a/src/version.c
+++ b/src/version.c
@@ -762,6 +762,8 @@ static char *(features[]) =
static int included_patches[] =
{ /* Add new patch number below this line */
/**/
+ 1790,
+/**/
1789,
/**/
1788,
diff --git a/src/window.c b/src/window.c
index 333663ba3..d3ec4cd02 100644
--- a/src/window.c
+++ b/src/window.c
@@ -2740,12 +2740,14 @@ winframe_remove(
}
/*
- * Find out which frame is going to get the freed up space when "win" is
- * closed.
- * if 'splitbelow'/'splitleft' the space goes to the window above/left.
- * if 'nosplitbelow'/'nosplitleft' the space goes to the window below/right.
- * This makes opening a window and closing it immediately keep the same window
- * layout.
+ * Return a pointer to the frame that will receive the empty screen space that
+ * is left over after "win" is closed.
+ *
+ * If 'splitbelow' or 'splitright' is set, the space goes above or to the left
+ * by default. Otherwise, the free space goes below or to the right. The
+ * result is that opening a window and then immediately closing it will
+ * preserve the initial window layout. The 'wfh' and 'wfw' settings are
+ * respected when possible.
*/
static frame_T *
win_altframe(
@@ -2753,20 +2755,40 @@ win_altframe(
tabpage_T *tp) /* tab page "win" is in, NULL for current */
{
frame_T *frp;
- int b;
+ frame_T *other_fr, *target_fr;
if (tp == NULL ? ONE_WINDOW : tp->tp_firstwin == tp->tp_lastwin)
- /* Last window in this tab page, will go to next tab page. */
return alt_tabpage()->tp_curwin->w_frame;
frp = win->w_frame;
+
+ if (frp->fr_prev == NULL)
+ return frp->fr_next;
+ if (frp->fr_next == NULL)
+ return frp->fr_prev;
+
+ target_fr = frp->fr_next;
+ other_fr = frp->fr_prev;
+ if (p_spr || p_sb)
+ {
+ target_fr = frp->fr_prev;
+ other_fr = frp->fr_next;
+ }
+
+ /* If 'wfh' or 'wfw' is set for the target and not for the alternate
+ * window, reverse the selection. */
if (frp->fr_parent != NULL && frp->fr_parent->fr_layout == FR_ROW)
- b = p_spr;
+ {
+ if (frame_fixed_width(target_fr) && !frame_fixed_width(other_fr))
+ target_fr = other_fr;
+ }
else
- b = p_sb;
- if ((!b && frp->fr_next != NULL) || frp->fr_prev == NULL)
- return frp->fr_next;
- return frp->fr_prev;
+ {
+ if (frame_fixed_height(target_fr) && !frame_fixed_height(other_fr))
+ target_fr = other_fr;
+ }
+
+ return target_fr;
}
/*