2023-08-01 11:06:35 -05:00
-- SPDX-License-Identifier: GPL-3.0-or-later
2023-06-30 21:34:07 -05:00
2023-08-28 19:47:29 -05:00
---Tag management.
---
---This module provides utilities for creating and manipulating tags.
---
---A tag is a sort of marker for each of your windows. It allows you to present windows in ways that
---traditional workspaces cannot.
---
---More specifically:
2023-09-07 20:48:41 -05:00
---
2023-08-28 19:47:29 -05:00
--- - A window can have multiple tags.
2023-09-07 20:48:41 -05:00
--- - This means that you can have one window show up across multiple "workspaces" if you come
2023-08-28 19:47:29 -05:00
--- something like i3.
--- - An output can display multiple tags at once.
2023-09-07 20:48:41 -05:00
--- - This allows you to toggle a tag and have windows on both tags display at once.
2023-08-28 19:47:29 -05:00
--- This is helpful if you, say, want to reference a browser window while coding; you toggle your
--- browser's tag and temporarily reference it while you work without having to change screens.
---
---Many of the functions in this module take Tag|TagTable|TagTableNamed|string.
---This is a convenience so you don't have to get a tag object every time you want to do
---something with tags.
---
---Instead, you can pass in either:
2023-09-07 20:48:41 -05:00
---
2023-08-28 19:47:29 -05:00
--- - A string of the tag's name (ex. "1")
2023-09-07 20:48:41 -05:00
--- - This will get the first tag with that name on the focused output.
2023-08-28 19:47:29 -05:00
--- - A table where [1] is the name and [2] is the output (or its name) (ex. { "1", output.get_by_name("DP-1") })
2023-09-07 20:48:41 -05:00
--- - This will get the first tag with that name on the specified output.
2023-08-28 19:47:29 -05:00
--- - The same table as above, but keyed with `name` and `output` (ex. { name = "1", output = "DP-1" })
2023-09-07 20:48:41 -05:00
--- - This is simply for those who want more clarity in their config.
2023-08-28 19:47:29 -05:00
---
---If you need to get tags beyond the first with the same name, use a `get` function and find what you need.
2023-07-21 21:02:02 -05:00
---@class TagModule
local tag_module = { }
2023-07-18 12:37:40 -05:00
2023-07-12 18:50:41 -05:00
---@alias Layout
---| "MasterStack" # One master window on the left with all other windows stacked to the right.
---| "Dwindle" # Windows split in half towards the bottom right corner.
---| "Spiral" # Windows split in half in a spiral.
2023-07-17 20:40:56 -05:00
---| "CornerTopLeft" # One main corner window in the top left with a column of windows on the right and a row on the bottom.
---| "CornerTopRight" # One main corner window in the top right with a column of windows on the left and a row on the bottom.
---| "CornerBottomLeft" # One main corner window in the bottom left with a column of windows on the right and a row on the top.
---| "CornerBottomRight" # One main corner window in the bottom right with a column of windows on the left and a row on the top.
2023-07-12 18:50:41 -05:00
2023-09-07 20:36:49 -05:00
---@alias TagTable { name: string, output: (string|Output)? }
---@alias TagConstructor Tag|TagTable|string
2023-08-21 20:17:27 -05:00
2023-08-28 19:47:29 -05:00
---A tag object.
---
---This can be retrieved through the various `get` functions in the `TagModule`.
2023-08-26 22:26:32 -05:00
---@classmod
2023-07-18 10:31:08 -05:00
---@class Tag
2023-07-20 15:22:22 -05:00
---@field private _id integer The internal id of this tag.
2023-07-21 14:36:32 -05:00
local tag = { }
2023-07-18 10:31:08 -05:00
2023-07-21 21:02:02 -05:00
---Create a tag from an id.
---The id is the unique identifier for each tag.
---@param id TagId
2023-07-18 10:31:08 -05:00
---@return Tag
2023-07-21 21:02:02 -05:00
local function create_tag ( id )
2023-07-20 15:22:22 -05:00
---@type Tag
2023-07-21 21:02:02 -05:00
local t = { _id = id }
2023-07-18 10:31:08 -05:00
-- Copy functions over
2023-07-21 14:36:32 -05:00
for k , v in pairs ( tag ) do
2023-07-20 15:22:22 -05:00
t [ k ] = v
2023-07-18 10:31:08 -05:00
end
2023-07-20 15:22:22 -05:00
return t
end
---Get this tag's internal id.
2023-07-21 21:02:02 -05:00
---***You probably won't need to use this.***
2023-07-20 15:22:22 -05:00
---@return integer
2023-07-21 14:36:32 -05:00
function tag : id ( )
2023-07-20 15:22:22 -05:00
return self._id
2023-07-18 10:31:08 -05:00
end
2023-07-18 12:37:40 -05:00
---Get this tag's active status.
2023-07-19 18:55:22 -05:00
---@return boolean|nil active `true` if the tag is active, `false` if not, and `nil` if the tag doesn't exist.
2023-07-21 21:44:56 -05:00
---@see TagModule.active — The corresponding module function
2023-07-21 14:36:32 -05:00
function tag : active ( )
2023-07-21 21:02:02 -05:00
return tag_module.active ( self )
2023-07-18 12:37:40 -05:00
end
2023-07-19 18:55:22 -05:00
---Get this tag's name.
---@return string|nil name The name of this tag, or nil if it doesn't exist.
2023-07-21 21:44:56 -05:00
---@see TagModule.name — The corresponding module function
2023-07-21 14:36:32 -05:00
function tag : name ( )
2023-07-21 21:02:02 -05:00
return tag_module.name ( self )
2023-07-18 12:37:40 -05:00
end
2023-07-19 18:55:22 -05:00
---Get this tag's output.
---@return Output|nil output The output this tag is on, or nil if the tag doesn't exist.
2023-07-21 21:44:56 -05:00
---@see TagModule.output — The corresponding module function
2023-07-21 14:36:32 -05:00
function tag : output ( )
2023-07-21 21:02:02 -05:00
return tag_module.output ( self )
2023-07-19 18:55:22 -05:00
end
2023-07-21 14:36:32 -05:00
---Switch to this tag.
2023-07-21 21:44:56 -05:00
---@see TagModule.switch_to — The corresponding module function
2023-07-21 14:36:32 -05:00
function tag : switch_to ( )
2023-07-21 21:02:02 -05:00
tag_module.switch_to ( self )
2023-07-21 14:36:32 -05:00
end
---Toggle this tag.
2023-07-21 21:44:56 -05:00
---@see TagModule.toggle — The corresponding module function
2023-07-21 14:36:32 -05:00
function tag : toggle ( )
2023-07-21 21:02:02 -05:00
tag_module.toggle ( self )
2023-07-21 14:36:32 -05:00
end
2023-07-18 12:37:40 -05:00
---Set this tag's layout.
---@param layout Layout
2023-07-21 21:44:56 -05:00
---@see TagModule.set_layout — The corresponding module function
2023-07-21 14:36:32 -05:00
function tag : set_layout ( layout )
2023-07-21 21:02:02 -05:00
tag_module.set_layout ( self , layout )
2023-07-18 12:37:40 -05:00
end
-----------------------------------------------------------
2023-06-30 21:34:07 -05:00
2023-07-20 16:54:26 -05:00
---Add tags to the specified output.
2023-06-30 21:34:07 -05:00
---
2023-07-20 16:54:26 -05:00
---### Examples
2023-06-30 21:34:07 -05:00
---```lua
2023-07-19 18:55:22 -05:00
---local op = output.get_by_name("DP-1")
---if op ~= nil then
--- tag.add(op, "1", "2", "3", "4", "5") -- Add tags with names 1-5
2023-07-11 11:59:38 -05:00
---end
2023-09-07 20:48:41 -05:00
--
--- -- You can also pass in a table.
2023-07-20 16:54:26 -05:00
---local tags = {"Terminal", "Browser", "Code", "Potato", "Email"}
2023-09-07 20:48:41 -05:00
---tag.add(op, tags)
2023-07-11 11:59:38 -05:00
---```
---@param output Output The output you want these tags to be added to.
2023-07-20 16:54:26 -05:00
---@param ... string The names of the new tags you want to add.
---@overload fun(output: Output, tag_names: string[])
2023-07-21 21:02:02 -05:00
---@see Output.add_tags — The corresponding object method
function tag_module . add ( output , ... )
2023-07-20 16:54:26 -05:00
local varargs = { ... }
if type ( varargs [ 1 ] ) == " string " then
local tag_names = varargs
tag_names [ " n " ] = nil -- remove the length to make it a true array for serializing
SendMsg ( {
AddTags = {
output_name = output : name ( ) ,
tag_names = tag_names ,
} ,
} )
else
local tag_names = varargs [ 1 ] --[=[@as string[]]=]
SendMsg ( {
AddTags = {
output_name = output : name ( ) ,
tag_names = tag_names ,
} ,
} )
end
2023-06-30 21:34:07 -05:00
end
2023-08-21 20:17:27 -05:00
---Toggle a tag on the specified output. If the output isn't specified, toggle it on the currently focused output instead.
2023-07-11 11:59:38 -05:00
---
2023-07-18 15:12:23 -05:00
---### Example
2023-07-11 11:59:38 -05:00
---
---```lua
2023-07-11 16:10:31 -05:00
---local op = output.get_by_name("DP-1")
2023-08-21 20:17:27 -05:00
---
---tag.toggle("1") -- Toggle tag 1 on the focused output
---
2023-09-09 04:09:28 -05:00
---tag.toggle({ name = "1", output = "DP-1" }) -- Toggle tag 1 on "DP-1"
---tag.toggle({ name = "1", output = op }) -- Same as above
2023-08-21 20:17:27 -05:00
---
2023-09-07 20:48:41 -05:00
--- -- Using a tag object
2023-08-21 20:17:27 -05:00
---local t = tag.get_by_name("1")[1] -- `t` is the first tag with the name "1"
---tag.toggle(t)
2023-07-11 11:59:38 -05:00
---```
2023-09-07 20:36:49 -05:00
---@param t TagConstructor
2023-07-21 21:02:02 -05:00
---@see Tag.toggle — The corresponding object method
2023-08-21 20:17:27 -05:00
function tag_module . toggle ( t )
2023-09-07 20:36:49 -05:00
local t = tag_module.get ( t )
2023-08-21 20:17:27 -05:00
if t then
2023-07-21 14:36:32 -05:00
SendMsg ( {
ToggleTag = {
2023-08-21 20:17:27 -05:00
tag_id = t : id ( ) ,
2023-07-21 14:36:32 -05:00
} ,
} )
end
2023-06-30 21:34:07 -05:00
end
2023-07-11 16:10:31 -05:00
---Switch to a tag on the specified output, deactivating any other active tags on it.
2023-08-21 20:17:27 -05:00
---If the output is not specified, this uses the currently focused output instead.
2023-07-11 11:59:38 -05:00
---
---This is used to replicate what a traditional workspace is on some other Wayland compositors.
---
2023-07-21 21:02:02 -05:00
---### Examples
2023-07-11 11:59:38 -05:00
---```lua
2023-08-21 20:17:27 -05:00
---local op = output.get_by_name("DP-1")
---
---tag.switch_to("1") -- Switch to tag 1 on the focused output
---
2023-09-09 04:09:28 -05:00
---tag.switch_to({ name = "1", output = "DP-1" }) -- Switch to tag 1 on "DP-1"
---tag.switch_to({ name = "1", output = op }) -- Same as above
2023-08-21 20:17:27 -05:00
---
2023-09-07 20:48:41 -05:00
--- -- Using a tag object
2023-08-21 20:17:27 -05:00
---local t = tag.get_by_name("1")[1] -- `t` is the first tag with the name "1"
---tag.switch_to(t)
2023-07-11 11:59:38 -05:00
---```
2023-09-07 20:36:49 -05:00
---@param t TagConstructor
2023-07-21 21:02:02 -05:00
---@see Tag.switch_to — The corresponding object method
2023-08-21 20:17:27 -05:00
function tag_module . switch_to ( t )
2023-09-07 20:36:49 -05:00
local t = tag_module.get ( t )
2023-08-21 20:17:27 -05:00
if t then
2023-07-21 14:36:32 -05:00
SendMsg ( {
SwitchToTag = {
2023-08-21 20:17:27 -05:00
tag_id = t : id ( ) ,
2023-07-21 14:36:32 -05:00
} ,
} )
end
2023-07-01 19:06:37 -05:00
end
2023-07-21 14:36:32 -05:00
---Set a layout for the tag on the specified output. If no output is provided, set it for the tag on the currently focused one.
2023-07-21 21:02:02 -05:00
---
---### Examples
---```lua
2023-08-21 20:17:27 -05:00
---local op = output.get_by_name("DP-1")
2023-07-21 21:02:02 -05:00
---
2023-08-21 20:17:27 -05:00
---tag.set_layout("1", "Dwindle") -- Set tag 1 on the focused output to "Dwindle"
---
2023-09-07 20:36:49 -05:00
---tag.set_layout({ name = "1", output = "DP-1" }, "Dwindle") -- Set tag 1 on "DP-1" to "Dwindle"
---tag.set_layout({ name = "1", output = op }, "Dwindle") -- Same as above
2023-08-21 20:17:27 -05:00
---
2023-09-07 20:48:41 -05:00
--- -- Using a tag object
2023-08-21 20:17:27 -05:00
---local t = tag.get_by_name("1")[1] -- `t` is the first tag with the name "1"
2023-07-21 21:02:02 -05:00
---tag.set_layout(t, "Dwindle")
---```
2023-08-21 20:17:27 -05:00
---
2023-09-07 20:36:49 -05:00
---@param t TagConstructor
2023-07-21 14:36:32 -05:00
---@param layout Layout The layout.
2023-07-21 21:02:02 -05:00
---@see Tag.set_layout — The corresponding object method
2023-08-21 20:17:27 -05:00
function tag_module . set_layout ( t , layout )
2023-09-07 20:36:49 -05:00
local t = tag_module.get ( t )
2023-08-21 20:17:27 -05:00
if t then
2023-07-21 14:36:32 -05:00
SendMsg ( {
SetLayout = {
2023-08-21 20:17:27 -05:00
tag_id = t : id ( ) ,
2023-07-21 14:36:32 -05:00
layout = layout ,
} ,
} )
end
2023-08-21 20:17:27 -05:00
end
2023-07-21 14:36:32 -05:00
2023-08-21 20:17:27 -05:00
---Get a tag with the specified name and optional output.
---
---If the output isn't specified, the focused one is used.
---
---If you have duplicate tags on an output, this returns the first one.
---If you need access to all duplicates, use `tag.get_on_output`, `tag.get_by_name`, or `tag.get_all`
---and filter for what you need.
---
---### Examples
---```lua
---local t = tag.get("1")
---local t = tag.get({ name = "3" })
2023-09-07 20:36:49 -05:00
---local t = tag.get({ name = "1", output = "HDMI-A-0" })
2023-08-21 20:17:27 -05:00
---
---local op = output.get_by_name("DP-2")
---if op ~= nil then
--- local t = tag.get({ name = "Code", output = op })
---end
---```
2023-09-07 20:36:49 -05:00
---@param params TagConstructor
2023-08-21 20:47:51 -05:00
---@return Tag|nil
2023-08-21 20:17:27 -05:00
---
---@see TagModule.get_on_output
---@see TagModule.get_by_name
---@see TagModule.get_all
function tag_module . get ( params )
2023-09-07 20:36:49 -05:00
-- If creating from a tag object, just return the obj
if params.id then
return params --[[@as Tag]]
end
-- string passed in
if type ( params ) == " string " then
local op = require ( " output " ) . get_focused ( )
if op == nil then
return nil
end
local tags = tag_module.get_by_name ( params )
for _ , t in pairs ( tags ) do
if t : output ( ) and t : output ( ) : name ( ) == op : name ( ) then
return t
end
end
return nil
end
-- TagTable was passed in
local params = params --[[@as TagTable]]
local tag_name = params.name
local op = params.output
if op == nil then
local o = require ( " output " ) . get_focused ( )
if o == nil then
return nil
end
op = o
elseif type ( op ) == " string " then
local o = require ( " output " ) . get_by_name ( op )
if o == nil then
return nil
end
op = o
end
local tags = tag_module.get_by_name ( tag_name )
for _ , t in pairs ( tags ) do
if t : output ( ) and t : output ( ) : name ( ) == op : name ( ) then
return t
end
end
return nil
2023-07-12 18:50:41 -05:00
end
2023-07-18 10:31:08 -05:00
---Get all tags on the specified output.
---
2023-07-21 21:02:02 -05:00
---### Example
2023-07-18 10:31:08 -05:00
---```lua
2023-07-21 21:02:02 -05:00
---local op = output.get_focused()
---if op ~= nil then
--- local tags = tag.get_on_output(op) -- All tags on the focused output
---end
2023-07-18 10:31:08 -05:00
---```
---@param output Output
---@return Tag[]
2023-07-21 21:02:02 -05:00
---
---@see Output.tags — The corresponding object method
function tag_module . get_on_output ( output )
2023-07-21 15:04:39 -05:00
local response = Request ( {
2023-07-20 15:22:22 -05:00
GetOutputProps = {
output_name = output : name ( ) ,
2023-07-18 10:31:08 -05:00
} ,
2023-07-21 15:04:39 -05:00
} )
2023-07-18 10:31:08 -05:00
2023-07-20 15:22:22 -05:00
local tag_ids = response.RequestResponse . response.OutputProps . tag_ids
2023-07-18 10:31:08 -05:00
---@type Tag[]
local tags = { }
2023-07-20 15:22:22 -05:00
if tag_ids == nil then
return tags
end
2023-07-18 15:12:23 -05:00
for _ , tag_id in pairs ( tag_ids ) do
2023-07-21 21:02:02 -05:00
table.insert ( tags , create_tag ( tag_id ) )
2023-07-18 10:31:08 -05:00
end
return tags
end
2023-07-19 18:55:22 -05:00
---Get all tags with this name across all outputs.
2023-07-21 21:02:02 -05:00
---
---### Example
---```lua
2023-09-07 20:48:41 -05:00
--- -- Given one monitor with the tags "OBS", "OBS", "VSCode", and "Spotify"...
2023-07-21 21:02:02 -05:00
---local tags = tag.get_by_name("OBS")
2023-09-07 20:48:41 -05:00
--- -- ...will have 2 tags in `tags`, while...
2023-07-21 21:02:02 -05:00
---local no_tags = tag.get_by_name("Firefox")
2023-09-07 20:48:41 -05:00
--- -- ...will have `no_tags` be empty.
2023-07-21 21:02:02 -05:00
---```
---@param name string The name of the tag(s) you want.
2023-07-19 18:55:22 -05:00
---@return Tag[]
2023-07-21 21:02:02 -05:00
function tag_module . get_by_name ( name )
local t_s = tag_module.get_all ( )
2023-07-20 16:05:26 -05:00
---@type Tag[]
local tags = { }
2023-07-21 14:36:32 -05:00
for _ , t in pairs ( t_s ) do
2023-07-20 16:05:26 -05:00
if t : name ( ) == name then
table.insert ( tags , t )
end
end
return tags
end
2023-07-21 21:02:02 -05:00
---Get all tags across all outputs.
---
---### Example
---```lua
2023-09-07 20:48:41 -05:00
--- -- With two monitors with the same tags: "1", "2", "3", "4", and "5"...
2023-07-21 21:02:02 -05:00
---local tags = tag.get_all()
2023-09-07 20:48:41 -05:00
--- -- ...`tags` should have 10 tags, with 5 pairs of those names across both outputs.
2023-07-21 21:02:02 -05:00
---```
2023-07-20 16:05:26 -05:00
---@return Tag[]
2023-07-21 21:02:02 -05:00
function tag_module . get_all ( )
2023-07-21 15:04:39 -05:00
local response = Request ( " GetTags " )
2023-07-19 18:55:22 -05:00
local tag_ids = response.RequestResponse . response.Tags . tag_ids
---@type Tag[]
local tags = { }
for _ , tag_id in pairs ( tag_ids ) do
2023-07-21 21:02:02 -05:00
table.insert ( tags , create_tag ( tag_id ) )
2023-07-19 18:55:22 -05:00
end
return tags
end
2023-07-21 21:02:02 -05:00
---Get the specified tag's name.
---
---### Example
---```lua
2023-09-07 20:48:41 -05:00
--- -- Assuming the tag `Terminal` exists...
2023-07-21 21:02:02 -05:00
---print(tag.name(tag.get_by_name("Terminal")[1]))
2023-09-07 20:48:41 -05:00
--- -- ...should print `Terminal`.
2023-07-21 21:02:02 -05:00
---```
---@param t Tag
---@return string|nil
---@see Tag.name — The corresponding object method
function tag_module . name ( t )
local response = Request ( {
GetTagProps = {
tag_id = t : id ( ) ,
} ,
} )
local name = response.RequestResponse . response.TagProps . name
return name
end
---Get whether or not the specified tag is active.
---@param t Tag
---@return boolean|nil
---@see Tag.active — The corresponding object method
function tag_module . active ( t )
local response = Request ( {
GetTagProps = {
tag_id = t : id ( ) ,
} ,
} )
local active = response.RequestResponse . response.TagProps . active
return active
end
---Get the output the specified tag is on.
---@param t Tag
---@return Output|nil
2023-07-21 21:44:56 -05:00
---@see OutputModule.get_for_tag — The called function
2023-07-21 21:02:02 -05:00
---@see Tag.output — The corresponding object method
function tag_module . output ( t )
return require ( " output " ) . get_for_tag ( t )
end
2023-09-09 04:01:55 -05:00
---@class LayoutCycler
---@field next fun(output: (Output|OutputName)?) Change the first active tag on `output` to its next layout. If `output` is empty, the focused output is used.
---@field prev fun(output: (Output|OutputName)?) Change the first active tag on `output` to its previous layout. If `output` is empty, the focused output is used.
---Given an array of layouts, this will create two functions; one will cycle forward the layout
---for the provided tag, and one will cycle backward.
2023-09-09 20:52:52 -05:00
---
--- ### Example
---```lua
---local layout_cycler = tag.layout_cycler({ "Dwindle", "Spiral", "MasterStack" })
---
---layout_cycler.next() -- Go to the next layout on the first tag of the focused output
---layout_cycler.prev() -- Go to the previous layout on the first tag of the focused output
---
---layout_cycler.next("DP-1") -- Do the above but on "DP-1" instead
---layout_cycler.prev(output.get_by_name("DP-1")) -- With an output object
---```
2023-09-09 04:01:55 -05:00
---@param layouts Layout[] The available layouts.
---@return LayoutCycler layout_cycler A table with the functions `next` and `prev`, which will cycle layouts for the given tag.
function tag_module . layout_cycler ( layouts )
local indices = { }
-- Return empty functions if layouts is empty
if # layouts == 0 then
return {
next = function ( _ ) end ,
prev = function ( _ ) end ,
}
end
return {
---@param output (Output|OutputName)?
next = function ( output )
if type ( output ) == " string " then
output = require ( " output " ) . get_by_name ( output )
end
output = output or require ( " output " ) . get_focused ( )
if output == nil then
return
end
local tags = output : tags ( )
for _ , tg in pairs ( tags ) do
if tg : active ( ) then
local id = tg : id ( )
if id == nil then
return
end
if # layouts == 1 then
indices [ id ] = 1
elseif indices [ id ] == nil then
indices [ id ] = 2
else
if indices [ id ] + 1 > # layouts then
indices [ id ] = 1
else
indices [ id ] = indices [ id ] + 1
end
end
tg : set_layout ( layouts [ indices [ id ] ] )
break
end
end
end ,
---@param output (Output|OutputName)?
prev = function ( output )
if type ( output ) == " string " then
output = require ( " output " ) . get_by_name ( output )
end
output = output or require ( " output " ) . get_focused ( )
if output == nil then
return
end
local tags = output : tags ( )
for _ , tg in pairs ( tags ) do
if tg : active ( ) then
local id = tg : id ( )
if id == nil then
return
end
if # layouts == 1 then
indices [ id ] = 1
elseif indices [ id ] == nil then
indices [ id ] = # layouts - 1
else
if indices [ id ] - 1 < 1 then
indices [ id ] = # layouts
else
indices [ id ] = indices [ id ] - 1
end
end
tg : set_layout ( layouts [ indices [ id ] ] )
break
end
end
end ,
}
end
2023-07-21 21:02:02 -05:00
return tag_module