File: MoveToSubmodel.cpp

package info (click to toggle)
freespace2 24.2.0%2Brepack-3
  • links: PTS, VCS
  • area: non-free
  • in suites: forky, sid
  • size: 43,740 kB
  • sloc: cpp: 595,005; ansic: 21,741; python: 1,174; sh: 457; makefile: 243; xml: 181
file content (79 lines) | stat: -rw-r--r-- 2,341 bytes parent folder | download
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

#include "MoveToSubmodel.h"

#include "math/vecmat.h"
#include "parse/parselo.h"
#include "ship/ship.h"

#include <utility>

namespace actions {
namespace types {

flagset<ProgramContextFlags> MoveToSubmodel::getRequiredExecutionContextFlags()
{
	return flagset<ProgramContextFlags>{ProgramContextFlags::HasObject, ProgramContextFlags::HasSubobject};
}

MoveToSubmodel::MoveToSubmodel(expression::TypedActionExpression<ValueType> subObjectExpression)
	: m_subObjectExpression(std::move(subObjectExpression))
{
}
MoveToSubmodel::~MoveToSubmodel() = default;

ActionResult MoveToSubmodel::execute(ProgramLocals& locals) const
{
	// The calling code should ensure that this never happens
	Assertion(locals.hostSubobject >= 0, "Did not have a valid host subobject.");

	auto instance = object_get_model_instance(locals.host.objp());
	Assertion(instance != -1, "Model instances are required if a host subobject is specified.");

	auto pmi = model_get_instance(instance);
	auto pm = model_get(pmi->model_num);

	const auto destinationSubObject = m_subObjectExpression.execute(locals.variables);
	bool goToParent = false;
	if (destinationSubObject == "<parent>") {
		goToParent = true;
	}

	if (goToParent) {
		// Special case when we simply want to go to our parent without explicitly specifying its name
		auto parentSubobject = pm->submodel[locals.hostSubobject].parent;

		if (parentSubobject < 0) {
			Warning(LOCATION,
				"Current program subobject %s on model %s has no parent!",
				pm->submodel[locals.hostSubobject].name,
				pm->filename);
			return ActionResult::Errored;
		}

		// We have a valid parent so this move will work properly
		locals.hostSubobject = parentSubobject;
		return ActionResult::Finished;
	}

	// We need to do a linear search for the right subobject
	for (int i = 0; i < pm->n_models; ++i) {
		const auto submodel = pm->submodel[i];

		if (subsystem_stricmp(destinationSubObject.c_str(), submodel.name) == 0) {
			// Found something!
			locals.hostSubobject = i;
			return ActionResult::Finished;
		}
	}

	Warning(LOCATION, "Could not find subobject %s in model %s!", destinationSubObject.c_str(), pm->filename);
	return ActionResult::Errored;
}

std::unique_ptr<Action> MoveToSubmodel::clone() const
{
	return std::unique_ptr<Action>(new MoveToSubmodel(*this));
}

} // namespace types
} // namespace actions