File: RocketFileInterface.cpp

package info (click to toggle)
freespace2 24.2.0%2Brepack-1
  • links: PTS, VCS
  • area: non-free
  • in suites: forky, sid
  • size: 43,716 kB
  • sloc: cpp: 595,001; ansic: 21,741; python: 1,174; sh: 457; makefile: 248; xml: 181
file content (118 lines) | stat: -rw-r--r-- 3,124 bytes parent folder | download | duplicates (2)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
//
//

#include "RocketFileInterface.h"

#include "cfile/cfile.h"
#include "mod_table/mod_table.h"

// Our Assert conflicts with the definitions inside libRocket
#pragma push_macro("Assert")
#undef Assert

#include <Rocket/Core/Log.h>

#pragma pop_macro("Assert")

using namespace Rocket::Core;

namespace scpui {

RocketFileInterface::RocketFileInterface() = default;
FileHandle RocketFileInterface::Open(const String& path)
{
	// In order to use libRocket correctly we need to make sure that we only have Unicode data. This is done here since
	// a mod that uses libRocket will always open a file through this function but a mod that doesn't will not call this
	if (!Unicode_text_mode) {
		Error(LOCATION,
		      "libRocket was used without Unicode text mode being enabled! The new UI system requires Unicode text.");
	}

	int dir_type;
	SCP_string name;

	if (!getCFilePath(path, name, dir_type)) {
		return 0;
	}

	return (FileHandle)cfopen(name.c_str(), "rb", CFILE_NORMAL, dir_type);
}
void RocketFileInterface::Close(FileHandle file) { cfclose((CFILE*)file); }
size_t RocketFileInterface::Read(void* buffer, size_t size, FileHandle file)
{
	auto fp = (CFILE*)file;

	auto read = cfread(buffer, 1, (int)size, fp);

	return (size_t)read;
}
bool RocketFileInterface::Seek(FileHandle file, long offset, int origin)
{
	auto fp = (CFILE*)file;

	int cf_seek_mode;
	switch (origin) {
	case SEEK_SET:
		cf_seek_mode = CF_SEEK_SET;
		break;
	case SEEK_CUR:
		cf_seek_mode = CF_SEEK_CUR;
		break;
	case SEEK_END:
		cf_seek_mode = CF_SEEK_END;
		break;
	default:
		Assertion(false, "Invalid seek mode encountered!");
		return false;
	}

	return cfseek(fp, (int)offset, cf_seek_mode) == 0;
}
size_t RocketFileInterface::Tell(FileHandle file) { return (size_t)cftell((CFILE*)file); }
size_t RocketFileInterface::Length(FileHandle file) { return (size_t)cfilelength((CFILE*)file); }
bool RocketFileInterface::getCFilePath(const String& path_in, SCP_string& name, int& dir_type)
{
	String path = path_in;
	if (path.Find("\\") != String::npos) {
		// We don't allow these in our paths
		Log::Message(
		    Log::LT_WARNING,
		    "The path '%s' contains the invalid character '\\'. This is not allowed! You should use '/' instead.",
		    path.CString());
		path.Replace("\\", "/");
	}

	if (path.Length() > 0 && path[path.Length() - 1] == '/') {
		// Trailing slashes are not allowed!
		Log::Message(Log::LT_WARNING,
		             "The path '%s' has trailing slashes. This is not allowed! The trailing slashes have been removed.",
		             path.CString());

		while (path.Length() > 0 && path[path.Length() - 1] == '/') {
			path = path.Substring(0, path.Length() - 1);
		}
	}

	auto slashPos = path.RFind("/");

	if (slashPos == String::npos) {
		// No path specified, search everywhere
		dir_type = CF_TYPE_ANY;
		name     = path.CString();
		return true;
	}

	auto dir_path = path.Substring(0, slashPos);

	auto type = cfile_get_path_type(dir_path.CString());

	if (type == CF_TYPE_INVALID) {
		return false;
	}

	dir_type = type;
	name     = path.Substring(slashPos + 1).CString();

	return true;
}
} // namespace scpui