screen: Add diagonal size (in inches and millimeters).

The use case for this will be to detech which screen is connected to
an output from the screen rules.

It is in millimeters because this is what the output provides and in
inches because the DPI is based on that unit and screens are sold with
the size in inches on the box.
This commit is contained in:
Emmanuel Lepage Vallee 2019-07-21 03:49:44 -04:00
parent 93799e8be3
commit cb88776980
2 changed files with 72 additions and 12 deletions

View File

@ -740,6 +740,26 @@ end
-- @property preferred_dpi -- @property preferred_dpi
-- @param number the DPI value. -- @param number the DPI value.
--- The maximum diagonal size in millimeters.
--
-- @property mm_maximum_size
-- @param number
--- The minimum diagonal size in millimeters.
--
-- @property mm_minimum_size
-- @param number
--- The maximum diagonal size in inches.
--
-- @property inch_maximum_size
-- @param number
--- The minimum diagonal size in inches.
--
-- @property inch_minimum_size
-- @param number
--- Emitted when a new screen is added. --- Emitted when a new screen is added.
-- --
-- The handler(s) of this signal are responsible of adding elements such as -- The handler(s) of this signal are responsible of adding elements such as

View File

@ -48,29 +48,65 @@ local function dpi_for_output(viewport, output)
end end
local function dpis_for_outputs(viewport) local function dpis_for_outputs(viewport)
local max, min = 0, math.huge local max_dpi, min_dpi, max_size, min_size = 0, math.huge, 0, math.huge
for _, o in pairs(viewport.outputs) do for _, o in pairs(viewport.outputs) do
local dpi = dpi_for_output(viewport, o) local dpi = dpi_for_output(viewport, o)
o.dpi = dpi o.dpi = dpi
max = math.max(max, dpi)
min = math.min(min, dpi) max_dpi = math.max(max_dpi, dpi)
min_dpi = math.min(min_dpi, dpi)
-- Compute the diagonal size.
if o.mm_width and o.mm_height then
o.mm_size = math.sqrt(o.mm_width^2 + o.mm_height^2)
o.inch_size = o.mm_size/mm_per_inch
max_size = math.max(max_size, o.mm_size)
min_size = math.min(min_size, o.mm_size)
end
end end
-- When there is no output. -- When there is no output.
if min == math.huge then if min_dpi == math.huge then
min = get_fallback_dpi() min_dpi = get_fallback_dpi()
max = min max_dpi = min_dpi
end end
--TODO Some output may have a lower resolution than the viewport, so their --TODO Some output may have a lower resolution than the viewport, so their
-- DPI is currently wrong. Once fixed, the preferred DPI can become -- DPI is currently wrong. Once fixed, the preferred DPI can become
-- different from the minimum one. -- different from the minimum one.
local pref = min local pref_dpi = min_dpi
viewport.minimum_dpi, viewport.maximum_dpi, viewport.preferred_dpi = min, max, pref viewport.minimum_dpi = min_dpi
viewport.maximum_dpi = max_dpi
viewport.preferred_dpi = pref_dpi
return max, min, pref -- Guess the diagonal size using the DPI.
if min_size == math.huge then
for _, o in pairs(viewport.outputs) do
local geo = o.geometry
if geo then
o.mm_size = math.sqrt(geo.width^2 + geo.height^2)/o.dpi
max_size = math.max(max_size, o.mm_size)
min_size = math.min(min_size, o.mm_size)
end
end
-- In case there is no output information.
if min_size == math.huge then
local geo = viewport.geometry
local size = math.sqrt(geo.width^2 + geo.height^2)/max_dpi
max_size, min_size = size, size
end
end
viewport.mm_minimum_size = min_size
viewport.mm_maximum_size = max_size
viewport.inch_minimum_size = min_size/mm_per_inch
viewport.inch_maximum_size = max_size/mm_per_inch
return max_dpi, min_dpi, pref_dpi
end end
local function update_outputs(old_viewport, new_viewport) local function update_outputs(old_viewport, new_viewport)
@ -360,15 +396,19 @@ return function(screen, d)
ascreen.object.set_dpi = set_dpi ascreen.object.set_dpi = set_dpi
ascreen.object.get_dpi = get_dpi ascreen.object.get_dpi = get_dpi
for _, prop in ipairs {"minimum", "maximum", "preferred" } do for _, prop in ipairs {"minimum_dpi" , "maximum_dpi" ,
screen.object["get_"..prop.."_dpi"] = function(s) "mm_maximum_width" , "mm_minimum_width" ,
"inch_maximum_width", "inch_minimum_width",
"preferred_dpi" } do
screen.object["get_"..prop] = function(s)
if not s.data.viewport then if not s.data.viewport then
update_screen_viewport(s) update_screen_viewport(s)
end end
local a = s.data.viewport or {} local a = s.data.viewport or {}
return a[prop.."_dpi"] or nil return a[prop] or nil
end end
end end
end end