#!/usr/bin/env python3 ''' xpybar – xmobar replacement written in python Copyright © 2014 Mattias Andrée (maandree@member.fsf.org) 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 3 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, see <http://www.gnu.org/licenses/>. ''' import subprocess import Xlib.display, Xlib.Xatom, Xlib.ext.randr, Xlib.X display = None ''' The connection the X display ''' screen = None ''' The screen the panel is placed in ''' screen_i = None ''' The index of the screen `screen` ''' def open_x(screen_no = None): ''' Open connection the X and select screen @param screen_no:int? The index of the screen to use, `None` for default ''' global display, screen, screen_i display = Xlib.display.Display() screen_i = screen_no if screen_no is not None else display.get_default_screen() screen = display.screen(screen_i) def get_monitors(): ''' Returns the pixel size and position of the monitors in the screen @return :list<(width:int, height:int, left:int, top:int)> The width, height, left position and top position of each monitor, starting with the primary output ''' global screen_i p = subprocess.Popen(['xrandr'], stdout = subprocess.PIPE) p = p.communicate()[0].decode('utf-8', 'replace') s = -1 rc = [] prim = None for line in p.split('\n'): if line.startswith('Screen '): s = int(line[len('Screen '):].split(':')[0]) elif s == screen_i: if ' connected ' in line: m = line.replace('-', '+-').replace('++', '+') p = ' primary ' in m m = m.replace(' primary ', ' ') m = m.split(' ')[2].replace('+', 'x').split('x') m = [int(x) for x in m] if p and (prim is None): prim = m else: rc.append(m) if prim is not None: rc = [prim] + rc return rc def get_display(): ''' Returns the X display ''' global display return display def get_screen(): ''' Returns the X screen ''' global screen return screen def create_panel(width, height, left, ypos, panel_height, at_top): ''' Create a docked panel, not mapped yet @param width:int The width of the output @param height:int The height of the output @param left:int The left position of the output @param ypos:int The position of the panel in relation to either the top or bottom edge of the output @param panel_height:int The height of the panel @param at_top:bool Whether the panel is to be docked to the top of the output, otherwise to the bottom @return The window ''' global display, screen ypos = ypos if at_top else (height - ypos - panel_height) window = screen.root.create_window(left, ypos, width, panel_height, 0, Xlib.X.CopyFromParent, Xlib.X.InputOutput, Xlib.X.CopyFromParent, event_mask = ( Xlib.X.StructureNotifyMask | Xlib.X.ButtonReleaseMask | Xlib.X.ExposureMask ), colormap = Xlib.X.CopyFromParent) top_ = lambda x, y, w, h : [0, 0, y + h, 0, 0, 0, 0, 0, x, x + w, 0, 0] bottom_ = lambda x, y, w, h : [0, 0, 0, y + h, 0, 0, 0, 0, 0, 0, x, x + w] window.set_wm_name('xpybar') window.set_wm_icon_name('xpybar') window.set_wm_class('bar', 'xpybar') _CARD = display.intern_atom("CARDINAL") _PSTRUT = display.intern_atom("_NET_WM_STRUT_PARTIAL") window.change_property(_PSTRUT, _CARD, 32, (top_ if at_top else bottom_)(left, ypos, width, panel_height)) _ATOM = display.intern_atom("ATOM") _TYPE = display.intern_atom("_NET_WM_WINDOW_TYPE") _DOCK = display.intern_atom("_NET_WM_WINDOW_TYPE_DOCK") window.change_property(_TYPE, _ATOM, 32, [_DOCK]) return window def draw_text(window, gc, x, y, text): ''' Draw a text on a window @param window The window @param gc The window's graphics context @param x:int The left position of the text @param y:int The Y position of the bottom of the text @param text:str The text to draw ''' text_ = text.encode('utf-16')[2:] text = [] for i in range(len(text_)): if (i & 1) == 0: text.append(text_[i]) else: text[-1] |= text_[i] << 8 window.image_text_16(gc, x, y, text) def close_x(): ''' Closes the connection to X, but flushes it first ''' global display display.flush() display.close()