2013-09-01 16:27:21 +02:00
|
|
|
/*
|
|
|
|
** filesystem.cpp
|
|
|
|
**
|
|
|
|
** This file is part of mkxp.
|
|
|
|
**
|
|
|
|
** Copyright (C) 2013 Jonas Kulla <Nyocurio@gmail.com>
|
|
|
|
**
|
|
|
|
** mkxp 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 2 of the License, or
|
|
|
|
** (at your option) any later version.
|
|
|
|
**
|
|
|
|
** mkxp 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 mkxp. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "filesystem.h"
|
|
|
|
|
Font: Overhaul font asset discovery
Previously, any font names requested by RGSS would be translated
directly to filenames by lowercasing and replacing spaces with
underscores (and finally doing some extension substitution).
To make this whole thing work smoother as well as get closer to
how font discovery is done in VX, we now scan the "Fonts/" folder
at startup and index all present font assets by their family name;
now, if an "Open Sans" font is present in "Fonts/", it will be
used regardless of filename.
Font assets with "Regular" style are preferred, but in their
absence, mkxp will make use of any other style it can find for
the respective family. This is not the exact same behavior as
VX, but it should cover 95% of use cases.
Previously, one could substitute fonts via filenames, ie. to
substitute "Arial" with "Open Sans", one would just rename
"OpenSans.ttf" to "arial.ttf" and put it in "Fonts/". With the
above change, this is no longer possible. As an alternative, one
can now explicitly specify font family substitutions via mkxp.conf;
eg. for the above case, one would add
fontSub=Arial>Open Sans
to the configuration file. Multiple such rules can be specified.
In the process, I also added the ability to provide
'Font.(default_)name' with an array of font families to search
for the first existing one instead of a plain string.
This makes the behavior closer to RMXP; however, it doesn't
work 100% the same: when a reference to the 'Font.name' array is
held and additional strings are added to it without re-assignig
the array to 'Font.name', those will be ignored.
2014-04-11 13:37:14 +02:00
|
|
|
#include "font.h"
|
2013-09-01 16:27:21 +02:00
|
|
|
#include "util.h"
|
|
|
|
#include "exception.h"
|
2013-12-11 20:46:54 +01:00
|
|
|
#include "boost-hash.h"
|
|
|
|
#include "debugwriter.h"
|
2013-09-01 16:27:21 +02:00
|
|
|
|
2013-12-04 17:48:37 +01:00
|
|
|
#include <physfs.h>
|
2013-09-01 16:27:21 +02:00
|
|
|
|
2013-11-30 12:00:11 +01:00
|
|
|
#include <SDL_sound.h>
|
|
|
|
|
2013-12-04 17:48:37 +01:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
2013-12-08 13:19:22 +01:00
|
|
|
#include <algorithm>
|
2013-12-26 20:18:33 +01:00
|
|
|
#include <vector>
|
2013-09-01 16:27:21 +02:00
|
|
|
|
|
|
|
struct RGSS_entryData
|
|
|
|
{
|
2013-09-04 13:30:14 +02:00
|
|
|
int64_t offset;
|
|
|
|
uint64_t size;
|
|
|
|
uint32_t startMagic;
|
2013-09-01 16:27:21 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
struct RGSS_entryHandle
|
|
|
|
{
|
|
|
|
RGSS_entryData data;
|
2013-09-04 13:30:14 +02:00
|
|
|
uint32_t currentMagic;
|
|
|
|
uint64_t currentOffset;
|
2013-09-01 16:27:21 +02:00
|
|
|
PHYSFS_Io *io;
|
|
|
|
|
2013-10-06 05:10:49 +02:00
|
|
|
RGSS_entryHandle(const RGSS_entryData &data, PHYSFS_Io *archIo)
|
2013-09-01 16:27:21 +02:00
|
|
|
: data(data),
|
|
|
|
currentMagic(data.startMagic),
|
|
|
|
currentOffset(0)
|
2013-10-06 05:10:49 +02:00
|
|
|
{
|
|
|
|
io = archIo->duplicate(archIo);
|
|
|
|
}
|
2013-09-01 16:27:21 +02:00
|
|
|
|
2013-10-06 05:10:49 +02:00
|
|
|
~RGSS_entryHandle()
|
|
|
|
{
|
|
|
|
io->destroy(io);
|
|
|
|
}
|
2013-09-01 16:27:21 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
struct RGSS_archiveData
|
|
|
|
{
|
|
|
|
PHYSFS_Io *archiveIo;
|
2013-11-01 08:24:14 +01:00
|
|
|
|
|
|
|
/* Maps: file path
|
|
|
|
* to: entry data */
|
2013-12-11 20:46:54 +01:00
|
|
|
BoostHash<std::string, RGSS_entryData> entryHash;
|
2013-11-01 08:24:14 +01:00
|
|
|
|
|
|
|
/* Maps: directory path,
|
|
|
|
* to: list of contained entries */
|
2013-12-11 20:46:54 +01:00
|
|
|
BoostHash<std::string, BoostSet<std::string> > dirHash;
|
2013-09-01 16:27:21 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
static bool
|
2013-09-04 13:30:14 +02:00
|
|
|
readUint32(PHYSFS_Io *io, uint32_t &result)
|
2013-09-01 16:27:21 +02:00
|
|
|
{
|
|
|
|
char buff[4];
|
|
|
|
PHYSFS_sint64 count = io->read(io, buff, 4);
|
|
|
|
|
|
|
|
result = ((buff[0] << 0x00) & 0x000000FF) |
|
|
|
|
((buff[1] << 0x08) & 0x0000FF00) |
|
|
|
|
((buff[2] << 0x10) & 0x00FF0000) |
|
|
|
|
((buff[3] << 0x18) & 0xFF000000) ;
|
|
|
|
|
|
|
|
return (count == 4);
|
|
|
|
}
|
|
|
|
|
|
|
|
#define RGSS_HEADER_1 0x53534752
|
2013-09-04 17:03:00 +02:00
|
|
|
#define RGSS_HEADER_2 0x01004441
|
2013-09-01 16:27:21 +02:00
|
|
|
|
|
|
|
#define RGSS_MAGIC 0xDEADCAFE
|
|
|
|
|
|
|
|
#define PHYSFS_ALLOC(type) \
|
|
|
|
static_cast<type*>(PHYSFS_getAllocator()->Malloc(sizeof(type)))
|
|
|
|
|
2013-09-04 13:30:14 +02:00
|
|
|
static inline uint32_t
|
|
|
|
advanceMagic(uint32_t &magic)
|
2013-09-01 16:27:21 +02:00
|
|
|
{
|
2013-09-04 13:30:14 +02:00
|
|
|
uint32_t old = magic;
|
2013-09-01 16:27:21 +02:00
|
|
|
|
|
|
|
magic = magic * 7 + 3;
|
|
|
|
|
|
|
|
return old;
|
|
|
|
}
|
|
|
|
|
|
|
|
static PHYSFS_sint64
|
|
|
|
RGSS_ioRead(PHYSFS_Io *self, void *buffer, PHYSFS_uint64 len)
|
|
|
|
{
|
|
|
|
RGSS_entryHandle *entry = static_cast<RGSS_entryHandle*>(self->opaque);
|
|
|
|
|
2013-10-04 07:52:31 +02:00
|
|
|
PHYSFS_Io *io = entry->io;
|
|
|
|
|
2013-12-08 13:19:22 +01:00
|
|
|
uint64_t toRead = std::min<uint64_t>(entry->data.size - entry->currentOffset, len);
|
2013-09-04 13:30:14 +02:00
|
|
|
uint64_t offs = entry->currentOffset;
|
2013-09-01 16:27:21 +02:00
|
|
|
|
2013-10-04 07:52:31 +02:00
|
|
|
io->seek(io, entry->data.offset + offs);
|
|
|
|
|
|
|
|
/* We divide up the bytes to be read in 3 categories:
|
|
|
|
*
|
|
|
|
* preAlign: If the current read address is not dword
|
|
|
|
* aligned, this is the number of bytes to read til
|
|
|
|
* we reach alignment again (therefore can only be
|
|
|
|
* 3 or less).
|
|
|
|
*
|
|
|
|
* align: The number of aligned dwords we can read
|
|
|
|
* times 4 (= number of bytes).
|
|
|
|
*
|
|
|
|
* postAlign: The number of bytes to read after the
|
|
|
|
* last aligned dword. Always 3 or less.
|
|
|
|
*
|
|
|
|
* Treating the pre- and post aligned reads specially,
|
|
|
|
* we can read all aligned dwords in one syscall directly
|
|
|
|
* into the write buffer and then run the xor chain on
|
|
|
|
* it afterwards. */
|
|
|
|
|
|
|
|
uint8_t preAlign = 4 - (offs % 4);
|
|
|
|
|
|
|
|
if (preAlign == 4)
|
|
|
|
preAlign = 0;
|
|
|
|
else
|
2013-12-08 13:19:22 +01:00
|
|
|
preAlign = std::min<uint64_t>(preAlign, len);
|
2013-10-04 07:52:31 +02:00
|
|
|
|
|
|
|
uint8_t postAlign = (len > preAlign) ? (offs + len) % 4 : 0;
|
|
|
|
|
|
|
|
uint64_t align = len - (preAlign + postAlign);
|
|
|
|
|
|
|
|
/* Byte buffer pointer */
|
|
|
|
uint8_t *bBufferP = static_cast<uint8_t*>(buffer);
|
|
|
|
|
|
|
|
if (preAlign > 0)
|
2013-09-01 16:27:21 +02:00
|
|
|
{
|
2013-10-04 07:52:31 +02:00
|
|
|
uint32_t dword;
|
|
|
|
io->read(io, &dword, preAlign);
|
2013-09-01 16:27:21 +02:00
|
|
|
|
2013-10-04 07:52:31 +02:00
|
|
|
/* Need to align the bytes with the
|
|
|
|
* magic before xoring */
|
|
|
|
dword <<= 8 * (offs % 4);
|
|
|
|
dword ^= entry->currentMagic;
|
2013-09-01 16:27:21 +02:00
|
|
|
|
2013-10-04 07:52:31 +02:00
|
|
|
/* Shift them back to normal */
|
|
|
|
dword >>= 8 * (offs % 4);
|
|
|
|
memcpy(bBufferP, &dword, preAlign);
|
2013-09-01 16:27:21 +02:00
|
|
|
|
2013-10-04 07:52:31 +02:00
|
|
|
bBufferP += preAlign;
|
|
|
|
|
|
|
|
/* Only advance the magic if we actually
|
|
|
|
* reached the next alignment */
|
|
|
|
if ((offs+preAlign) % 4 == 0)
|
2013-09-01 16:27:21 +02:00
|
|
|
advanceMagic(entry->currentMagic);
|
|
|
|
}
|
|
|
|
|
2013-10-04 07:52:31 +02:00
|
|
|
if (align > 0)
|
|
|
|
{
|
|
|
|
/* Double word buffer pointer */
|
|
|
|
uint32_t *dwBufferP = reinterpret_cast<uint32_t*>(bBufferP);
|
|
|
|
|
|
|
|
/* Read aligned dwords in one go */
|
|
|
|
io->read(io, bBufferP, align);
|
|
|
|
|
|
|
|
/* Then xor them */
|
|
|
|
for (uint64_t i = 0; i < (align / 4); ++i)
|
|
|
|
dwBufferP[i] ^= advanceMagic(entry->currentMagic);
|
|
|
|
|
|
|
|
bBufferP += align;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (postAlign > 0)
|
|
|
|
{
|
|
|
|
uint32_t dword;
|
|
|
|
io->read(io, &dword, postAlign);
|
|
|
|
|
|
|
|
/* Bytes are already aligned with magic */
|
|
|
|
dword ^= entry->currentMagic;
|
|
|
|
memcpy(bBufferP, &dword, postAlign);
|
|
|
|
}
|
|
|
|
|
2013-09-01 16:27:21 +02:00
|
|
|
entry->currentOffset += toRead;
|
|
|
|
|
|
|
|
return toRead;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
RGSS_ioSeek(PHYSFS_Io *self, PHYSFS_uint64 offset)
|
|
|
|
{
|
|
|
|
RGSS_entryHandle *entry = static_cast<RGSS_entryHandle*>(self->opaque);
|
|
|
|
|
|
|
|
if (offset == entry->currentOffset)
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
if (offset > entry->data.size-1)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
/* If rewinding, we need to rewind to begining */
|
|
|
|
if (offset < entry->currentOffset)
|
|
|
|
{
|
|
|
|
entry->currentOffset = 0;
|
|
|
|
entry->currentMagic = entry->data.startMagic;
|
|
|
|
}
|
|
|
|
|
2013-10-04 07:52:31 +02:00
|
|
|
/* For each overstepped alignment, advance magic */
|
2013-10-03 22:09:01 +02:00
|
|
|
uint64_t currentDword = entry->currentOffset / 4;
|
2013-10-04 07:52:31 +02:00
|
|
|
uint64_t targetDword = offset / 4;
|
|
|
|
uint64_t dwordsSought = targetDword - currentDword;
|
2013-10-03 22:09:01 +02:00
|
|
|
|
2013-09-04 13:30:14 +02:00
|
|
|
for (uint64_t i = 0; i < dwordsSought; ++i)
|
2013-09-01 16:27:21 +02:00
|
|
|
advanceMagic(entry->currentMagic);
|
|
|
|
|
|
|
|
entry->currentOffset = offset;
|
|
|
|
entry->io->seek(entry->io, entry->data.offset + entry->currentOffset);
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static PHYSFS_sint64
|
|
|
|
RGSS_ioTell(PHYSFS_Io *self)
|
|
|
|
{
|
|
|
|
RGSS_entryHandle *entry = static_cast<RGSS_entryHandle*>(self->opaque);
|
|
|
|
|
|
|
|
return entry->currentOffset;
|
|
|
|
}
|
|
|
|
|
|
|
|
static PHYSFS_sint64
|
|
|
|
RGSS_ioLength(PHYSFS_Io *self)
|
|
|
|
{
|
|
|
|
RGSS_entryHandle *entry = static_cast<RGSS_entryHandle*>(self->opaque);
|
|
|
|
|
|
|
|
return entry->data.size;
|
|
|
|
}
|
|
|
|
|
|
|
|
static PHYSFS_Io*
|
|
|
|
RGSS_ioDuplicate(PHYSFS_Io *self)
|
|
|
|
{
|
|
|
|
RGSS_entryHandle *entry = static_cast<RGSS_entryHandle*>(self->opaque);
|
|
|
|
RGSS_entryHandle *entryDup = new RGSS_entryHandle(*entry);
|
|
|
|
|
|
|
|
PHYSFS_Io *dup = PHYSFS_ALLOC(PHYSFS_Io);
|
|
|
|
*dup = *self;
|
|
|
|
dup->opaque = entryDup;
|
|
|
|
|
|
|
|
return dup;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
RGSS_ioDestroy(PHYSFS_Io *self)
|
|
|
|
{
|
|
|
|
RGSS_entryHandle *entry = static_cast<RGSS_entryHandle*>(self->opaque);
|
|
|
|
|
|
|
|
delete entry;
|
|
|
|
|
|
|
|
PHYSFS_getAllocator()->Free(self);
|
|
|
|
}
|
|
|
|
|
|
|
|
static const PHYSFS_Io RGSS_IoTemplate =
|
|
|
|
{
|
|
|
|
0, /* version */
|
|
|
|
0, /* opaque */
|
|
|
|
RGSS_ioRead,
|
|
|
|
0, /* write */
|
|
|
|
RGSS_ioSeek,
|
|
|
|
RGSS_ioTell,
|
|
|
|
RGSS_ioLength,
|
|
|
|
RGSS_ioDuplicate,
|
|
|
|
0, /* flush */
|
|
|
|
RGSS_ioDestroy
|
|
|
|
};
|
|
|
|
|
|
|
|
static void*
|
|
|
|
RGSS_openArchive(PHYSFS_Io *io, const char *, int forWrite)
|
|
|
|
{
|
|
|
|
if (forWrite)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
/* Check header */
|
2013-09-04 13:30:14 +02:00
|
|
|
uint32_t header1, header2;
|
2014-01-16 01:06:59 +01:00
|
|
|
|
|
|
|
if (!readUint32(io, header1))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (!readUint32(io, header2))
|
|
|
|
return 0;
|
2013-09-01 16:27:21 +02:00
|
|
|
|
|
|
|
if (header1 != RGSS_HEADER_1 || header2 != RGSS_HEADER_2)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
RGSS_archiveData *data = new RGSS_archiveData;
|
|
|
|
data->archiveIo = io;
|
|
|
|
|
2013-09-04 13:30:14 +02:00
|
|
|
uint32_t magic = RGSS_MAGIC;
|
2013-09-01 16:27:21 +02:00
|
|
|
|
2013-11-01 08:24:14 +01:00
|
|
|
/* Top level entry list */
|
2014-01-15 01:51:56 +01:00
|
|
|
BoostSet<std::string> &topLevel = data->dirHash[""];
|
2013-11-01 08:24:14 +01:00
|
|
|
|
2013-09-01 16:27:21 +02:00
|
|
|
while (true)
|
|
|
|
{
|
|
|
|
/* Read filename length,
|
|
|
|
* if nothing was read, no files remain */
|
2013-09-04 13:30:14 +02:00
|
|
|
uint32_t nameLen;
|
2014-01-01 12:56:45 +01:00
|
|
|
|
2013-09-01 16:27:21 +02:00
|
|
|
if (!readUint32(io, nameLen))
|
|
|
|
break;
|
|
|
|
|
|
|
|
nameLen ^= advanceMagic(magic);
|
|
|
|
|
|
|
|
static char nameBuf[512];
|
|
|
|
uint i;
|
|
|
|
for (i = 0; i < nameLen; ++i)
|
|
|
|
{
|
|
|
|
char c;
|
|
|
|
io->read(io, &c, 1);
|
|
|
|
nameBuf[i] = c ^ (advanceMagic(magic) & 0xFF);
|
|
|
|
if (nameBuf[i] == '\\')
|
|
|
|
nameBuf[i] = '/';
|
|
|
|
}
|
|
|
|
nameBuf[i] = 0;
|
|
|
|
|
2013-09-04 13:30:14 +02:00
|
|
|
uint32_t entrySize;
|
2013-09-01 16:27:21 +02:00
|
|
|
readUint32(io, entrySize);
|
|
|
|
entrySize ^= advanceMagic(magic);
|
|
|
|
|
|
|
|
RGSS_entryData entry;
|
|
|
|
entry.offset = io->tell(io);
|
|
|
|
entry.size = entrySize;
|
|
|
|
entry.startMagic = magic;
|
|
|
|
|
|
|
|
data->entryHash.insert(nameBuf, entry);
|
|
|
|
|
2013-11-01 08:24:14 +01:00
|
|
|
/* Check for top level entries */
|
|
|
|
for (i = 0; i < nameLen; ++i)
|
|
|
|
{
|
|
|
|
bool slash = nameBuf[i] == '/';
|
|
|
|
if (!slash && i+1 < nameLen)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (slash)
|
|
|
|
nameBuf[i] = '\0';
|
|
|
|
|
|
|
|
topLevel.insert(nameBuf);
|
|
|
|
|
|
|
|
if (slash)
|
|
|
|
nameBuf[i] = '/';
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check for more entries */
|
2013-09-01 16:27:21 +02:00
|
|
|
for (i = nameLen; i > 0; i--)
|
|
|
|
if (nameBuf[i] == '/')
|
|
|
|
{
|
|
|
|
nameBuf[i] = '\0';
|
2013-11-01 08:24:14 +01:00
|
|
|
|
|
|
|
const char *dir = nameBuf;
|
|
|
|
const char *entry = &nameBuf[i+1];
|
|
|
|
|
2013-12-11 20:46:54 +01:00
|
|
|
BoostSet<std::string> &entryList = data->dirHash[dir];
|
2013-11-01 08:24:14 +01:00
|
|
|
entryList.insert(entry);
|
2013-09-01 16:27:21 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
io->seek(io, entry.offset + entry.size);
|
|
|
|
}
|
|
|
|
|
|
|
|
return data;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
RGSS_enumerateFiles(void *opaque, const char *dirname,
|
|
|
|
PHYSFS_EnumFilesCallback cb,
|
|
|
|
const char *origdir, void *callbackdata)
|
|
|
|
{
|
|
|
|
RGSS_archiveData *data = static_cast<RGSS_archiveData*>(opaque);
|
|
|
|
|
2013-12-11 20:46:54 +01:00
|
|
|
std::string _dirname(dirname);
|
2013-09-01 16:27:21 +02:00
|
|
|
|
2013-11-01 08:24:14 +01:00
|
|
|
if (!data->dirHash.contains(_dirname))
|
|
|
|
return;
|
2013-09-01 16:27:21 +02:00
|
|
|
|
2013-12-11 20:46:54 +01:00
|
|
|
const BoostSet<std::string> &entries = data->dirHash[_dirname];
|
2013-09-01 16:27:21 +02:00
|
|
|
|
2013-12-11 20:46:54 +01:00
|
|
|
BoostSet<std::string>::const_iterator iter;
|
|
|
|
for (iter = entries.cbegin(); iter != entries.cend(); ++iter)
|
|
|
|
cb(callbackdata, origdir, iter->c_str());
|
2013-09-01 16:27:21 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static PHYSFS_Io*
|
|
|
|
RGSS_openRead(void *opaque, const char *filename)
|
|
|
|
{
|
|
|
|
RGSS_archiveData *data = static_cast<RGSS_archiveData*>(opaque);
|
|
|
|
|
|
|
|
if (!data->entryHash.contains(filename))
|
|
|
|
return 0;
|
|
|
|
|
2013-10-06 05:10:49 +02:00
|
|
|
RGSS_entryHandle *entry =
|
|
|
|
new RGSS_entryHandle(data->entryHash[filename], data->archiveIo);
|
2013-09-01 16:27:21 +02:00
|
|
|
|
|
|
|
PHYSFS_Io *io = PHYSFS_ALLOC(PHYSFS_Io);
|
|
|
|
|
|
|
|
*io = RGSS_IoTemplate;
|
|
|
|
io->opaque = entry;
|
|
|
|
|
|
|
|
return io;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
RGSS_stat(void *opaque, const char *filename, PHYSFS_Stat *stat)
|
|
|
|
{
|
|
|
|
RGSS_archiveData *data = static_cast<RGSS_archiveData*>(opaque);
|
|
|
|
|
|
|
|
bool hasFile = data->entryHash.contains(filename);
|
|
|
|
bool hasDir = data->dirHash.contains(filename);
|
|
|
|
|
|
|
|
if (!hasFile && !hasDir)
|
|
|
|
{
|
|
|
|
PHYSFS_setErrorCode(PHYSFS_ERR_NOT_FOUND);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
stat->modtime =
|
|
|
|
stat->createtime =
|
|
|
|
stat->accesstime = 0;
|
|
|
|
stat->readonly = 1;
|
|
|
|
|
|
|
|
if (hasFile)
|
|
|
|
{
|
|
|
|
RGSS_entryData &entry = data->entryHash[filename];
|
|
|
|
|
|
|
|
stat->filesize = entry.size;
|
|
|
|
stat->filetype = PHYSFS_FILETYPE_REGULAR;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
stat->filesize = 0;
|
|
|
|
stat->filetype = PHYSFS_FILETYPE_DIRECTORY;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
RGSS_closeArchive(void *opaque)
|
|
|
|
{
|
|
|
|
RGSS_archiveData *data = static_cast<RGSS_archiveData*>(opaque);
|
|
|
|
|
|
|
|
delete data;
|
|
|
|
}
|
|
|
|
|
|
|
|
static PHYSFS_Io*
|
|
|
|
RGSS_noop1(void*, const char*)
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int
|
|
|
|
RGSS_noop2(void*, const char*)
|
|
|
|
{
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static const PHYSFS_Archiver RGSS_Archiver =
|
|
|
|
{
|
|
|
|
0,
|
2014-04-16 13:06:16 +02:00
|
|
|
{
|
|
|
|
"RGSSAD",
|
|
|
|
"RGSS encrypted archive format",
|
|
|
|
"Jonas Kulla <Nyocurio@gmail.com>",
|
|
|
|
"http://k-du.de/rgss/rgss.html",
|
|
|
|
0 /* symlinks not supported */
|
|
|
|
},
|
|
|
|
RGSS_openArchive,
|
|
|
|
RGSS_enumerateFiles,
|
|
|
|
RGSS_openRead,
|
|
|
|
RGSS_noop1, /* openWrite */
|
|
|
|
RGSS_noop1, /* openAppend */
|
|
|
|
RGSS_noop2, /* remove */
|
|
|
|
RGSS_noop2, /* mkdir */
|
|
|
|
RGSS_stat,
|
|
|
|
RGSS_closeArchive
|
2013-09-01 16:27:21 +02:00
|
|
|
};
|
|
|
|
|
Font: Overhaul font asset discovery
Previously, any font names requested by RGSS would be translated
directly to filenames by lowercasing and replacing spaces with
underscores (and finally doing some extension substitution).
To make this whole thing work smoother as well as get closer to
how font discovery is done in VX, we now scan the "Fonts/" folder
at startup and index all present font assets by their family name;
now, if an "Open Sans" font is present in "Fonts/", it will be
used regardless of filename.
Font assets with "Regular" style are preferred, but in their
absence, mkxp will make use of any other style it can find for
the respective family. This is not the exact same behavior as
VX, but it should cover 95% of use cases.
Previously, one could substitute fonts via filenames, ie. to
substitute "Arial" with "Open Sans", one would just rename
"OpenSans.ttf" to "arial.ttf" and put it in "Fonts/". With the
above change, this is no longer possible. As an alternative, one
can now explicitly specify font family substitutions via mkxp.conf;
eg. for the above case, one would add
fontSub=Arial>Open Sans
to the configuration file. Multiple such rules can be specified.
In the process, I also added the ability to provide
'Font.(default_)name' with an array of font families to search
for the first existing one instead of a plain string.
This makes the behavior closer to RMXP; however, it doesn't
work 100% the same: when a reference to the 'Font.name' array is
held and additional strings are added to it without re-assignig
the array to 'Font.name', those will be ignored.
2014-04-11 13:37:14 +02:00
|
|
|
|
|
|
|
static inline PHYSFS_File *sdlPHYS(SDL_RWops *ops)
|
|
|
|
{
|
|
|
|
return static_cast<PHYSFS_File*>(ops->hidden.unknown.data1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Sint64 SDL_RWopsSize(SDL_RWops *ops)
|
|
|
|
{
|
|
|
|
PHYSFS_File *f = sdlPHYS(ops);
|
|
|
|
|
|
|
|
if (!f)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return PHYSFS_fileLength(f);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Sint64 SDL_RWopsSeek(SDL_RWops *ops, int64_t offset, int whence)
|
|
|
|
{
|
|
|
|
PHYSFS_File *f = sdlPHYS(ops);
|
|
|
|
|
|
|
|
if (!f)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
int64_t base;
|
|
|
|
|
|
|
|
switch (whence)
|
|
|
|
{
|
|
|
|
default:
|
|
|
|
case RW_SEEK_SET :
|
|
|
|
base = 0;
|
|
|
|
break;
|
|
|
|
case RW_SEEK_CUR :
|
|
|
|
base = PHYSFS_tell(f);
|
|
|
|
break;
|
|
|
|
case RW_SEEK_END :
|
|
|
|
base = PHYSFS_fileLength(f);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
int result = PHYSFS_seek(f, base + offset);
|
|
|
|
|
|
|
|
return (result != 0) ? PHYSFS_tell(f) : -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static size_t SDL_RWopsRead(SDL_RWops *ops, void *buffer, size_t size, size_t maxnum)
|
|
|
|
{
|
|
|
|
PHYSFS_File *f = sdlPHYS(ops);
|
|
|
|
|
|
|
|
if (!f)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
PHYSFS_sint64 result = PHYSFS_readBytes(f, buffer, size*maxnum);
|
|
|
|
|
|
|
|
return (result != -1) ? (result / size) : 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static size_t SDL_RWopsWrite(SDL_RWops *ops, const void *buffer, size_t size, size_t num)
|
|
|
|
{
|
|
|
|
PHYSFS_File *f = sdlPHYS(ops);
|
|
|
|
|
|
|
|
if (!f)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
PHYSFS_sint64 result = PHYSFS_writeBytes(f, buffer, size*num);
|
|
|
|
|
|
|
|
return (result != -1) ? (result / size) : 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int SDL_RWopsClose(SDL_RWops *ops)
|
|
|
|
{
|
|
|
|
PHYSFS_File *f = sdlPHYS(ops);
|
|
|
|
|
|
|
|
if (!f)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
int result = PHYSFS_close(f);
|
|
|
|
|
|
|
|
f = 0;
|
|
|
|
|
|
|
|
return (result != 0) ? 0 : -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int SDL_RWopsCloseFree(SDL_RWops *ops)
|
|
|
|
{
|
|
|
|
int result = SDL_RWopsClose(ops);
|
|
|
|
|
|
|
|
SDL_FreeRW(ops);
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
const Uint32 SDL_RWOPS_PHYSFS = SDL_RWOPS_UNKNOWN+10;
|
|
|
|
|
2013-11-30 12:00:11 +01:00
|
|
|
struct FileSystemPrivate
|
2013-09-01 16:27:21 +02:00
|
|
|
{
|
2014-01-16 04:19:22 +01:00
|
|
|
/* Maps: lower case filename, To: actual (mixed case) filename.
|
|
|
|
* This is for compatibility with games that take Windows'
|
|
|
|
* case insensitivity for granted */
|
2013-12-11 20:46:54 +01:00
|
|
|
BoostHash<std::string, std::string> pathCache;
|
2014-01-16 04:19:22 +01:00
|
|
|
bool havePathCache;
|
2013-09-01 16:27:21 +02:00
|
|
|
|
2014-01-01 07:07:18 +01:00
|
|
|
std::vector<std::string> extensions[FileSystem::Undefined+1];
|
|
|
|
|
2013-11-30 12:00:11 +01:00
|
|
|
/* Attempt to locate an extension string in a filename.
|
|
|
|
* Either a pointer into the input string pointing at the
|
|
|
|
* extension, or null is returned */
|
|
|
|
const char *findExt(const char *filename)
|
2013-09-01 16:27:21 +02:00
|
|
|
{
|
2014-01-16 04:19:22 +01:00
|
|
|
size_t len;
|
2013-09-01 16:27:21 +02:00
|
|
|
|
2013-11-30 12:00:11 +01:00
|
|
|
for (len = strlen(filename); len > 0; --len)
|
|
|
|
{
|
|
|
|
if (filename[len] == '/')
|
|
|
|
return 0;
|
2013-09-01 16:27:21 +02:00
|
|
|
|
2013-11-30 12:00:11 +01:00
|
|
|
if (filename[len] == '.')
|
|
|
|
return &filename[len+1];
|
|
|
|
}
|
2013-09-01 16:27:21 +02:00
|
|
|
|
2013-11-30 12:00:11 +01:00
|
|
|
return 0;
|
|
|
|
}
|
2013-09-01 16:27:21 +02:00
|
|
|
|
2014-01-16 04:19:22 +01:00
|
|
|
/* Complete filename via regular physfs lookup */
|
|
|
|
bool completeFilenameReg(const char *filename,
|
|
|
|
FileSystem::FileType type,
|
|
|
|
char *outBuffer,
|
|
|
|
size_t outN,
|
|
|
|
const char **foundExt)
|
2013-09-01 16:27:21 +02:00
|
|
|
{
|
2014-01-16 04:19:22 +01:00
|
|
|
/* Try supplementing extensions to find an existing path */
|
|
|
|
const std::vector<std::string> &extList = extensions[type];
|
|
|
|
|
|
|
|
for (size_t i = 0; i < extList.size(); ++i)
|
2014-01-01 07:07:18 +01:00
|
|
|
{
|
2014-01-16 04:19:22 +01:00
|
|
|
const char *ext = extList[i].c_str();
|
2014-01-01 07:07:18 +01:00
|
|
|
|
2014-01-16 04:19:22 +01:00
|
|
|
snprintf(outBuffer, outN, "%s.%s", filename, ext);
|
2014-01-01 07:07:18 +01:00
|
|
|
|
2014-01-16 04:19:22 +01:00
|
|
|
if (PHYSFS_exists(outBuffer))
|
2014-01-01 07:07:18 +01:00
|
|
|
{
|
2014-01-16 04:19:22 +01:00
|
|
|
if (foundExt)
|
|
|
|
*foundExt = ext;
|
2014-01-01 07:07:18 +01:00
|
|
|
|
2014-01-16 04:19:22 +01:00
|
|
|
return true;
|
2014-01-01 07:07:18 +01:00
|
|
|
}
|
2014-01-16 04:19:22 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Doing the check without supplemented extension
|
|
|
|
* fits the usage pattern of RMXP games */
|
|
|
|
if (PHYSFS_exists(filename))
|
|
|
|
{
|
|
|
|
strncpy(outBuffer, filename, outN);
|
2014-01-01 07:07:18 +01:00
|
|
|
|
|
|
|
if (foundExt)
|
2014-01-16 04:19:22 +01:00
|
|
|
*foundExt = findExt(filename);
|
2014-01-01 07:07:18 +01:00
|
|
|
|
2014-01-16 04:19:22 +01:00
|
|
|
return true;
|
2014-01-01 07:07:18 +01:00
|
|
|
}
|
|
|
|
|
2014-01-16 04:19:22 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Complete filename via path cache */
|
|
|
|
bool completeFilenamePC(const char *filename,
|
|
|
|
FileSystem::FileType type,
|
|
|
|
char *outBuffer,
|
|
|
|
size_t outN,
|
|
|
|
const char **foundExt)
|
|
|
|
{
|
2013-09-30 21:19:43 +02:00
|
|
|
size_t i;
|
2014-01-16 04:19:22 +01:00
|
|
|
char lowCase[512];
|
2013-09-30 21:19:43 +02:00
|
|
|
|
2014-01-16 04:19:22 +01:00
|
|
|
for (i = 0; i < sizeof(lowCase)-1 && filename[i]; ++i)
|
|
|
|
lowCase[i] = tolower(filename[i]);
|
2013-09-30 21:19:43 +02:00
|
|
|
|
2014-01-16 04:19:22 +01:00
|
|
|
lowCase[i] = '\0';
|
2013-09-30 21:19:43 +02:00
|
|
|
|
2014-01-16 04:19:22 +01:00
|
|
|
std::string key;
|
2013-09-30 21:19:43 +02:00
|
|
|
|
2014-01-16 04:19:22 +01:00
|
|
|
const std::vector<std::string> &extList = extensions[type];
|
|
|
|
|
|
|
|
for (size_t i = 0; i < extList.size(); ++i)
|
2013-11-30 12:00:11 +01:00
|
|
|
{
|
2014-01-16 04:19:22 +01:00
|
|
|
const char *ext = extList[i].c_str();
|
|
|
|
|
|
|
|
snprintf(outBuffer, outN, "%s.%s", lowCase, ext);
|
|
|
|
key = outBuffer;
|
2013-11-30 12:00:11 +01:00
|
|
|
|
2014-01-16 04:19:22 +01:00
|
|
|
if (pathCache.contains(key))
|
|
|
|
{
|
|
|
|
strncpy(outBuffer, pathCache[key].c_str(), outN);
|
|
|
|
|
|
|
|
if (foundExt)
|
|
|
|
*foundExt = ext;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
2013-11-30 12:00:11 +01:00
|
|
|
}
|
2013-09-30 21:19:43 +02:00
|
|
|
|
2014-01-16 04:19:22 +01:00
|
|
|
key = lowCase;
|
2013-09-30 21:19:43 +02:00
|
|
|
|
2014-01-16 04:19:22 +01:00
|
|
|
if (pathCache.contains(key))
|
2013-09-01 16:27:21 +02:00
|
|
|
{
|
2014-01-16 04:19:22 +01:00
|
|
|
strncpy(outBuffer, pathCache[key].c_str(), outN);
|
2013-11-30 12:00:11 +01:00
|
|
|
|
2014-01-16 04:19:22 +01:00
|
|
|
if (foundExt)
|
|
|
|
*foundExt = findExt(filename);
|
2013-11-30 12:00:11 +01:00
|
|
|
|
2014-01-16 04:19:22 +01:00
|
|
|
return true;
|
2013-09-01 16:27:21 +02:00
|
|
|
}
|
2013-09-30 21:19:43 +02:00
|
|
|
|
2014-01-16 04:19:22 +01:00
|
|
|
return false;
|
|
|
|
}
|
2013-11-30 12:00:11 +01:00
|
|
|
|
2014-01-16 04:19:22 +01:00
|
|
|
/* Try to complete 'filename' with file extensions
|
|
|
|
* based on 'type'. If no combination could be found,
|
|
|
|
* returns false, and 'foundExt' is untouched */
|
|
|
|
bool completeFileName(const char *filename,
|
|
|
|
FileSystem::FileType type,
|
|
|
|
char *outBuffer,
|
|
|
|
size_t outN,
|
|
|
|
const char **foundExt)
|
|
|
|
{
|
|
|
|
if (havePathCache)
|
|
|
|
return completeFilenamePC(filename, type, outBuffer, outN, foundExt);
|
|
|
|
else
|
|
|
|
return completeFilenameReg(filename, type, outBuffer, outN, foundExt);
|
2013-09-01 16:27:21 +02:00
|
|
|
}
|
|
|
|
|
2014-01-16 04:19:22 +01:00
|
|
|
PHYSFS_File *openReadHandle(const char *filename,
|
|
|
|
FileSystem::FileType type,
|
|
|
|
const char **foundExt)
|
2013-09-30 21:19:43 +02:00
|
|
|
{
|
2014-01-16 04:19:22 +01:00
|
|
|
char found[512];
|
2013-09-30 21:19:43 +02:00
|
|
|
|
2014-01-16 04:19:22 +01:00
|
|
|
if (!completeFileName(filename, type, found, sizeof(found), foundExt))
|
2014-01-15 01:47:16 +01:00
|
|
|
throw Exception(Exception::NoFileError, "%s", filename);
|
2013-09-30 21:19:43 +02:00
|
|
|
|
2014-01-16 04:19:22 +01:00
|
|
|
PHYSFS_File *handle = PHYSFS_openRead(found);
|
|
|
|
|
2013-09-30 21:19:43 +02:00
|
|
|
if (!handle)
|
|
|
|
throw Exception(Exception::PHYSFSError, "PhysFS: %s", PHYSFS_getLastError());
|
|
|
|
|
|
|
|
return handle;
|
|
|
|
}
|
Font: Overhaul font asset discovery
Previously, any font names requested by RGSS would be translated
directly to filenames by lowercasing and replacing spaces with
underscores (and finally doing some extension substitution).
To make this whole thing work smoother as well as get closer to
how font discovery is done in VX, we now scan the "Fonts/" folder
at startup and index all present font assets by their family name;
now, if an "Open Sans" font is present in "Fonts/", it will be
used regardless of filename.
Font assets with "Regular" style are preferred, but in their
absence, mkxp will make use of any other style it can find for
the respective family. This is not the exact same behavior as
VX, but it should cover 95% of use cases.
Previously, one could substitute fonts via filenames, ie. to
substitute "Arial" with "Open Sans", one would just rename
"OpenSans.ttf" to "arial.ttf" and put it in "Fonts/". With the
above change, this is no longer possible. As an alternative, one
can now explicitly specify font family substitutions via mkxp.conf;
eg. for the above case, one would add
fontSub=Arial>Open Sans
to the configuration file. Multiple such rules can be specified.
In the process, I also added the ability to provide
'Font.(default_)name' with an array of font families to search
for the first existing one instead of a plain string.
This makes the behavior closer to RMXP; however, it doesn't
work 100% the same: when a reference to the 'Font.name' array is
held and additional strings are added to it without re-assignig
the array to 'Font.name', those will be ignored.
2014-04-11 13:37:14 +02:00
|
|
|
|
|
|
|
void initReadOps(PHYSFS_File *handle,
|
|
|
|
SDL_RWops &ops,
|
|
|
|
bool freeOnClose)
|
|
|
|
{
|
|
|
|
ops.size = SDL_RWopsSize;
|
|
|
|
ops.seek = SDL_RWopsSeek;
|
|
|
|
ops.read = SDL_RWopsRead;
|
|
|
|
ops.write = SDL_RWopsWrite;
|
|
|
|
|
|
|
|
if (freeOnClose)
|
|
|
|
ops.close = SDL_RWopsCloseFree;
|
|
|
|
else
|
|
|
|
ops.close = SDL_RWopsClose;
|
|
|
|
|
|
|
|
ops.type = SDL_RWOPS_PHYSFS;
|
|
|
|
ops.hidden.unknown.data1 = handle;
|
|
|
|
}
|
2013-09-30 21:19:43 +02:00
|
|
|
};
|
|
|
|
|
2013-10-20 22:38:46 +02:00
|
|
|
FileSystem::FileSystem(const char *argv0,
|
|
|
|
bool allowSymlinks)
|
2013-09-30 21:19:43 +02:00
|
|
|
{
|
|
|
|
p = new FileSystemPrivate;
|
|
|
|
|
2014-01-01 07:07:18 +01:00
|
|
|
p->havePathCache = false;
|
|
|
|
|
2013-11-30 12:00:11 +01:00
|
|
|
/* Image extensions */
|
2013-12-26 20:18:33 +01:00
|
|
|
p->extensions[Image].push_back("jpg");
|
|
|
|
p->extensions[Image].push_back("png");
|
2013-11-30 12:00:11 +01:00
|
|
|
|
|
|
|
/* Audio extensions */
|
|
|
|
const Sound_DecoderInfo **di;
|
|
|
|
for (di = Sound_AvailableDecoders(); *di; ++di)
|
|
|
|
{
|
|
|
|
const char **ext;
|
|
|
|
for (ext = (*di)->extensions; *ext; ++ext)
|
|
|
|
{
|
2013-12-15 10:19:22 +01:00
|
|
|
/* All reported extensions are uppercase,
|
2013-11-30 12:00:11 +01:00
|
|
|
* so we need to hammer them down first */
|
|
|
|
char buf[16];
|
|
|
|
for (size_t i = 0; i < sizeof(buf); ++i)
|
|
|
|
{
|
|
|
|
buf[i] = tolower((*ext)[i]);
|
|
|
|
|
|
|
|
if (!buf[i])
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2013-12-26 20:18:33 +01:00
|
|
|
p->extensions[Audio].push_back(buf);
|
2013-11-30 12:00:11 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Font extensions */
|
2013-12-26 20:18:33 +01:00
|
|
|
p->extensions[Font].push_back("ttf");
|
|
|
|
p->extensions[Font].push_back("otf");
|
2013-11-30 12:00:11 +01:00
|
|
|
|
2013-09-30 21:19:43 +02:00
|
|
|
PHYSFS_init(argv0);
|
|
|
|
PHYSFS_registerArchiver(&RGSS_Archiver);
|
2013-10-20 22:38:46 +02:00
|
|
|
|
|
|
|
if (allowSymlinks)
|
|
|
|
PHYSFS_permitSymbolicLinks(1);
|
2013-09-30 21:19:43 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
FileSystem::~FileSystem()
|
|
|
|
{
|
|
|
|
delete p;
|
|
|
|
|
|
|
|
if (PHYSFS_deinit() == 0)
|
2013-12-11 20:46:54 +01:00
|
|
|
Debug() << "PhyFS failed to deinit.";
|
2013-09-30 21:19:43 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void FileSystem::addPath(const char *path)
|
|
|
|
{
|
|
|
|
PHYSFS_mount(path, 0, 1);
|
2013-09-01 16:27:21 +02:00
|
|
|
}
|
|
|
|
|
2013-09-30 21:19:43 +02:00
|
|
|
static void cacheEnumCB(void *d, const char *origdir,
|
|
|
|
const char *fname)
|
2013-09-01 16:27:21 +02:00
|
|
|
{
|
2013-09-30 21:19:43 +02:00
|
|
|
FileSystemPrivate *p = static_cast<FileSystemPrivate*>(d);
|
|
|
|
|
|
|
|
char buf[512];
|
|
|
|
|
2014-01-15 01:51:56 +01:00
|
|
|
if (*origdir == '\0')
|
2013-09-30 21:19:43 +02:00
|
|
|
strncpy(buf, fname, sizeof(buf));
|
2014-01-15 01:51:56 +01:00
|
|
|
else
|
|
|
|
snprintf(buf, sizeof(buf), "%s/%s", origdir, fname);
|
|
|
|
|
|
|
|
char *ptr = buf;
|
|
|
|
|
|
|
|
/* Trim leading slash */
|
|
|
|
if (*ptr == '/')
|
|
|
|
++ptr;
|
2013-09-30 21:19:43 +02:00
|
|
|
|
2014-01-15 01:51:56 +01:00
|
|
|
std::string mixedCase(ptr);
|
2013-09-01 16:27:21 +02:00
|
|
|
|
2013-11-01 08:24:14 +01:00
|
|
|
for (char *p = buf; *p; ++p)
|
2013-09-30 21:19:43 +02:00
|
|
|
*p = tolower(*p);
|
2013-09-01 16:27:21 +02:00
|
|
|
|
2014-01-15 01:51:56 +01:00
|
|
|
std::string lowerCase(ptr);
|
2013-11-01 08:24:14 +01:00
|
|
|
|
2013-09-30 21:19:43 +02:00
|
|
|
p->pathCache.insert(lowerCase, mixedCase);
|
2013-09-01 16:27:21 +02:00
|
|
|
|
2013-12-11 20:46:54 +01:00
|
|
|
PHYSFS_enumerateFilesCallback(mixedCase.c_str(), cacheEnumCB, p);
|
2013-09-30 21:19:43 +02:00
|
|
|
}
|
2013-09-01 16:27:21 +02:00
|
|
|
|
2013-09-30 21:19:43 +02:00
|
|
|
void FileSystem::createPathCache()
|
|
|
|
{
|
2014-01-15 01:51:56 +01:00
|
|
|
PHYSFS_enumerateFilesCallback("", cacheEnumCB, p);
|
2014-01-01 07:07:18 +01:00
|
|
|
|
|
|
|
p->havePathCache = true;
|
2013-09-01 16:27:21 +02:00
|
|
|
}
|
|
|
|
|
Font: Overhaul font asset discovery
Previously, any font names requested by RGSS would be translated
directly to filenames by lowercasing and replacing spaces with
underscores (and finally doing some extension substitution).
To make this whole thing work smoother as well as get closer to
how font discovery is done in VX, we now scan the "Fonts/" folder
at startup and index all present font assets by their family name;
now, if an "Open Sans" font is present in "Fonts/", it will be
used regardless of filename.
Font assets with "Regular" style are preferred, but in their
absence, mkxp will make use of any other style it can find for
the respective family. This is not the exact same behavior as
VX, but it should cover 95% of use cases.
Previously, one could substitute fonts via filenames, ie. to
substitute "Arial" with "Open Sans", one would just rename
"OpenSans.ttf" to "arial.ttf" and put it in "Fonts/". With the
above change, this is no longer possible. As an alternative, one
can now explicitly specify font family substitutions via mkxp.conf;
eg. for the above case, one would add
fontSub=Arial>Open Sans
to the configuration file. Multiple such rules can be specified.
In the process, I also added the ability to provide
'Font.(default_)name' with an array of font families to search
for the first existing one instead of a plain string.
This makes the behavior closer to RMXP; however, it doesn't
work 100% the same: when a reference to the 'Font.name' array is
held and additional strings are added to it without re-assignig
the array to 'Font.name', those will be ignored.
2014-04-11 13:37:14 +02:00
|
|
|
static void strToLower(std::string &str)
|
2013-09-01 16:27:21 +02:00
|
|
|
{
|
Font: Overhaul font asset discovery
Previously, any font names requested by RGSS would be translated
directly to filenames by lowercasing and replacing spaces with
underscores (and finally doing some extension substitution).
To make this whole thing work smoother as well as get closer to
how font discovery is done in VX, we now scan the "Fonts/" folder
at startup and index all present font assets by their family name;
now, if an "Open Sans" font is present in "Fonts/", it will be
used regardless of filename.
Font assets with "Regular" style are preferred, but in their
absence, mkxp will make use of any other style it can find for
the respective family. This is not the exact same behavior as
VX, but it should cover 95% of use cases.
Previously, one could substitute fonts via filenames, ie. to
substitute "Arial" with "Open Sans", one would just rename
"OpenSans.ttf" to "arial.ttf" and put it in "Fonts/". With the
above change, this is no longer possible. As an alternative, one
can now explicitly specify font family substitutions via mkxp.conf;
eg. for the above case, one would add
fontSub=Arial>Open Sans
to the configuration file. Multiple such rules can be specified.
In the process, I also added the ability to provide
'Font.(default_)name' with an array of font families to search
for the first existing one instead of a plain string.
This makes the behavior closer to RMXP; however, it doesn't
work 100% the same: when a reference to the 'Font.name' array is
held and additional strings are added to it without re-assignig
the array to 'Font.name', those will be ignored.
2014-04-11 13:37:14 +02:00
|
|
|
for (size_t i = 0; i < str.size(); ++i)
|
|
|
|
str[i] = tolower(str[i]);
|
2013-09-01 16:27:21 +02:00
|
|
|
}
|
|
|
|
|
Font: Overhaul font asset discovery
Previously, any font names requested by RGSS would be translated
directly to filenames by lowercasing and replacing spaces with
underscores (and finally doing some extension substitution).
To make this whole thing work smoother as well as get closer to
how font discovery is done in VX, we now scan the "Fonts/" folder
at startup and index all present font assets by their family name;
now, if an "Open Sans" font is present in "Fonts/", it will be
used regardless of filename.
Font assets with "Regular" style are preferred, but in their
absence, mkxp will make use of any other style it can find for
the respective family. This is not the exact same behavior as
VX, but it should cover 95% of use cases.
Previously, one could substitute fonts via filenames, ie. to
substitute "Arial" with "Open Sans", one would just rename
"OpenSans.ttf" to "arial.ttf" and put it in "Fonts/". With the
above change, this is no longer possible. As an alternative, one
can now explicitly specify font family substitutions via mkxp.conf;
eg. for the above case, one would add
fontSub=Arial>Open Sans
to the configuration file. Multiple such rules can be specified.
In the process, I also added the ability to provide
'Font.(default_)name' with an array of font families to search
for the first existing one instead of a plain string.
This makes the behavior closer to RMXP; however, it doesn't
work 100% the same: when a reference to the 'Font.name' array is
held and additional strings are added to it without re-assignig
the array to 'Font.name', those will be ignored.
2014-04-11 13:37:14 +02:00
|
|
|
struct FontSetsCBData
|
2013-09-01 16:27:21 +02:00
|
|
|
{
|
Font: Overhaul font asset discovery
Previously, any font names requested by RGSS would be translated
directly to filenames by lowercasing and replacing spaces with
underscores (and finally doing some extension substitution).
To make this whole thing work smoother as well as get closer to
how font discovery is done in VX, we now scan the "Fonts/" folder
at startup and index all present font assets by their family name;
now, if an "Open Sans" font is present in "Fonts/", it will be
used regardless of filename.
Font assets with "Regular" style are preferred, but in their
absence, mkxp will make use of any other style it can find for
the respective family. This is not the exact same behavior as
VX, but it should cover 95% of use cases.
Previously, one could substitute fonts via filenames, ie. to
substitute "Arial" with "Open Sans", one would just rename
"OpenSans.ttf" to "arial.ttf" and put it in "Fonts/". With the
above change, this is no longer possible. As an alternative, one
can now explicitly specify font family substitutions via mkxp.conf;
eg. for the above case, one would add
fontSub=Arial>Open Sans
to the configuration file. Multiple such rules can be specified.
In the process, I also added the ability to provide
'Font.(default_)name' with an array of font families to search
for the first existing one instead of a plain string.
This makes the behavior closer to RMXP; however, it doesn't
work 100% the same: when a reference to the 'Font.name' array is
held and additional strings are added to it without re-assignig
the array to 'Font.name', those will be ignored.
2014-04-11 13:37:14 +02:00
|
|
|
FileSystemPrivate *p;
|
|
|
|
SharedFontState *sfs;
|
|
|
|
};
|
2013-09-01 16:27:21 +02:00
|
|
|
|
Font: Overhaul font asset discovery
Previously, any font names requested by RGSS would be translated
directly to filenames by lowercasing and replacing spaces with
underscores (and finally doing some extension substitution).
To make this whole thing work smoother as well as get closer to
how font discovery is done in VX, we now scan the "Fonts/" folder
at startup and index all present font assets by their family name;
now, if an "Open Sans" font is present in "Fonts/", it will be
used regardless of filename.
Font assets with "Regular" style are preferred, but in their
absence, mkxp will make use of any other style it can find for
the respective family. This is not the exact same behavior as
VX, but it should cover 95% of use cases.
Previously, one could substitute fonts via filenames, ie. to
substitute "Arial" with "Open Sans", one would just rename
"OpenSans.ttf" to "arial.ttf" and put it in "Fonts/". With the
above change, this is no longer possible. As an alternative, one
can now explicitly specify font family substitutions via mkxp.conf;
eg. for the above case, one would add
fontSub=Arial>Open Sans
to the configuration file. Multiple such rules can be specified.
In the process, I also added the ability to provide
'Font.(default_)name' with an array of font families to search
for the first existing one instead of a plain string.
This makes the behavior closer to RMXP; however, it doesn't
work 100% the same: when a reference to the 'Font.name' array is
held and additional strings are added to it without re-assignig
the array to 'Font.name', those will be ignored.
2014-04-11 13:37:14 +02:00
|
|
|
static void fontSetEnumCB(void *data, const char *,
|
|
|
|
const char *fname)
|
2013-09-01 16:27:21 +02:00
|
|
|
{
|
Font: Overhaul font asset discovery
Previously, any font names requested by RGSS would be translated
directly to filenames by lowercasing and replacing spaces with
underscores (and finally doing some extension substitution).
To make this whole thing work smoother as well as get closer to
how font discovery is done in VX, we now scan the "Fonts/" folder
at startup and index all present font assets by their family name;
now, if an "Open Sans" font is present in "Fonts/", it will be
used regardless of filename.
Font assets with "Regular" style are preferred, but in their
absence, mkxp will make use of any other style it can find for
the respective family. This is not the exact same behavior as
VX, but it should cover 95% of use cases.
Previously, one could substitute fonts via filenames, ie. to
substitute "Arial" with "Open Sans", one would just rename
"OpenSans.ttf" to "arial.ttf" and put it in "Fonts/". With the
above change, this is no longer possible. As an alternative, one
can now explicitly specify font family substitutions via mkxp.conf;
eg. for the above case, one would add
fontSub=Arial>Open Sans
to the configuration file. Multiple such rules can be specified.
In the process, I also added the ability to provide
'Font.(default_)name' with an array of font families to search
for the first existing one instead of a plain string.
This makes the behavior closer to RMXP; however, it doesn't
work 100% the same: when a reference to the 'Font.name' array is
held and additional strings are added to it without re-assignig
the array to 'Font.name', those will be ignored.
2014-04-11 13:37:14 +02:00
|
|
|
FontSetsCBData *d = static_cast<FontSetsCBData*>(data);
|
|
|
|
FileSystemPrivate *p = d->p;
|
2013-09-01 16:27:21 +02:00
|
|
|
|
Font: Overhaul font asset discovery
Previously, any font names requested by RGSS would be translated
directly to filenames by lowercasing and replacing spaces with
underscores (and finally doing some extension substitution).
To make this whole thing work smoother as well as get closer to
how font discovery is done in VX, we now scan the "Fonts/" folder
at startup and index all present font assets by their family name;
now, if an "Open Sans" font is present in "Fonts/", it will be
used regardless of filename.
Font assets with "Regular" style are preferred, but in their
absence, mkxp will make use of any other style it can find for
the respective family. This is not the exact same behavior as
VX, but it should cover 95% of use cases.
Previously, one could substitute fonts via filenames, ie. to
substitute "Arial" with "Open Sans", one would just rename
"OpenSans.ttf" to "arial.ttf" and put it in "Fonts/". With the
above change, this is no longer possible. As an alternative, one
can now explicitly specify font family substitutions via mkxp.conf;
eg. for the above case, one would add
fontSub=Arial>Open Sans
to the configuration file. Multiple such rules can be specified.
In the process, I also added the ability to provide
'Font.(default_)name' with an array of font families to search
for the first existing one instead of a plain string.
This makes the behavior closer to RMXP; however, it doesn't
work 100% the same: when a reference to the 'Font.name' array is
held and additional strings are added to it without re-assignig
the array to 'Font.name', those will be ignored.
2014-04-11 13:37:14 +02:00
|
|
|
/* Only consider filenames with font extensions */
|
|
|
|
const char *ext = p->findExt(fname);
|
2013-09-01 16:27:21 +02:00
|
|
|
|
Font: Overhaul font asset discovery
Previously, any font names requested by RGSS would be translated
directly to filenames by lowercasing and replacing spaces with
underscores (and finally doing some extension substitution).
To make this whole thing work smoother as well as get closer to
how font discovery is done in VX, we now scan the "Fonts/" folder
at startup and index all present font assets by their family name;
now, if an "Open Sans" font is present in "Fonts/", it will be
used regardless of filename.
Font assets with "Regular" style are preferred, but in their
absence, mkxp will make use of any other style it can find for
the respective family. This is not the exact same behavior as
VX, but it should cover 95% of use cases.
Previously, one could substitute fonts via filenames, ie. to
substitute "Arial" with "Open Sans", one would just rename
"OpenSans.ttf" to "arial.ttf" and put it in "Fonts/". With the
above change, this is no longer possible. As an alternative, one
can now explicitly specify font family substitutions via mkxp.conf;
eg. for the above case, one would add
fontSub=Arial>Open Sans
to the configuration file. Multiple such rules can be specified.
In the process, I also added the ability to provide
'Font.(default_)name' with an array of font families to search
for the first existing one instead of a plain string.
This makes the behavior closer to RMXP; however, it doesn't
work 100% the same: when a reference to the 'Font.name' array is
held and additional strings are added to it without re-assignig
the array to 'Font.name', those will be ignored.
2014-04-11 13:37:14 +02:00
|
|
|
if (!ext)
|
|
|
|
return;
|
2013-09-01 16:27:21 +02:00
|
|
|
|
Font: Overhaul font asset discovery
Previously, any font names requested by RGSS would be translated
directly to filenames by lowercasing and replacing spaces with
underscores (and finally doing some extension substitution).
To make this whole thing work smoother as well as get closer to
how font discovery is done in VX, we now scan the "Fonts/" folder
at startup and index all present font assets by their family name;
now, if an "Open Sans" font is present in "Fonts/", it will be
used regardless of filename.
Font assets with "Regular" style are preferred, but in their
absence, mkxp will make use of any other style it can find for
the respective family. This is not the exact same behavior as
VX, but it should cover 95% of use cases.
Previously, one could substitute fonts via filenames, ie. to
substitute "Arial" with "Open Sans", one would just rename
"OpenSans.ttf" to "arial.ttf" and put it in "Fonts/". With the
above change, this is no longer possible. As an alternative, one
can now explicitly specify font family substitutions via mkxp.conf;
eg. for the above case, one would add
fontSub=Arial>Open Sans
to the configuration file. Multiple such rules can be specified.
In the process, I also added the ability to provide
'Font.(default_)name' with an array of font families to search
for the first existing one instead of a plain string.
This makes the behavior closer to RMXP; however, it doesn't
work 100% the same: when a reference to the 'Font.name' array is
held and additional strings are added to it without re-assignig
the array to 'Font.name', those will be ignored.
2014-04-11 13:37:14 +02:00
|
|
|
std::string lower(ext);
|
|
|
|
strToLower(lower);
|
2013-09-01 16:27:21 +02:00
|
|
|
|
Font: Overhaul font asset discovery
Previously, any font names requested by RGSS would be translated
directly to filenames by lowercasing and replacing spaces with
underscores (and finally doing some extension substitution).
To make this whole thing work smoother as well as get closer to
how font discovery is done in VX, we now scan the "Fonts/" folder
at startup and index all present font assets by their family name;
now, if an "Open Sans" font is present in "Fonts/", it will be
used regardless of filename.
Font assets with "Regular" style are preferred, but in their
absence, mkxp will make use of any other style it can find for
the respective family. This is not the exact same behavior as
VX, but it should cover 95% of use cases.
Previously, one could substitute fonts via filenames, ie. to
substitute "Arial" with "Open Sans", one would just rename
"OpenSans.ttf" to "arial.ttf" and put it in "Fonts/". With the
above change, this is no longer possible. As an alternative, one
can now explicitly specify font family substitutions via mkxp.conf;
eg. for the above case, one would add
fontSub=Arial>Open Sans
to the configuration file. Multiple such rules can be specified.
In the process, I also added the ability to provide
'Font.(default_)name' with an array of font families to search
for the first existing one instead of a plain string.
This makes the behavior closer to RMXP; however, it doesn't
work 100% the same: when a reference to the 'Font.name' array is
held and additional strings are added to it without re-assignig
the array to 'Font.name', those will be ignored.
2014-04-11 13:37:14 +02:00
|
|
|
if (!contains(p->extensions[FileSystem::Font], lower))
|
|
|
|
return;
|
2013-09-01 16:27:21 +02:00
|
|
|
|
Font: Overhaul font asset discovery
Previously, any font names requested by RGSS would be translated
directly to filenames by lowercasing and replacing spaces with
underscores (and finally doing some extension substitution).
To make this whole thing work smoother as well as get closer to
how font discovery is done in VX, we now scan the "Fonts/" folder
at startup and index all present font assets by their family name;
now, if an "Open Sans" font is present in "Fonts/", it will be
used regardless of filename.
Font assets with "Regular" style are preferred, but in their
absence, mkxp will make use of any other style it can find for
the respective family. This is not the exact same behavior as
VX, but it should cover 95% of use cases.
Previously, one could substitute fonts via filenames, ie. to
substitute "Arial" with "Open Sans", one would just rename
"OpenSans.ttf" to "arial.ttf" and put it in "Fonts/". With the
above change, this is no longer possible. As an alternative, one
can now explicitly specify font family substitutions via mkxp.conf;
eg. for the above case, one would add
fontSub=Arial>Open Sans
to the configuration file. Multiple such rules can be specified.
In the process, I also added the ability to provide
'Font.(default_)name' with an array of font families to search
for the first existing one instead of a plain string.
This makes the behavior closer to RMXP; however, it doesn't
work 100% the same: when a reference to the 'Font.name' array is
held and additional strings are added to it without re-assignig
the array to 'Font.name', those will be ignored.
2014-04-11 13:37:14 +02:00
|
|
|
std::string filename("Fonts/");
|
|
|
|
filename += fname;
|
2013-09-01 16:27:21 +02:00
|
|
|
|
Font: Overhaul font asset discovery
Previously, any font names requested by RGSS would be translated
directly to filenames by lowercasing and replacing spaces with
underscores (and finally doing some extension substitution).
To make this whole thing work smoother as well as get closer to
how font discovery is done in VX, we now scan the "Fonts/" folder
at startup and index all present font assets by their family name;
now, if an "Open Sans" font is present in "Fonts/", it will be
used regardless of filename.
Font assets with "Regular" style are preferred, but in their
absence, mkxp will make use of any other style it can find for
the respective family. This is not the exact same behavior as
VX, but it should cover 95% of use cases.
Previously, one could substitute fonts via filenames, ie. to
substitute "Arial" with "Open Sans", one would just rename
"OpenSans.ttf" to "arial.ttf" and put it in "Fonts/". With the
above change, this is no longer possible. As an alternative, one
can now explicitly specify font family substitutions via mkxp.conf;
eg. for the above case, one would add
fontSub=Arial>Open Sans
to the configuration file. Multiple such rules can be specified.
In the process, I also added the ability to provide
'Font.(default_)name' with an array of font families to search
for the first existing one instead of a plain string.
This makes the behavior closer to RMXP; however, it doesn't
work 100% the same: when a reference to the 'Font.name' array is
held and additional strings are added to it without re-assignig
the array to 'Font.name', those will be ignored.
2014-04-11 13:37:14 +02:00
|
|
|
PHYSFS_File *handle = PHYSFS_openRead(filename.c_str());
|
2013-09-01 16:27:21 +02:00
|
|
|
|
Font: Overhaul font asset discovery
Previously, any font names requested by RGSS would be translated
directly to filenames by lowercasing and replacing spaces with
underscores (and finally doing some extension substitution).
To make this whole thing work smoother as well as get closer to
how font discovery is done in VX, we now scan the "Fonts/" folder
at startup and index all present font assets by their family name;
now, if an "Open Sans" font is present in "Fonts/", it will be
used regardless of filename.
Font assets with "Regular" style are preferred, but in their
absence, mkxp will make use of any other style it can find for
the respective family. This is not the exact same behavior as
VX, but it should cover 95% of use cases.
Previously, one could substitute fonts via filenames, ie. to
substitute "Arial" with "Open Sans", one would just rename
"OpenSans.ttf" to "arial.ttf" and put it in "Fonts/". With the
above change, this is no longer possible. As an alternative, one
can now explicitly specify font family substitutions via mkxp.conf;
eg. for the above case, one would add
fontSub=Arial>Open Sans
to the configuration file. Multiple such rules can be specified.
In the process, I also added the ability to provide
'Font.(default_)name' with an array of font families to search
for the first existing one instead of a plain string.
This makes the behavior closer to RMXP; however, it doesn't
work 100% the same: when a reference to the 'Font.name' array is
held and additional strings are added to it without re-assignig
the array to 'Font.name', those will be ignored.
2014-04-11 13:37:14 +02:00
|
|
|
if (!handle)
|
|
|
|
return;
|
2013-09-01 16:27:21 +02:00
|
|
|
|
Font: Overhaul font asset discovery
Previously, any font names requested by RGSS would be translated
directly to filenames by lowercasing and replacing spaces with
underscores (and finally doing some extension substitution).
To make this whole thing work smoother as well as get closer to
how font discovery is done in VX, we now scan the "Fonts/" folder
at startup and index all present font assets by their family name;
now, if an "Open Sans" font is present in "Fonts/", it will be
used regardless of filename.
Font assets with "Regular" style are preferred, but in their
absence, mkxp will make use of any other style it can find for
the respective family. This is not the exact same behavior as
VX, but it should cover 95% of use cases.
Previously, one could substitute fonts via filenames, ie. to
substitute "Arial" with "Open Sans", one would just rename
"OpenSans.ttf" to "arial.ttf" and put it in "Fonts/". With the
above change, this is no longer possible. As an alternative, one
can now explicitly specify font family substitutions via mkxp.conf;
eg. for the above case, one would add
fontSub=Arial>Open Sans
to the configuration file. Multiple such rules can be specified.
In the process, I also added the ability to provide
'Font.(default_)name' with an array of font families to search
for the first existing one instead of a plain string.
This makes the behavior closer to RMXP; however, it doesn't
work 100% the same: when a reference to the 'Font.name' array is
held and additional strings are added to it without re-assignig
the array to 'Font.name', those will be ignored.
2014-04-11 13:37:14 +02:00
|
|
|
SDL_RWops ops;
|
|
|
|
p->initReadOps(handle, ops, false);
|
2013-09-01 16:27:21 +02:00
|
|
|
|
Font: Overhaul font asset discovery
Previously, any font names requested by RGSS would be translated
directly to filenames by lowercasing and replacing spaces with
underscores (and finally doing some extension substitution).
To make this whole thing work smoother as well as get closer to
how font discovery is done in VX, we now scan the "Fonts/" folder
at startup and index all present font assets by their family name;
now, if an "Open Sans" font is present in "Fonts/", it will be
used regardless of filename.
Font assets with "Regular" style are preferred, but in their
absence, mkxp will make use of any other style it can find for
the respective family. This is not the exact same behavior as
VX, but it should cover 95% of use cases.
Previously, one could substitute fonts via filenames, ie. to
substitute "Arial" with "Open Sans", one would just rename
"OpenSans.ttf" to "arial.ttf" and put it in "Fonts/". With the
above change, this is no longer possible. As an alternative, one
can now explicitly specify font family substitutions via mkxp.conf;
eg. for the above case, one would add
fontSub=Arial>Open Sans
to the configuration file. Multiple such rules can be specified.
In the process, I also added the ability to provide
'Font.(default_)name' with an array of font families to search
for the first existing one instead of a plain string.
This makes the behavior closer to RMXP; however, it doesn't
work 100% the same: when a reference to the 'Font.name' array is
held and additional strings are added to it without re-assignig
the array to 'Font.name', those will be ignored.
2014-04-11 13:37:14 +02:00
|
|
|
d->sfs->initFontSetCB(ops, filename);
|
2013-09-04 13:32:11 +02:00
|
|
|
|
Font: Overhaul font asset discovery
Previously, any font names requested by RGSS would be translated
directly to filenames by lowercasing and replacing spaces with
underscores (and finally doing some extension substitution).
To make this whole thing work smoother as well as get closer to
how font discovery is done in VX, we now scan the "Fonts/" folder
at startup and index all present font assets by their family name;
now, if an "Open Sans" font is present in "Fonts/", it will be
used regardless of filename.
Font assets with "Regular" style are preferred, but in their
absence, mkxp will make use of any other style it can find for
the respective family. This is not the exact same behavior as
VX, but it should cover 95% of use cases.
Previously, one could substitute fonts via filenames, ie. to
substitute "Arial" with "Open Sans", one would just rename
"OpenSans.ttf" to "arial.ttf" and put it in "Fonts/". With the
above change, this is no longer possible. As an alternative, one
can now explicitly specify font family substitutions via mkxp.conf;
eg. for the above case, one would add
fontSub=Arial>Open Sans
to the configuration file. Multiple such rules can be specified.
In the process, I also added the ability to provide
'Font.(default_)name' with an array of font families to search
for the first existing one instead of a plain string.
This makes the behavior closer to RMXP; however, it doesn't
work 100% the same: when a reference to the 'Font.name' array is
held and additional strings are added to it without re-assignig
the array to 'Font.name', those will be ignored.
2014-04-11 13:37:14 +02:00
|
|
|
SDL_RWclose(&ops);
|
2013-09-01 16:27:21 +02:00
|
|
|
}
|
|
|
|
|
Font: Overhaul font asset discovery
Previously, any font names requested by RGSS would be translated
directly to filenames by lowercasing and replacing spaces with
underscores (and finally doing some extension substitution).
To make this whole thing work smoother as well as get closer to
how font discovery is done in VX, we now scan the "Fonts/" folder
at startup and index all present font assets by their family name;
now, if an "Open Sans" font is present in "Fonts/", it will be
used regardless of filename.
Font assets with "Regular" style are preferred, but in their
absence, mkxp will make use of any other style it can find for
the respective family. This is not the exact same behavior as
VX, but it should cover 95% of use cases.
Previously, one could substitute fonts via filenames, ie. to
substitute "Arial" with "Open Sans", one would just rename
"OpenSans.ttf" to "arial.ttf" and put it in "Fonts/". With the
above change, this is no longer possible. As an alternative, one
can now explicitly specify font family substitutions via mkxp.conf;
eg. for the above case, one would add
fontSub=Arial>Open Sans
to the configuration file. Multiple such rules can be specified.
In the process, I also added the ability to provide
'Font.(default_)name' with an array of font families to search
for the first existing one instead of a plain string.
This makes the behavior closer to RMXP; however, it doesn't
work 100% the same: when a reference to the 'Font.name' array is
held and additional strings are added to it without re-assignig
the array to 'Font.name', those will be ignored.
2014-04-11 13:37:14 +02:00
|
|
|
void FileSystem::initFontSets(SharedFontState &sfs)
|
2013-09-10 04:18:58 +02:00
|
|
|
{
|
Font: Overhaul font asset discovery
Previously, any font names requested by RGSS would be translated
directly to filenames by lowercasing and replacing spaces with
underscores (and finally doing some extension substitution).
To make this whole thing work smoother as well as get closer to
how font discovery is done in VX, we now scan the "Fonts/" folder
at startup and index all present font assets by their family name;
now, if an "Open Sans" font is present in "Fonts/", it will be
used regardless of filename.
Font assets with "Regular" style are preferred, but in their
absence, mkxp will make use of any other style it can find for
the respective family. This is not the exact same behavior as
VX, but it should cover 95% of use cases.
Previously, one could substitute fonts via filenames, ie. to
substitute "Arial" with "Open Sans", one would just rename
"OpenSans.ttf" to "arial.ttf" and put it in "Fonts/". With the
above change, this is no longer possible. As an alternative, one
can now explicitly specify font family substitutions via mkxp.conf;
eg. for the above case, one would add
fontSub=Arial>Open Sans
to the configuration file. Multiple such rules can be specified.
In the process, I also added the ability to provide
'Font.(default_)name' with an array of font families to search
for the first existing one instead of a plain string.
This makes the behavior closer to RMXP; however, it doesn't
work 100% the same: when a reference to the 'Font.name' array is
held and additional strings are added to it without re-assignig
the array to 'Font.name', those will be ignored.
2014-04-11 13:37:14 +02:00
|
|
|
FontSetsCBData d = { p, &sfs };
|
2013-09-10 04:18:58 +02:00
|
|
|
|
Font: Overhaul font asset discovery
Previously, any font names requested by RGSS would be translated
directly to filenames by lowercasing and replacing spaces with
underscores (and finally doing some extension substitution).
To make this whole thing work smoother as well as get closer to
how font discovery is done in VX, we now scan the "Fonts/" folder
at startup and index all present font assets by their family name;
now, if an "Open Sans" font is present in "Fonts/", it will be
used regardless of filename.
Font assets with "Regular" style are preferred, but in their
absence, mkxp will make use of any other style it can find for
the respective family. This is not the exact same behavior as
VX, but it should cover 95% of use cases.
Previously, one could substitute fonts via filenames, ie. to
substitute "Arial" with "Open Sans", one would just rename
"OpenSans.ttf" to "arial.ttf" and put it in "Fonts/". With the
above change, this is no longer possible. As an alternative, one
can now explicitly specify font family substitutions via mkxp.conf;
eg. for the above case, one would add
fontSub=Arial>Open Sans
to the configuration file. Multiple such rules can be specified.
In the process, I also added the ability to provide
'Font.(default_)name' with an array of font families to search
for the first existing one instead of a plain string.
This makes the behavior closer to RMXP; however, it doesn't
work 100% the same: when a reference to the 'Font.name' array is
held and additional strings are added to it without re-assignig
the array to 'Font.name', those will be ignored.
2014-04-11 13:37:14 +02:00
|
|
|
PHYSFS_enumerateFilesCallback("Fonts", fontSetEnumCB, &d);
|
2013-09-10 04:18:58 +02:00
|
|
|
}
|
|
|
|
|
2013-09-01 16:27:21 +02:00
|
|
|
void FileSystem::openRead(SDL_RWops &ops,
|
|
|
|
const char *filename,
|
2013-09-10 04:18:58 +02:00
|
|
|
FileType type,
|
2013-11-30 12:00:11 +01:00
|
|
|
bool freeOnClose,
|
|
|
|
const char **foundExt)
|
2013-09-01 16:27:21 +02:00
|
|
|
{
|
2014-01-16 04:19:22 +01:00
|
|
|
PHYSFS_File *handle = p->openReadHandle(filename, type, foundExt);
|
2013-09-01 16:27:21 +02:00
|
|
|
|
Font: Overhaul font asset discovery
Previously, any font names requested by RGSS would be translated
directly to filenames by lowercasing and replacing spaces with
underscores (and finally doing some extension substitution).
To make this whole thing work smoother as well as get closer to
how font discovery is done in VX, we now scan the "Fonts/" folder
at startup and index all present font assets by their family name;
now, if an "Open Sans" font is present in "Fonts/", it will be
used regardless of filename.
Font assets with "Regular" style are preferred, but in their
absence, mkxp will make use of any other style it can find for
the respective family. This is not the exact same behavior as
VX, but it should cover 95% of use cases.
Previously, one could substitute fonts via filenames, ie. to
substitute "Arial" with "Open Sans", one would just rename
"OpenSans.ttf" to "arial.ttf" and put it in "Fonts/". With the
above change, this is no longer possible. As an alternative, one
can now explicitly specify font family substitutions via mkxp.conf;
eg. for the above case, one would add
fontSub=Arial>Open Sans
to the configuration file. Multiple such rules can be specified.
In the process, I also added the ability to provide
'Font.(default_)name' with an array of font families to search
for the first existing one instead of a plain string.
This makes the behavior closer to RMXP; however, it doesn't
work 100% the same: when a reference to the 'Font.name' array is
held and additional strings are added to it without re-assignig
the array to 'Font.name', those will be ignored.
2014-04-11 13:37:14 +02:00
|
|
|
p->initReadOps(handle, ops, freeOnClose);
|
|
|
|
}
|
2013-09-10 04:18:58 +02:00
|
|
|
|
Font: Overhaul font asset discovery
Previously, any font names requested by RGSS would be translated
directly to filenames by lowercasing and replacing spaces with
underscores (and finally doing some extension substitution).
To make this whole thing work smoother as well as get closer to
how font discovery is done in VX, we now scan the "Fonts/" folder
at startup and index all present font assets by their family name;
now, if an "Open Sans" font is present in "Fonts/", it will be
used regardless of filename.
Font assets with "Regular" style are preferred, but in their
absence, mkxp will make use of any other style it can find for
the respective family. This is not the exact same behavior as
VX, but it should cover 95% of use cases.
Previously, one could substitute fonts via filenames, ie. to
substitute "Arial" with "Open Sans", one would just rename
"OpenSans.ttf" to "arial.ttf" and put it in "Fonts/". With the
above change, this is no longer possible. As an alternative, one
can now explicitly specify font family substitutions via mkxp.conf;
eg. for the above case, one would add
fontSub=Arial>Open Sans
to the configuration file. Multiple such rules can be specified.
In the process, I also added the ability to provide
'Font.(default_)name' with an array of font families to search
for the first existing one instead of a plain string.
This makes the behavior closer to RMXP; however, it doesn't
work 100% the same: when a reference to the 'Font.name' array is
held and additional strings are added to it without re-assignig
the array to 'Font.name', those will be ignored.
2014-04-11 13:37:14 +02:00
|
|
|
void FileSystem::openReadRaw(SDL_RWops &ops,
|
|
|
|
const char *filename,
|
|
|
|
bool freeOnClose)
|
|
|
|
{
|
|
|
|
PHYSFS_File *handle = PHYSFS_openRead(filename);
|
|
|
|
assert(handle);
|
2013-09-01 16:27:21 +02:00
|
|
|
|
Font: Overhaul font asset discovery
Previously, any font names requested by RGSS would be translated
directly to filenames by lowercasing and replacing spaces with
underscores (and finally doing some extension substitution).
To make this whole thing work smoother as well as get closer to
how font discovery is done in VX, we now scan the "Fonts/" folder
at startup and index all present font assets by their family name;
now, if an "Open Sans" font is present in "Fonts/", it will be
used regardless of filename.
Font assets with "Regular" style are preferred, but in their
absence, mkxp will make use of any other style it can find for
the respective family. This is not the exact same behavior as
VX, but it should cover 95% of use cases.
Previously, one could substitute fonts via filenames, ie. to
substitute "Arial" with "Open Sans", one would just rename
"OpenSans.ttf" to "arial.ttf" and put it in "Fonts/". With the
above change, this is no longer possible. As an alternative, one
can now explicitly specify font family substitutions via mkxp.conf;
eg. for the above case, one would add
fontSub=Arial>Open Sans
to the configuration file. Multiple such rules can be specified.
In the process, I also added the ability to provide
'Font.(default_)name' with an array of font families to search
for the first existing one instead of a plain string.
This makes the behavior closer to RMXP; however, it doesn't
work 100% the same: when a reference to the 'Font.name' array is
held and additional strings are added to it without re-assignig
the array to 'Font.name', those will be ignored.
2014-04-11 13:37:14 +02:00
|
|
|
p->initReadOps(handle, ops, freeOnClose);
|
2013-09-01 16:27:21 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
bool FileSystem::exists(const char *filename, FileType type)
|
|
|
|
{
|
2014-01-16 04:19:22 +01:00
|
|
|
char found[512];
|
2013-09-01 16:27:21 +02:00
|
|
|
|
2014-01-16 04:19:22 +01:00
|
|
|
return p->completeFileName(filename, type, found, sizeof(found), 0);
|
2013-09-01 16:27:21 +02:00
|
|
|
}
|