2
0
mirror of https://github.com/koreader/koreader synced 2024-11-04 12:00:25 +00:00
koreader/plugins/goodreads.koplugin/doublekeyvaluepage.lua
2016-11-22 09:09:05 +01:00

352 lines
12 KiB
Lua

local InputContainer = require("ui/widget/container/inputcontainer")
local FrameContainer = require("ui/widget/container/framecontainer")
local VerticalGroup = require("ui/widget/verticalgroup")
local VerticalSpan = require("ui/widget/verticalspan")
local OverlapGroup = require("ui/widget/overlapgroup")
local LeftContainer = require("ui/widget/container/topcontainer")
local HorizontalSpan = require("ui/widget/horizontalspan")
local LineWidget = require("ui/widget/linewidget")
local Blitbuffer = require("ffi/blitbuffer")
local CloseButton = require("ui/widget/closebutton")
local UIManager = require("ui/uimanager")
local TextWidget = require("ui/widget/textwidget")
local GestureRange = require("ui/gesturerange")
local RenderText = require("ui/rendertext")
local Geom = require("ui/geometry")
local Font = require("ui/font")
local Device = require("device")
local Screen = Device.screen
local GoodreadsApi = require("goodreadsapi")
local LuaSettings = require("luasettings")
local DataStorage = require("datastorage")
local _ = require("gettext")
local InfoMessage = require("ui/widget/infomessage")
local DoubleKeyValueTitle = VerticalGroup:new{
kv_page = nil,
title = "",
tface = Font:getFace("tfont"),
align = "left",
}
function DoubleKeyValueTitle:init()
self.close_button = CloseButton:new{ window = self }
local btn_width = self.close_button:getSize().w
local title_txt_width = RenderText:sizeUtf8Text(
0, self.width, self.tface, self.title).x
local show_title_txt
if self.width < (title_txt_width + btn_width) then
show_title_txt = RenderText:truncateTextByWidth(
self.title, self.tface, self.width - btn_width)
else
show_title_txt = self.title
end
-- title and close button
table.insert(self, OverlapGroup:new{
dimen = { w = self.width },
TextWidget:new{
text = show_title_txt,
face = self.tface,
},
self.close_button,
})
-- page count and separation line
self.page_cnt = FrameContainer:new{
padding = 4,
margin = 0,
bordersize = 0,
background = Blitbuffer.COLOR_WHITE,
-- overlap offset x will be updated in setPageCount method
overlap_offset = {0, -15},
TextWidget:new{
text = "", -- page count
fgcolor = Blitbuffer.COLOR_GREY,
face = Font:getFace("ffont", 20),
},
}
self.title_bottom = OverlapGroup:new{
dimen = { w = self.width, h = Screen:scaleBySize(2) },
LineWidget:new{
dimen = Geom:new{ w = self.width, h = Screen:scaleBySize(2) },
background = Blitbuffer.COLOR_GREY,
style = "solid",
},
self.page_cnt,
}
table.insert(self, self.title_bottom)
table.insert(self, VerticalSpan:new{ width = Screen:scaleBySize(5) })
end
function DoubleKeyValueTitle:setPageCount(curr, total)
if total == 1 then
-- remove page count if there is only one page
table.remove(self.title_bottom, 2)
return
end
self.page_cnt[1]:setText(curr .. "/" .. total)
self.page_cnt.overlap_offset[1] = (self.width - self.page_cnt:getSize().w - 10)
self.title_bottom[2] = self.page_cnt
end
function DoubleKeyValueTitle:onClose()
self.kv_page:onClose()
return true
end
local DoubleKeyValueItem = InputContainer:new{
key = nil,
value = nil,
cface_up = Font:getFace("cfont", 22),
cface_down = Font:getFace("cfont", 18),
width = nil,
height = nil,
align = "left",
}
function DoubleKeyValueItem:init()
self.dimen = Geom:new{align = "left", w = self.width, h = self.height}
if self.callback and Device:isTouchDevice() then
self.ges_events.Tap = {
GestureRange:new{
ges = "tap",
range = self.dimen,
}
}
end
local key_w = RenderText:sizeUtf8Text(0, self.width, self.cface_down, self.key).x
local value_w = RenderText:sizeUtf8Text(0, self.width, self.cface_up, self.value).x
if key_w > self.width then
self.show_key = RenderText:truncateTextByWidth(self.key, self.cface_down, self.width)
else
self.show_key = self.key
end
if value_w > self.width then
self.show_value = RenderText:truncateTextByWidth(self.value, self.cface_up, self.width)
else
self.show_value = self.value
end
local h = self.dimen.h / 2
local w = self.dimen.w
self[1] = FrameContainer:new{
padding = 10,
bordersize = 0,
VerticalGroup:new{
dimen = Geom:new{ h = h, w = w },
padding = 10,
LeftContainer:new{
padding = 10,
dimen = Geom:new{ h = h, w = w },
TextWidget:new{
text = self.show_value,
padding = 10,
face = self.cface_up,
}
},
LeftContainer:new{
padding = 10,
dimen = Geom:new{ h = h / 5 , w = w },
HorizontalSpan:new{ width = Screen:scaleBySize(15), height = 3 }
},
LeftContainer:new{
padding = 10,
dimen = Geom:new{ h = h, w = w },
TextWidget:new{
text = self.show_key,
padding = 10,
face = self.cface_down,
}
}
}
}
end
function DoubleKeyValueItem:onTap()
local info = InfoMessage:new{text = _("Please wait...")}
UIManager:show(info)
UIManager:forceRePaint()
self.callback()
UIManager:close(info)
return true
end
local DoubleKeyValuePage = InputContainer:new{
title = "",
width = nil,
height = nil,
show_page = 1,
text_input = "",
pages = 1,
goodreads_key = "",
}
function DoubleKeyValuePage:readGRSettings()
self.gr_settings = LuaSettings:open(DataStorage:getSettingsDir().."/goodreadssettings.lua")
return self.gr_settings
end
function DoubleKeyValuePage:saveGRSettings(setting)
if not self.gr_settings then self:readGRSettings() end
self.gr_settings:saveSetting("goodreads", setting)
self.gr_settings:flush()
end
function DoubleKeyValuePage:init()
self.screen_width = Screen:getSize().w
self.screen_height = Screen:getSize().h
local gr_sett = self:readGRSettings().data
if gr_sett.goodreads then
self.goodreads_key = gr_sett.goodreads.key
self.goodreads_secret = gr_sett.goodreads.secret
end
self.kv_pairs = GoodreadsApi:showData(self.text_input, self.search_type, 1, self.goodreads_key)
self.total_res = GoodreadsApi:getTotalResults()
if self.total_res == nil then
self.total_res = 0
end
self.total_res = tonumber(self.total_res)
if self.kv_pairs == nil then
self.kv_pairs = {}
end
self.dimen = Geom:new{
w = self.width or self.screen_width,
h = self.height or self.screen_height,
}
if Device:isTouchDevice() then
self.ges_events.Swipe = {
GestureRange:new{
ges = "swipe",
range = self.dimen,
}
}
end
local padding = Screen:scaleBySize(10)
self.item_width = self.dimen.w - 2 * padding
self.item_height = Screen:scaleBySize(45)
-- setup title bar
self.title_bar = DoubleKeyValueTitle:new{
title = self.title,
width = self.item_width,
height = self.item_height,
kv_page = self,
}
-- setup main content
self.item_margin = self.item_height / 4
local line_height = self.item_height + 2 * self.item_margin
local content_height = self.dimen.h - self.title_bar:getSize().h
self.max_loaded_pages = 1
self.items_per_page = math.floor(content_height / line_height)
self.pages = math.ceil(self.total_res / self.items_per_page)
self.main_content = VerticalGroup:new{}
self:_populateItems()
-- assemble page
self[1] = FrameContainer:new{
height = self.dimen.h,
padding = padding,
bordersize = 0,
background = Blitbuffer.COLOR_WHITE,
VerticalGroup:new{
self.title_bar,
self.main_content,
},
}
end
function DoubleKeyValuePage:nextPage()
local new_page = math.min(self.show_page + 1, self.pages)
if (new_page * self.items_per_page > #self.kv_pairs) and (self.max_loaded_pages < new_page)
and #self.kv_pairs < self.total_res then
local api_page = math.floor(new_page * self.items_per_page / 20 ) + 1
-- load new portion of data
local new_pair = GoodreadsApi:showData(self.text_input, self.search_type, api_page, self.goodreads_key )
if new_pair == nil then return end
for _, v in pairs(new_pair) do
table.insert(self.kv_pairs, v)
end
end
if new_page > self.show_page then
if self.max_loaded_pages == self.show_page then
self.max_loaded_pages = self.max_loaded_pages + 1
end
self.show_page = new_page
self:_populateItems()
end
end
function DoubleKeyValuePage: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
-- make sure self.item_margin and self.item_height are set before calling this
function DoubleKeyValuePage:_populateItems()
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
table.insert(self.main_content,
VerticalSpan:new{ align = "left", width = self.item_margin })
if type(entry) == "table" then
table.insert(
self.main_content,
DoubleKeyValueItem:new{
height = self.item_height,
width = self.item_width,
key = entry[1],
value = entry[2],
align = "left",
callback = entry.callback,
}
)
elseif type(entry) == "string" then
local c = string.sub(entry, 1, 1)
if c == "-" then
table.insert(self.main_content, LineWidget:new{
background = Blitbuffer.COLOR_LIGHT_GREY,
dimen = Geom:new{
w = self.item_width,
h = Screen:scaleBySize(2)
},
style = "solid",
})
end
end
table.insert(self.main_content,
VerticalSpan:new{ width = self.item_margin })
end
self.title_bar:setPageCount(self.show_page, self.pages)
UIManager:setDirty(self, function()
return "ui", self.dimen
end)
end
function DoubleKeyValuePage:onSwipe(arg, ges_ev)
if ges_ev.direction == "west" then
local new_page = math.min(self.show_page + 1, self.pages)
if (new_page * self.items_per_page > #self.kv_pairs) and (self.max_loaded_pages < new_page)
and #self.kv_pairs < self.total_res then
local info = InfoMessage:new{text = _("Please wait...")}
UIManager:show(info)
UIManager:forceRePaint()
self:nextPage()
UIManager:close(info)
else
self:nextPage()
end
return true
elseif ges_ev.direction == "east" then
self:prevPage()
return true
end
end
function DoubleKeyValuePage:onClose()
UIManager:close(self)
return true
end
return DoubleKeyValuePage