mirror of https://github.com/koreader/koreader
You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
101 lines
3.7 KiB
Lua
101 lines
3.7 KiB
Lua
--[[--
|
|
AlphaContainer will paint its content (1 widget) onto lower levels using
|
|
a transparency (0..1)
|
|
|
|
Example:
|
|
|
|
local alpha
|
|
alpha = AlphaContainer:new{
|
|
alpha = 0.7,
|
|
|
|
FrameContainer:new{
|
|
background = Blitbuffer.COLOR_WHITE,
|
|
bordersize = Size.border.default,
|
|
margin = 0,
|
|
padding = Size.padding.default
|
|
}
|
|
}
|
|
--]]
|
|
|
|
local Blitbuffer = require("ffi/blitbuffer")
|
|
local WidgetContainer = require("ui/widget/container/widgetcontainer")
|
|
|
|
local AlphaContainer = WidgetContainer:new{
|
|
alpha = 1,
|
|
-- we cache a blitbuffer object for re-use here:
|
|
private_bb = nil,
|
|
-- we save the underlying area here:
|
|
background_bb = nil,
|
|
background_bb_x = nil,
|
|
background_bb_y = nil
|
|
}
|
|
|
|
function AlphaContainer:paintTo(bb, x, y)
|
|
local contentSize = self[1]:getSize()
|
|
local private_bb = self.private_bb
|
|
|
|
if self.background_bb then
|
|
-- NOTE: Best as I can tell, this was an attempt at avoiding alpha layering issues when an AlphaContainer is repainted *at the same coordinates* AND *over the same background*.
|
|
-- Unfortunately, those are hard constraints to respect, and, while we can take care of the first by invalidating the cache if coordinates have changed,
|
|
-- we can't do anything about the second (and that's exactly what happens in ReaderUI when paging around, for example: that'll obviously have changed what's below AlphaContainer ;)).
|
|
-- FWIW, MovableContainer's alpha handling rely on callers using setDirty("all") to force a repaint of the whole stack to avoid layering issues.
|
|
-- A better approach would probably involve letting UIManager handle it: if it finds a dirty translucent widget, mark all the widgets below it dirty, too...
|
|
if self.background_bb_x == x and self.background_bb_y == y then
|
|
bb:blitFrom(self.background_bb, self.background_bb_x, self.background_bb_y)
|
|
else
|
|
-- We moved, invalidate the bg cache.
|
|
self.background_bb:free()
|
|
self.background_bb = nil
|
|
self.background_bb_x = nil
|
|
self.background_bb_y = nil
|
|
end
|
|
end
|
|
|
|
if not private_bb
|
|
or private_bb:getWidth() ~= contentSize.w
|
|
or private_bb:getHeight() ~= contentSize.h
|
|
then
|
|
if private_bb then
|
|
private_bb:free() -- free the one we're going to replace
|
|
end
|
|
-- create private blitbuffer for our child widget to paint to
|
|
private_bb = Blitbuffer.new(contentSize.w, contentSize.h, bb:getType())
|
|
self.private_bb = private_bb
|
|
|
|
-- save what is below our painting area
|
|
if not self.background_bb
|
|
or self.background_bb:getWidth() ~= contentSize.w
|
|
or self.background_bb:getHeight() ~= contentSize.h
|
|
then
|
|
if self.background_bb then
|
|
self.background_bb:free() -- free the one we're going to replace
|
|
end
|
|
self.background_bb = Blitbuffer.new(contentSize.w, contentSize.h, bb:getType())
|
|
end
|
|
self.background_bb:blitFrom(bb, 0, 0, x, y)
|
|
self.background_bb_x = x
|
|
self.background_bb_y = y
|
|
end
|
|
|
|
-- now have our child widget paint to the private blitbuffer
|
|
private_bb:fill(Blitbuffer.COLOR_WHITE)
|
|
self[1]:paintTo(private_bb, 0, 0)
|
|
|
|
-- blit the private blitbuffer to our parent blitbuffer
|
|
bb:addblitFrom(private_bb, x, y, nil, nil, nil, nil, self.alpha)
|
|
end
|
|
|
|
function AlphaContainer:onCloseWidget()
|
|
if self.private_bb then
|
|
self.private_bb:free()
|
|
self.private_bb = nil
|
|
end
|
|
if self.background_bb then
|
|
self.background_bb:free()
|
|
self.background_bb = nil
|
|
end
|
|
end
|
|
|
|
|
|
return AlphaContainer
|