TegraExplorer/source/fs/fscopy.c
suchmememanyskill 5898c861ec Add Folder copy/move/delete
Also implement the current folder menu
2020-12-29 01:05:33 +01:00

179 lines
No EOL
4.3 KiB
C

#include "fscopy.h"
#include <libs/fatfs/ff.h>
#include <utils/btn.h>
#include "../tegraexplorer/tconf.h"
#include "../gfx/gfx.h"
#include <mem/heap.h>
#include <string.h>
#include "../gfx/gfxutils.h"
#include "fsutils.h"
#include "readers/folderReader.h"
ErrCode_t FileCopy(const char *locin, const char *locout, u8 options){
FIL in, out;
FILINFO in_info;
u64 sizeRemaining, toCopy;
u8 *buff;
u32 x, y;
ErrCode_t err = newErrCode(0);
int res = 0;
gfx_con_getpos(&x, &y);
if (!strcmp(locin, locout)){
return newErrCode(TE_ERR_SAME_LOC);
}
if ((res = f_open(&in, locin, FA_READ | FA_OPEN_EXISTING))){
return newErrCode(res);
}
if ((res = f_stat(locin, &in_info))){
return newErrCode(res);
}
if ((res = f_open(&out, locout, FA_CREATE_ALWAYS | FA_WRITE))){
return newErrCode(res);
}
if (options & COPY_MODE_PRINT){
gfx_printf("[ ]");
x += 16;
gfx_con_setpos(x, y);
}
buff = malloc(TConf.FSBuffSize);
sizeRemaining = f_size(&in);
const u64 totalsize = sizeRemaining;
while (sizeRemaining > 0){
toCopy = MIN(sizeRemaining, TConf.FSBuffSize);
if ((res = f_read(&in, buff, toCopy, NULL))){
err = newErrCode(res);
break;
}
if ((res = f_write(&out, buff, toCopy, NULL))){
err = newErrCode(res);
break;
}
sizeRemaining -= toCopy;
if (options & COPY_MODE_PRINT){
gfx_printf("%3d%%", (u32)(((totalsize - sizeRemaining) * 100) / totalsize));
gfx_con_setpos(x, y);
}
if (options & COPY_MODE_CANCEL && btn_read() & (BTN_VOL_DOWN | BTN_VOL_UP)){
f_unlink(locout);
break;
}
}
f_close(&in);
f_close(&out);
free(buff);
f_chmod(locout, in_info.fattrib, 0x3A);
//f_stat(locin, &in_info); //somehow stops fatfs from being weird
return err;
}
void BoxRestOfScreen(){
u32 tempX, tempY;
gfx_con_getpos(&tempX, &tempY);
gfx_boxGrey(tempX, tempY, YLEFT, tempY + 16, 0x1B);
}
ErrCode_t FolderCopy(const char *locin, const char *locout){
char *dstPath = CombinePaths(locout, strrchr(locin, '/') + 1);
int res = 0;
ErrCode_t ret = newErrCode(0);
u32 x, y;
gfx_con_getpos(&x, &y);
Vector_t fileVec = ReadFolder(locin, &res);
if (res){
ret = newErrCode(res);
}
else {
vecDefArray(FSEntry_t *, fs, fileVec);
f_mkdir(dstPath);
for (int i = 0; i < fileVec.count && !ret.err; i++){
char *temp = CombinePaths(locin, fs[i].name);
if (fs[i].isDir){
ret = FolderCopy(temp, dstPath);
}
else {
gfx_puts_limit(fs[i].name, (YLEFT - x) / 16 - 10);
BoxRestOfScreen();
char *tempDst = CombinePaths(dstPath, fs[i].name);
ret = FileCopy(temp, tempDst, COPY_MODE_PRINT);
free(tempDst);
gfx_con_setpos(x, y);
}
free(temp);
}
}
FILINFO fno;
if (!ret.err){
res = f_stat(locin, &fno);
if (res)
ret = newErrCode(res);
else
ret = newErrCode(f_chmod(dstPath, fno.fattrib, 0x3A));
}
free(dstPath);
clearFileVector(&fileVec);
return ret;
}
ErrCode_t FolderDelete(const char *path){
int res = 0;
ErrCode_t ret = newErrCode(0);
u32 x, y;
gfx_con_getpos(&x, &y);
Vector_t fileVec = ReadFolder(path, &res);
if (res){
ret = newErrCode(res);
}
else {
vecDefArray(FSEntry_t *, fs, fileVec);
for (int i = 0; i < fileVec.count && !ret.err; i++){
char *temp = CombinePaths(path, fs[i].name);
if (fs[i].isDir){
ret = FolderDelete(temp);
}
else {
gfx_puts_limit(fs[i].name, (YLEFT - x) / 16 - 10);
BoxRestOfScreen();
res = f_unlink(temp);
if (res){
ret = newErrCode(res);
}
gfx_con_setpos(x, y);
}
free(temp);
}
}
if (!ret.err){
res = f_unlink(path);
if (res)
ret = newErrCode(res);
}
clearFileVector(&fileVec);
return ret;
}