mirror of
https://github.com/koreader/koreader.git
synced 2025-08-10 00:52:38 +00:00
Merge branch 'master' of git://github.com/hwhw/kindlepdfviewer.git
Conflicts: reader.lua unireader.lua
This commit is contained in:
43
blitbuffer.c
43
blitbuffer.c
@@ -93,26 +93,53 @@ static int blitToBuffer(lua_State *L) {
|
||||
int x, y;
|
||||
|
||||
// check bounds
|
||||
if(ydest < 0) {
|
||||
// negative ydest, try to compensate
|
||||
if(ydest + h > 0) {
|
||||
// shrink h by negative dest offset
|
||||
h += ydest;
|
||||
// extend source offset
|
||||
yoffs += -ydest;
|
||||
ydest = 0;
|
||||
} else {
|
||||
// effectively no height
|
||||
return 0;
|
||||
}
|
||||
} else if(ydest >= dst->h) {
|
||||
// we're told to paint to off-bound target coords
|
||||
return 0;
|
||||
}
|
||||
if(ydest + h > dst->h) {
|
||||
// clamp height if too large for target size
|
||||
h = dst->h - ydest;
|
||||
}
|
||||
if(yoffs >= src->h) {
|
||||
// recalculated source offset is out of bounds
|
||||
return 0;
|
||||
} else if(yoffs + h > src->h) {
|
||||
// clamp height if too large for source size
|
||||
h = src->h - yoffs;
|
||||
}
|
||||
if(ydest >= dst->h) {
|
||||
// same stuff for x coords:
|
||||
if(xdest < 0) {
|
||||
if(xdest + w > 0) {
|
||||
w += xdest;
|
||||
xoffs += -xdest;
|
||||
xdest = 0;
|
||||
} else {
|
||||
return 0;
|
||||
}
|
||||
} else if(xdest >= dst->w) {
|
||||
return 0;
|
||||
} else if(ydest + h > dst->h) {
|
||||
h = dst->h - ydest;
|
||||
}
|
||||
if(xdest + w > dst->w) {
|
||||
w = dst->w - xdest;
|
||||
}
|
||||
if(xoffs >= src->w) {
|
||||
return 0;
|
||||
} else if(xoffs + w > src->w) {
|
||||
w = src->w - xoffs;
|
||||
}
|
||||
if(xdest >= dst->w) {
|
||||
return 0;
|
||||
} else if(xdest + w > dst->w) {
|
||||
w = dst->w - xdest;
|
||||
}
|
||||
|
||||
uint8_t *dstptr;
|
||||
uint8_t *srcptr;
|
||||
|
||||
56
djvu.c
56
djvu.c
@@ -15,6 +15,7 @@
|
||||
You should have received a copy of the GNU General Public License
|
||||
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
#include <libdjvu/miniexp.h>
|
||||
#include <libdjvu/ddjvuapi.h>
|
||||
|
||||
#include "string.h"
|
||||
@@ -119,15 +120,60 @@ static int getNumberOfPages(lua_State *L) {
|
||||
return 1;
|
||||
}
|
||||
|
||||
/* not supported yet, so return empty table */
|
||||
static int getTableOfContent(lua_State *L) {
|
||||
/*int count = 1;*/
|
||||
static int walkTableOfContent(lua_State *L, miniexp_t r, int *count, int depth) {
|
||||
depth++;
|
||||
|
||||
miniexp_t lista = miniexp_cdr(r); // go inside bookmars in the list
|
||||
|
||||
int length = miniexp_length(r);
|
||||
int counter = 0;
|
||||
char page_number[6];
|
||||
|
||||
while(counter < length-1) {
|
||||
lua_pushnumber(L, *count);
|
||||
lua_newtable(L);
|
||||
|
||||
lua_pushstring(L, "page");
|
||||
strcpy(page_number,miniexp_to_str(miniexp_car(miniexp_cdr(miniexp_nth(counter, lista)))));
|
||||
/* page numbers appear as #11, set # to 0 so strtol works */
|
||||
page_number[0]= '0';
|
||||
lua_pushnumber(L, strtol(page_number, NULL, 10)+1);
|
||||
lua_settable(L, -3);
|
||||
|
||||
lua_pushstring(L, "depth");
|
||||
lua_pushnumber(L, depth);
|
||||
lua_settable(L, -3);
|
||||
|
||||
lua_pushstring(L, "title");
|
||||
lua_pushstring(L, miniexp_to_str(miniexp_car(miniexp_nth(counter, lista))));
|
||||
lua_settable(L, -3);
|
||||
|
||||
lua_settable(L, -3);
|
||||
|
||||
(*count)++;
|
||||
|
||||
if (miniexp_length(miniexp_cdr(miniexp_nth(counter, lista))) > 1) {
|
||||
walkTableOfContent(L, miniexp_cdr(miniexp_nth(counter, lista)), count, depth);
|
||||
}
|
||||
counter++;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
||||
static int getTableOfContent(lua_State *L) {
|
||||
DjvuDocument *doc = (DjvuDocument*) luaL_checkudata(L, 1, "djvudocument");
|
||||
/*ol = djvu_load_outline(doc->doc_ref);*/
|
||||
miniexp_t r;
|
||||
int count = 1;
|
||||
|
||||
while ((r=ddjvu_document_get_outline(doc->doc_ref))==miniexp_dummy)
|
||||
handle(L, doc->context, True);
|
||||
|
||||
//printf("lista: %s\n", miniexp_to_str(miniexp_car(miniexp_nth(1, miniexp_cdr(r)))));
|
||||
|
||||
lua_newtable(L);
|
||||
/*walkTableOfContent(L, ol, &count, 0);*/
|
||||
walkTableOfContent(L, r, &count, 0);
|
||||
|
||||
return 1;
|
||||
}
|
||||
|
||||
|
||||
24
einkfb.c
24
einkfb.c
@@ -180,6 +180,29 @@ static int einkUpdate(lua_State *L) {
|
||||
return 0;
|
||||
}
|
||||
|
||||
/* NOTICE!!! You must close and reopen framebuffer after called this method.
|
||||
* Otherwise, screen resolution will not be updated! */
|
||||
static int einkSetOrientation(lua_State *L) {
|
||||
#ifndef EMULATE_READER
|
||||
FBInfo *fb = (FBInfo*) luaL_checkudata(L, 1, "einkfb");
|
||||
int mode = luaL_optint(L, 2, 0);
|
||||
|
||||
if (mode < 0 || mode > 3) {
|
||||
return luaL_error(L, "Wrong rotation mode %d given!", mode);
|
||||
}
|
||||
|
||||
/* ioctl has a different definition for rotation mode. */
|
||||
if (mode == 1)
|
||||
mode = 2;
|
||||
else if (mode == 2)
|
||||
mode = 1;
|
||||
|
||||
ioctl(fb->fd, FBIO_EINK_SET_DISPLAY_ORIENTATION, mode);
|
||||
#endif
|
||||
return 0;
|
||||
}
|
||||
|
||||
|
||||
static const struct luaL_reg einkfb_func[] = {
|
||||
{"open", openFrameBuffer},
|
||||
{NULL, NULL}
|
||||
@@ -189,6 +212,7 @@ static const struct luaL_reg einkfb_meth[] = {
|
||||
{"close", closeFrameBuffer},
|
||||
{"__gc", closeFrameBuffer},
|
||||
{"refresh", einkUpdate},
|
||||
{"setOrientation", einkSetOrientation},
|
||||
{"getSize", getSize},
|
||||
{NULL, NULL}
|
||||
};
|
||||
|
||||
@@ -1,26 +1,13 @@
|
||||
require "rendertext"
|
||||
require "keys"
|
||||
require "graphics"
|
||||
require "fontchooser"
|
||||
require "font"
|
||||
require "filesearcher"
|
||||
require "inputbox"
|
||||
require "selectmenu"
|
||||
|
||||
FileChooser = {
|
||||
-- Class vars:
|
||||
-- font for displaying toc item names
|
||||
fsize = 25,
|
||||
face = nil,
|
||||
fhash = nil,
|
||||
--face = freetype.newBuiltinFace("sans", 25),
|
||||
--fhash = "s25",
|
||||
|
||||
-- font for paging display
|
||||
ffsize = 16,
|
||||
fface = nil,
|
||||
ffhash = nil,
|
||||
--sface = freetype.newBuiltinFace("sans", 16),
|
||||
--sfhash = "s16",
|
||||
|
||||
-- spacing between lines
|
||||
spacing = 40,
|
||||
@@ -93,17 +80,6 @@ function FileChooser:setPath(newPath)
|
||||
end
|
||||
end
|
||||
|
||||
function FileChooser:updateFont()
|
||||
if self.fhash ~= FontChooser.cfont..self.fsize then
|
||||
self.face = freetype.newBuiltinFace(FontChooser.cfont, self.fsize)
|
||||
self.fhash = FontChooser.cfont..self.fsize
|
||||
end
|
||||
if self.ffhash ~= FontChooser.ffont..self.ffsize then
|
||||
self.fface = freetype.newBuiltinFace(FontChooser.ffont, self.ffsize)
|
||||
self.ffhash = FontChooser.ffont..self.ffsize
|
||||
end
|
||||
end
|
||||
|
||||
function FileChooser:choose(ypos, height)
|
||||
local perpage = math.floor(height / self.spacing) - 2
|
||||
local pagedirty = true
|
||||
@@ -139,7 +115,9 @@ function FileChooser:choose(ypos, height)
|
||||
end
|
||||
|
||||
while true do
|
||||
self:updateFont()
|
||||
local cface, cfhash= Font:getFaceAndHash(25)
|
||||
local fface, ffhash = Font:getFaceAndHash(16, Font.ffont)
|
||||
|
||||
if pagedirty then
|
||||
fb.bb:paintRect(0, ypos, fb.bb:getWidth(), height, 0)
|
||||
local c
|
||||
@@ -147,17 +125,17 @@ function FileChooser:choose(ypos, height)
|
||||
local i = (self.page - 1) * perpage + c
|
||||
if i <= #self.dirs then
|
||||
-- resembles display in midnight commander: adds "/" prefix for directories
|
||||
renderUtf8Text(fb.bb, 39, ypos + self.spacing*c, self.face, self.fhash, "/", true)
|
||||
renderUtf8Text(fb.bb, 50, ypos + self.spacing*c, self.face, self.fhash, self.dirs[i], true)
|
||||
renderUtf8Text(fb.bb, 39, ypos + self.spacing*c, cface, cfhash, "/", true)
|
||||
renderUtf8Text(fb.bb, 50, ypos + self.spacing*c, cface, cfhash, self.dirs[i], true)
|
||||
elseif i <= self.items then
|
||||
renderUtf8Text(fb.bb, 50, ypos + self.spacing*c, self.face, self.fhash, self.files[i-#self.dirs], true)
|
||||
renderUtf8Text(fb.bb, 50, ypos + self.spacing*c, cface, cfhash, self.files[i-#self.dirs], true)
|
||||
end
|
||||
end
|
||||
renderUtf8Text(fb.bb, 5, ypos + self.spacing * perpage + 42, self.fface, self.ffhash,
|
||||
renderUtf8Text(fb.bb, 5, ypos + self.spacing * perpage + 42, fface, ffhash,
|
||||
"Page "..self.page.." of "..(math.floor(self.items / perpage)+1), true)
|
||||
local msg = self.exception_message and self.exception_message:match("[^%:]+:%d+: (.*)") or "Path: "..self.path
|
||||
self.exception_message = nil
|
||||
renderUtf8Text(fb.bb, 5, ypos + self.spacing * (perpage+1) + 27, self.fface, self.ffhash, msg, true)
|
||||
renderUtf8Text(fb.bb, 5, ypos + self.spacing * (perpage+1) + 27, fface, ffhash, msg, true)
|
||||
markerdirty = true
|
||||
end
|
||||
if markerdirty then
|
||||
@@ -190,21 +168,29 @@ function FileChooser:choose(ypos, height)
|
||||
elseif ev.code == KEY_F then -- invoke fontchooser menu
|
||||
fonts_menu = SelectMenu:new{
|
||||
menu_title = "Fonts Menu",
|
||||
item_array = FontChooser.fonts,
|
||||
item_array = Font.fonts,
|
||||
}
|
||||
local re = fonts_menu:choose(0, height)
|
||||
if re then
|
||||
FontChooser.cfont = FontChooser.fonts[re]
|
||||
FontChooser:init()
|
||||
Font.cfont = Font.fonts[re]
|
||||
Font:update()
|
||||
end
|
||||
pagedirty = true
|
||||
elseif ev.code == KEY_S then -- invoke search input
|
||||
keywords = InputBox:input(height-100, 100, "Search:")
|
||||
if keywords then -- display search result according to keywords
|
||||
FileSearcher:init( self.path )
|
||||
file = FileSearcher:choose(ypos, height, keywords)
|
||||
if file then
|
||||
return file
|
||||
if keywords then
|
||||
-- call FileSearcher
|
||||
--[[
|
||||
This might looks a little bit dirty for using callback.
|
||||
But I cannot come up with a better solution for renewing
|
||||
the height arguemtn according to screen rotation mode.
|
||||
|
||||
The callback might also be useful for calling system
|
||||
settings menu in the future.
|
||||
--]]
|
||||
return nil, function()
|
||||
FileSearcher:init( self.path )
|
||||
FileSearcher:choose(ypos, height, keywords)
|
||||
end
|
||||
end
|
||||
pagedirty = true
|
||||
|
||||
@@ -1,22 +1,9 @@
|
||||
require "rendertext"
|
||||
require "keys"
|
||||
require "graphics"
|
||||
require "fontchooser"
|
||||
require "font"
|
||||
|
||||
FileSearcher = {
|
||||
-- font for displaying toc item names
|
||||
fsize = 25,
|
||||
face = nil,
|
||||
fhash = nil,
|
||||
-- font for page title
|
||||
tfsize = 30,
|
||||
tface = nil,
|
||||
tfhash = nil,
|
||||
-- font for paging display
|
||||
ffsize = 16,
|
||||
fface = nil,
|
||||
ffhash = nil,
|
||||
|
||||
-- title height
|
||||
title_H = 45,
|
||||
-- spacing between lines
|
||||
@@ -70,23 +57,6 @@ function FileSearcher:setPath(newPath)
|
||||
return true
|
||||
end
|
||||
|
||||
function FileSearcher:updateFont()
|
||||
if self.fhash ~= FontChooser.cfont..self.fsize then
|
||||
self.face = freetype.newBuiltinFace(FontChooser.cfont, self.fsize)
|
||||
self.fhash = FontChooser.cfont..self.fsize
|
||||
end
|
||||
|
||||
if self.tfhash ~= FontChooser.tfont..self.tfsize then
|
||||
self.tface = freetype.newBuiltinFace(FontChooser.tfont, self.tfsize)
|
||||
self.tfhash = FontChooser.tfont..self.tfsize
|
||||
end
|
||||
|
||||
if self.ffhash ~= FontChooser.ffont..self.ffsize then
|
||||
self.fface = freetype.newBuiltinFace(FontChooser.ffont, self.ffsize)
|
||||
self.ffhash = FontChooser.ffont..self.ffsize
|
||||
end
|
||||
end
|
||||
|
||||
function FileSearcher:setSearchResult(keywords)
|
||||
self.result = {}
|
||||
if keywords == " " then -- one space to show all files
|
||||
@@ -153,22 +123,25 @@ function FileSearcher:choose(ypos, height, keywords)
|
||||
end
|
||||
|
||||
while true do
|
||||
self:updateFont()
|
||||
local cface, cfhash = Font:getFaceAndHash(22)
|
||||
local tface, tfhash = Font:getFaceAndHash(25, Font.tfont)
|
||||
local fface, ffhash = Font:getFaceAndHash(16, Font.ffont)
|
||||
|
||||
if pagedirty then
|
||||
markerdirty = true
|
||||
fb.bb:paintRect(0, ypos, fb.bb:getWidth(), height, 0)
|
||||
|
||||
-- draw menu title
|
||||
renderUtf8Text(fb.bb, 30, ypos + self.title_H, self.tface, self.tfhash,
|
||||
renderUtf8Text(fb.bb, 30, ypos + self.title_H, tface, tfhash,
|
||||
"Search Result for: "..keywords, true)
|
||||
|
||||
-- draw results
|
||||
local c
|
||||
if self.items == 0 then -- nothing found
|
||||
y = ypos + self.title_H + self.spacing * 2
|
||||
renderUtf8Text(fb.bb, 20, y, self.face, self.fhash,
|
||||
renderUtf8Text(fb.bb, 20, y, cface, cfhash,
|
||||
"Sorry, no match found.", true)
|
||||
renderUtf8Text(fb.bb, 20, y + self.spacing, self.face, self.fhash,
|
||||
renderUtf8Text(fb.bb, 20, y + self.spacing, cface, cfhash,
|
||||
"Please try a different keyword.", true)
|
||||
markerdirty = false
|
||||
else -- found something, draw it
|
||||
@@ -176,7 +149,7 @@ function FileSearcher:choose(ypos, height, keywords)
|
||||
local i = (self.page - 1) * perpage + c
|
||||
if i <= self.items then
|
||||
y = ypos + self.title_H + (self.spacing * c)
|
||||
renderUtf8Text(fb.bb, 50, y, self.face, self.fhash,
|
||||
renderUtf8Text(fb.bb, 50, y, cface, cfhash,
|
||||
self.result[i].name, true)
|
||||
end
|
||||
end
|
||||
@@ -186,7 +159,7 @@ function FileSearcher:choose(ypos, height, keywords)
|
||||
y = ypos + self.title_H + (self.spacing * perpage) + self.foot_H
|
||||
x = (fb.bb:getWidth() / 2) - 50
|
||||
all_page = (math.floor(self.items / perpage)+1)
|
||||
renderUtf8Text(fb.bb, x, y, self.fface, self.ffhash,
|
||||
renderUtf8Text(fb.bb, x, y, fface, ffhash,
|
||||
"Page "..self.page.." of "..all_page, true)
|
||||
end
|
||||
|
||||
@@ -251,19 +224,31 @@ function FileSearcher:choose(ypos, height, keywords)
|
||||
elseif ev.code == KEY_F then -- invoke fontchooser menu
|
||||
fonts_menu = SelectMenu:new{
|
||||
menu_title = "Fonts Menu",
|
||||
item_array = FontChooser.fonts,
|
||||
item_array = Font.fonts,
|
||||
}
|
||||
local re = fonts_menu:choose(0, height)
|
||||
if re then
|
||||
FontChooser.cfont = FontChooser.fonts[re]
|
||||
FontChooser:init()
|
||||
Font.cfont = Font.fonts[re]
|
||||
Font:update()
|
||||
end
|
||||
pagedirty = true
|
||||
elseif ev.code == KEY_ENTER or ev.code == KEY_FW_PRESS then
|
||||
file_entry = self.result[perpage*(self.page-1)+self.current]
|
||||
file_full_path = file_entry.dir .. "/" .. file_entry.name
|
||||
|
||||
-- rotation mode might be changed while reading, so
|
||||
-- record height_percent here
|
||||
local height_percent = height/fb.bb:getHeight()
|
||||
openFile(file_full_path)
|
||||
|
||||
--reset height and item index if screen has been rotated
|
||||
local old_perpage = perpage
|
||||
height = math.floor(fb.bb:getHeight()*height_percent)
|
||||
perpage = math.floor(height / self.spacing) - 2
|
||||
self.current = (old_perpage * (self.page - 1) +
|
||||
self.current) % perpage
|
||||
self.page = math.floor(self.items / perpage) + 1
|
||||
|
||||
pagedirty = true
|
||||
elseif ev.code == KEY_BACK or ev.code == KEY_HOME then
|
||||
return nil
|
||||
|
||||
47
font.lua
Normal file
47
font.lua
Normal file
@@ -0,0 +1,47 @@
|
||||
|
||||
Font = {
|
||||
-- default font for menu contents
|
||||
cfont = "sans",
|
||||
-- default font for title
|
||||
tfont = "Helvetica-BoldOblique",
|
||||
-- default font for footer
|
||||
ffont = "sans",
|
||||
|
||||
-- built in fonts
|
||||
fonts = {"sans", "cjk", "mono",
|
||||
"Courier", "Courier-Bold", "Courier-Oblique", "Courier-BoldOblique",
|
||||
"Helvetica", "Helvetica-Oblique", "Helvetica-BoldOblique",
|
||||
"Times-Roman", "Times-Bold", "Times-Italic", "Times-BoldItalic",},
|
||||
|
||||
-- face table
|
||||
faces = {},
|
||||
}
|
||||
|
||||
function Font:getFaceAndHash(size, font)
|
||||
if not font then
|
||||
-- default to content font
|
||||
font = self.cfont
|
||||
end
|
||||
|
||||
local face = self.faces[font..size]
|
||||
-- build face if not found
|
||||
if not face then
|
||||
for _k,_v in ipairs(self.fonts) do
|
||||
if font == _v then
|
||||
face = freetype.newBuiltinFace(font, size)
|
||||
self.faces[font..size] = face
|
||||
end
|
||||
end
|
||||
if not face then
|
||||
print("#! Font "..font.." not supported!!")
|
||||
return nil
|
||||
end
|
||||
end
|
||||
return face, font..size
|
||||
end
|
||||
|
||||
function Font:update()
|
||||
self.faces = {}
|
||||
clearglyphcache()
|
||||
end
|
||||
|
||||
@@ -1,20 +0,0 @@
|
||||
|
||||
FontChooser = {
|
||||
-- font name for menu contents
|
||||
cfont = "sans",
|
||||
-- font name for title
|
||||
tfont = "Helvetica-BoldOblique",
|
||||
-- font name for footer
|
||||
ffont = "sans",
|
||||
|
||||
-- state buffer
|
||||
fonts = {"sans", "cjk", "mono",
|
||||
"Courier", "Courier-Bold", "Courier-Oblique", "Courier-BoldOblique",
|
||||
"Helvetica", "Helvetica-Oblique", "Helvetica-BoldOblique",
|
||||
"Times-Roman", "Times-Bold", "Times-Italic", "Times-BoldItalic",},
|
||||
}
|
||||
|
||||
function FontChooser:init()
|
||||
clearglyphcache()
|
||||
end
|
||||
|
||||
22
graphics.lua
22
graphics.lua
@@ -5,3 +5,25 @@ blitbuffer.paintBorder = function (bb, x, y, w, h, bw, c)
|
||||
bb:paintRect(x, y+bw, bw, h - 2*bw, c)
|
||||
bb:paintRect(x+w-bw, y+bw, bw, h - 2*bw, c)
|
||||
end
|
||||
|
||||
--[[
|
||||
Draw a progress bar according to following args:
|
||||
|
||||
@x: start position in x axis
|
||||
@y: start position in y axis
|
||||
@w: width for progress bar
|
||||
@h: height for progress bar
|
||||
@load_m_w: width margin for loading bar
|
||||
@load_m_h: height margin for loading bar
|
||||
@load_percent: progress in percent
|
||||
@c: color for loading bar
|
||||
--]]
|
||||
blitbuffer.progressBar = function (bb, x, y, w, h,
|
||||
load_m_w, load_m_h, load_percent, c)
|
||||
if load_m_h*2 > h then
|
||||
load_m_h = h/2
|
||||
end
|
||||
blitbuffer.paintBorder(fb.bb, x, y, w, h, 2, 15)
|
||||
fb.bb:paintRect(x+load_m_w, y+load_m_h,
|
||||
(w-2*load_m_w)*load_percent, (h-2*load_m_h), c)
|
||||
end
|
||||
|
||||
16
inputbox.lua
16
inputbox.lua
@@ -25,8 +25,6 @@ InputBox = {
|
||||
function InputBox:setDefaultInput(text)
|
||||
self.input_string = ""
|
||||
self:addString(text)
|
||||
--renderUtf8Text(fb.bb, self.input_start_x, self.input_start_y,
|
||||
--self.face, self.fhash, text, true)
|
||||
--self.input_cur_x = self.input_start_x + (string.len(text) * self.fwidth)
|
||||
--self.input_string = text
|
||||
end
|
||||
@@ -181,19 +179,21 @@ function InputBox:input(ypos, height, title, d_text)
|
||||
elseif ev.code == KEY_PGBCK then
|
||||
elseif ev.code == KEY_ENTER or ev.code == KEY_FW_PRESS then
|
||||
if self.input_string == "" then
|
||||
return nil
|
||||
else
|
||||
return self.input_string
|
||||
self.input_string = nil
|
||||
end
|
||||
break
|
||||
elseif ev.code == KEY_DEL then
|
||||
self:delChar()
|
||||
elseif ev.code == KEY_BACK then
|
||||
return nil
|
||||
self.input_string = nil
|
||||
break
|
||||
end
|
||||
|
||||
--local nsecs, nusecs = util.gettime()
|
||||
--local dur = (nsecs - secs) * 1000000 + nusecs - usecs
|
||||
--print("E: T="..ev.type.." V="..ev.value.." C="..ev.code.." DUR="..dur)
|
||||
end
|
||||
end
|
||||
end -- if
|
||||
end -- while
|
||||
|
||||
return self.input_string
|
||||
end
|
||||
|
||||
40
keys.lua
40
keys.lua
@@ -186,35 +186,6 @@ function set_emu_keycodes()
|
||||
KEY_VMINUS = 96 -- F12
|
||||
end
|
||||
|
||||
function getRotationMode()
|
||||
--[[
|
||||
return code for four kinds of rotation mode:
|
||||
|
||||
0 for no rotation,
|
||||
1 for landscape with bottom on the right side of screen, etc.
|
||||
|
||||
2
|
||||
+-----------+
|
||||
| +-------+ |
|
||||
| | | |
|
||||
| | | |
|
||||
| | | |
|
||||
3 | | | | 1
|
||||
| | | |
|
||||
| | | |
|
||||
| +-------+ |
|
||||
| |
|
||||
+-----------+
|
||||
0
|
||||
--]]
|
||||
if KEY_FW_DOWN == 116 then -- in EMU mode always return 0
|
||||
return 0
|
||||
end
|
||||
orie_fd = assert(io.open("/sys/module/eink_fb_hal_broads/parameters/bs_orientation", "r"))
|
||||
updown_fd = assert(io.open("/sys/module/eink_fb_hal_broads/parameters/bs_upside_down", "r"))
|
||||
mode = orie_fd:read() + (updown_fd:read() * 2)
|
||||
return mode
|
||||
end
|
||||
|
||||
function adjustKeyEvents(ev)
|
||||
if ev.type == EV_KEY and ev.value == EVENT_VALUE_KEY_PRESS then
|
||||
@@ -233,9 +204,9 @@ function adjustKeyEvents(ev)
|
||||
|
||||
-- adjust five way key according to rotation mode
|
||||
local code = ev.code
|
||||
if getRotationMode() == 0 then
|
||||
if Screen.cur_rotation_mode == 0 then
|
||||
return code
|
||||
elseif getRotationMode() == 1 then
|
||||
elseif Screen.cur_rotation_mode == 1 then
|
||||
if code == KEY_FW_UP then
|
||||
return KEY_FW_RIGHT
|
||||
elseif code == KEY_FW_RIGHT then
|
||||
@@ -247,7 +218,7 @@ function adjustKeyEvents(ev)
|
||||
else
|
||||
return code
|
||||
end
|
||||
elseif getRotationMode() == 2 then
|
||||
elseif Screen.cur_rotation_mode == 2 then
|
||||
if code == KEY_FW_UP then
|
||||
return KEY_FW_DOWN
|
||||
elseif code == KEY_FW_RIGHT then
|
||||
@@ -259,7 +230,7 @@ function adjustKeyEvents(ev)
|
||||
else
|
||||
return code
|
||||
end
|
||||
elseif getRotationMode() == 3 then
|
||||
elseif Screen.cur_rotation_mode == 3 then
|
||||
if code == KEY_FW_UP then
|
||||
return KEY_FW_LEFT
|
||||
elseif code == KEY_FW_RIGHT then
|
||||
@@ -272,4 +243,7 @@ function adjustKeyEvents(ev)
|
||||
return code
|
||||
end
|
||||
end
|
||||
-- This should not happen.
|
||||
print("# Unrecognizable rotation mode "..Screen.cur_rotation_mode.."!")
|
||||
return nil
|
||||
end
|
||||
|
||||
26
reader.lua
26
reader.lua
@@ -22,8 +22,7 @@ require "pdfreader"
|
||||
require "djvureader"
|
||||
require "filechooser"
|
||||
require "settings"
|
||||
require "keys"
|
||||
require "commands"
|
||||
require "screen"
|
||||
|
||||
-- option parsing:
|
||||
longopts = {
|
||||
@@ -112,12 +111,15 @@ end
|
||||
|
||||
fb = einkfb.open("/dev/fb0")
|
||||
width, height = fb:getSize()
|
||||
-- read current rotation mode
|
||||
Screen:updateRotationMode()
|
||||
origin_rotation_mode = Screen.cur_rotation_mode
|
||||
|
||||
-- set up reader's setting: font
|
||||
reader_settings = DocSettings:open(".reader")
|
||||
r_cfont = reader_settings:readsetting("cfont")
|
||||
if r_cfont ~=nil then
|
||||
FontChooser.cfont = r_cfont
|
||||
Font.cfont = r_cfont
|
||||
end
|
||||
|
||||
-- initialize global settings shared among all readers
|
||||
@@ -132,11 +134,15 @@ if ARGV[optind] and lfs.attributes(ARGV[optind], "mode") == "directory" then
|
||||
local running = true
|
||||
FileChooser:setPath(ARGV[optind])
|
||||
while running do
|
||||
local file = FileChooser:choose(0,height)
|
||||
if file ~= nil then
|
||||
running = openFile(file)
|
||||
local file, callback = FileChooser:choose(0,height)
|
||||
if callback then
|
||||
callback()
|
||||
else
|
||||
running = false
|
||||
if file ~= nil then
|
||||
running = openFile(file)
|
||||
else
|
||||
running = false
|
||||
end
|
||||
end
|
||||
end
|
||||
elseif ARGV[optind] and lfs.attributes(ARGV[optind], "mode") == "file" then
|
||||
@@ -149,9 +155,13 @@ end
|
||||
|
||||
|
||||
-- save reader settings
|
||||
reader_settings:savesetting("cfont", FontChooser.cfont)
|
||||
reader_settings:savesetting("cfont", Font.cfont)
|
||||
reader_settings:close()
|
||||
|
||||
-- @TODO dirty workaround, find a way to force native system poll
|
||||
-- screen orientation and upside down mode 09.03 2012
|
||||
fb:setOrientation(origin_rotation_mode)
|
||||
|
||||
input.closeAll()
|
||||
--os.execute('test -e /proc/keypad && echo "send '..KEY_HOME..'" > /proc/keypad ')
|
||||
if optarg["d"] ~= "emu" then
|
||||
|
||||
74
screen.lua
Normal file
74
screen.lua
Normal file
@@ -0,0 +1,74 @@
|
||||
--[[
|
||||
Copyright (C) 2011 Hans-Werner Hilse <hilse@web.de>
|
||||
|
||||
This program is free software: you can redistribute it and/or modify
|
||||
it under the terms of the GNU General Public License as published by
|
||||
the Free Software Foundation, either version 3 of the License, or
|
||||
(at your option) any later version.
|
||||
|
||||
This program is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
GNU General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU General Public License
|
||||
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
]]--
|
||||
|
||||
--[[
|
||||
Codes for rotation modes:
|
||||
|
||||
1 for no rotation,
|
||||
2 for landscape with bottom on the right side of screen, etc.
|
||||
|
||||
2
|
||||
+--------------+
|
||||
| +----------+ |
|
||||
| | | |
|
||||
| | Freedom! | |
|
||||
| | | |
|
||||
| | | |
|
||||
3 | | | | 1
|
||||
| | | |
|
||||
| | | |
|
||||
| +----------+ |
|
||||
| |
|
||||
| |
|
||||
+--------------+
|
||||
0
|
||||
--]]
|
||||
|
||||
|
||||
Screen = {
|
||||
cur_rotation_mode = 0,
|
||||
}
|
||||
|
||||
-- @orien: 1 for clockwise rotate, -1 for anti-clockwise
|
||||
-- Remember to reread screen resolution after this function call
|
||||
function Screen:screenRotate(orien)
|
||||
if orien == "clockwise" then
|
||||
orien = -1
|
||||
elseif orien == "anticlockwise" then
|
||||
orien = 1
|
||||
else
|
||||
return
|
||||
end
|
||||
|
||||
self.cur_rotation_mode = (self.cur_rotation_mode + orien) % 4
|
||||
-- you have to reopen framebuffer after rotate
|
||||
fb:setOrientation(self.cur_rotation_mode)
|
||||
fb:close()
|
||||
fb = einkfb.open("/dev/fb0")
|
||||
end
|
||||
|
||||
function Screen:updateRotationMode()
|
||||
if KEY_FW_DOWN == 116 then -- in EMU mode always set to 0
|
||||
self.cur_rotation_mode = 0
|
||||
else
|
||||
orie_fd = assert(io.open("/sys/module/eink_fb_hal_broads/parameters/bs_orientation", "r"))
|
||||
updown_fd = assert(io.open("/sys/module/eink_fb_hal_broads/parameters/bs_upside_down", "r"))
|
||||
self.cur_rotation_mode = orie_fd:read() + (updown_fd:read() * 2)
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
@@ -1,21 +1,15 @@
|
||||
require "rendertext"
|
||||
require "keys"
|
||||
require "graphics"
|
||||
require "fontchooser"
|
||||
require "font"
|
||||
|
||||
SelectMenu = {
|
||||
-- font for displaying item names
|
||||
fsize = 22,
|
||||
face = nil,
|
||||
fhash = nil,
|
||||
-- font for page title
|
||||
tfsize = 25,
|
||||
tface = nil,
|
||||
tfhash = nil,
|
||||
-- font for paging display
|
||||
ffsize = 16,
|
||||
fface = nil,
|
||||
ffhash = nil,
|
||||
-- font for item shortcut
|
||||
sface = freetype.newBuiltinFace("mono", 22),
|
||||
sfhash = "mono22",
|
||||
@@ -59,23 +53,6 @@ function SelectMenu:new(o)
|
||||
return o
|
||||
end
|
||||
|
||||
function SelectMenu:updateFont()
|
||||
if self.fhash ~= FontChooser.cfont..self.fsize then
|
||||
self.face = freetype.newBuiltinFace(FontChooser.cfont, self.fsize)
|
||||
self.fhash = FontChooser.cfont..self.fsize
|
||||
end
|
||||
|
||||
if self.tfhash ~= FontChooser.tfont..self.tfsize then
|
||||
self.tface = freetype.newBuiltinFace(FontChooser.tfont, self.tfsize)
|
||||
self.tfhash = FontChooser.tfont..self.tfsize
|
||||
end
|
||||
|
||||
if self.ffhash ~= FontChooser.ffont..self.ffsize then
|
||||
self.fface = freetype.newBuiltinFace(FontChooser.ffont, self.ffsize)
|
||||
self.ffhash = FontChooser.ffont..self.ffsize
|
||||
end
|
||||
end
|
||||
|
||||
function SelectMenu:getItemIndexByShortCut(c, perpage)
|
||||
if c == nil then return end -- unused key
|
||||
for _k,_v in ipairs(self.item_shortcuts) do
|
||||
@@ -92,7 +69,6 @@ function SelectMenu:choose(ypos, height)
|
||||
local perpage = math.floor(height / self.spacing) - 2
|
||||
local pagedirty = true
|
||||
local markerdirty = false
|
||||
self:updateFont()
|
||||
|
||||
local prevItem = function ()
|
||||
if self.current == 1 then
|
||||
@@ -126,25 +102,28 @@ function SelectMenu:choose(ypos, height)
|
||||
self.last_shortcut = 0
|
||||
|
||||
while true do
|
||||
local cface, cfhash = Font:getFaceAndHash(22)
|
||||
local tface, tfhash = Font:getFaceAndHash(25, Font.tfont)
|
||||
local fface, ffhash = Font:getFaceAndHash(16, Font.ffont)
|
||||
|
||||
if pagedirty then
|
||||
markerdirty = true
|
||||
-- draw menu title
|
||||
fb.bb:paintRect(0, ypos, fb.bb:getWidth(), self.title_H + 10, 0)
|
||||
fb.bb:paintRect(10, ypos + 10, fb.bb:getWidth() - 20, self.title_H, 5)
|
||||
|
||||
x = 20
|
||||
y = ypos + self.title_H
|
||||
renderUtf8Text(fb.bb, x, y, self.tface, self.tfhash,
|
||||
self.menu_title, true)
|
||||
local x = 20
|
||||
local y = ypos + self.title_H
|
||||
renderUtf8Text(fb.bb, x, y, tface, tfhash, self.menu_title, true)
|
||||
|
||||
-- draw items
|
||||
fb.bb:paintRect(0, ypos + self.title_H + 10, fb.bb:getWidth(), height - self.title_H, 0)
|
||||
if self.items == 0 then
|
||||
y = ypos + self.title_H + (self.spacing * 2)
|
||||
renderUtf8Text(fb.bb, 30, y, self.face, self.fhash,
|
||||
renderUtf8Text(fb.bb, 30, y, cface, cfhash,
|
||||
"Oops... Bad news for you:", true)
|
||||
y = y + self.spacing
|
||||
renderUtf8Text(fb.bb, 30, y, self.face, self.fhash,
|
||||
renderUtf8Text(fb.bb, 30, y, cface, cfhash,
|
||||
self.no_item_msg, true)
|
||||
markerdirty = false
|
||||
else
|
||||
@@ -160,8 +139,10 @@ function SelectMenu:choose(ypos, height)
|
||||
else
|
||||
fb.bb:paintRect(10, y-22, 29, 29, 3)
|
||||
end
|
||||
if self.item_shortcuts[c] ~= nil and string.len(self.item_shortcuts[c]) == 3 then
|
||||
renderUtf8Text(fb.bb, 13, y, self.fface, self.ffhash,
|
||||
if self.item_shortcuts[c] ~= nil and
|
||||
string.len(self.item_shortcuts[c]) == 3 then
|
||||
-- print "Del", "Sym and "Ent"
|
||||
renderUtf8Text(fb.bb, 13, y, fface, ffhash,
|
||||
self.item_shortcuts[c], true)
|
||||
else
|
||||
renderUtf8Text(fb.bb, 18, y, self.sface, self.sfhash,
|
||||
@@ -170,7 +151,7 @@ function SelectMenu:choose(ypos, height)
|
||||
|
||||
self.last_shortcut = c
|
||||
|
||||
renderUtf8Text(fb.bb, 50, y, self.face, self.fhash,
|
||||
renderUtf8Text(fb.bb, 50, y, cface, cfhash,
|
||||
self.item_array[i], true)
|
||||
end
|
||||
end
|
||||
@@ -179,7 +160,7 @@ function SelectMenu:choose(ypos, height)
|
||||
-- draw footer
|
||||
y = ypos + self.title_H + (self.spacing * perpage) + self.foot_H + 5
|
||||
x = (fb.bb:getWidth() / 2) - 50
|
||||
renderUtf8Text(fb.bb, x, y, self.fface, self.ffhash,
|
||||
renderUtf8Text(fb.bb, x, y, fface, ffhash,
|
||||
"Page "..self.page.." of "..(math.floor(self.items / perpage)+1), true)
|
||||
end
|
||||
|
||||
|
||||
288
unireader.lua
288
unireader.lua
@@ -70,6 +70,9 @@ UniReader = {
|
||||
-- tile cache state:
|
||||
cache_current_memsize = 0,
|
||||
cache = {},
|
||||
|
||||
pagehash = nil,
|
||||
|
||||
jump_stack = {},
|
||||
toc = nil,
|
||||
|
||||
@@ -137,6 +140,16 @@ function UniReader:initGlobalSettings(settings)
|
||||
end
|
||||
-- initialize commands
|
||||
self:add_all_commands()
|
||||
|
||||
local cache_max_memsize = settings:readsetting("cache_max_memsize")
|
||||
if cache_max_memsize then
|
||||
self.cache_max_memsize = cache_max_memsize
|
||||
end
|
||||
|
||||
local cache_max_ttl = settings:readsetting("cache_max_ttl")
|
||||
if cache_max_ttl then
|
||||
self.cache_max_ttl = cache_max_ttl
|
||||
end
|
||||
end
|
||||
|
||||
-- guarantee that we have enough memory in cache
|
||||
@@ -163,34 +176,111 @@ function UniReader:cacheclaim(size)
|
||||
return true
|
||||
end
|
||||
|
||||
function UniReader:draworcache(no, zoom, offset_x, offset_y, width, height, gamma, rotate)
|
||||
-- hash draw state
|
||||
local hash = self:cachehash(no, zoom, offset_x, offset_y, width, height, gamma, rotate)
|
||||
if self.cache[hash] == nil then
|
||||
-- not in cache, so prepare cache slot...
|
||||
self:cacheclaim(width * height / 2);
|
||||
self.cache[hash] = {
|
||||
ttl = self.cache_max_ttl,
|
||||
size = width * height / 2,
|
||||
bb = Blitbuffer.new(width, height)
|
||||
}
|
||||
-- and draw the page
|
||||
local page = self.doc:openPage(no)
|
||||
local dc = self:setzoom(page, hash)
|
||||
page:draw(dc, self.cache[hash].bb, 0, 0)
|
||||
page:close()
|
||||
else
|
||||
-- we have the page in our cache,
|
||||
-- so give it more ttl.
|
||||
self.cache[hash].ttl = self.cache_max_ttl
|
||||
end
|
||||
return hash
|
||||
end
|
||||
function UniReader:draworcache(no, preCache)
|
||||
-- our general caching strategy is as follows:
|
||||
-- #1 goal: we must render the needed area.
|
||||
-- #2 goal: we render as much of the requested page as we can
|
||||
-- #3 goal: we render the full page
|
||||
-- #4 goal: we render next page, too. (TODO)
|
||||
|
||||
-- calculate a hash for our current state
|
||||
function UniReader:cachehash(no, zoom, offset_x, offset_y, width, height, gamma, rotate)
|
||||
-- TODO (?): make this a "real" hash...
|
||||
return no..'_'..zoom..'_'..offset_x..','..offset_y..'-'..width..'x'..height..'_'..gamma..'_'..rotate
|
||||
-- ideally, this should be factored out and only be called when needed (TODO)
|
||||
local page = self.doc:openPage(no)
|
||||
local dc = self:setzoom(page)
|
||||
|
||||
-- offset_x_in_page & offset_y_in_page is the offset within zoomed page
|
||||
-- they are always positive.
|
||||
-- you can see self.offset_x_& self.offset_y as the offset within
|
||||
-- draw space, which includes the page. So it can be negative and positive.
|
||||
local offset_x_in_page = -self.offset_x
|
||||
local offset_y_in_page = -self.offset_y
|
||||
if offset_x_in_page < 0 then offset_x_in_page = 0 end
|
||||
if offset_y_in_page < 0 then offset_y_in_page = 0 end
|
||||
|
||||
-- check if we have relevant cache contents
|
||||
local pagehash = no..'_'..self.globalzoom..'_'..self.globalrotate..'_'..self.globalgamma
|
||||
if self.cache[pagehash] ~= nil then
|
||||
-- we have something in cache, check if it contains the requested part
|
||||
if self.cache[pagehash].x <= offset_x_in_page
|
||||
and self.cache[pagehash].y <= offset_y_in_page
|
||||
and ( self.cache[pagehash].x + self.cache[pagehash].w >= offset_x_in_page + width
|
||||
or self.cache[pagehash].w >= self.fullwidth - 1)
|
||||
and ( self.cache[pagehash].y + self.cache[pagehash].h >= offset_y_in_page + height
|
||||
or self.cache[pagehash].h >= self.fullheight - 1)
|
||||
then
|
||||
-- requested part is within cached tile
|
||||
-- ...so properly clean page
|
||||
page:close()
|
||||
-- ...and give it more time to live (ttl), except if we're precaching
|
||||
if not preCache then
|
||||
self.cache[pagehash].ttl = self.cache_max_ttl
|
||||
end
|
||||
-- ...and return blitbuffer plus offset into it
|
||||
return pagehash,
|
||||
offset_x_in_page - self.cache[pagehash].x,
|
||||
offset_y_in_page - self.cache[pagehash].y
|
||||
end
|
||||
end
|
||||
-- okay, we do not have it in cache yet.
|
||||
-- so render now.
|
||||
-- start off with the requested area
|
||||
local tile = { x = offset_x_in_page, y = offset_y_in_page,
|
||||
w = width, h = height }
|
||||
-- can we cache the full page?
|
||||
local max_cache = self.cache_max_memsize
|
||||
if preCache then
|
||||
max_cache = max_cache - self.cache[self.pagehash].size
|
||||
end
|
||||
if (self.fullwidth * self.fullheight / 2) <= max_cache then
|
||||
-- yes we can, so do this with offset 0, 0
|
||||
tile.x = 0
|
||||
tile.y = 0
|
||||
tile.w = self.fullwidth
|
||||
tile.h = self.fullheight
|
||||
elseif (tile.w*tile.h / 2) > max_cache then
|
||||
-- no, we can't. so generate a tile as big as we can go
|
||||
-- grow area in steps of 10px
|
||||
while ((tile.w+10) * (tile.h+10) / 2) < max_cache do
|
||||
if tile.x > 0 then
|
||||
tile.x = tile.x - 5
|
||||
tile.w = tile.w + 5
|
||||
end
|
||||
if tile.x + tile.w < self.fullwidth then
|
||||
tile.w = tile.w + 5
|
||||
end
|
||||
if tile.y > 0 then
|
||||
tile.y = tile.y - 5
|
||||
tile.h = tile.h + 5
|
||||
end
|
||||
if tile.y + tile.h < self.fullheigth then
|
||||
tile.h = tile.h + 5
|
||||
end
|
||||
end
|
||||
else
|
||||
if not preCache then
|
||||
print("E: not enough memory in cache left, probably a bug.")
|
||||
end
|
||||
return nil
|
||||
end
|
||||
self:cacheclaim(tile.w * tile.h / 2);
|
||||
self.cache[pagehash] = {
|
||||
x = tile.x,
|
||||
y = tile.y,
|
||||
w = tile.w,
|
||||
h = tile.h,
|
||||
ttl = self.cache_max_ttl,
|
||||
size = tile.w * tile.h / 2,
|
||||
bb = Blitbuffer.new(tile.w, tile.h)
|
||||
}
|
||||
--print ("# new biltbuffer:"..dump(self.cache[pagehash]))
|
||||
dc:setOffset(-tile.x, -tile.y)
|
||||
print("# rendering: page="..no)
|
||||
page:draw(dc, self.cache[pagehash].bb, 0, 0)
|
||||
page:close()
|
||||
|
||||
-- return hash and offset within blitbuffer
|
||||
return pagehash,
|
||||
offset_x_in_page - tile.x,
|
||||
offset_y_in_page - tile.y
|
||||
end
|
||||
|
||||
-- blank the cache
|
||||
@@ -329,7 +419,6 @@ function UniReader:setzoom(page)
|
||||
self.globalzoom_orig = self.globalzoom
|
||||
|
||||
dc:setRotate(self.globalrotate);
|
||||
dc:setOffset(self.offset_x, self.offset_y)
|
||||
self.fullwidth, self.fullheight = page:getSize(dc)
|
||||
self.min_offset_x = fb.bb:getWidth() - self.fullwidth
|
||||
self.min_offset_y = fb.bb:getHeight() - self.fullheight
|
||||
@@ -352,13 +441,34 @@ end
|
||||
|
||||
-- render and blit a page
|
||||
function UniReader:show(no)
|
||||
local slot
|
||||
if self.globalzoommode ~= self.ZOOM_BY_VALUE then
|
||||
slot = self:draworcache(no,self.globalzoommode,self.offset_x,self.offset_y,width,height,self.globalgamma,self.globalrotate)
|
||||
else
|
||||
slot = self:draworcache(no,self.globalzoom,self.offset_x,self.offset_y,width,height,self.globalgamma,self.globalrotate)
|
||||
local pagehash, offset_x, offset_y = self:draworcache(no)
|
||||
self.pagehash = pagehash
|
||||
local bb = self.cache[pagehash].bb
|
||||
local dest_x = 0
|
||||
local dest_y = 0
|
||||
if bb:getWidth() - offset_x < width then
|
||||
-- we can't fill the whole output width, center the content
|
||||
dest_x = (width - (bb:getWidth() - offset_x)) / 2
|
||||
end
|
||||
fb.bb:blitFullFrom(self.cache[slot].bb)
|
||||
if bb:getHeight() - offset_y < height and
|
||||
self.globalzoommode ~= self.ZOOM_FIT_TO_CONTENT_WIDTH_PAN then
|
||||
-- we can't fill the whole output height and not in
|
||||
-- ZOOM_FIT_TO_CONTENT_WIDTH_PAN mode, center the content
|
||||
dest_y = (height - (bb:getHeight() - offset_y)) / 2
|
||||
elseif self.globalzoommode == self.ZOOM_FIT_TO_CONTENT_WIDTH_PAN and
|
||||
self.offset_y > 0 then
|
||||
-- if we are in ZOOM_FIT_TO_CONTENT_WIDTH_PAN mode and turning to
|
||||
-- the top of the page, we might leave an empty space between the
|
||||
-- page top and screen top.
|
||||
dest_y = self.offset_y
|
||||
end
|
||||
if dest_x or dest_y then
|
||||
fb.bb:paintRect(0, 0, width, height, 8)
|
||||
end
|
||||
print("# blitFrom dest_off:("..dest_x..", "..dest_y..
|
||||
"), src_off:("..offset_x..", "..offset_y.."), "..
|
||||
"width:"..width..", height:"..height)
|
||||
fb.bb:blitFrom(bb, dest_x, dest_y, offset_x, offset_y, width, height)
|
||||
if self.rcount == self.rcountmax then
|
||||
print("full refresh")
|
||||
self.rcount = 1
|
||||
@@ -439,14 +549,12 @@ function UniReader:goto(no)
|
||||
self.pageno = no
|
||||
self:show(no)
|
||||
|
||||
-- TODO: move the following to a more appropriate place
|
||||
-- into the caching section
|
||||
if no < self.doc:getPages() then
|
||||
if self.globalzoommode ~= self.ZOOM_BY_VALUE then
|
||||
if #self.bbox == 0 or not self.bbox.enabled then
|
||||
-- pre-cache next page, but if we will modify bbox don't!
|
||||
self:draworcache(no+1,self.globalzoommode,self.offset_x,self.offset_y,width,height,self.globalgamma,self.globalrotate)
|
||||
end
|
||||
else
|
||||
self:draworcache(no,self.globalzoom,self.offset_x,self.offset_y,width,height,self.globalgamma,self.globalrotate)
|
||||
if #self.bbox == 0 or not self.bbox.enabled then
|
||||
-- pre-cache next page, but if we will modify bbox don't!
|
||||
self:draworcache(no+1, true)
|
||||
end
|
||||
end
|
||||
end
|
||||
@@ -531,6 +639,14 @@ function UniReader:setrotate(rotate)
|
||||
self:goto(self.pageno)
|
||||
end
|
||||
|
||||
-- @ orien: 1 for clockwise rotate, -1 for anti-clockwise
|
||||
function UniReader:screenRotate(orien)
|
||||
Screen:screenRotate(orien)
|
||||
width, height = fb:getSize()
|
||||
self:clearcache()
|
||||
self:goto(self.pageno)
|
||||
end
|
||||
|
||||
function UniReader:cleanUpTOCTitle(title)
|
||||
return title:gsub("\13", "")
|
||||
end
|
||||
@@ -544,13 +660,20 @@ function UniReader:getTOCTitleByPage(pageno)
|
||||
-- build toc when needed.
|
||||
self:fillTOC()
|
||||
end
|
||||
|
||||
for _k,_v in ipairs(self.toc) do
|
||||
if _v.page >= pageno then
|
||||
return self:cleanUpTOCTitle(_v.title)
|
||||
end
|
||||
|
||||
-- no table of content
|
||||
if #self.toc == 0 then
|
||||
return ""
|
||||
end
|
||||
return ""
|
||||
|
||||
local pre_entry = self.toc[1]
|
||||
for _k,_v in ipairs(self.toc) do
|
||||
if _v.page > pageno then
|
||||
break
|
||||
end
|
||||
pre_entry = _v
|
||||
end
|
||||
return self:cleanUpTOCTitle(pre_entry.title)
|
||||
end
|
||||
|
||||
function UniReader:showTOC()
|
||||
@@ -603,6 +726,37 @@ function UniReader:showJumpStack()
|
||||
end
|
||||
end
|
||||
|
||||
function UniReader:showMenu()
|
||||
local ypos = height - 50
|
||||
local load_percent = (self.pageno / self.doc:getPages())
|
||||
|
||||
fb.bb:paintRect(0, ypos, width, 50, 0)
|
||||
|
||||
ypos = ypos + 15
|
||||
local face, fhash = Font:getFaceAndHash(22)
|
||||
local cur_section = self:getTOCTitleByPage(self.pageno)
|
||||
if cur_section ~= "" then
|
||||
cur_section = "Section: "..cur_section
|
||||
end
|
||||
renderUtf8Text(fb.bb, 10, ypos+6, face, fhash,
|
||||
"Page: "..self.pageno.."/"..self.doc:getPages()..
|
||||
" "..cur_section, true)
|
||||
|
||||
ypos = ypos + 15
|
||||
blitbuffer.progressBar(fb.bb, 10, ypos, width-20, 15,
|
||||
5, 4, load_percent, 8)
|
||||
fb:refresh(1)
|
||||
while 1 do
|
||||
local ev = input.waitForEvent()
|
||||
ev.code = adjustKeyEvents(ev)
|
||||
if ev.type == EV_KEY and ev.value == EVENT_VALUE_KEY_PRESS then
|
||||
if ev.code == KEY_BACK or ev.code == KEY_MENU then
|
||||
return
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
function UniReader:odd_even(number)
|
||||
print("## odd_even "..number)
|
||||
if number % 2 == 1 then
|
||||
@@ -697,6 +851,17 @@ function UniReader:inputloop()
|
||||
else
|
||||
self:setglobalzoommode(self.ZOOM_FIT_TO_CONTENT_HALF_WIDTH_MARGIN)
|
||||
end
|
||||
elseif ev.code == KEY_G then
|
||||
local page = InputBox:input(height-100, 100, "Page:")
|
||||
-- convert string to number
|
||||
if not pcall(function () page = page + 0 end) then
|
||||
page = self.pageno
|
||||
else
|
||||
if page < 1 or page > self.doc:getPages() then
|
||||
page = self.pageno
|
||||
end
|
||||
end
|
||||
self:goto(page)
|
||||
elseif ev.code == KEY_T then
|
||||
self:showTOC()
|
||||
elseif ev.code == KEY_B then
|
||||
@@ -706,9 +871,17 @@ function UniReader:inputloop()
|
||||
self:showJumpStack()
|
||||
end
|
||||
elseif ev.code == KEY_J then
|
||||
self:setrotate( self.globalrotate + 10 )
|
||||
if Keys.shiftmode then
|
||||
self:screenRotate("clockwise")
|
||||
else
|
||||
self:setrotate( self.globalrotate + 10 )
|
||||
end
|
||||
elseif ev.code == KEY_K then
|
||||
self:setrotate( self.globalrotate - 10 )
|
||||
if Keys.shiftmode then
|
||||
self:screenRotate("anticlockwise")
|
||||
else
|
||||
self:setrotate( self.globalrotate - 10 )
|
||||
end
|
||||
elseif ev.code == KEY_HOME then
|
||||
if Keys.shiftmode or Keys.altmode then
|
||||
-- signal quit
|
||||
@@ -734,15 +907,9 @@ function UniReader:inputloop()
|
||||
elseif ev.code == KEY_Z and Keys.altmode then
|
||||
self.bbox.enabled = not self.bbox.enabled;
|
||||
print("# bbox override: ", self.bbox.enabled);
|
||||
end
|
||||
|
||||
-- switch to ZOOM_BY_VALUE to enable panning on fiveway move
|
||||
if ev.code == KEY_FW_LEFT
|
||||
or ev.code == KEY_FW_RIGHT
|
||||
or ev.code == KEY_FW_UP
|
||||
or ev.code == KEY_FW_DOWN
|
||||
then
|
||||
self.globalzoommode = self.ZOOM_BY_VALUE
|
||||
elseif ev.code == KEY_MENU then
|
||||
self:showMenu()
|
||||
self:goto(self.pageno)
|
||||
end
|
||||
]]
|
||||
-- switch to ZOOM_BY_VALUE to enable panning on fiveway move
|
||||
@@ -791,7 +958,7 @@ function UniReader:inputloop()
|
||||
self.offset_x = 0
|
||||
end
|
||||
elseif ev.code == KEY_FW_RIGHT then
|
||||
print("# KEY_FW_RIGHT "..self.offset_x.." - "..x.." < "..self.min_offset_x);
|
||||
print("# KEY_FW_RIGHT "..self.offset_x.." - "..x.." < "..self.min_offset_x.." - "..self.pan_margin);
|
||||
self.offset_x = self.offset_x - x
|
||||
if self.pan_by_page then
|
||||
if self.offset_x < self.min_offset_x - self.pan_margin and self.pageno < self.doc:getPages() then
|
||||
@@ -853,7 +1020,6 @@ function UniReader:inputloop()
|
||||
self.settings:savesetting("last_page", self.pageno)
|
||||
self.settings:savesetting("gamma", self.globalgamma)
|
||||
self.settings:savesetting("jumpstack", self.jump_stack)
|
||||
--self.settings:savesetting("pan_overlap_vertical", self.pan_overlap_vertical)
|
||||
self.settings:savesetting("bbox", self.bbox)
|
||||
self.settings:savesetting("globalzoom", self.globalzoom)
|
||||
self.settings:savesetting("globalzoommode", self.globalzoommode)
|
||||
@@ -955,4 +1121,4 @@ function UniReader:add_all_commands()
|
||||
end)
|
||||
--print commands
|
||||
for k,v in pairs(self.commands.map) do print(v) end
|
||||
end
|
||||
end
|
||||
|
||||
Reference in New Issue
Block a user