mirror of
https://github.com/0intro/wmii
synced 2024-11-25 15:20:15 +03:00
548 lines
13 KiB
Ruby
548 lines
13 KiB
Ruby
# DSL for wmiirc configuration.
|
|
#--
|
|
# Copyright protects this work.
|
|
# See LICENSE file for details.
|
|
#++
|
|
|
|
require 'shellwords'
|
|
require 'pathname'
|
|
require 'yaml'
|
|
|
|
require 'rubygems'
|
|
gem 'rumai', '~> 3'
|
|
require 'rumai'
|
|
|
|
include Rumai
|
|
|
|
class Handler < Hash
|
|
def initialize
|
|
super {|h,k| h[k] = [] }
|
|
end
|
|
|
|
##
|
|
# If a block is given, registers a handler
|
|
# for the given key and returns the handler.
|
|
#
|
|
# Otherwise, executes all handlers registered for the given key.
|
|
#
|
|
def handle key, *args, &block
|
|
if block
|
|
self[key] << block
|
|
|
|
elsif key? key
|
|
self[key].each do |block|
|
|
block.call(*args)
|
|
end
|
|
end
|
|
|
|
block
|
|
end
|
|
end
|
|
|
|
EVENTS = Handler.new
|
|
ACTIONS = Handler.new
|
|
KEYS = Handler.new
|
|
|
|
##
|
|
# If a block is given, registers a handler
|
|
# for the given event and returns the handler.
|
|
#
|
|
# Otherwise, executes all handlers for the given event.
|
|
#
|
|
def event *a, &b
|
|
EVENTS.handle(*a, &b)
|
|
end
|
|
|
|
##
|
|
# Returns a list of registered event names.
|
|
#
|
|
def events
|
|
EVENTS.keys
|
|
end
|
|
|
|
##
|
|
# If a block is given, registers a handler for
|
|
# the given action and returns the handler.
|
|
#
|
|
# Otherwise, executes all handlers for the given action.
|
|
#
|
|
def action *a, &b
|
|
ACTIONS.handle(*a, &b)
|
|
end
|
|
|
|
##
|
|
# Returns a list of registered action names.
|
|
#
|
|
def actions
|
|
ACTIONS.keys
|
|
end
|
|
|
|
##
|
|
# If a block is given, registers a handler for
|
|
# the given keypress and returns the handler.
|
|
#
|
|
# Otherwise, executes all handlers for the given keypress.
|
|
#
|
|
def key *a, &b
|
|
KEYS.handle(*a, &b)
|
|
end
|
|
|
|
##
|
|
# Returns a list of registered action names.
|
|
#
|
|
def keys
|
|
KEYS.keys
|
|
end
|
|
|
|
##
|
|
# Shows a menu (where the user must press keys on their keyboard to
|
|
# make a choice) with the given items and returns the chosen item.
|
|
#
|
|
# If nothing was chosen, then nil is returned.
|
|
#
|
|
# ==== Parameters
|
|
#
|
|
# [prompt]
|
|
# Instruction on what the user should enter or choose.
|
|
#
|
|
def key_menu choices, prompt = nil
|
|
words = ['dmenu', '-fn', CONFIG['display']['font']]
|
|
|
|
# show menu at the same location as the status bar
|
|
words << '-b' if CONFIG['display']['bar'] == 'bottom'
|
|
|
|
words.concat %w[-nf -nb -sf -sb].zip(
|
|
[
|
|
CONFIG['display']['color']['normal'],
|
|
CONFIG['display']['color']['focus'],
|
|
|
|
].map {|c| c.to_s.split[0,2] }.flatten
|
|
|
|
).flatten
|
|
|
|
words.push '-p', prompt if prompt
|
|
|
|
command = words.shelljoin
|
|
IO.popen(command, 'r+') do |menu|
|
|
menu.puts choices
|
|
menu.close_write
|
|
|
|
choice = menu.read
|
|
choice unless choice.empty?
|
|
end
|
|
end
|
|
|
|
##
|
|
# Shows a menu (where the user must click a menu
|
|
# item using their mouse to make a choice) with
|
|
# the given items and returns the chosen item.
|
|
#
|
|
# If nothing was chosen, then nil is returned.
|
|
#
|
|
# ==== Parameters
|
|
#
|
|
# [choices]
|
|
# List of choices to display in the menu.
|
|
#
|
|
# [initial]
|
|
# The choice that should be initially selected.
|
|
#
|
|
# If this choice is not included in the list
|
|
# of choices, then this item will be made
|
|
# into a makeshift title-bar for the menu.
|
|
#
|
|
def click_menu choices, initial = nil
|
|
words = ['wmii9menu']
|
|
|
|
if initial
|
|
words << '-i'
|
|
|
|
unless choices.include? initial
|
|
initial = "<<#{initial}>>:"
|
|
words << initial
|
|
end
|
|
|
|
words << initial
|
|
end
|
|
|
|
words.concat choices
|
|
command = words.shelljoin
|
|
|
|
choice = `#{command}`.chomp
|
|
choice unless choice.empty?
|
|
end
|
|
|
|
##
|
|
# Shows a key_menu() containing the given
|
|
# clients and returns the chosen client.
|
|
#
|
|
# If nothing was chosen, then nil is returned.
|
|
#
|
|
# ==== Parameters
|
|
#
|
|
# [prompt]
|
|
# Instruction on what the user should enter or choose.
|
|
#
|
|
# [clients]
|
|
# List of clients to present as choices to the user.
|
|
#
|
|
# If this parameter is not specified,
|
|
# its default value will be a list of
|
|
# all currently available clients.
|
|
#
|
|
def client_menu prompt = nil, clients = Rumai.clients
|
|
choices = []
|
|
|
|
clients.each_with_index do |c, i|
|
|
choices << "%d. [%s] %s" % [i, c[:tags].read, c[:label].read.downcase]
|
|
end
|
|
|
|
if target = key_menu(choices, prompt)
|
|
clients[target.scan(/\d+/).first.to_i]
|
|
end
|
|
end
|
|
|
|
##
|
|
# Returns the basenames of executable files present in the given directories.
|
|
#
|
|
def find_programs *dirs
|
|
dirs.flatten.
|
|
map {|d| Pathname.new(d).expand_path.children rescue [] }.flatten.
|
|
map {|f| f.basename.to_s if f.file? and f.executable? }.compact.uniq.sort
|
|
end
|
|
|
|
##
|
|
# Launches the command built from the given words in the background.
|
|
#
|
|
def launch *words
|
|
command = words.shelljoin
|
|
system "#{command} &"
|
|
end
|
|
|
|
##
|
|
# A button on a bar.
|
|
#
|
|
class Button < Thread
|
|
##
|
|
# Creates a new button at the given node and updates its label
|
|
# according to the given refresh rate (measured in seconds). The
|
|
# given block is invoked to calculate the label of the button.
|
|
#
|
|
# The return value of the given block can be either an
|
|
# array (whose first item is a wmii color sequence for the
|
|
# button, and the remaining items compose the label of the
|
|
# button) or a string containing the label of the button.
|
|
#
|
|
# If the given block raises a standard exception, then that will be
|
|
# rescued and displayed (using error colors) as the button's label.
|
|
#
|
|
def initialize fs_bar_node, refresh_rate, &button_label
|
|
raise ArgumentError, 'block must be given' unless block_given?
|
|
|
|
super(fs_bar_node) do |button|
|
|
while true
|
|
label =
|
|
begin
|
|
Array(button_label.call)
|
|
rescue Exception => e
|
|
LOG.error e
|
|
[CONFIG['display']['color']['error'], e]
|
|
end
|
|
|
|
# provide default color
|
|
unless label.first =~ /(?:#[[:xdigit:]]{6} ?){3}/
|
|
label.unshift CONFIG['display']['color']['normal']
|
|
end
|
|
|
|
button.create unless button.exist?
|
|
button.write label.join(' ')
|
|
sleep refresh_rate
|
|
end
|
|
end
|
|
end
|
|
|
|
##
|
|
# Refreshes the label of this button.
|
|
#
|
|
alias refresh wakeup
|
|
end
|
|
|
|
##
|
|
# Loads the given YAML configuration file.
|
|
#
|
|
def load_config config_file
|
|
Object.const_set :CONFIG, YAML.load_file(config_file)
|
|
|
|
# script
|
|
eval CONFIG['script']['before'].to_s, TOPLEVEL_BINDING,
|
|
"#{config_file}:script:before"
|
|
|
|
# display
|
|
fo = ENV['WMII_FONT'] = CONFIG['display']['font']
|
|
fc = ENV['WMII_FOCUSCOLORS'] = CONFIG['display']['color']['focus']
|
|
nc = ENV['WMII_NORMCOLORS'] = CONFIG['display']['color']['normal']
|
|
|
|
settings = {
|
|
'font' => fo,
|
|
'focuscolors' => fc,
|
|
'normcolors' => nc,
|
|
'border' => CONFIG['display']['border'],
|
|
'bar on' => CONFIG['display']['bar'],
|
|
'colmode' => CONFIG['display']['column']['mode'],
|
|
'grabmod' => CONFIG['control']['grab'],
|
|
}
|
|
|
|
begin
|
|
fs.ctl.write settings.map {|pair| pair.join(' ') }.join("\n")
|
|
|
|
rescue Rumai::IXP::Error => e
|
|
#
|
|
# settings that are not supported in a particular wmii version
|
|
# are ignored, and those that are supported are (silently)
|
|
# applied. but a "bad command" error is raised nevertheless!
|
|
#
|
|
warn e.inspect
|
|
warn e.backtrace.join("\n")
|
|
end
|
|
|
|
launch 'xsetroot', '-solid', CONFIG['display']['background']
|
|
|
|
# column
|
|
fs.colrules.write CONFIG['display']['column']['rule']
|
|
|
|
# client
|
|
event 'CreateClient' do |client_id|
|
|
client = Client.new(client_id)
|
|
|
|
unless defined? @client_tags_by_regexp
|
|
@client_tags_by_regexp = CONFIG['display']['client'].map {|hash|
|
|
k, v = hash.to_a.first
|
|
[eval(k, TOPLEVEL_BINDING, "#{config_file}:display:client"), v]
|
|
}
|
|
end
|
|
|
|
if label = client.props.read rescue nil
|
|
catch :found do
|
|
@client_tags_by_regexp.each do |regexp, tags|
|
|
if label =~ regexp
|
|
client.tags = tags
|
|
throw :found
|
|
end
|
|
end
|
|
|
|
# force client onto current view
|
|
begin
|
|
client.tags = curr_tag
|
|
client.focus
|
|
rescue
|
|
# ignore
|
|
end
|
|
end
|
|
end
|
|
end
|
|
|
|
# status
|
|
action 'status' do
|
|
fs.rbar.clear
|
|
|
|
unless defined? @status_button_by_name
|
|
@status_button_by_name = {}
|
|
@status_button_by_file = {}
|
|
@on_click_by_status_button = {}
|
|
|
|
CONFIG['display']['status'].each_with_index do |hash, position|
|
|
name, defn = hash.to_a.first
|
|
|
|
# buttons appear in ASCII order of their IXP file name
|
|
file = "#{position}-#{name}"
|
|
|
|
button = eval(
|
|
"Button.new(fs.rbar[#{file.inspect}], #{defn['refresh']}) { #{defn['content']} }",
|
|
TOPLEVEL_BINDING, "#{config_file}:display:status:#{name}"
|
|
)
|
|
|
|
@status_button_by_name[name] = button
|
|
@status_button_by_file[file] = button
|
|
|
|
# mouse click handler
|
|
if code = defn['click']
|
|
@on_click_by_status_button[button] = eval(
|
|
"lambda {|mouse_button| #{code} }", TOPLEVEL_BINDING,
|
|
"#{config_file}:display:status:#{name}:click"
|
|
)
|
|
end
|
|
end
|
|
end
|
|
|
|
@status_button_by_name.each_value {|b| b.refresh }
|
|
|
|
end
|
|
|
|
##
|
|
# Returns the status button associated with the given name.
|
|
#
|
|
# ==== Parameters
|
|
#
|
|
# [name]
|
|
# Either the the user-defined name of
|
|
# the status button or the basename
|
|
# of the status button's IXP file.
|
|
#
|
|
def status_button name
|
|
@status_button_by_name[name] || @status_button_by_file[name]
|
|
end
|
|
|
|
##
|
|
# Refreshes the content of the status button with the given name.
|
|
#
|
|
# ==== Parameters
|
|
#
|
|
# [name]
|
|
# Either the the user-defined name of
|
|
# the status button or the basename
|
|
# of the status button's IXP file.
|
|
#
|
|
def status name
|
|
if button = status_button(name)
|
|
button.refresh
|
|
end
|
|
end
|
|
|
|
##
|
|
# Invokes the mouse click handler for the given mouse
|
|
# button on the status button that has the given name.
|
|
#
|
|
# ==== Parameters
|
|
#
|
|
# [name]
|
|
# Either the the user-defined name of
|
|
# the status button or the basename
|
|
# of the status button's IXP file.
|
|
#
|
|
# [mouse_button]
|
|
# The identification number of
|
|
# the mouse button (as defined
|
|
# by X server) that was clicked.
|
|
#
|
|
def status_click name, mouse_button
|
|
if button = status_button(name) and
|
|
handle = @on_click_by_status_button[button]
|
|
then
|
|
handle.call mouse_button.to_i
|
|
end
|
|
end
|
|
|
|
# control
|
|
action 'reload' do
|
|
# reload this wmii configuration
|
|
reload_config
|
|
end
|
|
|
|
action 'rehash' do
|
|
# scan for available programs and actions
|
|
@programs = find_programs(ENV['PATH'].squeeze(':').split(':'))
|
|
end
|
|
|
|
# kill all currently open clients
|
|
action 'clear' do
|
|
# firefox's restore session feature does not
|
|
# work unless the whole process is killed.
|
|
system 'killall firefox firefox-bin thunderbird thunderbird-bin'
|
|
|
|
# gnome-panel refuses to die by any other means
|
|
system 'killall -s TERM gnome-panel'
|
|
|
|
Thread.pass until clients.each do |c|
|
|
begin
|
|
c.focus # XXX: client must be on current view in order to be killed
|
|
c.kill
|
|
rescue
|
|
# ignore
|
|
end
|
|
end.empty?
|
|
end
|
|
|
|
# kill the window manager only; do not touch the clients!
|
|
action 'kill' do
|
|
fs.ctl.write 'quit'
|
|
end
|
|
|
|
# kill both clients and window manager
|
|
action 'quit' do
|
|
action 'clear'
|
|
action 'kill'
|
|
end
|
|
|
|
event 'Unresponsive' do |client_id|
|
|
client = Client.new(client_id)
|
|
|
|
IO.popen('xmessage -nearmouse -file - -buttons Kill,Wait -print', 'w+') do |f|
|
|
f.puts 'The following client is not responding.', ''
|
|
f.puts client.inspect
|
|
f.puts client.label.read
|
|
|
|
f.puts '', 'What would you like to do?'
|
|
f.close_write
|
|
|
|
if f.read.chomp == 'Kill'
|
|
client.slay
|
|
end
|
|
end
|
|
end
|
|
|
|
event 'Notice' do |*argv|
|
|
unless defined? @notice_mutex
|
|
require 'thread'
|
|
@notice_mutex = Mutex.new
|
|
end
|
|
|
|
Thread.new do
|
|
# prevent notices from overwriting each other
|
|
@notice_mutex.synchronize do
|
|
button = fs.rbar['!notice']
|
|
button.create unless button.exist?
|
|
|
|
# display the notice
|
|
message = argv.join(' ')
|
|
|
|
LOG.info message # also log it in case the user is AFK
|
|
button.write "#{CONFIG['display']['color']['notice']} #{message}"
|
|
|
|
# clear the notice
|
|
sleep [1, CONFIG['display']['notice'].to_i].max
|
|
button.remove
|
|
end
|
|
end
|
|
end
|
|
|
|
%w[key action event].each do |param|
|
|
if settings = CONFIG['control'][param]
|
|
settings.each do |name, code|
|
|
if param == 'key'
|
|
# expand ${...} expressions in shortcut key sequences
|
|
name = name.gsub(/\$\{(.+?)\}/) { CONFIG['control'][$1] }
|
|
end
|
|
|
|
eval "#{param}(#{name.inspect}) {|*argv| #{code} }",
|
|
TOPLEVEL_BINDING, "#{config_file}:control:#{param}:#{name}"
|
|
end
|
|
end
|
|
end
|
|
|
|
# script
|
|
action 'status'
|
|
action 'rehash'
|
|
|
|
eval CONFIG['script']['after'].to_s, TOPLEVEL_BINDING,
|
|
"#{config_file}:script:after"
|
|
|
|
end
|
|
|
|
##
|
|
# Reloads the entire wmii configuration.
|
|
#
|
|
def reload_config
|
|
LOG.info 'reload'
|
|
exec $0
|
|
end
|