mirror of
https://github.com/koreader/koreader.git
synced 2025-08-10 00:52:38 +00:00
For consistency with BookMap and PageBrowser widgets where swipe south (the usual gesture to quick close) can't be used for closing and we had to use any multiswipe instead, allow any multiswipe to close these other fullscreen widgets too: Menu (ToC, Bookmarks), KeyValuePage, ImageViewer, BookStatusWidget, ReaderProgress, CalendarView.
665 lines
23 KiB
Lua
665 lines
23 KiB
Lua
--[[--
|
|
Widget that presents a multi-page to show key value pairs.
|
|
|
|
Example:
|
|
|
|
local Foo = KeyValuePage:new{
|
|
title = "Statistics",
|
|
kv_pairs = {
|
|
{"Current period", "00:00:00"},
|
|
-- single or more "-" will generate a solid line
|
|
"----------------------------",
|
|
{"Page to read", "5"},
|
|
{"Time to read", "00:01:00"},
|
|
{"Press me", "will invoke the callback",
|
|
callback = function() print("hello") end },
|
|
},
|
|
}
|
|
UIManager:show(Foo)
|
|
|
|
]]
|
|
|
|
local BD = require("ui/bidi")
|
|
local Blitbuffer = require("ffi/blitbuffer")
|
|
local BottomContainer = require("ui/widget/container/bottomcontainer")
|
|
local Button = require("ui/widget/button")
|
|
local Device = require("device")
|
|
local Font = require("ui/font")
|
|
local FrameContainer = require("ui/widget/container/framecontainer")
|
|
local Geom = require("ui/geometry")
|
|
local GestureRange = require("ui/gesturerange")
|
|
local HorizontalGroup = require("ui/widget/horizontalgroup")
|
|
local HorizontalSpan = require("ui/widget/horizontalspan")
|
|
local InputContainer = require("ui/widget/container/inputcontainer")
|
|
local LeftContainer = require("ui/widget/container/leftcontainer")
|
|
local LineWidget = require("ui/widget/linewidget")
|
|
local OverlapGroup = require("ui/widget/overlapgroup")
|
|
local Size = require("ui/size")
|
|
local TextViewer = require("ui/widget/textviewer")
|
|
local TextWidget = require("ui/widget/textwidget")
|
|
local TitleBar = require("ui/widget/titlebar")
|
|
local UIManager = require("ui/uimanager")
|
|
local VerticalGroup = require("ui/widget/verticalgroup")
|
|
local VerticalSpan = require("ui/widget/verticalspan")
|
|
local Input = Device.input
|
|
local Screen = Device.screen
|
|
local T = require("ffi/util").template
|
|
local _ = require("gettext")
|
|
|
|
local KeyValueItem = InputContainer:new{
|
|
key = nil,
|
|
value = nil,
|
|
value_lang = nil,
|
|
font_size = 20, -- will be adjusted depending on keyvalues_per_page
|
|
key_font_name = "smallinfofontbold",
|
|
value_font_name = "smallinfofont",
|
|
width = nil,
|
|
height = nil,
|
|
textviewer_width = nil,
|
|
textviewer_height = nil,
|
|
value_overflow_align = "left",
|
|
-- "right": only align right if value overflow 1/2 width
|
|
-- "right_always": align value right even when small and
|
|
-- only key overflows 1/2 width
|
|
}
|
|
|
|
function KeyValueItem:init()
|
|
self.dimen = Geom:new{w = self.width, h = self.height}
|
|
|
|
-- self.value may contain some control characters (\n \t...) that would
|
|
-- be rendered as a square. Replace them with a shorter and nicer '|'.
|
|
-- (Let self.value untouched, as with Hold, the original value can be
|
|
-- displayed correctly in TextViewer.)
|
|
local tvalue = tostring(self.value)
|
|
tvalue = tvalue:gsub("[\n\t]", "|")
|
|
|
|
local frame_padding = Size.padding.default
|
|
local frame_internal_width = self.width - frame_padding * 2
|
|
local middle_padding = Size.padding.default -- min enforced padding between key and value
|
|
local available_width = frame_internal_width - middle_padding
|
|
|
|
-- Default widths (and position of value widget) if each text fits in 1/2 screen width
|
|
local key_w = math.floor(frame_internal_width / 2 - middle_padding)
|
|
local value_w = math.floor(frame_internal_width / 2)
|
|
|
|
local key_widget = TextWidget:new{
|
|
text = self.key,
|
|
max_width = available_width,
|
|
face = Font:getFace(self.key_font_name, self.font_size),
|
|
}
|
|
local value_widget = TextWidget:new{
|
|
text = tvalue,
|
|
max_width = available_width,
|
|
face = Font:getFace(self.value_font_name, self.font_size),
|
|
lang = self.value_lang,
|
|
}
|
|
local key_w_rendered = key_widget:getWidth()
|
|
local value_w_rendered = value_widget:getWidth()
|
|
|
|
-- As both key_widget and value_width will be in a HorizontalGroup,
|
|
-- and key is always left aligned, we can just tweak the key width
|
|
-- to position the value_widget
|
|
local value_align_right = false
|
|
local fit_right_align = true -- by default, really right align
|
|
|
|
if key_w_rendered > key_w or value_w_rendered > value_w then
|
|
-- One (or both) does not fit in 1/2 width
|
|
if key_w_rendered + value_w_rendered > available_width then
|
|
-- Both do not fit: one has to be truncated so they fit
|
|
if key_w_rendered >= value_w_rendered then
|
|
-- Rare case: key larger than value.
|
|
-- We should have kept our keys small, smaller than 1/2 width.
|
|
-- If it is larger than value, it's that value is kinda small,
|
|
-- so keep the whole value, and truncate the key
|
|
key_w = available_width - value_w_rendered
|
|
else
|
|
-- Usual case: value larger than key.
|
|
-- Keep our small key, fit the value in the remaining width.
|
|
key_w = key_w_rendered
|
|
end
|
|
value_align_right = true -- so the ellipsis touches the screen right border
|
|
if self.value_align ~= "right" and self.value_overflow_align ~= "right"
|
|
and self.value_overflow_align ~= "right_always" then
|
|
-- Don't adjust the ellipsis to the screen right border,
|
|
-- so the left of text is aligned with other truncated texts
|
|
fit_right_align = false
|
|
end
|
|
-- Allow for displaying the non-truncated text with Hold
|
|
if Device:isTouchDevice() then
|
|
self.ges_events.Hold = {
|
|
GestureRange:new{
|
|
ges = "hold",
|
|
range = self.dimen,
|
|
}
|
|
}
|
|
-- If no tap callback, allow for displaying the non-truncated
|
|
-- text with Tap too
|
|
if not self.callback then
|
|
self.callback = function()
|
|
self:onHold()
|
|
end
|
|
end
|
|
end
|
|
else
|
|
-- Both can fit: break the 1/2 widths
|
|
if self.value_align == "right" or self.value_overflow_align == "right_always"
|
|
or (self.value_overflow_align == "right" and value_w_rendered > value_w) then
|
|
key_w = available_width - value_w_rendered
|
|
value_align_right = true
|
|
else
|
|
key_w = key_w_rendered
|
|
end
|
|
end
|
|
-- In all the above case, we set the right key_w to include any
|
|
-- needed additional in-between padding: value_w is what's left.
|
|
value_w = available_width - key_w
|
|
else
|
|
if self.value_align == "right" then
|
|
key_w = available_width - value_w_rendered
|
|
value_w = value_w_rendered
|
|
value_align_right = true
|
|
end
|
|
end
|
|
|
|
-- Adjust widgets' max widths if needed
|
|
value_widget:setMaxWidth(value_w)
|
|
if fit_right_align and value_align_right and value_widget:getWidth() < value_w then
|
|
-- Because of truncation at glyph boundaries, value_widget
|
|
-- may be a tad smaller than the specified value_w:
|
|
-- adjust key_w so value is pushed to the screen right border
|
|
value_w = value_widget:getWidth()
|
|
key_w = available_width - value_w
|
|
end
|
|
key_widget:setMaxWidth(key_w)
|
|
|
|
-- For debugging positioning:
|
|
-- value_widget = FrameContainer:new{ padding=0, margin=0, bordersize=1, value_widget }
|
|
|
|
if self.callback and Device:isTouchDevice() then
|
|
self.ges_events.Tap = {
|
|
GestureRange:new{
|
|
ges = "tap",
|
|
range = self.dimen,
|
|
}
|
|
}
|
|
end
|
|
|
|
self[1] = FrameContainer:new{
|
|
padding = frame_padding,
|
|
padding_top = 0,
|
|
padding_bottom = 0,
|
|
bordersize = 0,
|
|
background = Blitbuffer.COLOR_WHITE,
|
|
HorizontalGroup:new{
|
|
dimen = self.dimen:copy(),
|
|
LeftContainer:new{
|
|
dimen = {
|
|
w = key_w,
|
|
h = self.height
|
|
},
|
|
key_widget,
|
|
},
|
|
HorizontalSpan:new{
|
|
width = middle_padding,
|
|
},
|
|
LeftContainer:new{
|
|
dimen = {
|
|
w = value_w,
|
|
h = self.height
|
|
},
|
|
value_widget,
|
|
}
|
|
}
|
|
}
|
|
end
|
|
|
|
function KeyValueItem:onTap()
|
|
if self.callback then
|
|
if G_reader_settings:isFalse("flash_ui") then
|
|
self.callback()
|
|
else
|
|
-- c.f., ui/widget/iconbutton for the canonical documentation about the flash_ui code flow
|
|
|
|
-- Highlight
|
|
--
|
|
self[1].invert = true
|
|
UIManager:widgetInvert(self[1], self[1].dimen.x, self[1].dimen.y)
|
|
UIManager:setDirty(nil, "fast", self[1].dimen)
|
|
|
|
UIManager:forceRePaint()
|
|
UIManager:yieldToEPDC()
|
|
|
|
-- Unhighlight
|
|
--
|
|
self[1].invert = false
|
|
UIManager:widgetInvert(self[1], self[1].dimen.x, self[1].dimen.y)
|
|
UIManager:setDirty(nil, "ui", self[1].dimen)
|
|
|
|
-- Callback
|
|
--
|
|
self.callback()
|
|
|
|
UIManager:forceRePaint()
|
|
end
|
|
end
|
|
return true
|
|
end
|
|
|
|
function KeyValueItem:onHold()
|
|
local textviewer = TextViewer:new{
|
|
title = self.key,
|
|
text = self.value,
|
|
text_face = Font:getFace("x_smallinfofont", self.font_size),
|
|
lang = self.value_lang,
|
|
width = self.textviewer_width,
|
|
height = self.textviewer_height,
|
|
}
|
|
UIManager:show(textviewer)
|
|
return true
|
|
end
|
|
|
|
|
|
local KeyValuePage = InputContainer:new{
|
|
title = "",
|
|
width = nil,
|
|
height = nil,
|
|
values_lang = nil,
|
|
-- index for the first item to show
|
|
show_page = 1,
|
|
-- aligment of value when key or value overflows its reserved width (for
|
|
-- now: 50%): "left" (stick to key), "right" (stick to scren right border)
|
|
value_overflow_align = "left",
|
|
single_page = nil, -- show all items on one single page (and make them small)
|
|
}
|
|
|
|
function KeyValuePage:init()
|
|
self.dimen = Geom:new{
|
|
w = self.width or Screen:getWidth(),
|
|
h = self.height or Screen:getHeight(),
|
|
}
|
|
if self.dimen.w == Screen:getWidth() and self.dimen.h == Screen:getHeight() then
|
|
self.covers_fullscreen = true -- hint for UIManager:_repaint()
|
|
end
|
|
|
|
if Device:hasKeys() then
|
|
self.key_events = {
|
|
Close = { {Input.group.Back}, doc = "close page" },
|
|
NextPage = {{Input.group.PgFwd}, doc = "next page"},
|
|
PrevPage = {{Input.group.PgBack}, doc = "prev page"},
|
|
}
|
|
end
|
|
if Device:isTouchDevice() then
|
|
self.ges_events.Swipe = {
|
|
GestureRange:new{
|
|
ges = "swipe",
|
|
range = self.dimen,
|
|
}
|
|
}
|
|
self.ges_events.MultiSwipe = {
|
|
GestureRange:new{
|
|
ges = "multiswipe",
|
|
range = self.dimen,
|
|
}
|
|
}
|
|
end
|
|
|
|
-- return button
|
|
--- @todo: alternative icon if BD.mirroredUILayout()
|
|
self.page_return_arrow = self.page_return_arrow or Button:new{
|
|
icon = BD.mirroredUILayout() and "back.top.rtl" or "back.top",
|
|
callback = function() self:onReturn() end,
|
|
bordersize = 0,
|
|
show_parent = self,
|
|
}
|
|
-- group for page info
|
|
local chevron_left = "chevron.left"
|
|
local chevron_right = "chevron.right"
|
|
local chevron_first = "chevron.first"
|
|
local chevron_last = "chevron.last"
|
|
if BD.mirroredUILayout() then
|
|
chevron_left, chevron_right = chevron_right, chevron_left
|
|
chevron_first, chevron_last = chevron_last, chevron_first
|
|
end
|
|
self.page_info_left_chev = self.page_info_left_chev or Button:new{
|
|
icon = chevron_left,
|
|
callback = function() self:prevPage() end,
|
|
bordersize = 0,
|
|
show_parent = self,
|
|
}
|
|
self.page_info_right_chev = self.page_info_right_chev or Button:new{
|
|
icon = chevron_right,
|
|
callback = function() self:nextPage() end,
|
|
bordersize = 0,
|
|
show_parent = self,
|
|
}
|
|
self.page_info_first_chev = self.page_info_first_chev or Button:new{
|
|
icon = chevron_first,
|
|
callback = function() self:goToPage(1) end,
|
|
bordersize = 0,
|
|
show_parent = self,
|
|
}
|
|
self.page_info_last_chev = self.page_info_last_chev or Button:new{
|
|
icon = chevron_last,
|
|
callback = function() self:goToPage(self.pages) end,
|
|
bordersize = 0,
|
|
show_parent = self,
|
|
}
|
|
self.page_info_spacer = HorizontalSpan:new{
|
|
width = Screen:scaleBySize(32),
|
|
}
|
|
|
|
if self.callback_return == nil and self.return_button == nil then
|
|
self.page_return_arrow:hide()
|
|
elseif self.callback_return == nil then
|
|
self.page_return_arrow:disable()
|
|
end
|
|
self.return_button = HorizontalGroup:new{
|
|
HorizontalSpan:new{
|
|
width = Size.span.horizontal_small,
|
|
},
|
|
self.page_return_arrow,
|
|
HorizontalSpan:new{
|
|
width = self.dimen.w - self.page_return_arrow:getSize().w - Size.span.horizontal_small,
|
|
},
|
|
}
|
|
|
|
self.page_info_left_chev:hide()
|
|
self.page_info_right_chev:hide()
|
|
self.page_info_first_chev:hide()
|
|
self.page_info_last_chev:hide()
|
|
|
|
self.page_info_text = self.page_info_text or Button:new{
|
|
text = "",
|
|
hold_input = {
|
|
title = _("Enter page number"),
|
|
type = "number",
|
|
hint_func = function()
|
|
return "(" .. "1 - " .. self.pages .. ")"
|
|
end,
|
|
deny_blank_input = true,
|
|
callback = function(input)
|
|
local page = tonumber(input)
|
|
if page and page >= 1 and page <= self.pages then
|
|
self:goToPage(page)
|
|
end
|
|
end,
|
|
ok_text = _("Go to page"),
|
|
},
|
|
call_hold_input_on_tap = true,
|
|
bordersize = 0,
|
|
text_font_face = "pgfont",
|
|
text_font_bold = false,
|
|
}
|
|
self.page_info = HorizontalGroup:new{
|
|
self.page_info_first_chev,
|
|
self.page_info_spacer,
|
|
self.page_info_left_chev,
|
|
self.page_info_spacer,
|
|
self.page_info_text,
|
|
self.page_info_spacer,
|
|
self.page_info_right_chev,
|
|
self.page_info_spacer,
|
|
self.page_info_last_chev,
|
|
}
|
|
|
|
local padding = Size.padding.large
|
|
self.item_width = self.dimen.w - 2 * padding
|
|
|
|
local footer = BottomContainer:new{
|
|
dimen = self.dimen:copy(),
|
|
self.page_info,
|
|
}
|
|
if self.single_page then
|
|
footer = nil
|
|
end
|
|
|
|
local page_return = BottomContainer:new{
|
|
dimen = self.dimen:copy(),
|
|
self.return_button,
|
|
}
|
|
|
|
self.title_bar = TitleBar:new{
|
|
title = self.title,
|
|
fullscreen = self.covers_fullscreen,
|
|
width = self.width,
|
|
align = "left",
|
|
with_bottom_line = true,
|
|
bottom_line_color = Blitbuffer.COLOR_DARK_GRAY,
|
|
bottom_line_h_padding = padding,
|
|
close_callback = function() self:onClose() end,
|
|
}
|
|
|
|
-- setup main content
|
|
local available_height = self.dimen.h
|
|
- self.title_bar:getHeight()
|
|
- Size.span.vertical_large -- for above page_info (as title_bar adds one itself)
|
|
- (self.single_page and 0 or self.page_info:getSize().h)
|
|
- 2*Size.line.thick
|
|
-- account for possibly 2 separator lines added
|
|
|
|
local force_items_per_page
|
|
if self.single_page then
|
|
force_items_per_page = math.max(#self.kv_pairs,
|
|
G_reader_settings:readSetting("keyvalues_per_page") or self:getDefaultKeyValuesPerPage())
|
|
end
|
|
|
|
self.items_per_page = force_items_per_page or
|
|
G_reader_settings:readSetting("keyvalues_per_page") or self:getDefaultKeyValuesPerPage()
|
|
self.item_height = math.floor(available_height / self.items_per_page)
|
|
-- Put half of the pixels lost by floor'ing between title and content
|
|
local content_height = self.items_per_page * self.item_height
|
|
local span_height = math.floor((available_height - content_height) / 2)
|
|
|
|
-- Font size is not configurable: we can get a good one from the following
|
|
local TextBoxWidget = require("ui/widget/textboxwidget")
|
|
local line_extra_height = 1.0 -- ~ 2em -- unscaled_size_check: ignore
|
|
-- (gives a font size similar to the fixed one from former implementation at 14 items per page)
|
|
self.items_font_size = TextBoxWidget:getFontSizeToFitHeight(self.item_height, 1, line_extra_height)
|
|
|
|
self.pages = math.ceil(#self.kv_pairs / self.items_per_page)
|
|
self.main_content = VerticalGroup:new{}
|
|
|
|
-- set textviewer height to let our title fully visible (but hide the bottom line)
|
|
self.textviewer_width = self.item_width
|
|
self.textviewer_height = self.dimen.h - 2 * (self.title_bar:getHeight() - Size.padding.default - Size.line.thick)
|
|
|
|
self:_populateItems()
|
|
|
|
local content = OverlapGroup:new{
|
|
allow_mirroring = false,
|
|
dimen = self.dimen:copy(),
|
|
VerticalGroup:new{
|
|
align = "left",
|
|
self.title_bar,
|
|
VerticalSpan:new{ width = span_height },
|
|
HorizontalGroup:new{
|
|
HorizontalSpan:new{ width = padding },
|
|
self.main_content,
|
|
}
|
|
},
|
|
page_return,
|
|
footer,
|
|
}
|
|
-- assemble page
|
|
self[1] = FrameContainer:new{
|
|
width = self.dimen.w,
|
|
height = self.dimen.h,
|
|
padding = 0,
|
|
margin = 0,
|
|
bordersize = 0,
|
|
background = Blitbuffer.COLOR_WHITE,
|
|
content,
|
|
}
|
|
end
|
|
|
|
function KeyValuePage:getDefaultKeyValuesPerPage()
|
|
-- Get a default according to Screen DPI (roughly following
|
|
-- the former implementation building logic)
|
|
local default_item_height = Size.item.height_default * 1.5 -- we were adding 1/2 as margin
|
|
local nb_items = math.floor(Screen:getHeight() / default_item_height)
|
|
nb_items = nb_items - 3 -- account for title and footer heights
|
|
return nb_items
|
|
end
|
|
|
|
function KeyValuePage:nextPage()
|
|
local new_page = math.min(self.show_page+1, self.pages)
|
|
if new_page > self.show_page then
|
|
self.show_page = new_page
|
|
self:_populateItems()
|
|
end
|
|
end
|
|
|
|
function KeyValuePage:prevPage()
|
|
local new_page = math.max(self.show_page-1, 1)
|
|
if new_page < self.show_page then
|
|
self.show_page = new_page
|
|
self:_populateItems()
|
|
end
|
|
end
|
|
|
|
function KeyValuePage:goToPage(page)
|
|
self.show_page = page
|
|
self:_populateItems()
|
|
end
|
|
|
|
-- make sure self.item_margin and self.item_height are set before calling this
|
|
function KeyValuePage:_populateItems()
|
|
self.page_info:resetLayout()
|
|
self.return_button:resetLayout()
|
|
self.main_content:clear()
|
|
local idx_offset = (self.show_page - 1) * self.items_per_page
|
|
for idx = 1, self.items_per_page do
|
|
local entry = self.kv_pairs[idx_offset + idx]
|
|
if entry == nil then break end
|
|
|
|
if type(entry) == "table" then
|
|
table.insert(self.main_content, KeyValueItem:new{
|
|
height = self.item_height,
|
|
width = self.item_width,
|
|
font_size = self.items_font_size,
|
|
key = entry[1],
|
|
value = entry[2],
|
|
value_lang = self.values_lang,
|
|
callback = entry.callback,
|
|
callback_back = entry.callback_back,
|
|
textviewer_width = self.textviewer_width,
|
|
textviewer_height = self.textviewer_height,
|
|
value_overflow_align = self.value_overflow_align,
|
|
value_align = self.value_align,
|
|
show_parent = self,
|
|
})
|
|
if entry.separator then
|
|
table.insert(self.main_content, LineWidget:new{
|
|
background = Blitbuffer.COLOR_LIGHT_GRAY,
|
|
dimen = Geom:new{
|
|
w = self.item_width,
|
|
h = Size.line.thick
|
|
},
|
|
style = "solid",
|
|
})
|
|
end
|
|
elseif type(entry) == "string" then
|
|
-- deprecated, use separator=true on a regular k/v table
|
|
-- (kept in case some user plugins would use this)
|
|
local c = string.sub(entry, 1, 1)
|
|
if c == "-" then
|
|
table.insert(self.main_content, LineWidget:new{
|
|
background = Blitbuffer.COLOR_LIGHT_GRAY,
|
|
dimen = Geom:new{
|
|
w = self.item_width,
|
|
h = Size.line.thick
|
|
},
|
|
style = "solid",
|
|
})
|
|
end
|
|
end
|
|
end
|
|
|
|
-- update page information
|
|
if self.pages >= 1 then
|
|
self.page_info_text:setText(T(_("Page %1 of %2"), self.show_page, self.pages))
|
|
if self.pages > 1 then
|
|
self.page_info_text:enable()
|
|
else
|
|
self.page_info_text:disableWithoutDimming()
|
|
end
|
|
self.page_info_left_chev:show()
|
|
self.page_info_right_chev:show()
|
|
self.page_info_first_chev:show()
|
|
self.page_info_last_chev:show()
|
|
|
|
self.page_info_left_chev:enableDisable(self.show_page > 1)
|
|
self.page_info_right_chev:enableDisable(self.show_page < self.pages)
|
|
self.page_info_first_chev:enableDisable(self.show_page > 1)
|
|
self.page_info_last_chev:enableDisable(self.show_page < self.pages)
|
|
else
|
|
self.page_info_text:setText(_("No items"))
|
|
self.page_info_text:disableWithoutDimming()
|
|
|
|
self.page_info_left_chev:hide()
|
|
self.page_info_right_chev:hide()
|
|
self.page_info_first_chev:hide()
|
|
self.page_info_last_chev:hide()
|
|
end
|
|
|
|
UIManager:setDirty(self, function()
|
|
return "ui", self.dimen
|
|
end)
|
|
end
|
|
|
|
function KeyValuePage:onNextPage()
|
|
self:nextPage()
|
|
return true
|
|
end
|
|
|
|
function KeyValuePage:onPrevPage()
|
|
self:prevPage()
|
|
return true
|
|
end
|
|
|
|
function KeyValuePage:onSwipe(arg, ges_ev)
|
|
local direction = BD.flipDirectionIfMirroredUILayout(ges_ev.direction)
|
|
if direction == "west" then
|
|
self:nextPage()
|
|
return true
|
|
elseif direction == "east" then
|
|
self:prevPage()
|
|
return true
|
|
elseif direction == "south" then
|
|
-- Allow easier closing with swipe down
|
|
self:onClose()
|
|
elseif direction == "north" then
|
|
-- no use for now
|
|
do end -- luacheck: ignore 541
|
|
else -- diagonal swipe
|
|
-- trigger full refresh
|
|
UIManager:setDirty(nil, "full")
|
|
-- a long diagonal swipe may also be used for taking a screenshot,
|
|
-- so let it propagate
|
|
return false
|
|
end
|
|
end
|
|
|
|
function KeyValuePage:onMultiSwipe(arg, ges_ev)
|
|
-- For consistency with other fullscreen widgets where swipe south can't be
|
|
-- used to close and where we then allow any multiswipe to close, allow any
|
|
-- multiswipe to close this widget too.
|
|
self:onClose()
|
|
return true
|
|
end
|
|
|
|
function KeyValuePage:onClose()
|
|
UIManager:close(self)
|
|
return true
|
|
end
|
|
|
|
function KeyValuePage:onReturn()
|
|
if self.callback_return then
|
|
self:callback_return()
|
|
UIManager:close(self)
|
|
UIManager:setDirty(nil, "ui")
|
|
end
|
|
end
|
|
|
|
return KeyValuePage
|