awesome/layouts/tile.c

190 lines
5.2 KiB
C

/*
* tile.c - tile layout
*
* Copyright © 2007 Julien Danjou <julien@danjou.info>
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License along
* with this program; if not, write to the Free Software Foundation, Inc.,
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
*
*/
#include <stdio.h>
#include "util.h"
#include "screen.h"
#include "awesome.h"
#include "tag.h"
#include "layout.h"
#include "layouts/tile.h"
extern awesome_config globalconf;
void
uicb_tag_setnmaster(int screen, const char * arg)
{
Tag *curtag = get_current_tag(screen);
Layout *curlay = curtag->layout;
if(!arg || (curlay->arrange != layout_tile && curlay->arrange != layout_tileleft))
return;
if((curtag->nmaster = (int) compute_new_value_from_arg(arg, (double) curtag->nmaster)) < 0)
curtag->nmaster = 0;
arrange(screen);
}
void
uicb_tag_setncol(int screen, const char * arg)
{
Tag *curtag = get_current_tag(screen);
Layout *curlay = curtag->layout;
if(!arg || (curlay->arrange != layout_tile && curlay->arrange != layout_tileleft))
return;
if((curtag->ncol = (int) compute_new_value_from_arg(arg, (double) curtag->ncol)) < 1)
curtag->ncol = 1;
arrange(screen);
}
void
uicb_tag_setmwfact(int screen, const char *arg)
{
char *newarg;
Tag *curtag = get_current_tag(screen);
Layout *curlay = curtag->layout;
if(!arg || (curlay->arrange != layout_tile && curlay->arrange != layout_tileleft))
return;
newarg = a_strdup(arg);
if(curlay->arrange == layout_tileleft)
{
if(newarg[0] == '+')
newarg[0] = '-';
else if(arg[0] == '-')
newarg[0] = '+';
}
if((curtag->mwfact = compute_new_value_from_arg(newarg, curtag->mwfact)) < 0.1)
curtag->mwfact = 0.1;
else if(curtag->mwfact > 0.9)
curtag->mwfact = 0.9;
arrange(screen);
p_delete(&newarg);
}
static void
_tile(int screen, const Bool right)
{
/* windows area geometry */
int wah = 0, waw = 0, wax = 0, way = 0;
/* new coordinates */
unsigned int nx, ny, nw, nh;
/* master size */
unsigned int mw = 0, mh = 0;
int n, i, masterwin = 0, otherwin = 0;
int real_ncol = 1, win_by_col = 1, current_col = 0;
ScreenInfo *screens_info = NULL;
Client *c;
Tag *curtag = get_current_tag(screen);
screens_info = get_screen_info(globalconf.display, screen, &globalconf.screens[screen].statusbar, &globalconf.screens[screen].padding);
for(n = 0, c = globalconf.clients; c; c = c->next)
if(IS_TILED(c, screen))
n++;
wah = screens_info[screen].height;
waw = screens_info[screen].width;
wax = screens_info[screen].x_org;
way = screens_info[screen].y_org;
masterwin = MIN(n, curtag->nmaster);
otherwin = n - masterwin;
if(otherwin < 0)
otherwin = 0;
if(curtag->nmaster)
{
mh = masterwin ? wah / masterwin : waw;
mw = otherwin ? waw * curtag->mwfact : waw;
}
else
mh = mw = 0;
real_ncol = curtag->ncol > 0 ? MIN(otherwin, curtag->ncol) : MIN(otherwin, 1);
for(i = 0, c = globalconf.clients; c; c = c->next)
{
if(!IS_TILED(c, screen))
continue;
c->ismax = False;
if(i < curtag->nmaster)
{ /* master */
ny = way + i * mh;
nx = wax + (right ? 0 : waw - mw);
client_resize(c, nx, ny, mw - 2 * c->border, mh - 2 * c->border, globalconf.screens[screen].resize_hints, False);
}
else
{ /* tile window */
if(real_ncol)
win_by_col = otherwin / real_ncol;
if((i - curtag->nmaster) && (i - curtag->nmaster) % win_by_col == 0 && current_col < real_ncol - 1)
current_col++;
if(current_col == real_ncol - 1)
win_by_col += otherwin % real_ncol;
if(otherwin <= real_ncol)
nh = wah - 2 * c->border;
else
nh = (wah / win_by_col) - 2 * c->border;
nw = (waw - mw) / real_ncol - 2 * c->border;
if(i == curtag->nmaster || otherwin <= real_ncol || (i - curtag->nmaster) % win_by_col == 0)
ny = way;
else
ny = way + ((i - curtag->nmaster) % win_by_col) * (nh + 2 * c->border);
nx = wax + current_col * (nw + 2 * c->border) + (right ? mw : 0);
client_resize(c, nx, ny, nw, nh, globalconf.screens[screen].resize_hints, False);
}
i++;
}
p_delete(&screens_info);
}
void
layout_tile(int screen)
{
_tile(screen, True);
}
void
layout_tileleft(int screen)
{
_tile(screen, False);
}
// vim: filetype=c:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:encoding=utf-8:textwidth=99