local setmetatable,type = setmetatable, type local ipairs, pairs = ipairs, pairs local button = require( "awful.button" ) local beautiful = require( "beautiful" ) local tag = require( "awful.tag" ) local client2 = require( "awful.client" ) local radical = require( "radical" ) local util = require( "awful.util" ) local wibox = require( "wibox" ) local tag_list = require( "radical.impl.taglist" ) local capi = { client = client, mouse = mouse, screen = screen} local module,pause_monitoring = {},false -- Using viewonly to change tag will create an unwanted -- focus event in the awful stack. While this module does -- not use it, it try to be a good citizen and avoid corrupting -- it. local lock_history = false local function awful_client_history_add(c) if not lock_history then client2.focus.history.add(c) end end -- Keep its own history instead of using awful.client.focus.history local focusIdx,focusTable = 1,setmetatable({}, { __mode = 'v' }) local focusTag = setmetatable({}, { __mode = 'v' }) local function push_focus(c) if c and not pause_monitoring then focusTable[c] = focusIdx focusIdx = focusIdx + 1 focusTag[c] = c.screen.selected_tag end end capi.client.connect_signal("focus", push_focus) -- Remove client when closed client.connect_signal("unmanage", function (c) focusTable[c] = nil end) local function compare(a,b) return a[1] > b[1] end local function get_history(screen) local result = {} for k,v in pairs(focusTable) do result[#result+1] = {v,k} end local orphanCount = -100 for k,v in ipairs(capi.client.get(screen or 1)) do if not focusTable[v] then result[#result+1] = setmetatable({orphanCount,v}, { __mode = 'v' }) orphanCount = orphanCount -1 end end table.sort(result,compare) return result end -- Simulate a titlebar local function button_group(args) local widget = wibox.widget.imagebox() widget:set_image( module.titlebar_path.. args.field .."_normal_"..(args.checked() and "active" or "inactive")..".png" ) widget:buttons( util.table.join(button({ }, 1 , args.onclick))) return widget end local function select_next(menu) local item = menu.next_item if not item then return end item.selected = true item.button1(nil,nil,nil,nil,true) return true end local function is_in_tag(t,c) for k,v in ipairs(c:tags()) do if t == v then return true end end return false end local function reload_infoshapes(client,item) local infoshapes = {} for k,v in ipairs(client:tags()) do infoshapes[#infoshapes+1] = {text = v.name} end if item then item.infoshapes = infoshapes end return infoshapes end local function reload_highlight(i) if i.selected then local hl = {} for k,v in ipairs(i.client:tags()) do hl[#hl+1] = v end tag_list.highlight(hl) i._internal.border_color_back = i._internal.border_color_back or i.client.border_color i.client.border_color = beautiful.bg_urgent elseif i._internal.border_color_back then i.client.border_color = i._internal.border_color_back end end --TODO save the settings somewhere local conf = { tags = true, screens = true, minimized = true, } local function toggle_all_tags(item) conf.tags = not conf.tags item.checked = conf.tags end local function toggle_all_screens(item) conf.screens = not conf.screens item.checked = conf.screens end local function toggle_maximized(item) conf.minimized = not conf.minimized item.checked = conf.minimized end local function new(args) local histo = get_history(--[[screen]]) if #histo == 0 then return end local scr = capi.client.focus and capi.client.focus.screen or capi.mouse.screen local t,auto_release = scr.selected_tag,args.auto_release local currentMenu = radical.box({filter = true, show_filter=not auto_release, autodiscard = true, disable_markup=true,fkeys_prefix=not auto_release,width=(((capi.screen[scr]).geometry.width)/2), icon_transformation = beautiful.alttab_icon_transformation,filter_underlay="Use [Shift] and [Control] to toggle clients",filter_underlay_color=beautiful.menu_bg_normal, filter_placeholder="Type to filter"}) currentMenu.margins.top = currentMenu.border_width currentMenu.margins.bottom = currentMenu.border_width if not auto_release then local pref_bg = wibox.container.background() local pref_l = wibox.layout.align.horizontal() pref_bg.fit = function(s,c,w,h) local w2,h2 = wibox.container.background.fit(s,c,w,h) return w2,currentMenu.item_height end pref_bg:set_bg(currentMenu.bg_alternate) local tb2= wibox.widget.textbox() tb2:set_markup(" "..#histo.." clients") pref_l:set_first(tb2) pref_bg:set_widget(pref_l) local pref_menu,pref_menu_l = radical.bar{item_style=radical.item.style.basic} pref_menu:add_widget(wibox.widget.textbox "Display: ") pref_menu:add_widget(radical.widgets.separator(pref_menu,radical.widgets.separator.VERTICAL)) pref_menu:add_item{text="All tags ", checkable = true, checked = true, button1 = toggle_all_tags} pref_menu:add_widget(radical.widgets.separator(pref_menu,radical.widgets.separator.VERTICAL)) pref_menu:add_item{text="Minimized ", checkable = true, checked = true, button1 = toggle_all_screens} pref_menu:add_widget(radical.widgets.separator(pref_menu,radical.widgets.separator.VERTICAL)) pref_menu:add_item{text="All Screens ", checkable = true, checked = true, button1 = toggle_maximized} -- pref_menu:add_widget(radical.widgets.separator(pref_menu,radical.widgets.separator.VERTICAL)) pref_l:set_third(pref_menu_l) currentMenu:add_prefix_widget(pref_bg) end currentMenu:add_key_hook({}, "Tab", "press", select_next) currentMenu:add_key_hook({}, "Shift_L", "press", function() local item = currentMenu._current_item item.checked = not item.checked local c = item.client if c.screen ~= scr then return end client2.toggletag (t, c) reload_infoshapes(c,item) if not auto_release then reload_highlight(item) end if item._internal.border_color_back then c.border_color = item._internal.border_color_back end return true end) currentMenu:add_key_hook({}, "Control_L", "press", function() local item = currentMenu._current_item item.checked = not item.checked local c = item.client if c.screen ~= scr then return end client2.movetotag(t, c) reload_infoshapes(c,item) if not auto_release then reload_highlight(item) end return true end) for k,v2 in ipairs(histo) do local l,v = wibox.layout.fixed.horizontal(),v2[2] if not auto_release and module.titlebar_path then l:add( button_group({client = v, field = "floating" , focus = false, checked = function() return v.floating end, onclick = function() v.floating = not v.floating end })) l:add( button_group({client = v, field = "maximized", focus = false, checked = function() return v.maximized end, onclick = function() v.maximized = not v.maximized end })) l:add( button_group({client = v, field = "sticky" , focus = false, checked = function() return v.sticky end, onclick = function() v.sticky = not v.sticky end })) l:add( button_group({client = v, field = "ontop" , focus = false, checked = function() return v.ontop end, onclick = function() v.ontop = not v.ontop end })) l:add( button_group({client = v, field = "close" , focus = false, checked = function() return false end, onclick = function() v:kill(); currentMenu.visible = false; end })) l.fit = function (s,c,w,h) return 5*h,h end end local underlays = reload_infoshapes(v) local i = currentMenu:add_item({ text = v.name, icon = v.icon or module.default_icon, suffix_widget = not auto_release and l or nil, selected = capi.client.focus and capi.client.focus == v, infoshapes = underlays, checkable = (not auto_release) and v.screen == scr, checked = v.screen == scr and (not auto_release and is_in_tag(t,v)) or nil, button1 = function(a,b,c,d,no_hide) local t = focusTag[v] or v:tags()[1] if t and t.selected == false and not util.table.hasitem(v:tags(),capi.screen[v.screen].selected_tag) then lock_history = true tag.viewonly(t) lock_history = false end capi.client.focus = v v:raise() if not no_hide then currentMenu.visible = false end end, }) i.client = v if not auto_release then i:connect_signal("selected::changed",reload_highlight) end end if auto_release then currentMenu:add_key_hook({}, "Alt_L", "release", function(_) currentMenu.visible = false return false end) select_next(currentMenu) end pause_monitoring,currentMenu.visible = true, true currentMenu:connect_signal("visible::changed",function(m) if not m.visible then pause_monitoring = false push_focus(capi.client.focus) if not auto_release then tag_list.highlight() end if currentMenu._current_item and currentMenu._current_item._internal.border_color_back then currentMenu._current_item.client.border_color = currentMenu._current_item._internal.border_color_back end end end) return currentMenu end function module.altTab(args) new({leap = 1,auto_release = (args or {}).auto_release}) end function module.altTabBack(args) new({leap = -1,auto_release = (args or {}).auto_release}) end -- Sometime need to lock .add capi.client.disconnect_signal("focus", client2.focus.history.add) capi.client.connect_signal("focus", awful_client_history_add) return setmetatable(module, { __call = function(_, ...) return new(...) end }) -- kate: space-indent on; indent-width 2; replace-tabs on;