wiki.xw3.org

Welcome to the xw3 Wiki! Powered by hanez

User Tools

Site Tools


awesomewm:drop-down_terminal

Differences

This shows you the differences between two versions of the page.

Link to this comparison view

Both sides previous revisionPrevious revision
Next revision
Previous revision
awesomewm:drop-down_terminal [2023-08-28] hanezawesomewm:drop-down_terminal [2025-03-18] (current) – ↷ Page moved from user:hanez:awesomewm:drop-down_terminal to awesomewm:drop-down_terminal hanez
Line 1: Line 1:
-===== Drop-down Terminal ====+===== AwesomeWM Drop-down Terminal ====
  
 ==== Introduction ==== ==== Introduction ====
Line 108: Line 108:
    end    end
 end</code> end</code>
 +
 +==== Another solution  ====
 +
 +The previous solution have two little quirks: the console window is detected as the first window being managed just after requesting one. There could be a race condition but it is unlikely. The second associated quirk is that when you restart awesome, you lose the fact that this window is a drop-down terminal. You now have one terminal which is sticky. This is a bit disturbing. Here is another solution that works around this by relying on a name given to the dropdown terminal: 
 +
 +<code>-- Quake like console on top
 +-- Similar to:
 +--   http://git.sysphere.org/awesome-configs/tree/scratch/drop.lua
 +
 +-- But uses a different implementation. The main difference is that we
 +-- are able to detect the Quake console from its name
 +-- (QuakeConsoleNeedsUniqueName by default).
 +
 +-- Use:
 +
 +-- local quake = require("quake")
 +-- local quakeconsole = {}
 +-- for s = 1, screen.count() do
 +--    quakeconsole[s] = quake({ terminal = config.terminal,
 +--         height = 0.3,
 +--                              screen = s })
 +-- end
 +
 +-- config.keys.global = awful.util.table.join(
 +--    config.keys.global,
 +--    awful.key({ modkey }, "`",
 +--      function () quakeconsole[mouse.screen]:toggle() end)
 +
 +-- If you have a rule like "awful.client.setslave" for your terminals,
 +-- ensure you use an exception for
 +-- QuakeConsoleNeedsUniqueName. Otherwise, you may run into problems
 +-- with focus.
 +
 +local setmetatable = setmetatable
 +local string = string
 +local awful  = require("awful")
 +local capi   = { mouse = mouse,
 + screen = screen,
 + client = client,
 + timer = timer }
 +
 +-- I use a namespace for my modules...
 +module("quake")
 +
 +local QuakeConsole = {}
 +
 +-- Display
 +function QuakeConsole:display()
 +   -- First, we locate the terminal
 +   local client = nil
 +   local i = 0
 +   for c in awful.client.cycle(function (c)
 +   -- c.name may be changed!
 +   return c.instance == self.name
 +        end,
 +        nil, self.screen) do
 +      i = i + 1
 +      if i == 1 then
 + client = c
 +      else
 + -- Additional matching clients, let's remove the sticky bit
 + -- which may persist between awesome restarts. We don't close
 + -- them as they may be valuable. They will just turn into a
 + -- classic terminal.
 + c.sticky = false
 + c.ontop = false
 + c.above = false
 +      end
 +   end
 +
 +   if not client and not self.visible then
 +      -- The terminal is not here yet but we don't want it yet. Just do nothing.
 +      return
 +   end
 +
 +   if not client then
 +      -- The client does not exist, we spawn it
 +      awful.util.spawn(self.terminal .. " " .. string.format(self.argname, self.name),
 +        false, self.screen)
 +      return
 +   end
 +
 +   -- Comptute size
 +   local geom = capi.screen[self.screen].workarea
 +   local width, height = self.width, self.height
 +   if width  <= 1 then width = geom.width * width end
 +   if height <= 1 then height = geom.height * height end
 +   local x, y
 +   if     self.horiz == "left"  then x = geom.x
 +   elseif self.horiz == "right" then x = geom.width + geom.x - width
 +   else   x = geom.x + (geom.width - width)/2 end
 +   if     self.vert == "top"    then y = geom.y
 +   elseif self.vert == "bottom" then y = geom.height + geom.y - height
 +   else   y = geom.y + (geom.height - height)/2 end
 +
 +   -- Resize
 +   awful.client.floating.set(client, true)
 +   client.border_width = 0
 +   client.size_hints_honor = false
 +   client:geometry({ x = x, y = y, width = width, height = height })
 +
 +   -- Sticky and on top
 +   client.ontop = true
 +   client.above = true
 +   client.skip_taskbar = true
 +   client.sticky = true
 +
 +   -- This is not a normal window, don't apply any specific keyboard stuff
 +   client:buttons({})
 +   client:keys({})
 +
 +   -- Toggle display
 +   if self.visible then
 +      client.hidden = false
 +      client:raise()
 +      capi.client.focus = client
 +   else
 +      client.hidden = true
 +   end
 +end
 +
 +-- Create a console
 +function QuakeConsole:new(config)
 +   -- The "console" object is just its configuration.
 +
 +   -- The application to be invoked is:
 +   --   config.terminal .. " " .. string.format(config.argname, config.name)
 +   config.terminal = config.terminal or "xterm" -- application to spawn
 +   config.name     = config.name     or "QuakeConsoleNeedsUniqueName" -- window name
 +   config.argname  = config.argname  or "-name %s"     -- how to specify window name
 +
 +   -- If width or height <= 1 this is a proportion of the workspace
 +   config.height   = config.height   or 0.25        -- height
 +   config.width    = config.width    or 1        -- width
 +   config.vert     = config.vert     or "top"        -- top, bottom or center
 +   config.horiz    = config.horiz    or "center"       -- left, right or center
 +
 +   config.screen   = config.screen or capi.mouse.screen
 +   config.visible  = config.visible or false -- Initially, not visible
 +
 +   local console = setmetatable(config, { __index = QuakeConsole })
 +   capi.client.add_signal("manage",
 +   function(c)
 +      if c.instance == console.name and c.screen == console.screen then
 + console:display()
 +      end
 +   end)
 +   capi.client.add_signal("unmanage",
 +   function(c)
 +      if c.instance == console.name and c.screen == console.screen then
 + console.visible = false
 +      end
 +   end)
 +
 +   -- "Reattach" currently running QuakeConsole. This is in case awesome is restarted.
 +   local reattach = capi.timer { timeout = 0 }
 +   reattach:add_signal("timeout",
 +        function()
 +   reattach:stop()
 +   console:display()
 +        end)
 +   reattach:start()
 +   return console
 +end
 +
 +-- Toggle the console
 +function QuakeConsole:toggle()
 +   self.visible = not self.visible
 +   self:display()
 +end
 +
 +setmetatable(_M, { __call = function(_, ...) return QuakeConsole:new(...) end })</code>
 +
 +This only works for applications that accept to be given a name through the command line (xterm, rxvt). Example of use : 
 +
 +<code>local quake = require("quake")
 +
 +local quakeconsole = {}
 +for s = 1, screen.count() do
 +   quakeconsole[s] = quake({ terminal = config.terminal,
 +      height = 0.3,
 +      screen = s })
 +end
 +
 +config.keys.global = awful.util.table.join(
 +   config.keys.global,
 +   awful.key({ modkey }, "`",
 +      function () quakeconsole[mouse.screen]:toggle() end)</code>
 +
 +Source: [[https://web.archive.org/web/20121101152450/http://awesome.naquadah.org/wiki/Drop-down_terminal]]
awesomewm/drop-down_terminal.1693229214.txt.gz · Last modified: 2023-08-28 by hanez