Compare commits

..

No commits in common. "bb837dd30eb42e600926d49100b31f20981f82bb" and "3c6f3ae2371d5ec4e1ca57f3fb9a47cf87fc8a32" have entirely different histories.

13 changed files with 42 additions and 257 deletions

View File

@ -97,11 +97,7 @@ func (api *API) registerWebAppRoutes() {
// Local / Offline Static Pages (No Template, No Auth) // Local / Offline Static Pages (No Template, No Auth)
api.Router.GET("/local", api.localDocuments) api.Router.GET("/local", api.localDocuments)
// Reader (Reader Page, Document Progress, Devices)
api.Router.GET("/reader", api.documentReader) api.Router.GET("/reader", api.documentReader)
api.Router.GET("/reader/devices", api.authWebAppMiddleware, api.getDevices)
api.Router.GET("/reader/progress/:document", api.authWebAppMiddleware, api.getDocumentProgress)
// Web App // Web App
api.Router.GET("/", api.authWebAppMiddleware, api.createAppResourcesRoute("home")) api.Router.GET("/", api.authWebAppMiddleware, api.createAppResourcesRoute("home"))
@ -110,6 +106,7 @@ func (api *API) registerWebAppRoutes() {
api.Router.GET("/documents/:document", api.authWebAppMiddleware, api.createAppResourcesRoute("document")) api.Router.GET("/documents/:document", api.authWebAppMiddleware, api.createAppResourcesRoute("document"))
api.Router.GET("/documents/:document/cover", api.authWebAppMiddleware, api.getDocumentCover) api.Router.GET("/documents/:document/cover", api.authWebAppMiddleware, api.getDocumentCover)
api.Router.GET("/documents/:document/file", api.authWebAppMiddleware, api.downloadDocument) api.Router.GET("/documents/:document/file", api.authWebAppMiddleware, api.downloadDocument)
api.Router.GET("/documents/:document/progress", api.authWebAppMiddleware, api.getDocumentProgress)
api.Router.GET("/login", api.createAppResourcesRoute("login")) api.Router.GET("/login", api.createAppResourcesRoute("login"))
api.Router.GET("/logout", api.authWebAppMiddleware, api.authLogout) api.Router.GET("/logout", api.authWebAppMiddleware, api.authLogout)
api.Router.GET("/register", api.createAppResourcesRoute("login", gin.H{"Register": true})) api.Router.GET("/register", api.createAppResourcesRoute("login", gin.H{"Register": true}))

View File

@ -113,9 +113,10 @@ func (api *API) createAppResourcesRoute(routeName string, args ...map[string]any
} }
templateVars["User"] = userID templateVars["User"] = userID
if routeName == "documents" { // Potential URL Parameters
qParams := bindQueryParams(c, 9) qParams := bindQueryParams(c)
if routeName == "documents" {
var query *string var query *string
if qParams.Search != nil && *qParams.Search != "" { if qParams.Search != nil && *qParams.Search != "" {
search := "%" + *qParams.Search + "%" search := "%" + *qParams.Search + "%"
@ -180,8 +181,6 @@ func (api *API) createAppResourcesRoute(routeName string, args ...map[string]any
templateVars["Data"] = document templateVars["Data"] = document
templateVars["TotalTimeLeftSeconds"] = int64((100.0 - document.Percentage) * float64(document.SecondsPerPercent)) templateVars["TotalTimeLeftSeconds"] = int64((100.0 - document.Percentage) * float64(document.SecondsPerPercent))
} else if routeName == "activity" { } else if routeName == "activity" {
qParams := bindQueryParams(c, 15)
activityFilter := database.GetActivityParams{ activityFilter := database.GetActivityParams{
UserID: userID, UserID: userID,
Offset: (*qParams.Page - 1) * *qParams.Limit, Offset: (*qParams.Page - 1) * *qParams.Limit,
@ -398,20 +397,6 @@ func (api *API) getDocumentProgress(c *gin.Context) {
}) })
} }
func (api *API) getDevices(c *gin.Context) {
rUser, _ := c.Get("AuthorizedUser")
devices, err := api.DB.Queries.GetDevices(api.DB.Ctx, rUser.(string))
if err != nil && err != sql.ErrNoRows {
log.Error("[getDevices] GetDevices DB Error:", err)
errorPage(c, http.StatusInternalServerError, fmt.Sprintf("GetDevices DB Error: %v", err))
return
}
c.JSON(http.StatusOK, devices)
}
func (api *API) uploadNewDocument(c *gin.Context) { func (api *API) uploadNewDocument(c *gin.Context) {
var rDocUpload requestDocumentUpload var rDocUpload requestDocumentUpload
if err := c.ShouldBind(&rDocUpload); err != nil { if err := c.ShouldBind(&rDocUpload); err != nil {
@ -996,12 +981,13 @@ func (api *API) getDocumentsWordCount(documents []database.GetDocumentsWithStats
return nil return nil
} }
func bindQueryParams(c *gin.Context, defaultLimit int64) queryParams { func bindQueryParams(c *gin.Context) queryParams {
var qParams queryParams var qParams queryParams
c.BindQuery(&qParams) c.BindQuery(&qParams)
if qParams.Limit == nil { if qParams.Limit == nil {
qParams.Limit = &defaultLimit var defaultValue int64 = 9
qParams.Limit = &defaultValue
} else if *qParams.Limit < 0 { } else if *qParams.Limit < 0 {
var zeroValue int64 = 0 var zeroValue int64 = 0
qParams.Limit = &zeroValue qParams.Limit = &zeroValue

View File

@ -49,7 +49,7 @@ func (api *API) opdsEntry(c *gin.Context) {
}, },
Links: []opds.Link{ Links: []opds.Link{
{ {
Href: "/api/opds/documents", Href: "/api/opds/documents?limit=100",
TypeLink: "application/atom+xml;type=feed;profile=opds-catalog", TypeLink: "application/atom+xml;type=feed;profile=opds-catalog",
}, },
}, },
@ -66,8 +66,8 @@ func (api *API) opdsDocuments(c *gin.Context) {
userID = rUser.(string) userID = rUser.(string)
} }
// Potential URL Parameters (Default Pagination - 100) // Potential URL Parameters
qParams := bindQueryParams(c, 100) qParams := bindQueryParams(c)
// Possible Query // Possible Query
var query *string var query *string
@ -160,7 +160,7 @@ func (api *API) opdsSearchDescription(c *gin.Context) {
rawXML := `<OpenSearchDescription xmlns="http://a9.com/-/spec/opensearch/1.1/"> rawXML := `<OpenSearchDescription xmlns="http://a9.com/-/spec/opensearch/1.1/">
<ShortName>Search AnthoLume</ShortName> <ShortName>Search AnthoLume</ShortName>
<Description>Search AnthoLume</Description> <Description>Search AnthoLume</Description>
<Url type="application/atom+xml;profile=opds-catalog;kind=acquisition" template="/api/opds/documents?search={searchTerms}"/> <Url type="application/atom+xml;profile=opds-catalog;kind=acquisition" template="/api/opds/documents?limit=100&search={searchTerms}"/>
</OpenSearchDescription>` </OpenSearchDescription>`
c.Data(http.StatusOK, "application/xml", []byte(rawXML)) c.Data(http.StatusOK, "application/xml", []byte(rawXML))
} }

1
assets/lib/platform.min.js vendored Normal file

File diff suppressed because one or more lines are too long

View File

@ -31,6 +31,7 @@
<script src="/assets/lib/jszip.min.js"></script> <script src="/assets/lib/jszip.min.js"></script>
<script src="/assets/lib/epub.min.js"></script> <script src="/assets/lib/epub.min.js"></script>
<script src="/assets/lib/idb-keyval.min.js"></script> <script src="/assets/lib/idb-keyval.min.js"></script>
<script src="/assets/lib/sw-helper.min.js"></script>
<!-- Local --> <!-- Local -->
<script src="/assets/common.js"></script> <script src="/assets/common.js"></script>

View File

@ -20,6 +20,7 @@
<link rel="stylesheet" href="/assets/style.css" /> <link rel="stylesheet" href="/assets/style.css" />
<!-- Libraries --> <!-- Libraries -->
<script src="/assets/lib/platform.min.js"></script>
<script src="/assets/lib/jszip.min.js"></script> <script src="/assets/lib/jszip.min.js"></script>
<script src="/assets/lib/epub.min.js"></script> <script src="/assets/lib/epub.min.js"></script>
<script src="/assets/lib/no-sleep.min.js"></script> <script src="/assets/lib/no-sleep.min.js"></script>
@ -70,10 +71,6 @@
#top-bar:not(.top-0) { #top-bar:not(.top-0) {
top: calc((8em + env(safe-area-inset-top)) * -1); top: calc((8em + env(safe-area-inset-top)) * -1);
} }
select:invalid {
color: gray;
}
</style> </style>
</head> </head>
<body class="bg-gray-100 dark:bg-gray-800"> <body class="bg-gray-100 dark:bg-gray-800">
@ -263,120 +260,5 @@
</div> </div>
<div id="viewer" class="w-full h-full"></div> <div id="viewer" class="w-full h-full"></div>
</main> </main>
<!-- Device Selector -->
<div
id="device-selector"
class="hidden absolute top-0 left-0 w-full h-full z-50"
>
<div
class="fixed top-0 left-0 bg-black opacity-50 w-screen h-screen"
></div>
<div
class="relative flex flex-col gap-4 p-4 max-h-[95%] w-5/6 md:w-1/2 bg-white dark:bg-gray-800 overflow-scroll -translate-x-2/4 -translate-y-2/4 top-1/2 left-1/2 overflow-hidden shadow rounded"
>
<div class="text-center flex flex-col gap-2">
<h3 class="text-lg font-bold leading-6 dark:text-gray-300">
Select Device
</h3>
<p class="text-xs text-gray-500 text-center">
This device appears to be new! Please either assume an existing
device, or create a new one.
</p>
</div>
<div
class="flex flex-col gap-2 grow p-4 rounded shadow-lg bg-white dark:bg-gray-700 text-gray-500 dark:text-white"
>
<div class="flex gap-4 flex-col">
<div class="flex relative min-w-[12em]">
<span
class="inline-flex items-center px-3 border-t bg-white border-l border-b border-gray-300 text-gray-500 shadow-sm text-sm"
>
<svg
width="15"
height="15"
viewBox="0 0 24 24"
fill="currentColor"
xmlns="http://www.w3.org/2000/svg"
>
<path
d="M5.65517 2.22732C5.2225 2.34037 4.9438 2.50021 4.72718 2.71244C4.42179 3.01165 4.22268 3.43172 4.11382 4.225C4.00176 5.04159 4 6.12387 4 7.67568V16.2442C4.38867 15.9781 4.82674 15.7756 5.29899 15.6517C5.41296 15.6217 5.53103 15.5983 5.65517 15.5799V2.22732Z"
/>
<path
d="M7.31034 15.5135C7.32206 15.5135 7.33382 15.5135 7.34563 15.5135L20 15.5135V7.67568C20 6.12387 19.9982 5.04159 19.8862 4.22499C19.7773 3.43172 19.5782 3.01165 19.2728 2.71244C18.9674 2.41324 18.5387 2.21816 17.729 2.11151C16.8955 2.00172 15.7908 2 14.2069 2H9.7931C8.79138 2 7.98133 2.00069 7.31034 2.02897V15.5135Z"
/>
<path
d="M7.47341 17.1351C6.39395 17.1351 6.01657 17.1421 5.72738 17.218C4.93365 17.4264 4.30088 18.0044 4.02952 18.7558C4.0463 19.1382 4.07259 19.4746 4.11382 19.775C4.22268 20.5683 4.42179 20.9884 4.72718 21.2876C5.03258 21.5868 5.46135 21.7818 6.27103 21.8885C7.10452 21.9983 8.2092 22 9.7931 22H14.2069C15.7908 22 16.8955 21.9983 17.729 21.8885C18.5387 21.7818 18.9674 21.5868 19.2728 21.2876C19.5782 20.9884 19.7773 20.5683 19.8862 19.775C19.9776 19.1088 19.9956 18.2657 19.9991 17.1351H7.47341Z"
/>
</svg>
</span>
<select
class="flex-1 appearance-none rounded-none border border-gray-300 w-full py-2 px-4 bg-white text-gray-700 placeholder-gray-400 shadow-sm text-base focus:outline-none focus:ring-2 focus:ring-purple-600 focus:border-transparent"
id="source"
name="source"
required
>
<option value="" disabled selected hidden>
Select Existing Device
</option>
</select>
</div>
<button
class="px-10 py-2 text-base font-semibold text-center text-white transition duration-200 ease-in bg-black shadow-md hover:text-black hover:bg-white focus:outline-none focus:ring-2"
>
<span class="w-full">Assume Device</span>
</button>
</div>
</div>
<div
class="flex flex-col gap-2 grow p-4 rounded shadow-lg bg-white dark:bg-gray-700 text-gray-500 dark:text-white"
>
<div class="flex gap-4 flex-col">
<div class="flex flex-col w-full grow">
<div class="flex relative">
<span
class="inline-flex items-center px-3 border-t bg-white border-l border-b border-gray-300 text-gray-500 shadow-sm text-sm"
>
<svg
width="15"
height="15"
viewBox="0 0 24 24"
fill="currentColor"
xmlns="http://www.w3.org/2000/svg"
>
<path
d="M5.65517 2.22732C5.2225 2.34037 4.9438 2.50021 4.72718 2.71244C4.42179 3.01165 4.22268 3.43172 4.11382 4.225C4.00176 5.04159 4 6.12387 4 7.67568V16.2442C4.38867 15.9781 4.82674 15.7756 5.29899 15.6517C5.41296 15.6217 5.53103 15.5983 5.65517 15.5799V2.22732Z"
/>
<path
d="M7.31034 15.5135C7.32206 15.5135 7.33382 15.5135 7.34563 15.5135L20 15.5135V7.67568C20 6.12387 19.9982 5.04159 19.8862 4.22499C19.7773 3.43172 19.5782 3.01165 19.2728 2.71244C18.9674 2.41324 18.5387 2.21816 17.729 2.11151C16.8955 2.00172 15.7908 2 14.2069 2H9.7931C8.79138 2 7.98133 2.00069 7.31034 2.02897V15.5135Z"
/>
<path
d="M7.47341 17.1351C6.39395 17.1351 6.01657 17.1421 5.72738 17.218C4.93365 17.4264 4.30088 18.0044 4.02952 18.7558C4.0463 19.1382 4.07259 19.4746 4.11382 19.775C4.22268 20.5683 4.42179 20.9884 4.72718 21.2876C5.03258 21.5868 5.46135 21.7818 6.27103 21.8885C7.10452 21.9983 8.2092 22 9.7931 22H14.2069C15.7908 22 16.8955 21.9983 17.729 21.8885C18.5387 21.7818 18.9674 21.5868 19.2728 21.2876C19.5782 20.9884 19.7773 20.5683 19.8862 19.775C19.9776 19.1088 19.9956 18.2657 19.9991 17.1351H7.47341Z"
/>
</svg>
</span>
<input
type="text"
id="name"
name="name"
class="flex-1 appearance-none rounded-none border border-gray-300 w-full py-2 px-4 bg-white text-gray-700 placeholder-gray-400 shadow-sm text-base focus:outline-none focus:ring-2 focus:ring-purple-600 focus:border-transparent"
placeholder="New Device Name"
/>
</div>
</div>
<button
class="px-10 py-2 text-base font-semibold text-center text-white transition duration-200 ease-in bg-black shadow-md hover:text-black hover:bg-white focus:outline-none focus:ring-2"
>
<span class="w-full">Create Device</span>
</button>
</div>
</div>
</div>
</div>
</body> </body>
</html> </html>

View File

@ -17,7 +17,7 @@ async function initReader() {
if (documentType == "REMOTE") { if (documentType == "REMOTE") {
// Get Server / Cached Document // Get Server / Cached Document
let progressResp = await fetch("/reader/progress/" + documentID); let progressResp = await fetch("/documents/" + documentID + "/progress");
documentData = await progressResp.json(); documentData = await progressResp.json();
// Update With Local Cache // Update With Local Cache
@ -145,60 +145,14 @@ class EBookReader {
); );
} }
// Device Already Set this.readerSettings.deviceName =
if (this.readerSettings.deviceID) return; this.readerSettings.deviceName ||
platform.os.toString() + " - " + platform.name;
// Get Elements this.readerSettings.deviceID = this.readerSettings.deviceID || randomID();
let devicePopup = document.querySelector("#device-selector");
let devSelector = devicePopup.querySelector("select");
let devInput = devicePopup.querySelector("input");
let [assumeButton, createButton] = devicePopup.querySelectorAll("button");
// Set Visible // Save Settings (Device ID)
devicePopup.classList.remove("hidden"); this.saveSettings();
// Add Devices
fetch("/reader/devices").then(async (r) => {
let data = await r.json();
data.forEach((item) => {
let optionEl = document.createElement("option");
optionEl.value = item.id;
optionEl.textContent = item.device_name;
devSelector.appendChild(optionEl);
});
});
assumeButton.addEventListener("click", () => {
let deviceID = devSelector.value;
if (deviceID == "") {
// TODO - Error Message
return;
}
let selectedOption = devSelector.children[devSelector.selectedIndex];
let deviceName = selectedOption.textContent;
this.readerSettings.deviceID = deviceID;
this.readerSettings.deviceName = deviceName;
this.saveSettings();
devicePopup.classList.add("hidden");
});
createButton.addEventListener("click", () => {
let deviceName = devInput.value.trim();
if (deviceName == "") {
// TODO - Error Message
return;
}
this.readerSettings.deviceID = randomID();
this.readerSettings.deviceName = deviceName;
this.saveSettings();
devicePopup.classList.add("hidden");
});
} }
/** /**

View File

@ -44,7 +44,7 @@ const ROUTES = [
type: CACHE_UPDATE_ASYNC, type: CACHE_UPDATE_ASYNC,
}, },
{ {
route: /^\/reader\/progress\/[a-zA-Z0-9]{32}$/, route: /^\/documents\/[a-zA-Z0-9]{32}\/progress$/,
type: CACHE_UPDATE_SYNC, type: CACHE_UPDATE_SYNC,
}, },
{ {
@ -74,6 +74,7 @@ const PRECACHE_ASSETS = [
"/assets/common.js", "/assets/common.js",
// Library Assets // Library Assets
"/assets/lib/platform.min.js",
"/assets/lib/jszip.min.js", "/assets/lib/jszip.min.js",
"/assets/lib/epub.min.js", "/assets/lib/epub.min.js",
"/assets/lib/no-sleep.min.js", "/assets/lib/no-sleep.min.js",
@ -119,9 +120,7 @@ async function handleFetch(event) {
// Find Directive // Find Directive
const directive = ROUTES.find( const directive = ROUTES.find(
(item) => (item) => url.match(item.route) || url == item.route
(item.route instanceof RegExp && url.match(item.route)) ||
url == item.route
) || { type: CACHE_NEVER }; ) || { type: CACHE_NEVER };
// Get Fallback // Get Fallback
@ -171,22 +170,12 @@ function handleMessage(event) {
caches.open(SW_CACHE_NAME).then(async (cache) => { caches.open(SW_CACHE_NAME).then(async (cache) => {
let allKeys = await cache.keys(); let allKeys = await cache.keys();
// Get Cached Resources
let docResources = allKeys let docResources = allKeys
.map((item) => new URL(item.url).pathname) .map((item) => new URL(item.url).pathname)
.filter( .filter((item) => item.startsWith("/documents/"));
(item) =>
item.startsWith("/documents/") ||
item.startsWith("/reader/progress/")
);
// Derive Unique IDs
let documentIDs = Array.from( let documentIDs = Array.from(
new Set( new Set(docResources.map((item) => item.split("/")[2]))
docResources
.filter((item) => item.startsWith("/documents/"))
.map((item) => item.split("/")[2])
)
); );
/** /**
@ -199,10 +188,10 @@ function handleMessage(event) {
.filter( .filter(
(id) => (id) =>
docResources.includes("/documents/" + id + "/file") && docResources.includes("/documents/" + id + "/file") &&
docResources.includes("/reader/progress/" + id) docResources.includes("/documents/" + id + "/progress")
) )
.map(async (id) => { .map(async (id) => {
let url = "/reader/progress/" + id; let url = "/documents/" + id + "/progress";
let currentCache = await caches.match(url); let currentCache = await caches.match(url);
let resp = await updateCache(url).catch((e) => currentCache); let resp = await updateCache(url).catch((e) => currentCache);
return resp.json(); return resp.json();
@ -212,12 +201,13 @@ function handleMessage(event) {
event.source.postMessage({ id, data: cachedDocuments }); event.source.postMessage({ id, data: cachedDocuments });
}); });
} else if (data.type === DEL_SW_CACHE) { } else if (data.type === DEL_SW_CACHE) {
let basePath = "/documents/" + data.id;
caches caches
.open(SW_CACHE_NAME) .open(SW_CACHE_NAME)
.then((cache) => .then((cache) =>
Promise.all([ Promise.all([
cache.delete("/documents/" + data.id + "/file"), cache.delete(basePath + "/file"),
cache.delete("/reader/progress/" + data.id), cache.delete(basePath + "/progress"),
]) ])
) )
.then(() => event.source.postMessage({ id, data: "SUCCESS" })) .then(() => event.source.postMessage({ id, data: "SUCCESS" }))

View File

@ -493,7 +493,7 @@ function SyncNinja:userLogin(username, password, menu)
self.settings.password = userkey self.settings.password = userkey
if menu then menu:updateItems() end if menu then menu:updateItems() end
UIManager:show(InfoMessage:new{ UIManager:show(InfoMessage:new{
text = _("Logged in to AnthoLume server.") text = _("Logged in to KOReader server.")
}) })
self:schedulePeriodicPush(0) self:schedulePeriodicPush(0)
@ -532,7 +532,7 @@ function SyncNinja:userRegister(username, password, menu)
self.settings.password = userkey self.settings.password = userkey
if menu then menu:updateItems() end if menu then menu:updateItems() end
UIManager:show(InfoMessage:new{ UIManager:show(InfoMessage:new{
text = _("Registered to AnthoLume server.") text = _("Registered to KOReader server.")
}) })
self:schedulePeriodicPush(0) self:schedulePeriodicPush(0)

View File

@ -40,12 +40,9 @@ WHERE id = $id;
WITH filtered_activity AS ( WITH filtered_activity AS (
SELECT SELECT
document_id, document_id,
device_id,
user_id, user_id,
start_time, start_time,
duration, duration,
ROUND(CAST(start_percentage AS REAL) * 100, 2) AS start_percentage,
ROUND(CAST(end_percentage AS REAL) * 100, 2) AS end_percentage,
ROUND(CAST(end_percentage - start_percentage AS REAL) * 100, 2) AS read_percentage ROUND(CAST(end_percentage - start_percentage AS REAL) * 100, 2) AS read_percentage
FROM activity FROM activity
WHERE WHERE
@ -63,13 +60,10 @@ WITH filtered_activity AS (
SELECT SELECT
document_id, document_id,
device_id,
CAST(STRFTIME('%Y-%m-%d %H:%M:%S', activity.start_time, users.time_offset) AS TEXT) AS start_time, CAST(STRFTIME('%Y-%m-%d %H:%M:%S', activity.start_time, users.time_offset) AS TEXT) AS start_time,
title, title,
author, author,
duration, duration,
start_percentage,
end_percentage,
read_percentage read_percentage
FROM filtered_activity AS activity FROM filtered_activity AS activity
LEFT JOIN documents ON documents.id = activity.document_id LEFT JOIN documents ON documents.id = activity.document_id
@ -134,7 +128,6 @@ WHERE id = $device_id LIMIT 1;
-- name: GetDevices :many -- name: GetDevices :many
SELECT SELECT
devices.id,
devices.device_name, devices.device_name,
CAST(STRFTIME('%Y-%m-%d %H:%M:%S', devices.created_at, users.time_offset) AS TEXT) AS created_at, CAST(STRFTIME('%Y-%m-%d %H:%M:%S', devices.created_at, users.time_offset) AS TEXT) AS created_at,
CAST(STRFTIME('%Y-%m-%d %H:%M:%S', devices.last_synced, users.time_offset) AS TEXT) AS last_synced CAST(STRFTIME('%Y-%m-%d %H:%M:%S', devices.last_synced, users.time_offset) AS TEXT) AS last_synced

View File

@ -150,12 +150,9 @@ const getActivity = `-- name: GetActivity :many
WITH filtered_activity AS ( WITH filtered_activity AS (
SELECT SELECT
document_id, document_id,
device_id,
user_id, user_id,
start_time, start_time,
duration, duration,
ROUND(CAST(start_percentage AS REAL) * 100, 2) AS start_percentage,
ROUND(CAST(end_percentage AS REAL) * 100, 2) AS end_percentage,
ROUND(CAST(end_percentage - start_percentage AS REAL) * 100, 2) AS read_percentage ROUND(CAST(end_percentage - start_percentage AS REAL) * 100, 2) AS read_percentage
FROM activity FROM activity
WHERE WHERE
@ -173,13 +170,10 @@ WITH filtered_activity AS (
SELECT SELECT
document_id, document_id,
device_id,
CAST(STRFTIME('%Y-%m-%d %H:%M:%S', activity.start_time, users.time_offset) AS TEXT) AS start_time, CAST(STRFTIME('%Y-%m-%d %H:%M:%S', activity.start_time, users.time_offset) AS TEXT) AS start_time,
title, title,
author, author,
duration, duration,
start_percentage,
end_percentage,
read_percentage read_percentage
FROM filtered_activity AS activity FROM filtered_activity AS activity
LEFT JOIN documents ON documents.id = activity.document_id LEFT JOIN documents ON documents.id = activity.document_id
@ -195,15 +189,12 @@ type GetActivityParams struct {
} }
type GetActivityRow struct { type GetActivityRow struct {
DocumentID string `json:"document_id"` DocumentID string `json:"document_id"`
DeviceID string `json:"device_id"` StartTime string `json:"start_time"`
StartTime string `json:"start_time"` Title *string `json:"title"`
Title *string `json:"title"` Author *string `json:"author"`
Author *string `json:"author"` Duration int64 `json:"duration"`
Duration int64 `json:"duration"` ReadPercentage float64 `json:"read_percentage"`
StartPercentage float64 `json:"start_percentage"`
EndPercentage float64 `json:"end_percentage"`
ReadPercentage float64 `json:"read_percentage"`
} }
func (q *Queries) GetActivity(ctx context.Context, arg GetActivityParams) ([]GetActivityRow, error) { func (q *Queries) GetActivity(ctx context.Context, arg GetActivityParams) ([]GetActivityRow, error) {
@ -223,13 +214,10 @@ func (q *Queries) GetActivity(ctx context.Context, arg GetActivityParams) ([]Get
var i GetActivityRow var i GetActivityRow
if err := rows.Scan( if err := rows.Scan(
&i.DocumentID, &i.DocumentID,
&i.DeviceID,
&i.StartTime, &i.StartTime,
&i.Title, &i.Title,
&i.Author, &i.Author,
&i.Duration, &i.Duration,
&i.StartPercentage,
&i.EndPercentage,
&i.ReadPercentage, &i.ReadPercentage,
); err != nil { ); err != nil {
return nil, err return nil, err
@ -402,7 +390,6 @@ func (q *Queries) GetDevice(ctx context.Context, deviceID string) (Device, error
const getDevices = `-- name: GetDevices :many const getDevices = `-- name: GetDevices :many
SELECT SELECT
devices.id,
devices.device_name, devices.device_name,
CAST(STRFTIME('%Y-%m-%d %H:%M:%S', devices.created_at, users.time_offset) AS TEXT) AS created_at, CAST(STRFTIME('%Y-%m-%d %H:%M:%S', devices.created_at, users.time_offset) AS TEXT) AS created_at,
CAST(STRFTIME('%Y-%m-%d %H:%M:%S', devices.last_synced, users.time_offset) AS TEXT) AS last_synced CAST(STRFTIME('%Y-%m-%d %H:%M:%S', devices.last_synced, users.time_offset) AS TEXT) AS last_synced
@ -413,7 +400,6 @@ ORDER BY devices.last_synced DESC
` `
type GetDevicesRow struct { type GetDevicesRow struct {
ID string `json:"id"`
DeviceName string `json:"device_name"` DeviceName string `json:"device_name"`
CreatedAt string `json:"created_at"` CreatedAt string `json:"created_at"`
LastSynced string `json:"last_synced"` LastSynced string `json:"last_synced"`
@ -428,12 +414,7 @@ func (q *Queries) GetDevices(ctx context.Context, userID string) ([]GetDevicesRo
var items []GetDevicesRow var items []GetDevicesRow
for rows.Next() { for rows.Next() {
var i GetDevicesRow var i GetDevicesRow
if err := rows.Scan( if err := rows.Scan(&i.DeviceName, &i.CreatedAt, &i.LastSynced); err != nil {
&i.ID,
&i.DeviceName,
&i.CreatedAt,
&i.LastSynced,
); err != nil {
return nil, err return nil, err
} }
items = append(items, i) items = append(items, i)

View File

@ -38,7 +38,7 @@
<p>{{ $activity.Duration }}</p> <p>{{ $activity.Duration }}</p>
</td> </td>
<td class="p-3 border-b border-gray-200"> <td class="p-3 border-b border-gray-200">
<p>{{ $activity.EndPercentage }}%</p> <p>{{ $activity.ReadPercentage }}%</p>
</td> </td>
</tr> </tr>
{{end}} {{end}}

View File

@ -35,7 +35,7 @@
type="text" type="text"
id="search" id="search"
name="search" name="search"
class="flex-1 appearance-none rounded-none border border-gray-300 w-full py-2 px-2 bg-white text-gray-700 placeholder-gray-400 shadow-sm text-base focus:outline-none focus:ring-2 focus:ring-purple-600 focus:border-transparent" class="flex-1 appearance-none rounded-none border border-gray-300 w-full py-2 px-4 bg-white text-gray-700 placeholder-gray-400 shadow-sm text-base focus:outline-none focus:ring-2 focus:ring-purple-600 focus:border-transparent"
placeholder="Search Author / Title" placeholder="Search Author / Title"
/> />
</div> </div>