2005-11-19 10:37:58 +00:00
|
|
|
# -*- Mode: Python -*-
|
|
|
|
# vi:si:et:sw=4:sts=4:ts=4
|
|
|
|
|
2005-07-13 10:55:18 +00:00
|
|
|
# gst-python
|
|
|
|
# Copyright (C) 2005 Fluendo S.L.
|
|
|
|
# Originally from the Flumotion streaming server.
|
|
|
|
#
|
|
|
|
# This library is free software; you can redistribute it and/or
|
|
|
|
# modify it under the terms of the GNU Library General Public
|
|
|
|
# License as published by the Free Software Foundation; either
|
|
|
|
# version 2 of the License, or (at your option) any later version.
|
|
|
|
#
|
|
|
|
# This library 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
|
|
|
|
# Library General Public License for more details.
|
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU Library General Public
|
|
|
|
# License along with this library; if not, write to the
|
2013-11-25 17:01:48 +00:00
|
|
|
# Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
|
2013-12-12 11:20:12 +00:00
|
|
|
# Boston, MA 02110-1301, USA.
|
2005-07-13 12:27:47 +00:00
|
|
|
#
|
|
|
|
# Author: Zaheer Merali <zaheermerali at gmail dot com>
|
2005-07-13 10:55:18 +00:00
|
|
|
|
|
|
|
import gtk
|
|
|
|
from gtk import gdk
|
|
|
|
import gobject
|
|
|
|
|
|
|
|
|
|
|
|
# this VUMeter respects IEC standard
|
|
|
|
# BS 6840-18:1996/IEC-268-18
|
|
|
|
# and is inspired by JACK's meterbridge dpm_meters.c
|
|
|
|
|
|
|
|
class FVUMeter(gtk.DrawingArea):
|
|
|
|
__gsignals__ = { 'expose-event' : 'override',
|
|
|
|
'size-allocate': 'override',
|
|
|
|
'size-request': 'override',
|
|
|
|
'realize' : 'override'
|
|
|
|
}
|
|
|
|
__gproperties__ = {
|
|
|
|
'peak' : (gobject.TYPE_FLOAT,
|
|
|
|
'peak volume level',
|
|
|
|
'peak volume level in dB',
|
|
|
|
-90.0,
|
|
|
|
0,
|
|
|
|
-90.0,
|
|
|
|
gobject.PARAM_READWRITE),
|
|
|
|
'decay' : (gobject.TYPE_FLOAT,
|
|
|
|
'decay volume level',
|
|
|
|
'decay volume level in dB',
|
|
|
|
-90.0,
|
|
|
|
0,
|
|
|
|
-90.0,
|
|
|
|
gobject.PARAM_READWRITE),
|
|
|
|
'orange-threshold': (gobject.TYPE_FLOAT,
|
|
|
|
'threshold for orange',
|
|
|
|
'threshold for orange use in dB',
|
|
|
|
-90.0,
|
|
|
|
0,
|
|
|
|
-10.0,
|
|
|
|
gobject.PARAM_READWRITE),
|
|
|
|
'red-threshold': (gobject.TYPE_FLOAT,
|
|
|
|
'threshold for red',
|
|
|
|
'threshold for red use in dB',
|
|
|
|
-90.0,
|
|
|
|
0,
|
|
|
|
-1.0,
|
|
|
|
gobject.PARAM_READWRITE)
|
|
|
|
|
|
|
|
}
|
|
|
|
green_gc = None
|
|
|
|
orange_gc = None
|
|
|
|
red_gc = None
|
|
|
|
yellow_gc = None
|
|
|
|
|
|
|
|
topborder = 7
|
|
|
|
peaklevel = -90.0
|
|
|
|
decaylevel = -90.0
|
|
|
|
orange_threshold = -10.0
|
|
|
|
red_threshold = -1.0
|
|
|
|
bottomborder = 25
|
|
|
|
leftborder = 15
|
|
|
|
rightborder = 65
|
|
|
|
|
|
|
|
# Returns the meter deflection percentage given a db value
|
|
|
|
def iec_scale(self, db):
|
|
|
|
pct = 0.0
|
|
|
|
|
|
|
|
if db < -70.0:
|
|
|
|
pct = 0.0
|
|
|
|
elif db < -60.0:
|
|
|
|
pct = (db + 70.0) * 0.25
|
|
|
|
elif db < -50.0:
|
|
|
|
pct = (db + 60.0) * 0.5 + 2.5
|
|
|
|
elif db < -40.0:
|
|
|
|
pct = (db + 50.0) * 0.75 + 7.5
|
|
|
|
elif db < -30.0:
|
|
|
|
pct = (db + 40.0) * 1.5 + 15.0
|
|
|
|
elif db < -20.0:
|
|
|
|
pct = (db + 30.0) * 2.0 + 30.0
|
|
|
|
elif db < 0.0:
|
|
|
|
pct = (db + 20.0) * 2.5 + 50.0
|
|
|
|
else:
|
|
|
|
pct = 100.0
|
|
|
|
|
|
|
|
return pct
|
|
|
|
|
|
|
|
def do_get_property(self, property):
|
|
|
|
if property.name == 'peak':
|
|
|
|
return self.peaklevel
|
|
|
|
elif property.name == 'decay':
|
|
|
|
return self.decaylevel
|
|
|
|
elif property.name == 'orange-threshold':
|
|
|
|
return self.orange_threshold
|
|
|
|
elif property.name == 'red-threshold':
|
|
|
|
return self.red_threshold
|
|
|
|
else:
|
|
|
|
raise AttributeError, 'unknown property %s' % property.name
|
|
|
|
|
|
|
|
def do_set_property(self, property, value):
|
|
|
|
if property.name == 'peak':
|
|
|
|
self.peaklevel = value
|
|
|
|
elif property.name == 'decay':
|
|
|
|
self.decaylevel = value
|
|
|
|
elif property.name == 'orange-threshold':
|
|
|
|
self.orange_threshold = value
|
|
|
|
elif property.name == 'red-threshold':
|
|
|
|
self.red_threshold = value
|
|
|
|
else:
|
|
|
|
raise AttributeError, 'unknown property %s' % property.name
|
|
|
|
|
|
|
|
self.queue_draw()
|
|
|
|
|
|
|
|
def do_size_request(self, requisition):
|
|
|
|
requisition.width = 250
|
|
|
|
requisition.height = 50
|
|
|
|
|
|
|
|
def do_size_allocate(self, allocation):
|
|
|
|
self.allocation = allocation
|
|
|
|
if self.flags() & gtk.REALIZED:
|
|
|
|
self.window.move_resize(*allocation)
|
|
|
|
|
|
|
|
def do_realize(self):
|
|
|
|
self.set_flags(self.flags() | gtk.REALIZED)
|
|
|
|
|
|
|
|
self.window = gdk.Window(self.get_parent_window(),
|
|
|
|
width=self.allocation.width,
|
|
|
|
height=self.allocation.height,
|
|
|
|
window_type=gdk.WINDOW_CHILD,
|
|
|
|
wclass=gdk.INPUT_OUTPUT,
|
|
|
|
event_mask=self.get_events() | gdk.EXPOSURE_MASK)
|
|
|
|
|
|
|
|
colormap = gtk.gdk.colormap_get_system()
|
|
|
|
green = colormap.alloc_color(0, 65535, 0)
|
|
|
|
orange = colormap.alloc_color(65535, 32768, 0)
|
|
|
|
red = colormap.alloc_color(65535, 0, 0)
|
|
|
|
yellow = colormap.alloc_color(65535, 65535, 0)
|
|
|
|
self.green_gc = gdk.GC(self.window, foreground=green)
|
|
|
|
self.orange_gc = gdk.GC(self.window, foreground=orange)
|
|
|
|
self.red_gc = gdk.GC(self.window, foreground=red)
|
|
|
|
self.yellow_gc = gdk.GC(self.window, foreground=yellow)
|
|
|
|
|
|
|
|
self.window.set_user_data(self)
|
|
|
|
self.style.attach(self.window)
|
|
|
|
self.style.set_background(self.window, gtk.STATE_NORMAL)
|
|
|
|
|
|
|
|
def do_expose_event(self, event):
|
|
|
|
self.chain(event)
|
|
|
|
|
|
|
|
x, y, w, h = self.allocation
|
|
|
|
vumeter_width = w - (self.leftborder + self.rightborder)
|
|
|
|
vumeter_height = h - (self.topborder + self.bottomborder)
|
|
|
|
self.window.draw_rectangle(self.style.black_gc, True,
|
|
|
|
self.leftborder, self.topborder,
|
|
|
|
vumeter_width,
|
|
|
|
vumeter_height)
|
|
|
|
# draw peak level
|
2005-09-24 12:17:59 +00:00
|
|
|
# 0 maps to width of 0, full scale maps to total width
|
2005-07-13 10:55:18 +00:00
|
|
|
peaklevelpct = self.iec_scale(self.peaklevel)
|
2005-09-24 12:17:59 +00:00
|
|
|
peakwidth = int(vumeter_width * (peaklevelpct / 100))
|
2005-07-13 10:55:18 +00:00
|
|
|
draw_gc = self.green_gc
|
|
|
|
if self.peaklevel >= self.orange_threshold:
|
|
|
|
draw_gc = self.orange_gc
|
|
|
|
if self.peaklevel >= self.red_threshold:
|
|
|
|
draw_gc = self.red_gc
|
2005-09-24 12:17:59 +00:00
|
|
|
if peakwidth > 0:
|
|
|
|
self.window.draw_rectangle(draw_gc, True,
|
|
|
|
self.leftborder, self.topborder,
|
|
|
|
peakwidth, vumeter_height)
|
|
|
|
|
2005-07-13 10:55:18 +00:00
|
|
|
# draw yellow decay level
|
|
|
|
if self.decaylevel > -90.0:
|
|
|
|
decaylevelpct = self.iec_scale(self.decaylevel)
|
2005-09-24 12:21:04 +00:00
|
|
|
decaywidth = int(vumeter_width * (decaylevelpct / 100))
|
2005-09-24 12:17:59 +00:00
|
|
|
# cheat the geometry by drawing 0% level at pixel 0,
|
|
|
|
# which is same position as just above 0%
|
|
|
|
if decaywidth == 0:
|
|
|
|
decaywidth = 1
|
2005-07-13 10:55:18 +00:00
|
|
|
self.window.draw_line(self.yellow_gc,
|
2005-09-24 12:17:59 +00:00
|
|
|
self.leftborder + decaywidth - 1,
|
2005-07-13 10:55:18 +00:00
|
|
|
self.topborder,
|
2005-09-24 12:17:59 +00:00
|
|
|
self.leftborder + decaywidth - 1,
|
|
|
|
self.topborder + vumeter_height - 1)
|
2005-07-13 10:55:18 +00:00
|
|
|
|
|
|
|
# draw tick marks
|
2005-09-23 21:10:36 +00:00
|
|
|
scalers = [
|
|
|
|
('-90', 0.0),
|
|
|
|
('-40', 0.15),
|
|
|
|
('-30', 0.30),
|
|
|
|
('-20', 0.50),
|
|
|
|
('-10', 0.75),
|
|
|
|
( '-5', 0.875),
|
|
|
|
( '0', 1.0),
|
|
|
|
]
|
|
|
|
for level, scale in scalers:
|
2005-09-24 12:17:59 +00:00
|
|
|
# tick mark, 6 pixels high
|
|
|
|
# we cheat again here by putting the 0 at the first pixel
|
2005-09-23 21:10:36 +00:00
|
|
|
self.window.draw_line(self.style.black_gc,
|
2005-09-24 12:17:59 +00:00
|
|
|
self.leftborder + int(scale * (vumeter_width - 1)),
|
2005-09-23 21:10:36 +00:00
|
|
|
h - self.bottomborder,
|
2005-09-24 12:17:59 +00:00
|
|
|
self.leftborder + int(scale * (vumeter_width - 1)),
|
2005-09-23 21:10:36 +00:00
|
|
|
h - self.bottomborder + 5)
|
2005-09-24 12:17:59 +00:00
|
|
|
# tick label
|
2005-09-23 21:10:36 +00:00
|
|
|
layout = self.create_pango_layout(level)
|
|
|
|
layout_width, layout_height = layout.get_pixel_size()
|
|
|
|
self.window.draw_layout(self.style.black_gc,
|
2005-09-24 12:17:59 +00:00
|
|
|
self.leftborder + int(scale * vumeter_width)
|
|
|
|
- int(layout_width / 2),
|
2005-09-23 21:10:36 +00:00
|
|
|
h - self.bottomborder + 7, layout)
|
|
|
|
|
|
|
|
# draw the peak level to the right
|
2005-07-13 10:55:18 +00:00
|
|
|
layout = self.create_pango_layout("%.2fdB" % self.peaklevel)
|
|
|
|
layout_width, layout_height = layout.get_pixel_size()
|
|
|
|
self.window.draw_layout(self.style.black_gc,
|
|
|
|
self.leftborder + vumeter_width + 5,
|
2005-09-24 12:17:59 +00:00
|
|
|
self.topborder + int(vumeter_height / 2 - layout_height / 2),
|
2005-07-13 10:55:18 +00:00
|
|
|
layout)
|
|
|
|
|
|
|
|
gobject.type_register(FVUMeter)
|