/*
* QCMA: Cross-platform content manager assistant for the PS Vita
*
* Copyright (C) 2013 Codestation
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see .
*/
#include "cmaobject.h"
#include "sforeader.h"
#include "database.h"
#include "cmautils.h"
#include
#include
#include
#include
int CMAObject::ohfi_count = OHFI_OFFSET;
CMAObject::CMAObject(CMAObject *obj_parent) :
parent(obj_parent), metadata()
{
}
CMAObject::~CMAObject()
{
free(metadata.name);
free(metadata.path);
if(MASK_SET(metadata.dataType, SaveData | Folder)) {
free(metadata.data.saveData.title);
free(metadata.data.saveData.detail);
free(metadata.data.saveData.dirName);
free(metadata.data.saveData.savedataTitle);
} else if(MASK_SET(metadata.dataType, Photo | File)) {
free(metadata.data.photo.title);
free(metadata.data.photo.fileName);
delete metadata.data.photo.tracks;
} else if(MASK_SET(metadata.dataType, Music | File)) {
free(metadata.data.music.title);
free(metadata.data.music.fileName);
free(metadata.data.music.album);
free(metadata.data.music.artist);
delete metadata.data.music.tracks;
} else if(MASK_SET(metadata.dataType, Video | File)) {
free(metadata.data.video.title);
free(metadata.data.video.explanation);
free(metadata.data.video.fileName);
free(metadata.data.video.copyright);
delete metadata.data.video.tracks;
}
}
void CMAObject::loadSfoMetadata(const QString &path)
{
QString sfo = QDir(path).absoluteFilePath("PARAM.SFO");
bool skipMetadata = QSettings().value("skipMetadata", false).toBool();
SfoReader reader;
if(!skipMetadata && reader.load(sfo)) {
metadata.data.saveData.title = strdup(reader.value("TITLE", ""));
//FIXME: disable savedata detail for now
//QString detail(reader.value("SAVEDATA_DETAIL", ""));
// libxml follow the spec and normalizes the newlines (and others) but
// the PS Vita chokes on contiguous codes and crashes the CMA app on
// the device. Of course, the "fix" would be that libxml doesn't
// normalize newlines but that is aganist the XML spec that the PS Vita
// doesn't respect >_>
// convert DOS to UNIX newlines
//detail.replace("\r\n", "\n");
// separate newlines from quotes
//detail.replace("\n\"", "\n \"");
//detail.replace("\"\n", "\" \n");
// merge consecutive newlines
//detail.replace("\n\n", "\n");
//while(detail.endsWith('\n')) {
// detail.chop(1);
//}
//metadata.data.saveData.detail = strdup(detail.toStdString().c_str());
metadata.data.saveData.detail = strdup("");
// remove newlines from savedata title
QString title(reader.value("SAVEDATA_TITLE", ""));
while(title.endsWith('\n')) {
title.chop(1);
}
metadata.data.saveData.savedataTitle = strdup(title.toStdString().c_str());
metadata.data.saveData.dateTimeUpdated = QFileInfo(sfo).birthTime().toUTC().toSecsSinceEpoch();
} else {
metadata.data.saveData.title = strdup(metadata.name);
metadata.data.saveData.detail = strdup("");
metadata.data.saveData.savedataTitle = strdup("");
metadata.data.saveData.dateTimeUpdated = 0;
}
}
void CMAObject::initObject(const QFileInfo &file, int obj_file_type)
{
metadata.name = strdup(file.fileName().toUtf8().data());
metadata.ohfiParent = parent->metadata.ohfi;
metadata.ohfi = ohfi_count++;
metadata.type = VITA_DIR_TYPE_MASK_REGULAR; // ignored for files
metadata.dateTimeCreated = file.birthTime().toUTC().toSecsSinceEpoch();
metadata.size = 0;
DataType type = file.isFile() ? File : Folder;
metadata.dataType = (DataType)(type | (parent->metadata.dataType & ~Folder));
// create additional metadata
if(MASK_SET(metadata.dataType, SaveData | Folder)) {
metadata.data.saveData.dirName = strdup(metadata.name);
metadata.data.saveData.statusType = 1;
loadSfoMetadata(file.absoluteFilePath());
} else if(MASK_SET(metadata.dataType, Music | File)) {
if(obj_file_type < 0) {
qWarning("Invalid file type for music: %i, setting it to zero", obj_file_type);
obj_file_type = 0;
}
metadata.data.music.fileName = strdup(metadata.name);
metadata.data.music.fileFormatType = audio_list[obj_file_type].file_format;
metadata.data.music.statusType = 1;
metadata.data.music.numTracks = 1;
metadata.data.music.tracks = new media_track();
metadata.data.music.tracks->type = VITA_TRACK_TYPE_AUDIO;
metadata.data.music.tracks->data.track_photo.codecType = audio_list[obj_file_type].file_codec;
Database::loadMusicMetadata(file.absoluteFilePath(), metadata);
} else if(MASK_SET(metadata.dataType, Video | File)) {
metadata.data.video.fileName = strdup(metadata.name);
metadata.data.video.dateTimeUpdated = file.birthTime().toUTC().toSecsSinceEpoch();
metadata.data.video.statusType = 1;
metadata.data.video.fileFormatType = FILE_FORMAT_MP4;
metadata.data.video.parentalLevel = 0;
metadata.data.video.numTracks = 1;
metadata.data.video.tracks = new media_track();
metadata.data.video.tracks->type = VITA_TRACK_TYPE_VIDEO;
Database::loadVideoMetadata(file.absoluteFilePath(), metadata);
} else if(MASK_SET(metadata.dataType, Photo | File)) {
if(obj_file_type < 0) {
qWarning("Invalid file type for photos: %i, setting it to zero", obj_file_type);
obj_file_type = 0;
}
metadata.data.photo.fileName = strdup(metadata.name);
metadata.data.photo.fileFormatType = photo_list[obj_file_type].file_format;
metadata.data.photo.statusType = 1;
metadata.data.photo.dateTimeOriginal = file.birthTime().toUTC().toSecsSinceEpoch();
metadata.data.photo.numTracks = 1;
metadata.data.photo.tracks = new media_track();
metadata.data.photo.tracks->type = VITA_TRACK_TYPE_PHOTO;
metadata.data.photo.tracks->data.track_photo.codecType = photo_list[obj_file_type].file_codec;
Database::loadPhotoMetadata(file.absoluteFilePath(), metadata);
}
m_path = file.absoluteFilePath();
if(parent->metadata.path == NULL) {
metadata.path = strdup(metadata.name);
} else {
QString newpath = QString(parent->metadata.path) + "/" + metadata.name;
metadata.path = strdup(newpath.toUtf8().data());
}
updateObjectSize(file.size());
}
void CMAObject::updateObjectSize(qint64 size)
{
if(parent) {
parent->updateObjectSize(size);
}
metadata.size += size;
}
void CMAObject::rename(const QString &newname)
{
free(metadata.name);
metadata.name = strdup(newname.toUtf8().data());
if(metadata.path) {
QStringList metadata_path(QString(metadata.path).split("/"));
metadata_path.replace(metadata_path.count() - 1, newname);
free(metadata.path);
metadata.path = strdup(metadata_path.join("/").toUtf8().data());
}
m_path = QFileInfo(m_path).absoluteDir().path() + "/" + newname;
}
void CMAObject::refreshPath()
{
if(parent) {
free(metadata.path);
QString newpath(QString(parent->metadata.path) + "/" + metadata.name);
metadata.path = strdup(newpath.toUtf8().data());
m_path = parent->m_path + "/" + metadata.name;
}
}
bool CMAObject::hasParent(const CMAObject *obj)
{
if(parent) {
if(metadata.ohfiParent == obj->metadata.ohfi) {
return true;
} else {
return parent->hasParent(obj);
}
}
return false;
}
bool CMAObject::operator==(const CMAObject &obj)
{
return metadata.ohfi == obj.metadata.ohfi;
}
bool CMAObject::operator!=(const CMAObject &obj)
{
return metadata.ohfi != obj.metadata.ohfi;
}
bool CMAObject::operator<(const CMAObject &obj)
{
return metadata.ohfi < obj.metadata.ohfi;
}