/* * 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; }