local args = ... local ldoc = nil -- luacheck: globals new_type -- Configuration file for ldoc project='awesome' -- luacheck: globals project title='awesome API documentation' -- luacheck: globals title description='API documentation for awesome, a highly configurable X window manager (version @AWESOME_VERSION@).' -- luacheck: globals description format='discount' -- luacheck: globals format dir='../doc' -- luacheck: globals dir -- Make the docs prettier pretty='lua' -- luacheck: globals pretty style=true -- luacheck: globals style template=true -- luacheck: globals template backtick_references=true -- luacheck: globals backtick_references merge=true -- luacheck: globals merge use_markdown_titles=true -- luacheck: globals use_markdown_titles wrap=true -- luacheck: globals wrap -- luacheck: globals full_description full_description = [[ Welcome to the documentation for the Awesome window manager. Below you find an overview of the individual parts which links to the full documentation. If you are a new user, you may want to read @{07-my-first-awesome.md} to get started. In @{05-awesomerc.md}, the default configuration is explained. If you already used awesome in the past, @{89-NEWS.md} and @{17-porting-tips.md} should be useful for you. ### Default configuration components name:
### Guides
@{07-my-first-awesome.md|Getting started} @{90-FAQ.md|FAQ} @{01-readme.md|Read Me} @{89-NEWS.md|NEWS}
@{03-declarative-layout.md|The widget system} @{09-options.md|Startup options} @{05-awesomerc.md|The default rc.lua} @{08-client-layout-system.md|Window management}
## Major libraries AwesomeWM ship multiple libraries. Here is an overview of the purpose and scope of those libraries.
Library Description
`gears`Utilities such as color parsing and objects
`wibox`Awesome own generic widget framework
`awful`Everything related to window managment
`awful.widget`Window management related widgets
`awful.layout`The default stateless client tiling module.
`ruled`Define declarative rules on various events
`naughty`Notifications
`menubar`XDG (application) menu implementation
`beautiful`Awesome theme module
]] topics={ '00-authors.md', '01-readme.md', '02-contributing.md', '03-declarative-layout.md', '04-new-widgets.md', '05-awesomerc.md', '06-appearance.md', '07-my-first-awesome.md', '08-client-layout-system.md', '09-options.md', '10-building-and-testing.md', '16-using-cairo.md', '17-porting-tips.md', '90-FAQ.md', '89-NEWS.md', } local databases, named_tags, item_id, is_init = {}, {}, 1, false local all_theme_vars, delayed_collect = nil, {} local function collect_item_common(tab, mod, item) tab[item.name] = tab[item.name] or {} local var = tab[item.name] var[mod.name] = var[mod.name] or {} var[mod.name][#var[mod.name] + 1] = item end --- Add another custom tag **value** to an existing item. -- @tparam string tag_data The raw/unparsed string for the -- tag. It will be parsed like any other tags. local function add_custom_tag_common(item, tag_name, meta_tag, tag_data) item.tags = item.tags or {} item.tags[tag_name] = item.tags[tag_name] or {} item.tags[tag_name][#item.tags[tag_name] + 1] = tag_data if meta_tag.collect_callback then delayed_collect[#delayed_collect+1] = {item, item.module, tag_name, tag_data} end end -- Wrap the global `new_type` to allow us to extend what types can do, the -- same is done below for custom tags. local function create_type(args) new_type(args.name, args.title, args.project_level, args.subfield_title) databases[args.name] = databases[args.name] or {} databases[args.name].collect = args.collect_callback databases[args.name].finish = args.finish_callback return args end -- Add the @usebeautiful from the optional/default value. local function auto_add_usebeautiful(item) local vars, var_names = {}, {} -- Datamine the default values to autogenerate the @usebeautiful. for parm in ldoc.modules.iter(item.params) do for p in ldoc.modules.iter(item:subparam(parm)) do local def = item:default_of_param(p) if def and def ~= true then for var in def:gmatch("beautiful[.][a-z_]+") do if not var_names[var] then vars[#vars+1] = var end var_names[var] = true end end end end for _, var in ipairs(vars) do named_tags.usebeautiful:add_to_item(item, var) end end -- The first stereotype are the constructors. create_type { name = "constructorfct", title = "Constructors", project_level = false, subfield_title = "Parameters", collect_callback = auto_add_usebeautiful, } -- Some constructors like `awful.key` have both a named parameter -- syntax and a multiple function parameter syntax. create_type { name = "constructorfct2", title = "ldoc_skip", project_level = false, subfield_title = "Parameters", } -- Hack to get the functions on top of the signals and properties create_type { name = "function", title = "Functions", project_level = false, subfield_title = "Parameters", } -- For "classes", use an explicit type for static functions. This allows -- @function and its implicit cousin to be banned in the CI. create_type { name = "staticfct", title = "Static module functions", project_level = false, subfield_title = "Parameters", } -- Documentation for objects properties create_type { name = "property", title = "Object properties", project_level = false, subfield_title = "Type constraints", collect_callback = auto_add_usebeautiful, finish_callback = function(item) -- All properties need to have a type. Otherwise they don't render -- properly. Also, because typed properties are kind of the point of -- switching to ldoc in the first place. if (not item.params) or not item.params[1] then print( "WARNING: The ".. item.name .." property from "..item.module.name.." is missing it's type." ) return end local param = item:subparam(item.params[1]) local type = item:type_of_param(param[1]) -- Force people to use @tparam because it is easier to lint and allows -- multiple types. if (not type) or type == "" then print( "WARNING: Property ".. item.name .." from "..item.module.name.." either ".. " doesn't have a type or uses @param instead of @tparam. @tparam is required".. " because it allows multi-type and better default value handling." ) end -- One of the repeated problem we have is the first word of the -- description being removed because it is used as the property name. -- This "rule" might be stupid, but it prevents it from accidentally -- happening again. if item.params[1] ~= item.name:match("[.]?("..item.params[1]..")$") then print( "WARNING: Property ".. item.name .." from "..item.module.name.." @tparam name".. " doesn't match the property name. For linting purpose, please fix this." ) end end } -- Documentation for objects deprecated properties create_type { name = "deprecatedproperty", title = "Deprecated object properties", project_level = false, subfield_title = "Type constraints", } -- Documentation for objects deprecated methods create_type { name = "deprecatedmethod", title = "Deprecated object methods", project_level = false, subfield_title = "Parameters", } -- Use a custom type for the methods to bypass the faulty ldoc built-in detection. -- (yes, the space after Methods *is* on purpose to avoid clashing with ldoc -- internal "methods" concept) create_type { name = "method", title = "Object methods ", project_level = false, subfield_title = "Parameters", } -- New type for signals. local all_signals all_signals = create_type { name = "signal", title = "Signals", project_level = false, subfield_title = "Arguments", collection = {}, collect_callback = function(item, mod) collect_item_common(all_signals.collection, mod, item) end, } -- Deprecated signals. create_type { name = "deprecatedsignal", title = "Deprecated signals", project_level = false, subfield_title = "Arguments", } -- New type for signals connections create_type { name = "signalhandler", title = "Request handlers", project_level = false, subfield_title = "Arguments", } -- Allow objects to define a set of beautiful properties affecting them all_theme_vars = create_type { name = "beautiful", title = "Theme variables", project_level = false, subfield_title = "Type constraints", collection = {}, collect_callback = function(item, mod) if item.name:sub(1,4) ~= "beau" then -- This would be a bug. if item.name:match("[.]") then print("WANRING: A beautiful variable is called `", item.name, "`. This name is invalid.") end -- This happens because the `ldoc` messes with the name. All variables if item.module.name == "beautiful" then item.name = "beautiful." .. item.name else print( "WARNING: All theme variable need a `beautiful.` prefix. It appears", item.name, " lacks it" ) end end collect_item_common(all_theme_vars.collection, mod, item) -- Those are global variables, there can be only one per name. if all_theme_vars.collection[item.name] then local mod = pairs(all_theme_vars.collection[item.name])(all_theme_vars.collection[item.name]) if mod ~= item.module.name then print("ERROR: "..item.name, "from", item.module.name, "is already defined in module ", mod) end end end, finish_callback = function(item) if item.tags["beautiful_used_by"] then return end -- Every variable should be consumed by something. Better "park" some -- variable in the constructors than leave them disconnected from the -- global model. print("WARNING: ", item.name, "is not used by anything, add @usebeautiful or @propbeautiful") end } -- Put deprecated methods in their own section create_type { name = "deprecated", title = "Deprecated functions", project_level = false, subfield_title = "Parameters", } -- For the legacy stateless layout related functions create_type { name = "legacylayout", title = "Layout related functions", project_level = false, subfield_title = "Parameters", } -- Have a category for the client layouts create_type { name = "clientlayout", title = "Client layouts", project_level = false, subfield_title = "Parameters", } -- Source functions for the taglist/tasklist/layoutlist create_type { name = "sourcefunction", title = "List source functions", project_level = false, } -- Document some callback prototypes create_type { name = "callback", title = "Callback functions prototype", project_level = false, subfield_title = "Parameters", } -- gears.matcher / awful.rules sources create_type { name = "rulesources", title = "Rule sources", project_level = false, subfield_title = "Parameters", } -- gears.matcher / awful.rules rule components create_type { name = "rulecomponent", title = "Rule components", project_level = false, subfield_title = "Type", } -- Filter functions for the taglist/tasklist/layoutlist create_type { name = "filterfunction", title = "List filters", project_level = false, } -- Extra client properties available only in awful.rules/spawn constructs create_type { name = "clientruleproperty", title = "Extra properties available in the rules", project_level = false, subfield_title = "Type", } -- Extra *matching* properties for rules. create_type { name = "matchingproperty", title = "Extra matching properties used in rules", project_level = false, subfield_title = "Type", } -- Simulate the default "params" parser format, except the optional "[]" section -- needs a space. local function parse_custom_tags(text, params) text = text:gmatch("[ ]*(.*)$")() local raw_extra = "" if text:sub(1,1) == '[' then local count = 1 -- Find the matching ']' for i=2, text:len() do local char = text:sub(i,i) if char == '[' then count = count + 1 elseif char == ']' then count = count - 1 end raw_extra = raw_extra..char if count == 0 then text = text:sub(i+2) break end end end -- Split the remaining text into words. local words, values, description = {}, {}, {} for word in text:gmatch("[^ \n\r]*") do if word ~= "" then words[#words+1] = word end end for idx, word in ipairs(words) do if idx <= #params then local name = params[idx].name values[name] = { name = name, title = params[idx].title or name, value = word } else description[#description+1] = word values.description = values.description and values.description.." "..word or word end end return values end -- Mimic the template classes. local function default_format_callback(self, params, _, md) local ret = "" if self.table then -- All columns are mandatory for _, p in ipairs(self.params) do local content = params[p.name] and params[p.name].value or "" ret = ret..""..(p.markdown and md("`"..content.."`") or content).."" end return ret .. ""..md(params.description).."" else if params.name then ret = ''.. md("`"..params.name.value.."`").. " " end if params.type then ret = ret .. ''.. md("`"..params.type.value.."`").. "" end return ret.." "..md(params.description) end end -- Generate a format function. local function default_format(self, callback, name) return function(raw, item, md) local p = name and item.parsed_tags and item.parsed_tags[name] and item.parsed_tags[name][raw] if not p then p = parse_custom_tags(raw, self.params or {}) end return (callback or default_format_callback)(self, p, item, md) end end -- Add a new @something which can be used in any types. -- @tparam table args -- @tparam string args.name The name. -- @tparam string args.hidden Show in the doc or for internal use only. -- @tparam table args.table Show the items in a table rather than a list. The -- content is the list of header names. -- @tparam table args.params The parameters (table with name, tilte, format). -- @tparam boolean[opt=true] args.auto_subtags Create the usage and tparams subtags. local add_custom_tag add_custom_tag = function(args) local name = args.name args.name, args[1] = nil, name custom_tags = custom_tags or {} databases[name] = databases[name] or {} databases[name].collect = args.collect_callback local f = args.format args.format = default_format(args, f, name) function args:add_to_item(item, tag_value) add_custom_tag_common(item, name, args, tag_value) local parsed = parse_custom_tags(tag_value, args.params or {}) if parsed then item.parsed_tags = item.parsed_tags or {} item.parsed_tags[name] = item.parsed_tags[name] or {} item.parsed_tags[name][tag_value] = parsed end item.has_show_more = item.has_show_more or (not args.hidden) end custom_tags[#custom_tags+1] = args named_tags[name] = args -- Auto create @name_tparams and @name_usage for each custom tags. if args.auto_subtags ~= false then add_custom_tag { name = name.."tparam", auto_params = true, parent = args, auto_subtags = false, params = { { name = "type" }, { name = "name" }, } } add_custom_tag { name = name.."usage", auto_usage = true, parent = args, auto_subtags = false, } end end -- When a type will request a permission. -- @emits class signal Message[...] add_custom_tag { name = "emits", title = "Emit signals", hidden = false, params = { { name = "name" } }, collect_callback = function(item, mod, tag, value) local parsed = parse_custom_tags(value, named_tags.emits.params) all_signals.collection[parsed.name.value] = all_signals.collection[parsed.name.value] or {} local var = all_signals.collection[parsed.name.value] var[mod.name] = var[mod.name] or {} var[mod.name][#var[mod.name] + 1] = item end } -- Avoid repetitive boilerplate code for property signals. -- Add true if the signal has the value or false if it doesn't. -- @propemits true/false true/false description[...] add_custom_tag { name = "propemits", title = "Emit signals", hidden = false, params = {{name = "new_value"}, {name = "old_value"}}, format = function(self, params, item, md) -- Add an automatic fallback description. local description = params.description ~= "" and params.description or "When the `"..item.name.."` value changes." local new_value = params.new_value.value == "true" local old_value = params.old_value.value == "true" -- Add the sub-tags. local subs = {} item.auto_params["propemitstparam_override"] = subs -- The first argument is always the object which changes. subs[1] = item.module.name.." self ".." The object which changed (".. "useful when connecting many object to the same callback)." -- Most signals also have the new value. if new_value then local type = item.params[1] or "unknown" subs[2] = type.." ".."new_value The new value affected to the property." end -- Some also have the old value. if old_value then local type = item.params[1] or "unknown" subs[3] = type.." ".."old_value The property's old value." end local new_params = { name = { name = item.name, value = "property::"..item.name }, description = description } return default_format_callback(self, new_params, item, md) end } -- List the beautiful variables used by the method or property fallbacks. -- @usebeautiful beautiful.varname usage[...] add_custom_tag { name = "usebeautiful", title = "Consumed theme variables", hidden = false, table = { "Variable", "Usage" }, params = { { name = "name", markdown = true, } }, collect_callback = function(item, mod, tag, value) local params = parse_custom_tags(value, named_tags.usebeautiful.params) if all_theme_vars.collection[params.name.value] then local sig_mods = all_theme_vars.collection[params.name.value] local _, themed_items = pairs(sig_mods)(sig_mods) local themed_item = themed_items[1] named_tags.beautiful_used_by:add_to_item(themed_item, item.name.." "..item.summary) -- Auto fill the description if its empty. if (not params.description) or params.description == "" then params.description = themed_item.summary end else print("WARNING: Could not find theme variable", params.name.value) end end } -- For all properties which have a standard `@beautiful` variable for them -- @propbeautiful fallback1 fallback2 fallback3 fallback4 add_custom_tag { name = "propbeautiful", title = "Consumed theme variables", params = { { name = "fallback1" }, { name = "fallback2" }, { name = "fallback3" }, { name = "fallback4" }, }, table = { "Variable", "Usage" }, format = function(self, p, item, md) local modname = item.module.name:gmatch("[^.]+$")() local last = "beautiful."..(modname.."_"..item.name):gsub("[.]", "_") local ret = ""..md("`"..last.."`").."Fallback when "..md("`"..item.name.."`").. " isn't set." for _, fallback in ipairs({p.fallback1, p.fallback2, p.fallback3, p.fallback4 }) do ret = ret .. "".. ""..md("`"..fallback.value.."`").."Fallback when "..md("`"..last.."`").. " isn't set." last = fallback.value end return ret end, finish_callback = function(item, raw, params, modules) local modname = item.module.name:gmatch("[^.]+$")() local name = "beautiful."..(modname.."_"..item.name):gsub("[.]", "_") if all_theme_vars.collection[name] then local sig_mods = all_theme_vars.collection[name] local _, themed_items = pairs(sig_mods)(sig_mods) local themed_item = themed_items[1] named_tags.beautiful_used_by:add_to_item(themed_item, item.name.." "..item.summary) end end } -- Define the base class where a method/property is implemented. -- @baseclass my_module.my_submodule.my_baseclass add_custom_tag { name = "baseclass", hidden = true } -- Define which interface a method or property implements. -- @interface container add_custom_tag { name = "interface", hidden = true } -- Define when a signal is only emitted on a class rather than on objects. add_custom_tag { name = "classsignal", hidden = true, } -- Specify when this an item was deprecated. -- @deprecatedin 4.4 Optional message. add_custom_tag { name = "deprecatedin", hidden = true, params = { { name = "api_level" }, }, } -- Specify when this an item was deprecated because it was renamed. -- @renamedin 4.4 new_name Optional message. add_custom_tag { name = "renamedin", hidden = true, params = { { name = "api_level" }, { name = "new_name" }, }, } -- Specify when this an item was deprecated because it was moved. -- @movedin 4.4 new_class new_name Optional message. add_custom_tag { name = "movedin", hidden = true, params = { { name = "api_level" }, { name = "new_class" }, { name = "new_name" }, }, } -- Specify when an item was added. -- @introducedin 4.4 add_custom_tag { name = "introducedin", hidden = true, params = { { name = "api_level" }, }, } add_custom_tag { name = "request", title = "Requested actions or permissions", params = { { name = "class" }, { name = "type" }, { name = "context" }, { name = "default" }, }, table = { "Class", "Permission", "Context", "Default", "Description" }, } -- Define the supermodule class. -- This tag should be used at the module level. All properties from the -- supermodule will be recursively added to the module by our ldoc template. -- @supermodule supermodule add_custom_tag { name = "supermodule", hidden = true, auto_subtags = false } -- Mark the item as hidden. -- This tag should be used to hide items from the documentation. -- @hidden add_custom_tag { name = "hidden", hidden = true, } -- Mark the item as readonly. -- This tag should be used to mark readonly properties. -- @readonly add_custom_tag { name = "readonly", hidden = true, auto_subtags = false } -- Forces every method and function to have either `@treturn` or `@noreturn`. -- This avoids the many case where the return value was forgotten. add_custom_tag { name = "noreturn", hidden = true, auto_subtags = false } add_custom_tag { name = "signalhandler", hidden = false, title = "Request handler", auto_subtags = false, params = { { name = "source" }, }, format = function(self, params, item, md) return ldoc.markup("`"..params.source.value.."`") .. " " .. params.description end } add_custom_tag { name = "beautiful_used_by", hidden = false, title = "Used by", auto_subtags = false, params = { { name = "source" }, }, format = function(self, params, item, md) return ldoc.markup("`"..params.source.value.."`") .. " " .. params.description end } -- Auto generate some code for replacing request handlers. add_custom_tag { name = "sourcesignal", title = "Source signal", hidden = false, params = { { name = "class" }, { name = "signal" }, }, format = function(self, params, item, md) return ldoc.markup("`"..params.signal.value.."` from the `"..params.class.value.."` module.").. " This code allows to disconnect this handler and implement your own:" end, finish_callback = function(item, raw, params, modules) if all_signals.collection[params.signal.value] then local sig_mods = all_signals.collection[params.signal.value] local _, signal_items = pairs(sig_mods)(sig_mods) local signal_item = signal_items[1] -- Auto generate some code. local code = "-- Disconnect the original handler.\n" code = code .. signal_item.module.name .. ".disconnect_signal('".. params.signal.value .."', "..item.name..")\n\n" code = code .. "-- Connect a custom handler.\n" code = code .. signal_item.module.name .. ".connect_signal('".. params.signal.value .."', function(" if signal_item.type == "signal" then for idx, parm in ipairs(signal_item.params) do code = code .. parm .. (idx == #signal_item.params and "" or ", ") end code = code .. ")\n -- code here\nend)" named_tags.signalhandler:add_to_item(signal_item, item.name.." "..item.summary) end item.tag_extra_data = item.tag_extra_data or {} item.tag_extra_data["sourcesignal"] = item.tag_extra_data["sourcesignal"] or {} item.tag_extra_data["sourcesignal"][raw] = item.tag_extra_data["sourcesignal"][raw] or {} item.tag_extra_data["sourcesignal"][raw].usage = ldoc.prettify(code) item.has_show_more = true end end } -- @rangestart -- @rangeend -- @minimumvalue -- @maximumvalue -- @tenumvalue -- @increments -- @tablekey or not? -- More fitting section names kind_names={topic='Documentation', module='Libraries', script='Sample files'} -- Sort modules alphabetically sort_modules=true -- Add more project level (left side index) types. new_type("coreclassmod", "Core_components" , true) new_type("inputmodule" , "Input_handling" , true) new_type("ruleslib" , "Declarative_rules", true) new_type("widgetmod" , "Widgets" , true) new_type("containermod", "Widget_containers", true) new_type("layoutmod" , "Widget_layouts" , true) new_type("popupmod" , "Popups_and_bars" , true) new_type("utillib" , "Utility_libraries", true) new_type("themelib" , "Theme_related_libraries", true) file = { -- C parts of libraries '../dbus.c', '../luaa.c', '../mouse.c', '../mousegrabber.c', '../root.c', '../selection.c', '../spawn.c', '../xkb.c', '../objects/client.c', '../objects/drawable.c', '../objects/screen.c', '../objects/tag.c', '../objects/window.c', -- LUA libraries '../lib/', -- Auto generated scripts '../script_files/', exclude = { -- exclude these modules, as they do not contain any written -- documentation '../lib/awful/autofocus.lua', '../lib/awful/client/shape.lua', '../lib/awful/dbus.lua', '../lib/awful/_compat.lua', '../lib/awful/init.lua', '../lib/awful/remote.lua', '../lib/awful/screen/dpi.lua', '../lib/awful/startup_notification.lua', '../lib/awful/mouse/drag_to_tag.lua', '../lib/awful/permissions/_common.lua', '../lib/awful/client/urgent.lua', '../lib/gears/init.lua', '../lib/wibox/layout/init.lua', '../lib/wibox/container/init.lua', '../lib/naughty/constants.lua', '../lib/naughty/dbus.lua', '../lib/beautiful/gtk.lua', '../lib/ruled/init.lua', -- Ignore some parts of the widget library '../lib/awful/widget/init.lua', '../lib/naughty/layout/init.lua', '../lib/naughty/widget/init.lua', '../lib/naughty/container/init.lua', '../lib/naughty/list/init.lua', '../lib/naughty/widget/_default.lua', -- Ignore components that provide no value to users and have confusing -- names '../objects/button.c', '../objects/key.c', -- Deprecated classes for one years or more don't deserve entries -- in the index '../lib/awful/widget/graph.lua', '../lib/awful/widget/progressbar.lua', '../lib/awful/widget/textclock.lua', '../lib/awful/wibox.lua', '../lib/awful/ewmh.lua', '../lib/wibox/layout/constraint.lua', '../lib/wibox/layout/margin.lua', '../lib/wibox/layout/mirror.lua', '../lib/wibox/layout/rotate.lua', '../lib/wibox/layout/scroll.lua', '../lib/wibox/widget/background.lua', '../lib/wibox/drawable.lua', } } -- Wrap the module name for the CSS highlight. local function wrap_modname(str, item) if (not item.module) or str:sub(1, #item.module.name+1) ~= item.module.name.."." then return str end return ""..item.module.name.."." .. str:sub(#item.module.name+2, 9999) end local named_args = { [ "(args)" ] = true, [ "([args])" ] = true, [ "([args=nil])" ] = true, [ "([args={}])" ] = true, [ '(args)' ] = true, } local param_name_whitelist = { Mod2 = true, Lock = true, } -- Sections which are hidden by default, but visible when clicked. local summarize = { is_deprecated = {index = 1, title = "deprecated" , count = false}, emits = {index = 2, title = "signal" , count = true }, propemits = {index = 3, title = "signal" , count = true }, usebeautiful = {index = 4, title = "theme variable" , count = true }, propbeautiful = {index = 5, title = "theme variable" , count = true }, request = {index = 6, title = "permission" , count = true }, classsignal = {index = 7, title = "Class level only", count = false}, readonly = {index = 8, title = "read only" , count = false}, } local delimiter_for_tag = { usebeautiful = { "table class='widget_list' border=1", "table", "tr", "tr", {"Theme variable", "Usage"}}, propbeautiful = { "table class='widget_list' border=1", "table", "tr", "tr", {"Theme variable", "Usage"}}, request = { "table class='widget_list' border=1", "table", "tr", "tr", {"Class", "Permission", "Context", "Default", "Description"}}, sourcesignal = { "div", "div", "span", "span" } } --- If the title is short and descritive, there is no point to bloat it. -- -- That test is mostly to catch empty or lazy description (such as the module -- name and nothing else). local short_title_whitelist = { [ "rc.lua" ] = true, [ "naughty.action" ] = true, [ "naughty.layout.box" ] = true, [ "naughty.widget.title" ] = true, [ "theme.lua" ] = true, [ "wibox.container.margin" ] = true, [ "wibox.container.arcchart"] = true, [ "wibox.widget.checkbox" ] = true, [ "wibox.widget.imagebox" ] = true, [ "wibox.widget.separator" ] = true, [ "wibox.widget.progressbar"] = true, [ "naughty" ] = true, [ "xproperties" ] = true, } -- Use the first word of the subtag content to map it to its tag. local function sort_subtags(item, tag, subtag, values) local ret = {} for _, value in ipairs(values) do local parsed = parse_custom_tags(value, {{name = "maps_to"}}) ret[parsed.maps_to.value] = ret[parsed.maps_to.value] or {} ret[parsed.maps_to.value][#ret[parsed.maps_to.value]+1] = parsed.description end return ret end -- Gather a summary of the tags hidden by default . local function generate_summary(item) local tgs = {} for k, v in pairs(summarize) do tgs[v.index] = {title=v.title, count=0, showcount=v.count} end for tag, value in pairs(item.tags) do if summarize[tag] then tgs[summarize[tag].index].count = #value end end local ret, has_show_more = {}, item.has_show_more or false for k, v in ipairs(tgs) do if v.count > 0 then if v.count > 1 then v.title = v.title .. "s" end ret[#ret+1] = v has_show_more = v.showcount or has_show_more end end item.extra_summary = #ret > 0 and ret or nil item.has_show_more = has_show_more end -- We have custom types, sub-types and different rendering. -- -- To avoid added too much business logic in the template, handle this now. -- Note that this works because the name handler is called when doing the table -- of content, which is before any custom types is used. local function init_custom_types(item) if item.is_init then return end generate_summary(item) -- Give each item an unique identifier so the JavaScript can locate them. item.uid, item_id = item_id, item_id + 1 item.delims, item.auto_usage, item.auto_params = {}, {}, {} local to_rm = {} for tag, values in pairs(item.tags) do -- Remove the sub-tags so they don't get rendered as top level ones. if named_tags[tag] and named_tags[tag].auto_usage then item.auto_usage[tag] = sort_subtags( item, named_tags[tag].parent, named_tags[tag], values ) to_rm[#to_rm+1] = tag elseif named_tags[tag] and named_tags[tag].auto_params then item.auto_params[tag] = sort_subtags( item, named_tags[tag].parent, named_tags[tag], values ) to_rm[#to_rm+1] = tag end end -- Remove from the top-level tag list. for _, rm in ipairs(to_rm) do item.tags[rm] = nil end -- Set the item base class. if item.tags["baseclass"] then item.baseclass = item.tags["baseclass"][1] end if not item.baseclass and item.module then item.baseclass = item.module.name end -- Some methods and properties can be inherited from parent classes. -- in those case, they need the explicit `@baseclass` tag. item.inherited = item.baseclass and item.module and item.module.name ~= item.baseclass function item.get_delim(tag) if delimiter_for_tag[tag] then return delimiter_for_tag[tag][1], delimiter_for_tag[tag][2], delimiter_for_tag[tag][3], delimiter_for_tag[tag][4], delimiter_for_tag[tag][5] else return "ul", "ul", "li", "li", nil end end -- Allow the template to fetch the right sub-tags. function item.get_auto_params(tag, value) local extra = (item.tag_extra_data and item.tag_extra_data[tag] and item.tag_extra_data[tag][value]) or {} -- Makes auto-generated subtags easier to implement. if item.auto_params[tag.."tparam_override"] then return item.auto_params[tag.."tparam_override"], named_tags[tag.."tparam"], extra end local parsed = nil if item.parsed_tags and item.parsed_tags[tag] and item.parsed_tags[tag][value] then parsed = item.parsed_tags[tag][value] end if not item.auto_params[tag.."tparam"] then return nil, nil, extra end if parsed.name and item.auto_params[tag.."tparam"][parsed.name.value] then return item.auto_params[tag.."tparam"][parsed.name.value], named_tags[tag.."tparam"], extra end return nil, nil, extra end item.is_init = true end -- Wrap the arguments for the CSS highlight. local function wrap_args(item) if not item.args then return "" end -- Display named args with `{}` and ordered args with `()` if named_args[item.args] then item.is_named_call = true return "{[args]}" end local new_args = item.args:sub(2, item.args:len()-1) return " ("..new_args..")" end -- Mimics the ldoc built-in method style, but better. -- -- This custom renderer exists because using ldoc built-in method detection -- turned out to be too unreliable and upstream is dead. local function render_methods(item) local ret = item.name -- Some methods will have it depending on the weather. Most wont. if not ret:find(":") then ret = ":"..ret end return ret .. " " .. wrap_args(item) end -- Replace the "|" in alternative types by "or". local function pipe_to_or(s) s = s:gsub("|"," or ") if s:sub(1,1) == "?" then s = s:gsub("?","nil or ") end return s end -- Parse the magic parameters to type in something the template eats. local function sanitize_type(item) for parm in ldoc.modules.iter(item.params) do local t = item:type_of_param(parm) -- Remove the value. t = t:gsub("(\\[[^\\]]])","") t = t:gsub("?","") -- Add " or " between alternatives t = pipe_to_or(t) -- Fallback. t = t == "" and parm or t t = t == "" and "N/A" or t item.display_type = ""..t.."" -- There is no value in repeating the type a second time. if item.params.map[parm] == "" then item.hide_params = true end if t ~= "N/A" then return end end -- It has to be set, otherwise the table will have different col count. item.display_type = "N/A" end -- Detect error prone markdown formatting. local function detect_markdown_footguns(string) for line in string:gmatch("[^\n]*") do if line:sub(1,4) ~= " " then local is_tick = false -- Detect unquoted code with 2 underscores. They will render as italic, -- which is definitively not wanted. Add "`" to fix. for part in line:gmatch("[^`]*") do if not is_tick then local count = 0 for under in part:gmatch("[^\\]_") do count = count + 1 end if count > 1 then return false end end is_tick = not is_tick end end end return true end -- Add parentheses when there is alternative return types. local function generate_return_tuple(item, group) local tuple = {} for r in group:iter() do local type = item:return_type(r); type = pipe_to_or(type) if type ~= nil and type ~= "" then tuple[#tuple+1] = type end end if #tuple == 1 then return tuple[1] else local ret = "(" for i, elem in ldoc.ipairs(tuple) do ret = ret..elem if i ~= #tuple then ret = ret..", " end end ret = ret..")" return ret end end -- Create a return type string. local function sanitize_return_type(item) local groups = item.retgroups item.display_type = "" if groups then item.display_type = " -> " for i,group in ldoc.ipairs(groups) do item.display_type = item.display_type .. generate_return_tuple( item, group, ldoc ) if i ~= #groups then item.display_type = item.display_type .. " or " end end item.display_type = item.display_type .. "" elseif not item.tags["noreturn"] then print( "WARNING:", item.name, "from", item.module.name, "doesn't have a return value or @noreturn" ) end item.compact_signature = true end -- Traverse the entire doc and reverse-map the links between our custom -- tags and the content it is pointing toward. From that, we add some -- "consumers"/"providers" tables to each (relevant) item. local function global_init(_ldoc) ldoc = _ldoc if is_init then return end is_init = true local module_by_name = {} -- First pass, gather the providers. for _, mod in ipairs(ldoc.modules) do module_by_name[mod.name] = mod if mod.type ~= "topic" and ((not mod.summary) or mod.summary == "" or mod.summary:lower():sub(1,8) == "awesome") then print("WARNING: Each module needs a good summary as its first line", mod.name, "doesn't") end local sum_count = 0 for line in (mod.summary or ""):gmatch("[^\n]*") do sum_count = sum_count + 1 end if sum_count > 1 and #mod.summary > 120 then print("WARNING:", mod.name, "summary has to be only 1 line.") end if mod.summary and #mod.summary < 30 and not short_title_whitelist[mod.name] then print("WARNING:", mod.name, "summary is too short.") end for tag, values in pairs(mod.tags) do for _, value in ipairs(values.gmatch and {values} or values) do -- Collect instances of some types. if databases[tag] and databases[tag].collect then delayed_collect[#delayed_collect+1] = {mod, mod, tag, value} end end end for kind, items in mod.kinds() do for item in items() do -- Decorate the item with our customizations. init_custom_types(item) -- print(item.description) if item.summary and not detect_markdown_footguns(item.summary) then print( "WARNING: "..item.name.." from "..item.module.name.." seems to have".. " ambiguous markdown in its summary, probably underscores. Please either".. " use '`' or HTML." ) end -- Collect instances of some types. if databases[item.type] and databases[item.type].collect then databases[item.type].collect(item, mod) end for tag, values in pairs(item.tags) do for _, value in ipairs(values.gmatch and {values} or values) do -- Collect instances of some types. if databases[tag] and databases[tag].collect then delayed_collect[#delayed_collect+1] = {item, mod, tag, value} end end end -- Lint the parameters too. for parm in ldoc.modules.iter(item.params) do local first = parm:match("^[ ]*(.)") -- If the first letter of a parameter is capital, the type is probably missing. -- That's what happen if you rename @param to @tparam without actually adding the type. if first and first ~= first:lower() and parm:upper() ~= parm and not param_name_whitelist[parm] then print( "ERROR: Parameter "..parm.." from ".. item.name .. " ".. item.module.name.. " starts with a capital letter. This is forbidden." ) end for p in ldoc.modules.iter(item:subparam(parm)) do local desc = item.params.map[p] if not detect_markdown_footguns(desc) then print( "WARNING: Always use '`' to encapsulate code in the parameters." .. p .. " from " .. item.name .. " " .. item.module.name .. " contains " .. "some ambiguous markup. If this is a false positive, use HTML." ) end -- It *sometime* works, but is at risk if being interpreted as code by markdown. if desc:match("^ [ ]+") then print( "WARNING: ", item.name, "from", item.module.name, "has a parameter description ".. "with leading spaces. This will not render correctly", "'"..desc.."'" ) end first = desc:match("^[ ]*([a-z])") if first and first ~= first:upper() then print("WARNING: Sentences starts with a capital letter. Please fix", item.module.name, item.name) end --TODO convert @propbeautiful to @usebeautiful and lint them --TODO check the module summary. end end end end end -- Second pass, collect the custom tags. This is delayed because -- the tags depend on the items. Not all items have been processed if -- this was called during the first pass. Also, collect callbacks can -- themselves create more delayed_collect entries recursively. for _, to_collect in ipairs(delayed_collect) do databases[to_collect[3]].collect(to_collect[1], to_collect[2], to_collect[3], to_collect[4]) end -- Third pass, parse every custom tags and fill the consumers tables. for _, mod in ipairs(ldoc.modules) do for kind, items in mod.kinds() do for item in items() do for tag_name, values in pairs(item.tags) do for _, tag in ipairs(values.gmatch and {values} or values) do local parsed = named_tags[tag_name] and named_tags[tag_name].params and parse_custom_tags(tag, named_tags[tag_name].params) -- Allow the tags to, ie, auto-generate some extra content. if named_tags[tag_name] and named_tags[tag_name].finish_callback then named_tags[tag_name].finish_callback(item, tag, parsed, module_by_name) end -- Save the parsed content to be used in get_auto_params. if parsed then item.parsed_tags = item.parsed_tags or {} item.parsed_tags[tag_name] = item.parsed_tags[tag_name] or {} item.parsed_tags[tag_name][tag] = parsed end end end -- Mostly for linting. if databases[item.type] and databases[item.type].finish then databases[item.type].finish(item, mod, module_by_name) end end end end end -- Work around the fact that tag/awful.tag client/awful.client pages -- are merged. local function compare_module_name(input, module) for _, mod in ipairs { module, "awful."..module } do if input:sub(1, #mod) == mod then return true end end return false end local no_prefix = { property = true, signal = true, clientruleproperty = true, deprecatedproperty = true, } -- These modules merge the doc of their `awful` siblings. local coreclassmap = { tag = "tag and awful.tag", screen = "screen and awful.screen", client = "client and awful.client", mouse = "mouse and awful.mouse", } -- Add the full module name in front. local add_mod = { ["function"] = true, constructorfct = true, constructorfct2 = true, staticfct = true, deprecated = true, field = true, signalhandler = true, table = true, } -- Add the arguments. local add_args = { constructorfct = true, constructorfct2 = true, staticfct = true, signalhandler = true, callback = true, deprecated = true, } -- Add a type column to the summary and type field in the description. local display_type = { property = true, beautiful = true, field = true, deprecatedproperty = true, clientruleproperty = true, matchingproperty = true, rulecomponent = true, } -- Add the `-> ret_type` annotation. local display_return_type = { method = true, deprecatedmethod = true, staticfct = true, } -- Show return values. local show_return = { ["function"] = true, constructorfct = true, constructorfct2 = true, legacylayout = true, staticfct = true, method = true, deprecated = true, deprecatedmethod = true, } -- The different type of deprecation. local is_deprecated = { deprecated = true, deprecatedproperty = true, deprecatedmethod = true, deprecatedsignal = true, } custom_display_name_handler = function(item, default_handler) item.global_init = global_init if is_deprecated[item.type] then item.tags.is_deprecated = {true} end init_custom_types(item) -- Do not use the default handler. It encodes the parameters using -- the optchain convention. Not everybody is confortable with this. -- It is also unreadable. item.args = "" if item.params then item.args = "(" for key, line in ipairs(item.params) do local name, comment = item:split_param(line) local def = item:default_of_param(line) if def then item.args = item.args .. '' .. name .. "" .. (key < #item.params and ", " or "") else item.args = item.args .. name .. (key < #item.params and ", " or "") end end item.args = item.args .. ")" end local ret = default_handler(item) -- Edit the input so the template is notified. if display_type[item.type] then -- Punch a hole in the sandbox and inject the `ldoc` object. item.sanitize_type = sanitize_type elseif display_return_type[item.type] then item.sanitize_type = sanitize_return_type end -- LDoc hardcode the "Returns" section for "function" only, fix that. if show_return[item.type] and item.tags["return"] then item.ret = item.tags["return"] item:build_return_groups() end -- Remove the "namespace" from the signals and properties if no_prefix[item.type] then local name = item.name:match("%.([^.]+)$") return name ~= "" and name or item.name end -- Handle the left sidebar modules. if item.type == "coreclassmod" and coreclassmap[item.name] then return coreclassmap[item.name] end -- Undocumented API to make the libraries and classmod "function" section -- more consistent. Right now some have their full awful.foo.bar while other -- have "just" `bar`. Given we use constructors from metatables, we have no -- choice but to use the full function name. It also makes copy/paste easier. if add_mod[item.type] then local is_field = item.type == "field" or item.type == "tfield" if (not ret:find(".", 1, true)) and (not ret:find(":", 1, true)) then ret = item.module.name .. "." .. ret elseif is_field and not compare_module_name(ret, item.module.name) then ret = item.module.name .. "." .. ret end end if item.type:match("method") then ret = render_methods(item) end -- Get rid of the "module:" in front of method names. It is either wrong or -- just redundant. ret = ret:gsub("([^:]*)(:[^:])","%2") -- Undocumented API to get rid of `module.rounded_rect` rather than -- `gears.shape.rounded_rect` if ret:sub(1, 7) == "module." and module then return ret:gsub("^module", item.module.name) end -- It isn't there by default. if add_args[item.type] then ret = ret .. " " .. wrap_args(item) end -- Beautify. ret = wrap_modname(ret, item) return ret end -- vim: filetype=lua:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:textwidth=80