/* * tile.c - tile layout * * Copyright © 2007 Julien Danjou * * 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 #include "util.h" #include "screen.h" #include "awesome.h" #include "tag.h" #include "layout.h" #include "layouts/tile.h" void uicb_tag_setnmaster(awesome_config *awesomeconf, int screen, const char * arg) { Tag *curtag = get_current_tag(awesomeconf->screens[screen].tags, awesomeconf->screens[screen].ntags); 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(awesomeconf, screen); } void uicb_tag_setncol(awesome_config *awesomeconf, int screen, const char * arg) { Tag *curtag = get_current_tag(awesomeconf->screens[screen].tags, awesomeconf->screens[screen].ntags); 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(awesomeconf, screen); } void uicb_tag_setmwfact(awesome_config * awesomeconf, int screen, const char *arg) { char *newarg; Tag *curtag = get_current_tag(awesomeconf->screens[screen].tags, awesomeconf->screens[screen].ntags); 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(awesomeconf, screen); p_delete(&newarg); } static void _tile(awesome_config *awesomeconf, 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(awesomeconf->screens[screen].tags, awesomeconf->screens[screen].ntags); screens_info = get_screen_info(awesomeconf->display, screen, &awesomeconf->screens[screen].statusbar, &awesomeconf->screens[screen].padding); for(n = 0, c = awesomeconf->clients; c; c = c->next) if(IS_TILED(c, screen, awesomeconf->screens[screen].tags, awesomeconf->screens[screen].ntags)) 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 = MIN(otherwin, curtag->ncol); for(i = 0, c = awesomeconf->clients; c; c = c->next) { if(!IS_TILED(c, screen, awesomeconf->screens[screen].tags, awesomeconf->screens[screen].ntags)) 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, awesomeconf, awesomeconf->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, awesomeconf, awesomeconf->screens[screen].resize_hints, False); } i++; } p_delete(&screens_info); } void layout_tile(awesome_config *awesomeconf, int screen) { _tile(awesomeconf, screen, True); } void layout_tileleft(awesome_config *awesomeconf, int screen) { _tile(awesomeconf, screen, False); } // vim: filetype=c:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:encoding=utf-8:textwidth=99