mirror of
https://github.com/koreader/koreader
synced 2024-11-10 01:10:34 +00:00
0aa19d72a0
docsettings : added hasSidecarDir() and made getSidecarDir() more robust widget/menu : allow items to specify themselves to be displayed in bold
180 lines
5.8 KiB
Lua
180 lines
5.8 KiB
Lua
local lfs = require("libs/libkoreader-lfs")
|
|
local DataStorage = require("datastorage")
|
|
local dump = require("dump")
|
|
local purgeDir = require("ffi/util").purgeDir
|
|
|
|
local DocSettings = {}
|
|
|
|
local HISTORY_DIR = DataStorage:getHistoryDir()
|
|
|
|
local function buildCandidate(file_path)
|
|
if lfs.attributes(file_path, "mode") == "file" then
|
|
return { file_path, lfs.attributes(file_path, "modification") }
|
|
else
|
|
return nil
|
|
end
|
|
end
|
|
|
|
-- Sidecar directory is the file without _last_ suffix.
|
|
function DocSettings:getSidecarDir(doc_path)
|
|
local file_without_suffix = doc_path:match("(.*)%.")
|
|
if file_without_suffix then
|
|
return file_without_suffix..".sdr"
|
|
end
|
|
-- We shouldn't be called with anything but files with registered
|
|
-- extensions, but in case we are, return something useful
|
|
return doc_path..".sdr"
|
|
end
|
|
|
|
function DocSettings:hasSidecarDir(doc_path)
|
|
-- We may be called with items from FileManager, which may not be a document
|
|
if lfs.attributes(doc_path, "mode") == "directory" then
|
|
return false
|
|
end
|
|
return lfs.attributes(self:getSidecarDir(doc_path), "mode") == "directory"
|
|
end
|
|
|
|
function DocSettings:getHistoryPath(fullpath)
|
|
return HISTORY_DIR .. "/[" .. fullpath:gsub("(.*/)([^/]+)","%1] %2"):gsub("/","#") .. ".lua"
|
|
end
|
|
|
|
function DocSettings:getPathFromHistory(hist_name)
|
|
if hist_name == nil or hist_name == '' then return nil end
|
|
-- 1. select everything included in brackets
|
|
local s = string.match(hist_name,"%b[]")
|
|
if s == nil or s == '' then return nil end
|
|
-- 2. crop the bracket-sign from both sides
|
|
-- 3. and finally replace decorative signs '#' to dir-char '/'
|
|
return string.gsub(string.sub(s,2,-3),"#","/")
|
|
end
|
|
|
|
function DocSettings:getNameFromHistory(hist_name)
|
|
if hist_name == nil or hist_name == '' then return nil end
|
|
local s = string.match(hist_name, "%b[]")
|
|
if s == nil or s == '' then return nil end
|
|
-- at first, search for path length
|
|
-- and return the rest of string without 4 last characters (".lua")
|
|
return string.sub(hist_name, string.len(s)+2, -5)
|
|
end
|
|
|
|
function DocSettings:open(docfile)
|
|
-- TODO(zijiehe): Remove history_path, use only sidecar.
|
|
local new = {}
|
|
local ok, stored
|
|
new.history_file = self:getHistoryPath(docfile)
|
|
|
|
local sidecar = self:getSidecarDir(docfile)
|
|
new.sidecar = sidecar
|
|
if lfs.attributes(sidecar, "mode") ~= "directory" then
|
|
lfs.mkdir(sidecar)
|
|
end
|
|
-- If there is a file which has a same name as the sidecar directory, or
|
|
-- the file system is read-only, we should not waste time to read it.
|
|
if lfs.attributes(sidecar, "mode") == "directory" then
|
|
-- New sidecar file name is metadata.{file last suffix}.lua. So we
|
|
-- can handle two files with only different suffixes.
|
|
new.sidecar_file = sidecar.."/metadata."..
|
|
docfile:match(".*%.(.+)")..".lua"
|
|
new.legacy_sidecar_file = sidecar.."/"..
|
|
docfile:match("([^%/]+%..+)")..".lua"
|
|
end
|
|
|
|
local candidates = {}
|
|
-- New sidecar file
|
|
table.insert(candidates, buildCandidate(new.sidecar_file))
|
|
-- Legacy sidecar file
|
|
table.insert(candidates, buildCandidate(new.legacy_sidecar_file))
|
|
-- Legacy history folder
|
|
table.insert(candidates, buildCandidate(new.history_file))
|
|
-- Legacy kpdfview setting
|
|
table.insert(candidates, buildCandidate(docfile..".kpdfview.lua"))
|
|
table.sort(candidates, function(l, r)
|
|
if l == nil then
|
|
return false
|
|
elseif r == nil then
|
|
return true
|
|
else
|
|
return l[2] > r[2]
|
|
end
|
|
end)
|
|
for _, k in pairs(candidates) do
|
|
ok, stored = pcall(dofile, k[1])
|
|
if ok then
|
|
break
|
|
end
|
|
end
|
|
if ok and stored then
|
|
new.data = stored
|
|
new.candidates = candidates
|
|
else
|
|
new.data = {}
|
|
end
|
|
|
|
return setmetatable(new, {__index = DocSettings})
|
|
end
|
|
|
|
function DocSettings:readSetting(key)
|
|
return self.data[key]
|
|
end
|
|
|
|
function DocSettings:saveSetting(key, value)
|
|
self.data[key] = value
|
|
end
|
|
|
|
function DocSettings:delSetting(key)
|
|
self.data[key] = nil
|
|
end
|
|
|
|
function DocSettings:flush()
|
|
-- write serialized version of the data table into one of
|
|
-- i) sidecar directory in the same directory of the document or
|
|
-- ii) history directory in root directory of KOReader
|
|
if not self.history_file and not self.sidecar_file then
|
|
return
|
|
end
|
|
|
|
-- If we can write to sidecar_file, we do not need to write to history_file
|
|
-- anymore.
|
|
local serials = { self.sidecar_file, self.history_file }
|
|
local s_out = dump(self.data)
|
|
os.setlocale('C', 'numeric')
|
|
for _, f in pairs(serials) do
|
|
local f_out = io.open(f, "w")
|
|
if f_out ~= nil then
|
|
f_out:write("-- we can read Lua syntax here!\nreturn ")
|
|
f_out:write(s_out)
|
|
f_out:write("\n")
|
|
f_out:close()
|
|
|
|
if self.candidates ~= nil
|
|
and not G_reader_settings:readSetting(
|
|
"preserve_legacy_docsetting") then
|
|
for _, k in pairs(self.candidates) do
|
|
if k[1] ~= f then
|
|
os.remove(k[1])
|
|
-- We should not remove sidecar folder, as it may
|
|
-- contain Kindle history files.
|
|
end
|
|
end
|
|
end
|
|
|
|
break
|
|
end
|
|
end
|
|
end
|
|
|
|
function DocSettings:close()
|
|
self:flush()
|
|
end
|
|
|
|
function DocSettings:purge()
|
|
if self.history_file then
|
|
os.remove(self.history_file)
|
|
end
|
|
if lfs.attributes(self.sidecar, "mode") == "directory" then
|
|
purgeDir(self.sidecar)
|
|
end
|
|
end
|
|
|
|
return DocSettings
|