265 lines
6.6 KiB
C
265 lines
6.6 KiB
C
|
/**************************************************************************
|
||
|
*
|
||
|
* Tint2 : common windows function
|
||
|
*
|
||
|
* Copyright (C) 2007 Pål Staurland (staura@gmail.com)
|
||
|
* Modified (C) 2008 thierry lorthiois (lorthiois@bbsoft.fr)
|
||
|
*
|
||
|
* This program is free software; you can redistribute it and/or
|
||
|
* modify it under the terms of the GNU General Public License version 2
|
||
|
* as published by the Free Software Foundation.
|
||
|
*
|
||
|
* 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 <X11/Xlib.h>
|
||
|
#include <X11/Xutil.h>
|
||
|
#include <X11/Xatom.h>
|
||
|
#include <stdio.h>
|
||
|
#include <stdlib.h>
|
||
|
#include <string.h>
|
||
|
#include <Imlib2.h>
|
||
|
|
||
|
#include "common.h"
|
||
|
#include "window.h"
|
||
|
#include "server.h"
|
||
|
|
||
|
|
||
|
|
||
|
void set_active (Window win)
|
||
|
{
|
||
|
send_event32 (win, server.atom._NET_ACTIVE_WINDOW, 2, 0);
|
||
|
}
|
||
|
|
||
|
|
||
|
void set_desktop (int desktop)
|
||
|
{
|
||
|
send_event32 (server.root_win, server.atom._NET_CURRENT_DESKTOP, desktop, 0);
|
||
|
}
|
||
|
|
||
|
|
||
|
void windows_set_desktop (Window win, int desktop)
|
||
|
{
|
||
|
send_event32 (win, server.atom._NET_WM_DESKTOP, desktop, 2);
|
||
|
}
|
||
|
|
||
|
|
||
|
void set_close (Window win)
|
||
|
{
|
||
|
send_event32 (win, server.atom._NET_CLOSE_WINDOW, 0, 2);
|
||
|
}
|
||
|
|
||
|
|
||
|
void window_toggle_shade (Window win)
|
||
|
{
|
||
|
send_event32 (win, server.atom._NET_WM_STATE, 2, 0);
|
||
|
}
|
||
|
|
||
|
|
||
|
int window_is_hidden (Window win)
|
||
|
{
|
||
|
Window window;
|
||
|
Atom *at;
|
||
|
int count, i;
|
||
|
|
||
|
if (XGetTransientForHint(server.dsp, win, &window) != 0) {
|
||
|
if (window) {
|
||
|
return 1;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
at = server_get_property (win, server.atom._NET_WM_STATE, XA_ATOM, &count);
|
||
|
for (i = 0; i < count; i++) {
|
||
|
if (at[i] == server.atom._NET_WM_STATE_SKIP_PAGER || at[i] == server.atom._NET_WM_STATE_SKIP_TASKBAR) {
|
||
|
XFree(at);
|
||
|
return 1;
|
||
|
}
|
||
|
}
|
||
|
XFree(at);
|
||
|
|
||
|
at = server_get_property (win, server.atom._NET_WM_WINDOW_TYPE, XA_ATOM, &count);
|
||
|
for (i = 0; i < count; i++) {
|
||
|
if (at[i] == server.atom._NET_WM_WINDOW_TYPE_DOCK || at[i] == server.atom._NET_WM_WINDOW_TYPE_DESKTOP || at[i] == server.atom._NET_WM_WINDOW_TYPE_TOOLBAR || at[i] == server.atom._NET_WM_WINDOW_TYPE_MENU || at[i] == server.atom._NET_WM_WINDOW_TYPE_SPLASH) {
|
||
|
XFree(at);
|
||
|
return 1;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// specification
|
||
|
// Windows with neither _NET_WM_WINDOW_TYPE nor WM_TRANSIENT_FOR set
|
||
|
// MUST be taken as top-level window.
|
||
|
XFree(at);
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
|
||
|
int window_get_desktop (Window win)
|
||
|
{
|
||
|
return get_property32(win, server.atom._NET_WM_DESKTOP, XA_CARDINAL);
|
||
|
}
|
||
|
|
||
|
|
||
|
int window_get_monitor (Window win)
|
||
|
{
|
||
|
int i, x, y;
|
||
|
Window src;
|
||
|
|
||
|
XTranslateCoordinates(server.dsp, win, server.root_win, 0, 0, &x, &y, &src);
|
||
|
for (i = 0; i < server.nb_monitor; i++) {
|
||
|
if (x >= server.monitor[i].x && x <= (server.monitor[i].x + server.monitor[i].width))
|
||
|
if (y >= server.monitor[i].y && y <= (server.monitor[i].y + server.monitor[i].height))
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
//printf("window %lx : ecran %d, (%d, %d)\n", win, i, x, y);
|
||
|
if (i == server.nb_monitor) return 0;
|
||
|
else return i;
|
||
|
}
|
||
|
|
||
|
|
||
|
int window_is_iconified (Window win)
|
||
|
{
|
||
|
return (IconicState == get_property32(win, server.atom.WM_STATE, server.atom.WM_STATE));
|
||
|
}
|
||
|
|
||
|
|
||
|
int server_get_number_of_desktop ()
|
||
|
{
|
||
|
return get_property32(server.root_win, server.atom._NET_NUMBER_OF_DESKTOPS, XA_CARDINAL);
|
||
|
}
|
||
|
|
||
|
|
||
|
int server_get_current_desktop ()
|
||
|
{
|
||
|
return get_property32(server.root_win, server.atom._NET_CURRENT_DESKTOP, XA_CARDINAL);
|
||
|
}
|
||
|
|
||
|
|
||
|
Window window_get_active ()
|
||
|
{
|
||
|
return get_property32(server.root_win, server.atom._NET_ACTIVE_WINDOW, XA_WINDOW);
|
||
|
}
|
||
|
|
||
|
|
||
|
int window_is_active (Window win)
|
||
|
{
|
||
|
return (win == get_property32(server.root_win, server.atom._NET_ACTIVE_WINDOW, XA_WINDOW));
|
||
|
}
|
||
|
|
||
|
|
||
|
int get_icon_count (long *data, int num)
|
||
|
{
|
||
|
int count, pos, w, h;
|
||
|
|
||
|
count = 0;
|
||
|
pos = 0;
|
||
|
while (pos < num) {
|
||
|
w = data[pos++];
|
||
|
h = data[pos++];
|
||
|
pos += w * h;
|
||
|
if (pos > num || w * h == 0) break;
|
||
|
count++;
|
||
|
}
|
||
|
|
||
|
return count;
|
||
|
}
|
||
|
|
||
|
|
||
|
long *get_best_icon (long *data, int icon_count, int num, int *iw, int *ih, int best_icon_size)
|
||
|
{
|
||
|
int width[icon_count], height[icon_count], pos, i, w, h;
|
||
|
long *icon_data[icon_count];
|
||
|
|
||
|
/* List up icons */
|
||
|
pos = 0;
|
||
|
i = icon_count;
|
||
|
while (i--) {
|
||
|
w = data[pos++];
|
||
|
h = data[pos++];
|
||
|
if (pos + w * h > num) break;
|
||
|
|
||
|
width[i] = w;
|
||
|
height[i] = h;
|
||
|
icon_data[i] = &data[pos];
|
||
|
|
||
|
pos += w * h;
|
||
|
}
|
||
|
|
||
|
/* Try to find exact size */
|
||
|
int icon_num = -1;
|
||
|
for (i = 0; i < icon_count; i++) {
|
||
|
if (width[i] == best_icon_size) {
|
||
|
icon_num = i;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* Take the biggest or whatever */
|
||
|
if (icon_num < 0) {
|
||
|
int highest = 0;
|
||
|
for (i = 0; i < icon_count; i++) {
|
||
|
if (width[i] > highest) {
|
||
|
icon_num = i;
|
||
|
highest = width[i];
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
*iw = width[icon_num];
|
||
|
*ih = height[icon_num];
|
||
|
return icon_data[icon_num];
|
||
|
}
|
||
|
|
||
|
|
||
|
void draw_rect(cairo_t *c, double x, double y, double w, double h, double r)
|
||
|
{
|
||
|
if (r > 0.0) {
|
||
|
double c1 = 0.55228475 * r;
|
||
|
|
||
|
cairo_move_to(c, x+r, y);
|
||
|
cairo_rel_line_to(c, w-2*r, 0);
|
||
|
cairo_rel_curve_to(c, c1, 0.0, r, c1, r, r);
|
||
|
cairo_rel_line_to(c, 0, h-2*r);
|
||
|
cairo_rel_curve_to(c, 0.0, c1, c1-r, r, -r, r);
|
||
|
cairo_rel_line_to (c, -w +2*r, 0);
|
||
|
cairo_rel_curve_to (c, -c1, 0, -r, -c1, -r, -r);
|
||
|
cairo_rel_line_to (c, 0, -h + 2 * r);
|
||
|
cairo_rel_curve_to (c, 0, -c1, r - c1, -r, r, -r);
|
||
|
}
|
||
|
else
|
||
|
cairo_rectangle(c, x, y, w, h);
|
||
|
}
|
||
|
|
||
|
|
||
|
void get_text_size(PangoFontDescription *font, int *height_ink, int *height, int panel_height, char *text, int len)
|
||
|
{
|
||
|
PangoRectangle rect_ink, rect;
|
||
|
|
||
|
Pixmap pmap = server_create_pixmap (panel_height, panel_height);
|
||
|
cairo_surface_t *cs = cairo_xlib_surface_create (server.dsp, pmap, server.visual, panel_height, panel_height);
|
||
|
cairo_t *c = cairo_create (cs);
|
||
|
|
||
|
PangoLayout *layout = pango_cairo_create_layout (c);
|
||
|
pango_layout_set_font_description (layout, font);
|
||
|
pango_layout_set_text (layout, text, len);
|
||
|
|
||
|
pango_layout_get_pixel_extents(layout, &rect_ink, &rect);
|
||
|
*height_ink = rect_ink.height;
|
||
|
*height = rect.height;
|
||
|
//printf("dimension : %d - %d\n", rect_ink.height, rect.height);
|
||
|
|
||
|
g_object_unref (layout);
|
||
|
cairo_destroy (c);
|
||
|
cairo_surface_destroy (cs);
|
||
|
XFreePixmap (server.dsp, pmap);
|
||
|
}
|
||
|
|
||
|
|
||
|
|