mirror of
https://github.com/Teh-Dango/Sarge-AI.git
synced 2024-08-30 16:32:11 +00:00
170 lines
4.8 KiB
Plaintext
170 lines
4.8 KiB
Plaintext
|
/* Select a random position based on anchor position, direction and distance.
|
||
|
In: [position,distance,direction,water,road,emptySpace]
|
||
|
Out: position
|
||
|
*/
|
||
|
private ["_org","_dst","_dir","_pos","_water","_road","_empty"];
|
||
|
_org = _this select 0;
|
||
|
_dst = _this select 1;
|
||
|
_dir = if (count _this > 2) then {_this select 2} else {random 360};
|
||
|
_water = if (count _this > 3) then {_this select 3} else {0};
|
||
|
_road = if (count _this > 4) then {_this select 4} else {[0,200]};
|
||
|
_empty = if (count _this > 5) then {_this select 5} else {[]};
|
||
|
|
||
|
// Object instead of position array given
|
||
|
if (typename _org == "OBJECT") then {_org = getpos _org};
|
||
|
|
||
|
// Distance given as an array of min and max. Pick a random between them.
|
||
|
if (typename _dst == "ARRAY") then {
|
||
|
private ["_min","_max"];
|
||
|
_min = _dst select 0;
|
||
|
_max = _dst select 1;
|
||
|
_dst = (_min + random(_max - _min));
|
||
|
};
|
||
|
|
||
|
// Direction given as an array of min and max. Pick a random dir between them.
|
||
|
if (typename _dir == "ARRAY") then {
|
||
|
private ["_min","_max","_ang"];
|
||
|
_min = _dir select 0;
|
||
|
_max = _dir select 1;
|
||
|
|
||
|
_ang = _max - _min;
|
||
|
|
||
|
// Min bigger than max, can happen with directions around north
|
||
|
if (_ang < 0) then { _ang = _ang + 360 };
|
||
|
|
||
|
_dir = (_min + random _ang);
|
||
|
};
|
||
|
|
||
|
_pos = [_org,_dst,_dir] call UPSMON_pos_fnc_getPos;
|
||
|
|
||
|
// Water position
|
||
|
if (typeName _water == "SCALAR") then {
|
||
|
switch _water do {
|
||
|
case 0: { // Water not allowed
|
||
|
if (surfaceIsWater _pos) then {
|
||
|
private ["_p","_d","_l"];
|
||
|
_d = 0; _l = true;
|
||
|
|
||
|
// Search for a land position starting from the randomly picked position and
|
||
|
// then going outwards from it in full circles in 20m steps.
|
||
|
while {_d = _d + 20; _l && _d < 5000} do {
|
||
|
for "_i" from 0 to 340 step 20 do {
|
||
|
_p = [_pos,_d,_i] call UPSMON_pos_fnc_getpos;
|
||
|
if (!surfaceIsWater _p) exitwith {_l = false};
|
||
|
};
|
||
|
};
|
||
|
_pos = _p;
|
||
|
};
|
||
|
};
|
||
|
case 1: { // Water allowed
|
||
|
|
||
|
};
|
||
|
case 2: { // Only water allowed
|
||
|
if !(surfaceIsWater _pos) then {
|
||
|
private ["_p","_d","_l"];
|
||
|
_d = 0; _l = true;
|
||
|
|
||
|
// Search for a water position starting from the randomly picked position and
|
||
|
// then going outwards from it in full circles in 20m steps.
|
||
|
while {_d = _d + 20; _l && _d < 5000} do {
|
||
|
for "_i" from 0 to 340 step 20 do {
|
||
|
_p = [_pos,_d,_i] call UPSMON_pos_fnc_getpos;
|
||
|
if (surfaceIsWater _p) exitwith {_l = false};
|
||
|
};
|
||
|
};
|
||
|
_pos = _p;
|
||
|
};
|
||
|
};
|
||
|
};
|
||
|
} else { // For backward compatibility
|
||
|
// Water position is not allowed
|
||
|
if !_water then {
|
||
|
if (surfaceIsWater _pos) then {
|
||
|
private ["_p","_d","_l"];
|
||
|
_d = 0; _l = true;
|
||
|
|
||
|
// Search for a land position starting from the randomly picked position and
|
||
|
// then going outwards from it in full circles in 20m steps.
|
||
|
while {_d = _d + 20; _l && _d < 5000} do {
|
||
|
for "_i" from 0 to 340 step 20 do {
|
||
|
_p = [_pos,_d,_i] call UPSMON_pos_fnc_getpos;
|
||
|
if (!surfaceIsWater _p) exitwith {_l = false};
|
||
|
};
|
||
|
};
|
||
|
_pos = _p;
|
||
|
};
|
||
|
};
|
||
|
};
|
||
|
|
||
|
// Road position.
|
||
|
if (count _road > 0) then {
|
||
|
if ((_road select 0) > 0) then {
|
||
|
private ["_mode","_range","_roads","_cnt","_p","_p2"];
|
||
|
_mode = _road select 0;
|
||
|
_range = _road select 1;
|
||
|
_roads = _pos nearroads _range;
|
||
|
_cnt = count _roads;
|
||
|
_p = [];
|
||
|
|
||
|
// Road position(s) found.
|
||
|
if (_cnt > 0) then {
|
||
|
_p = getpos (_roads select 0);
|
||
|
|
||
|
// Found more than one road position, return closest.
|
||
|
if (_cnt > 1) then {
|
||
|
for "_i" from 1 to (_cnt - 1) do {
|
||
|
_p2 = getpos (_roads select _i);
|
||
|
if ((_p2 distance _pos) < (_p distance _pos)) then {
|
||
|
_p = _p2;
|
||
|
};
|
||
|
};
|
||
|
};
|
||
|
};
|
||
|
|
||
|
switch _mode do {
|
||
|
// Road position preferred but not forced.
|
||
|
case 1: {
|
||
|
if (count _p > 0) then {
|
||
|
_pos = _p;
|
||
|
};
|
||
|
};
|
||
|
// Only accept road position, return empty array if none found.
|
||
|
case 2: {
|
||
|
if (count _p > 0) then {
|
||
|
_pos = _p;
|
||
|
} else {
|
||
|
_pos resize 0;
|
||
|
};
|
||
|
};
|
||
|
};
|
||
|
};
|
||
|
};
|
||
|
|
||
|
// Find empty position
|
||
|
private ["_dst","_veh","_p"];
|
||
|
|
||
|
_dst = 200;
|
||
|
_veh = "";
|
||
|
switch (typename _empty) do {
|
||
|
case "OBJECT": { _veh = typeof _empty }; // Only vehicle given, use default distance
|
||
|
case "SCALAR": {_dst = _empty;};
|
||
|
case "ARRAY": {
|
||
|
if (count _empty > 0) then {
|
||
|
_dst = _empty select 0;
|
||
|
_veh = _empty select 1;
|
||
|
if (typename _veh == typename objNull) then { _veh = typeof _veh };
|
||
|
};
|
||
|
};
|
||
|
};
|
||
|
|
||
|
_p = [];
|
||
|
if (count _pos > 0) then {_p = _pos findEmptyPosition [0,_dst,_veh];};
|
||
|
|
||
|
|
||
|
// If an empty position is found, use it. Otherwise, return the original position.
|
||
|
if (count _p > 0) then {
|
||
|
_pos = _p;
|
||
|
};
|
||
|
|
||
|
// Return position
|
||
|
_pos
|