Compare commits
5 Commits
Author | SHA1 | Date | |
---|---|---|---|
6c64557f89 | |||
cf297abff7 | |||
010e64b382 | |||
a87abaf976 | |||
16e986243c |
@ -6,6 +6,7 @@ A plugin that keeps track of how long you played your games, and an app to show
|
|||||||
- As it's rewritten as a kernel plugin, the files that record the playtime won't ever get corrupted.
|
- As it's rewritten as a kernel plugin, the files that record the playtime won't ever get corrupted.
|
||||||
- It only writes to file when you close/launch or suspend/resume the games so it doesn't murder your memory card, unlike the old one.
|
- It only writes to file when you close/launch or suspend/resume the games so it doesn't murder your memory card, unlike the old one.
|
||||||
- Complete support for Adrenaline and Adrenaline Bubbles.
|
- Complete support for Adrenaline and Adrenaline Bubbles.
|
||||||
|
- You can add apps/games you selected to a blacklist (located at ux0:/data/TrackPlug/blacklist.txt)
|
||||||
- The app works at 60FPS instead of 30FPS.
|
- The app works at 60FPS instead of 30FPS.
|
||||||
|
|
||||||
# How do I install this?
|
# How do I install this?
|
||||||
@ -17,11 +18,13 @@ ur0:tai/BetterTrackPlug.skprx
|
|||||||
install the vpk.
|
install the vpk.
|
||||||
Reboot.
|
Reboot.
|
||||||
|
|
||||||
# Features To Be Added
|
# How do I use this?
|
||||||
- Blacklist support so you don't have to see how much time you've spent on VitaShell.
|
- UP/DOWN to navigate.
|
||||||
|
- Triangle to prompt clearing playtime of the selected record, asks if you want to blacklist if you do clear the playtime.
|
||||||
|
|
||||||
# Notes
|
# Notes
|
||||||

|

