aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorMattias Andrée <maandree@operamail.com>2015-04-04 01:59:52 +0200
committerMattias Andrée <maandree@operamail.com>2015-04-04 01:59:52 +0200
commit754b671bdae0e4775bfa57524e037b1198fed536 (patch)
treed7a55feb00517e85ad3bb8abf4382ce469d9b9bc
parentupdate todo (diff)
downloadxpybar-754b671bdae0e4775bfa57524e037b1198fed536.tar.gz
xpybar-754b671bdae0e4775bfa57524e037b1198fed536.tar.bz2
xpybar-754b671bdae0e4775bfa57524e037b1198fed536.tar.xz
add image support
Signed-off-by: Mattias Andrée <maandree@operamail.com>
-rw-r--r--DEPENDENCIES3
-rw-r--r--examples/plugins/image40
-rw-r--r--src/plugins/image.py114
3 files changed, 157 insertions, 0 deletions
diff --git a/DEPENDENCIES b/DEPENDENCIES
index 08a58d3..441f2f0 100644
--- a/DEPENDENCIES
+++ b/DEPENDENCIES
@@ -18,6 +18,9 @@ OPTIONAL RUNTIME DEPENDENCIES:
inotify-tools: for inotify support
alarm: for limiting the time of a file search in locks
findutils: for file search in locks
+ imagemagic: for image support
+ file: for image support
+ librsvg: for image support
BUILD DEPENDENCIES:
diff --git a/examples/plugins/image b/examples/plugins/image
new file mode 100644
index 0000000..e81d324
--- /dev/null
+++ b/examples/plugins/image
@@ -0,0 +1,40 @@
+# -*- python -*-
+'''
+xpybar – xmobar replacement written in python
+Copyright © 2014, 2015 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/>.
+'''
+
+# A xpybar configuration example testing the features of plugins.image
+
+import time
+import threading
+
+from plugins.image import Image
+
+
+OUTPUT, HEIGHT, YPOS, TOP = 0, 24, 24, True
+
+
+#image_ = Image('/usr/share/icons/hicolor/scalable/apps/mate-panel-clock.svg', 'rgb(0, 0, 0)', 24)
+#image_ = Image('/usr/share/icons/oxygen/256x256/apps/clock.png', 'rgb(0, 0, 0)', 24)
+image_ = Image('/usr/share/icons/ContrastHigh/48x48/apps/clock.png', 'rgb(0, 0, 0)', 24)
+
+
+def redraw():
+ bar.clear()
+ image_.draw(bar, 0, 0)
+ image_.draw(bar, bar.width - image_.width, 0)
+
diff --git a/src/plugins/image.py b/src/plugins/image.py
new file mode 100644
index 0000000..6d40179
--- /dev/null
+++ b/src/plugins/image.py
@@ -0,0 +1,114 @@
+# -*- python -*-
+'''
+xpybar – xmobar replacement written in python
+Copyright © 2014, 2015 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/>.
+'''
+
+
+class Image:
+ '''
+ Images and icons
+ '''
+
+
+ def __init__(self, file, background = 'black', width = None, height = None):
+ '''
+ Constructor
+
+ @param file:str The pathname of the image
+ @param background:str ImageMagick understandable string for the background colour
+ @param width:int? The width the image should have, `None` to not resize or use `height`
+ @param height:int? The height the image should have, `None` to not resize or use `width`
+ '''
+ import Xlib.X, sys
+ from subprocess import Popen, PIPE
+
+ self.format = Xlib.X.ZPixmap
+ self.depth = 24
+
+ height = width if height is None else height
+ width = height if width is None else width
+ raster = None
+
+ convert = ['file', '-']
+ convert = Popen(convert, stdin = open(file, 'rb'), stdout = PIPE, stderr = sys.stderr)
+ if 'Scalable Vector Graphics' in convert.communicate()[0].decode('utf-8', 'replace'):
+ convert = ['rsvg-convert', '-f', 'png']
+ if width is not None:
+ convert += ['-w', str(width), '-h', str(height)]
+ convert = Popen(convert, stdin = open(file, 'rb'), stdout = PIPE, stderr = sys.stderr)
+ raster = convert.communicate()[0]
+ if not convert.wait() == 0:
+ raise Exception('Image could not be converted')
+
+ convert = ['convert', '-', '-background', background, '-alpha', 'remove', '-depth', '8']
+ if width is not None:
+ convert += ['-resize', '%ix%i!' % (width, height)]
+ convert += ['ppm:-']
+ convert = Popen(convert, stdin = PIPE if raster is not None else open(file, 'rb'),
+ stdout = PIPE, stderr = sys.stderr)
+ if raster is not None:
+ self.data = list(convert.communicate(raster)[0])
+ else:
+ self.data = list(convert.communicate()[0])
+ if not convert.wait() == 0:
+ raise Exception('Image could not be converted')
+
+ self.width, self.height, state, comment = [], [], 0, False
+ for i in range(len(self.data)):
+ b = self.data[i]
+ if comment:
+ if b == ord('\n'):
+ comment = False
+ elif b == ord('#'):
+ comment = True
+ elif b in (ord(' '), ord('\n')):
+ if (state & 1) == 0:
+ continue
+ state += 1
+ if state == 4 * 2:
+ break
+ else:
+ state += 1 - (state & 1)
+ if state == 2 * 1 + 1:
+ self.width.append(b)
+ elif state == 2 * 2 + 1:
+ self.height.append(b)
+
+ self.data, buf = [], self.data[i:]
+ self.width = int(bytes(self.width).decode('utf-8', 'strict'))
+ self.height = int(bytes(self.height).decode('utf-8', 'strict'))
+
+ i = 0
+ for _i in range(len(buf) // 3):
+ self.data.append(buf[i])
+ self.data.append(buf[i + 1])
+ self.data.append(buf[i + 2])
+ self.data.append(0)
+ i += 3
+ self.data = bytes(self.data)
+
+
+ def draw(self, bar, x, y):
+ '''
+ Draw the image on a bar
+
+ @param bar:Bar The bar to draw the image on
+ @param x:int The left position of the image
+ @param y:int The top position of the image
+ '''
+ bar.window.put_image(bar.gc, x, y, self.width, self.height, self.format, self.depth, 0, self.data)
+