2016-02-16 00:14:15 +00:00
|
|
|
--[[--
|
|
|
|
Font module.
|
|
|
|
]]
|
|
|
|
|
2014-08-14 11:49:42 +00:00
|
|
|
local lfs = require("libs/libkoreader-lfs")
|
|
|
|
local Freetype = require("ffi/freetype")
|
2014-10-30 18:42:18 +00:00
|
|
|
local Screen = require("device").screen
|
2015-06-06 09:21:39 +00:00
|
|
|
local Device = require("device")
|
2013-10-22 18:51:29 +00:00
|
|
|
local DEBUG = require("dbg")
|
2012-03-10 08:41:23 +00:00
|
|
|
|
2013-10-18 20:38:07 +00:00
|
|
|
local Font = {
|
2014-03-13 13:52:43 +00:00
|
|
|
fontmap = {
|
|
|
|
-- default font for menu contents
|
2014-07-08 00:11:17 +00:00
|
|
|
cfont = "noto/NotoSans-Regular.ttf",
|
2014-03-13 13:52:43 +00:00
|
|
|
-- default font for title
|
|
|
|
--tfont = "NimbusSanL-BoldItal.cff",
|
2014-07-08 00:11:17 +00:00
|
|
|
tfont = "noto/NotoSans-Bold.ttf",
|
2014-03-13 13:52:43 +00:00
|
|
|
-- default font for footer
|
2014-07-08 00:11:17 +00:00
|
|
|
ffont = "noto/NotoSans-Regular.ttf",
|
2014-03-13 13:52:43 +00:00
|
|
|
|
|
|
|
-- default font for reading position info
|
2014-07-08 00:11:17 +00:00
|
|
|
rifont = "noto/NotoSans-Regular.ttf",
|
2014-03-13 13:52:43 +00:00
|
|
|
|
|
|
|
-- default font for pagination display
|
2014-07-08 00:11:17 +00:00
|
|
|
pgfont = "noto/NotoSans-Regular.ttf",
|
2014-03-13 13:52:43 +00:00
|
|
|
|
|
|
|
-- selectmenu: font for item shortcut
|
|
|
|
scfont = "droid/DroidSansMono.ttf",
|
|
|
|
|
|
|
|
-- help page: font for displaying keys
|
|
|
|
hpkfont = "droid/DroidSansMono.ttf",
|
|
|
|
-- font for displaying help messages
|
2014-07-08 00:11:17 +00:00
|
|
|
hfont = "noto/NotoSans-Regular.ttf",
|
2014-03-13 13:52:43 +00:00
|
|
|
|
|
|
|
-- font for displaying input content
|
|
|
|
-- we have to use mono here for better distance controlling
|
|
|
|
infont = "droid/DroidSansMono.ttf",
|
|
|
|
|
|
|
|
-- font for info messages
|
2014-07-08 00:11:17 +00:00
|
|
|
infofont = "noto/NotoSans-Regular.ttf",
|
2014-03-13 13:52:43 +00:00
|
|
|
},
|
2016-02-15 23:00:38 +00:00
|
|
|
sizemap = {
|
|
|
|
cfont = 24,
|
|
|
|
tfont = 26,
|
|
|
|
ffont = 20,
|
|
|
|
pgfont = 20,
|
|
|
|
scfont = 20,
|
|
|
|
rifont = 16,
|
|
|
|
hpkfont = 20,
|
|
|
|
hfont = 24,
|
|
|
|
infont = 22,
|
|
|
|
infofont = 24,
|
|
|
|
},
|
2014-03-13 13:52:43 +00:00
|
|
|
fallbacks = {
|
2015-09-03 14:45:12 +00:00
|
|
|
[1] = "noto/NotoSansCJK-Regular.ttf",
|
2014-07-08 00:11:17 +00:00
|
|
|
[2] = "noto/NotoSans-Regular.ttf",
|
2015-09-03 14:45:12 +00:00
|
|
|
[3] = "freefont/FreeSans.ttf",
|
2014-03-13 13:52:43 +00:00
|
|
|
},
|
|
|
|
|
2015-01-25 08:52:01 +00:00
|
|
|
fontdir = "./fonts",
|
2014-03-13 13:52:43 +00:00
|
|
|
|
|
|
|
-- face table
|
|
|
|
faces = {},
|
2012-03-10 08:41:23 +00:00
|
|
|
}
|
|
|
|
|
2012-04-09 17:04:26 +00:00
|
|
|
function Font:getFace(font, size)
|
2015-09-09 03:10:17 +00:00
|
|
|
-- default to content font
|
|
|
|
if not font then font = self.cfont end
|
2014-03-13 13:52:43 +00:00
|
|
|
|
2016-02-15 23:00:38 +00:00
|
|
|
if not size then size = self.sizemap[font] end
|
2014-08-20 01:39:50 +00:00
|
|
|
-- original size before scaling by screen DPI
|
|
|
|
local orig_size = size
|
2016-02-16 02:08:04 +00:00
|
|
|
size = Screen:scaleBySize(size)
|
2014-03-13 13:52:43 +00:00
|
|
|
|
2015-09-09 03:10:17 +00:00
|
|
|
local hash = font..size
|
|
|
|
local face_obj = self.faces[hash]
|
2014-03-13 13:52:43 +00:00
|
|
|
-- build face if not found
|
2015-09-09 03:10:17 +00:00
|
|
|
if not face_obj then
|
2014-03-13 13:52:43 +00:00
|
|
|
local realname = self.fontmap[font]
|
|
|
|
if not realname then
|
|
|
|
realname = font
|
|
|
|
end
|
|
|
|
realname = self.fontdir.."/"..realname
|
2016-02-16 02:08:04 +00:00
|
|
|
local ok, face = pcall(Freetype.newFace, realname, size)
|
2014-03-13 13:52:43 +00:00
|
|
|
if not ok then
|
|
|
|
DEBUG("#! Font "..font.." ("..realname..") not supported: "..face)
|
|
|
|
return nil
|
|
|
|
end
|
2016-02-16 00:14:15 +00:00
|
|
|
--- Freetype font face wrapper object
|
|
|
|
-- @table FontFaceObj
|
|
|
|
-- @field size size of the font face (after scaled by screen size)
|
|
|
|
-- @field orig_size raw size of the font face (before scale)
|
|
|
|
-- @field ftface font face object from freetype
|
|
|
|
-- @field hash hash key for this font face
|
2015-09-09 03:10:17 +00:00
|
|
|
face_obj = {
|
|
|
|
size = size,
|
|
|
|
orig_size = orig_size,
|
|
|
|
ftface = face,
|
|
|
|
hash = hash
|
|
|
|
}
|
|
|
|
self.faces[hash] = face_obj
|
2014-03-13 13:52:43 +00:00
|
|
|
end
|
2015-09-09 03:10:17 +00:00
|
|
|
return face_obj
|
2012-04-09 17:04:26 +00:00
|
|
|
end
|
|
|
|
|
2015-10-22 15:15:41 +00:00
|
|
|
--[[
|
|
|
|
These fonts from Kindle system cannot be loaded by Freetype.
|
|
|
|
--]]
|
|
|
|
local kindle_fonts_blacklist = {
|
|
|
|
["HYGothicBold.ttf"] = true,
|
|
|
|
["HYGothicMedium.ttf"] = true,
|
|
|
|
["HYMyeongJoBold.ttf"] = true,
|
|
|
|
["HYMyeongJoMedium.ttf"] = true,
|
|
|
|
["MYingHeiTBold.ttf"] = true,
|
|
|
|
["MYingHeiTMedium.ttf"] = true,
|
|
|
|
["SongTBold.ttf"] = true,
|
|
|
|
["SongTMedium.ttf"] = true,
|
|
|
|
}
|
|
|
|
|
|
|
|
local function isInFontsBlacklist(f)
|
2015-10-22 13:19:34 +00:00
|
|
|
if Device:isKindle() then
|
2015-10-22 15:15:41 +00:00
|
|
|
return kindle_fonts_blacklist[f]
|
2015-10-22 09:02:21 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2015-01-25 08:52:01 +00:00
|
|
|
function Font:_readList(target, dir)
|
2015-01-27 18:17:54 +00:00
|
|
|
-- lfs.dir non-exsitent directory will give error, weird!
|
|
|
|
local ok, iter, dir_obj = pcall(lfs.dir, dir)
|
|
|
|
if not ok then return end
|
|
|
|
for f in iter, dir_obj do
|
2014-03-13 13:52:43 +00:00
|
|
|
if lfs.attributes(dir.."/"..f, "mode") == "directory" and f ~= "." and f ~= ".." then
|
2015-01-25 08:52:01 +00:00
|
|
|
self:_readList(target, dir.."/"..f)
|
2014-03-13 13:52:43 +00:00
|
|
|
else
|
|
|
|
local file_type = string.lower(string.match(f, ".+%.([^.]+)") or "")
|
2014-08-23 14:46:28 +00:00
|
|
|
if file_type == "ttf" or file_type == "ttc"
|
|
|
|
or file_type == "cff" or file_type == "otf" then
|
2015-10-22 15:15:41 +00:00
|
|
|
if not isInFontsBlacklist(f) then
|
2015-10-22 09:02:21 +00:00
|
|
|
table.insert(target, dir.."/"..f)
|
|
|
|
end
|
2014-03-13 13:52:43 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2012-04-09 17:04:26 +00:00
|
|
|
end
|
|
|
|
|
2015-06-06 09:21:39 +00:00
|
|
|
function Font:_getExternalFontDir()
|
|
|
|
if Device:isAndroid() then
|
|
|
|
return ANDROID_FONT_DIR
|
|
|
|
else
|
|
|
|
return os.getenv("EXT_FONT_DIR")
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2012-04-09 17:04:26 +00:00
|
|
|
function Font:getFontList()
|
2015-01-25 08:52:01 +00:00
|
|
|
local fontlist = {}
|
|
|
|
self:_readList(fontlist, self.fontdir)
|
2015-06-06 09:21:39 +00:00
|
|
|
-- multiple paths should be joined with semicolon
|
|
|
|
for dir in string.gmatch(self:_getExternalFontDir() or "", "([^;]+)") do
|
2015-01-25 08:52:01 +00:00
|
|
|
self:_readList(fontlist, dir)
|
|
|
|
end
|
2014-03-13 13:52:43 +00:00
|
|
|
table.sort(fontlist)
|
|
|
|
return fontlist
|
2012-03-10 08:41:23 +00:00
|
|
|
end
|
|
|
|
|
2013-10-18 20:38:07 +00:00
|
|
|
return Font
|