|
||||||
|
|
||||||
I would advise you to set up your bubbles in a way that their title ID's will be the same as the corresponding PSP game's title ID instead of the default PSPEMUXXX, this way, they will use the same file that stores the playtime and you won't see the game twice on the list if you launch it both from bubble or directly from adrenaline.
|
I would advise you to set up your bubbles in a way that their title ID's will be the same as the corresponding PSP game's title ID instead of the default PSPEMUXXX, this way, they will use the same file that stores the playtime and you won't see the game twice on the list if you launch it both from bubble or directly from adrenaline.
|
||||||
|
|
||||||
As it's currently not possible for me to extract icons from inside Adrenaline, games launched directly from Adrenaline won't have any icons if there is no corresponding bubble for the said game. I also don't think it would be efficient even if I knew how to do it.
|
As it's currently not possible for me to extract icons from inside Adrenaline, games launched directly from Adrenaline won't have any icons if there is no corresponding bubble for the said game. I also don't think it would be efficient even if I knew how to do it.
|
||||||
|
@ -3,21 +3,22 @@ System.createDirectory("ux0:/data/TrackPlug/")
|
|||||||
System.createDirectory("ux0:/data/TrackPlug/Records")
|
System.createDirectory("ux0:/data/TrackPlug/Records")
|
||||||
System.createDirectory("ux0:/data/TrackPlug/Config")
|
System.createDirectory("ux0:/data/TrackPlug/Config")
|
||||||
System.createDirectory("ux0:/data/TrackPlug/Assets")
|
System.createDirectory("ux0:/data/TrackPlug/Assets")
|
||||||
-- Scanning TrackPlug folder
|
-- Creating blacklist file if it doesn't exist
|
||||||
local tbl = System.listDirectory("ux0:/data/TrackPlug/Records")
|
if not System.doesFileExist("ux0:/data/TrackPlug/blacklist.txt") then
|
||||||
local blacklist = {}
|
local blacklist_file = System.openFile("ux0:/data/TrackPlug/blacklist.txt", FCREATE)
|
||||||
-- Removing blacklisted games
|
System.closeFile(blacklist_file)
|
||||||
--[[for i, file in pairs(tbl) do
|
|
||||||
local titleid = string.sub(file.name,1,-5)
|
|
||||||
for k, toberemoved in pairs(blacklist) do
|
|
||||||
if titleid == blacklist[k] then
|
|
||||||
System.deleteFile("ux0:/data/TrackPlug/Records"..file.name)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
end
|
end
|
||||||
-- Reset the table
|
-- Reading blacklist file
|
||||||
tbl = System.listDirectory("ux0:/data/TrackPlug/Records")]]--
|
local blacklist_file = System.openFile("ux0:/data/TrackPlug/blacklist.txt", FREAD)
|
||||||
|
local blacklist_fsize = System.sizeFile(blacklist_file)
|
||||||
|
local blacklist_var = System.readFile(blacklist_file, blacklist_fsize)
|
||||||
|
System.closeFile(blacklist_file)
|
||||||
|
-- Removing blacklisted titles
|
||||||
|
for line in blacklist_var:gmatch("([^\n]*)\n?") do
|
||||||
|
System.deleteFile("ux0:/data/TrackPlug/Records/" .. line .. ".bin")
|
||||||
|
end
|
||||||
|
-- Read entries to a table
|
||||||
|
local tbl = System.listDirectory("ux0:/data/TrackPlug/Records")
|
||||||
if tbl == nil then
|
if tbl == nil then
|
||||||
tbl = {}
|
tbl = {}
|
||||||
end
|
end
|
||||||
@ -259,6 +260,7 @@ end
|
|||||||
local sel_val = 128
|
local sel_val = 128
|
||||||
local decrease = true
|
local decrease = true
|
||||||
local freeze = false
|
local freeze = false
|
||||||
|
local freeze_blacklist = false
|
||||||
local mov_y = 0
|
local mov_y = 0
|
||||||
local mov_step = 0
|
local mov_step = 0
|
||||||
local new_list_idx = nil
|
local new_list_idx = nil
|
||||||
@ -311,7 +313,7 @@ function RenderList()
|
|||||||
if i ~= list_idx + 5 then
|
if i ~= list_idx + 5 then
|
||||||
Graphics.drawImage(5, y + mov_y, big_tbl[i].icon)
|
Graphics.drawImage(5, y + mov_y, big_tbl[i].icon)
|
||||||
end
|
end
|
||||||
Graphics.debugPrint(150, y + 35 + mov_y, big_tbl[i].title, Color.new(230,140,175))
|
Graphics.debugPrint(150, y + 35 + mov_y, string.gsub(big_tbl[i].title, "\n", " "), Color.new(230,140,175))
|
||||||
--Graphics.debugPrint(150, y + 45 + mov_y, "Title ID: " .. big_tbl[i].id, white)
|
--Graphics.debugPrint(150, y + 45 + mov_y, "Title ID: " .. big_tbl[i].id, white)
|
||||||
--Graphics.debugPrint(150, y + 65 + mov_y, "Region: " .. big_tbl[i].region, white)
|
--Graphics.debugPrint(150, y + 65 + mov_y, "Region: " .. big_tbl[i].region, white)
|
||||||
Graphics.debugPrint(150, y + 65 + mov_y, "Playtime: " .. big_tbl[i].ptime, white)
|
Graphics.debugPrint(150, y + 65 + mov_y, "Playtime: " .. big_tbl[i].ptime, white)
|
||||||
@ -319,7 +321,7 @@ function RenderList()
|
|||||||
if r_idx == 0 then
|
if r_idx == 0 then
|
||||||
r_idx = #tbl
|
r_idx = #tbl
|
||||||
end
|
end
|
||||||
Graphics.debugPrint(920, y + 100 + mov_y, "#" .. r_idx, white)
|
Graphics.debugPrint(910, y + 100 + mov_y, "#" .. r_idx, white)
|
||||||
y = y + 132
|
y = y + 132
|
||||||
if real_i <= 0 then
|
if real_i <= 0 then
|
||||||
i = real_i
|
i = real_i
|
||||||
@ -331,23 +333,32 @@ end
|
|||||||
|
|
||||||
-- Main loop
|
-- Main loop
|
||||||
local f_idx = 1
|
local f_idx = 1
|
||||||
|
local f_idx_2 = 1
|
||||||
local useless = 0
|
local useless = 0
|
||||||
|
local blacklisted_title = "kek"
|
||||||
|
local blacklisted_title_id = "kek"
|
||||||
local oldpad = Controls.read()
|
local oldpad = Controls.read()
|
||||||
while #tbl > 0 do
|
while #tbl > 0 do
|
||||||
Graphics.initBlend()
|
Graphics.initBlend()
|
||||||
Graphics.fillRect(0,960,0,544,Color.new(10,5,15))
|
Graphics.fillRect(0,960,0,544,Color.new(10,5,15))
|
||||||
|
Graphics.fillRect(0,960,4,140,Color.new(20,20,20))
|
||||||
wav:init()
|
wav:init()
|
||||||
RenderList()
|
RenderList()
|
||||||
if freeze then
|
if freeze then
|
||||||
showAlarm("Do you want to delete this record permanently? \n" .. tbl[list_idx].title, f_idx)
|
showAlarm("Do you want to clear playtime of this record? \n" .. string.gsub(tbl[list_idx].title, "\n", " "), f_idx)
|
||||||
end
|
end
|
||||||
|
if freeze_blacklist then
|
||||||
|
showAlarm("Do you want to blacklist this record? \n" .. blacklisted_title, f_idx_2)
|
||||||
|
end
|
||||||
Graphics.termBlend()
|
Graphics.termBlend()
|
||||||
Screen.flip()
|
Screen.flip()
|
||||||
Screen.waitVblankStart()
|
Screen.waitVblankStart()
|
||||||
local pad = Controls.read()
|
local pad = Controls.read()
|
||||||
if Controls.check(pad, SCE_CTRL_UP) and mov_y == 0 then
|
if Controls.check(pad, SCE_CTRL_UP) and mov_y == 0 then
|
||||||
if freeze then
|
if freeze then
|
||||||
f_idx = 1
|
f_idx = 1
|
||||||
|
elseif freeze_blacklist then
|
||||||
|
f_idx_2 = 1
|
||||||
else
|
else
|
||||||
new_list_idx = list_idx - 1
|
new_list_idx = list_idx - 1
|
||||||
if new_list_idx == 0 then
|
if new_list_idx == 0 then
|
||||||
@ -357,7 +368,9 @@ while #tbl > 0 do
|
|||||||
end
|
end
|
||||||
elseif Controls.check(pad, SCE_CTRL_DOWN) and mov_y == 0 then
|
elseif Controls.check(pad, SCE_CTRL_DOWN) and mov_y == 0 then
|
||||||
if freeze then
|
if freeze then
|
||||||
f_idx = 2
|
f_idx = 2
|
||||||
|
elseif freeze_blacklist then
|
||||||
|
f_idx_2 = 2
|
||||||
else
|
else
|
||||||
new_list_idx = list_idx + 1
|
new_list_idx = list_idx + 1
|
||||||
if new_list_idx > #tbl then
|
if new_list_idx > #tbl then
|
||||||
@ -366,17 +379,30 @@ while #tbl > 0 do
|
|||||||
mov_y = -11
|
mov_y = -11
|
||||||
end
|
end
|
||||||
elseif Controls.check(pad, SCE_CTRL_TRIANGLE) and not Controls.check(oldpad, SCE_CTRL_TRIANGLE) and not freeze then
|
elseif Controls.check(pad, SCE_CTRL_TRIANGLE) and not Controls.check(oldpad, SCE_CTRL_TRIANGLE) and not freeze then
|
||||||
freeze = true
|
freeze = true
|
||||||
f_idx = 1
|
f_idx = 2
|
||||||
elseif Controls.check(pad, SCE_CTRL_CROSS) and not Controls.check(oldpad, SCE_CTRL_CROSS) and freeze then
|
f_idx_2 = 2
|
||||||
freeze = false
|
elseif Controls.check(pad, SCE_CTRL_CROSS) and not Controls.check(oldpad, SCE_CTRL_CROSS) and freeze then
|
||||||
if f_idx == 1 then -- Delete
|
freeze = false
|
||||||
System.deleteFile("ux0:/data/TrackPlug/Records/" .. tbl[list_idx].id .. ".bin")
|
if f_idx == 1 then -- Delete
|
||||||
|
blacklisted_title_id = tbl[list_idx].id
|
||||||
|
blacklisted_title = string.gsub(tbl[list_idx].title, "\n", " ")
|
||||||
|
System.deleteFile("ux0:/data/TrackPlug/Records/" .. tbl[list_idx].id .. ".bin")
|
||||||
|
freeze_blacklist = true
|
||||||
table.remove(tbl, list_idx)
|
table.remove(tbl, list_idx)
|
||||||
big_tbl = {}
|
big_tbl = {}
|
||||||
list_idx = list_idx - 1
|
list_idx = list_idx - 1
|
||||||
end
|
end
|
||||||
end
|
elseif Controls.check(pad, SCE_CTRL_CROSS) and not Controls.check(oldpad, SCE_CTRL_CROSS) and freeze_blacklist then
|
||||||
|
freeze_blacklist = false
|
||||||
|
if f_idx_2 == 1 then -- Blacklist
|
||||||
|
local file = System.openFile("ux0:/data/TrackPlug/blacklist.txt", FRDWR)
|
||||||
|
local file_size = System.sizeFile(file)
|
||||||
|
local file_var = System.readFile(file, file_size)
|
||||||
|
System.writeFile(file, blacklisted_title_id .. "\n", string.len(blacklisted_title_id) + 1)
|
||||||
|
System.closeFile(file)
|
||||||
|
end
|
||||||
|
end
|
||||||
oldpad = pad
|
oldpad = pad
|
||||||
end
|
end
|
||||||
|
|
||||||
|
BIN
app/sce_sys/icon0.png
Normal file
BIN
app/sce_sys/icon0.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 6.6 KiB |
BIN
app/sce_sys/livearea/contents/bg.png
Normal file
BIN
app/sce_sys/livearea/contents/bg.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 82 KiB |
BIN
app/sce_sys/livearea/contents/startup.png
Normal file
BIN
app/sce_sys/livearea/contents/startup.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 18 KiB |
11
app/sce_sys/livearea/contents/template.xml
Normal file
11
app/sce_sys/livearea/contents/template.xml
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
|
||||||
|
<livearea style="a1" format-ver="01.00" content-rev="1">
|
||||||
|
<livearea-background>
|
||||||
|
<image>bg.png</image>
|
||||||
|
</livearea-background>
|
||||||
|
|
||||||
|
<gate>
|
||||||
|
<startup-image>startup.png</startup-image>
|
||||||
|
</gate>
|
||||||
|
</livearea>
|
BIN
app/sce_sys/param.sfo
Normal file
BIN
app/sce_sys/param.sfo
Normal file
Binary file not shown.
@ -1,15 +1,15 @@
|
|||||||
cmake_minimum_required(VERSION 2.8)
|
cmake_minimum_required(VERSION 2.8)
|
||||||
|
|
||||||
if(NOT DEFINED CMAKE_TOOLCHAIN_FILE)
|
if(NOT DEFINED CMAKE_TOOLCHAIN_FILE)
|
||||||
if(DEFINED ENV{VITASDK})
|
if(DEFINED ENV{DOLCESDK})
|
||||||
set(CMAKE_TOOLCHAIN_FILE "$ENV{VITASDK}/share/vita.toolchain.cmake" CACHE PATH "toolchain file")
|
set(CMAKE_TOOLCHAIN_FILE "$ENV{DOLCESDK}/share/dolce.toolchain.cmake" CACHE PATH "toolchain file")
|
||||||
else()
|
else()
|
||||||
message(FATAL_ERROR "Please define VITASDK to point to your SDK path!")
|
message(FATAL_ERROR "Please define DOLCESDK to point to your SDK path!")
|
||||||
endif()
|
endif()
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
project(BetterTrackPlug)
|
project(BetterTrackPlug)
|
||||||
include("$ENV{VITASDK}/share/vita.cmake" REQUIRED)
|
include("$ENV{DOLCESDK}/share/dolce.cmake" REQUIRED)
|
||||||
|
|
||||||
set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -Wl,-q -Wall -O3 -std=gnu99")
|
set(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -Wl,-q -Wall -O3 -std=gnu99")
|
||||||
set(CMAKE_EXE_LINKER_FLAGS "${CMAKE_EXE_LINKER_FLAGS} -nostdlib")
|
set(CMAKE_EXE_LINKER_FLAGS "${CMAKE_EXE_LINKER_FLAGS} -nostdlib")
|
||||||
@ -36,7 +36,7 @@ target_link_libraries(${PROJECT_NAME}
|
|||||||
gcc
|
gcc
|
||||||
-nostdlib
|
-nostdlib
|
||||||
)
|
)
|
||||||
vita_create_self(${PROJECT_NAME}.skprx ${PROJECT_NAME}
|
dolce_create_self(${PROJECT_NAME}.skprx ${PROJECT_NAME}
|
||||||
UNSAFE
|
UNSAFE
|
||||||
CONFIG ${CMAKE_SOURCE_DIR}/BetterTrackPlug.yml
|
CONFIG ${CMAKE_SOURCE_DIR}/BetterTrackPlug.yml
|
||||||
)
|
)
|
||||||
|
15
src/main.c
15
src/main.c
@ -1,12 +1,9 @@
|
|||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
#include <taihen.h>
|
#include <taihen.h>
|
||||||
#include <psp2/io/fcntl.h>
|
#include <psp2kern/kernel/sysmem.h>
|
||||||
#include <psp2kern/kernel/sysmem.h>
|
#include <psp2kern/kernel/modulemgr.h>
|
||||||
|
#include <psp2kern/kernel/iofilemgr.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include <stdarg.h>
|
|
||||||
#include <vitasdkkern.h>
|
|
||||||
#include <stdlib.h>
|
|
||||||
#include <psp2/appmgr.h>
|
|
||||||
|
|
||||||
#define printf ksceDebugPrintf
|
#define printf ksceDebugPrintf
|
||||||
#define SECOND 1000000
|
#define SECOND 1000000
|
||||||
@ -97,7 +94,7 @@ static void adrenaline_titlewriter(char *dummy_id,char *dummy_title){
|
|||||||
char pathfolder[128];
|
char pathfolder[128];
|
||||||
snprintf(path, 128, "ux0:/data/TrackPlug/Assets/%s/title.txt", dummy_id);
|
snprintf(path, 128, "ux0:/data/TrackPlug/Assets/%s/title.txt", dummy_id);
|
||||||
snprintf(pathfolder, 128, "ux0:/data/TrackPlug/Assets/%s/", dummy_id);
|
snprintf(pathfolder, 128, "ux0:/data/TrackPlug/Assets/%s/", dummy_id);
|
||||||
int dfd = ksceIoMkdir(pathfolder,6);
|
ksceIoMkdir(pathfolder,6);
|
||||||
SceUID fd = ksceIoOpen(path,
|
SceUID fd = ksceIoOpen(path,
|
||||||
SCE_O_WRONLY | SCE_O_CREAT | SCE_O_TRUNC, 0777);
|
SCE_O_WRONLY | SCE_O_CREAT | SCE_O_TRUNC, 0777);
|
||||||
if (fd < 0) return;
|
if (fd < 0) return;
|
||||||
@ -197,7 +194,7 @@ int event_handler(int pid, int ev, int a3, int a4, int *a5, int a6) {
|
|||||||
//Write games playtime and reset the variable. If Adrenaline was in XMB, no need
|
//Write games playtime and reset the variable. If Adrenaline was in XMB, no need
|
||||||
//to write it again as it will be handled by the thread.
|
//to write it again as it will be handled by the thread.
|
||||||
if (!isxmb) write_playtime(adrenaline_id);
|
if (!isxmb) write_playtime(adrenaline_id);
|
||||||
snprintf(adrenaline_id,12,"pleasePLEASE");
|
snprintf(adrenaline_id,12,"pleasePLEAS");
|
||||||
pspemu = 0;
|
pspemu = 0;
|
||||||
} else
|
} else
|
||||||
write_playtime(id);
|
write_playtime(id);
|
||||||
@ -243,4 +240,4 @@ int module_start(SceSize args, void *argp) {
|
|||||||
int module_stop(SceSize args, void *argp) {
|
int module_stop(SceSize args, void *argp) {
|
||||||
taiHookReleaseForKernel(hooks[0], event_handler_ref);
|
taiHookReleaseForKernel(hooks[0], event_handler_ref);
|
||||||
return SCE_KERNEL_STOP_SUCCESS;
|
return SCE_KERNEL_STOP_SUCCESS;
|
||||||
}
|
}
|
||||||
|
Reference in New Issue
Block a user