mirror of
https://github.com/thegeneralist01/Scene-Manager-DevRepo
synced 2026-01-12 16:10:29 +01:00
* Mouse can now be used to fully navigate menus * Added check for driver's current vehicle when releasing from Stop waypoint in case the ped is not in a vehicle. * Lines are now only drawn between waypoint markers under the same conditions that waypoint markers are drawn * Updated marker position to be player's mouse position * Version update * Added logic to update waypoint position during driving task if the waypoint position was changed before the driver arrived. * Removed unused usings. Refactored debug statements to use Game.LogTrivial instead of Logger.Log * Removed class * Removed Logger class * Consolidated all custom enums to this class. Added default waypoint settings from .ini * Modified a hint message * Fixed collector options not being enabled/disabled when Collector box is checked * Refactored AITasking into CollectedVehicle. * Updated reference to vehicle tasking based on AITasking refactor. Fixed speed zone radius not updating correctly. * Version update * Refactored AITasking into CollectedVehicle * Added check for CollectorRadius being more than SpeedZoneRadius. Added debug messages when values are reset to default. * Added hint message if player tries to edit waypoints while 3D waypoints are disabled * Fixed a bug where a 3D waypoint marker would be drawn even if 3D waypoints were disabled * Removed unnecessary property setting when a vehicle is being removed from a path. * Added a check for if the driver loses their task and reassigns it. * Fixed a bug where the 3D line between waypoints was still being drawn even though 3D waypoints were disabled in the settings menu. * Updated version * Added console command to show info about collected vehicles. * Added ConsoleCommand class, removed AITasking class. * Removed class after refactoring into CollectedVehicle * Update README.md * Update README.md * Added ini setting for Advanced Barrier Options. Added enum for TrafficLight state. Added debug message for invalid barriers. * Updated version * Added MousePositionInWorld and RNUIMouseInputHandler classes * Fixed a crash when a collected ped is arrested. * Refactored to implement RNUIMouseInputHandler class. * Removed unused method. * Updated version * Disabled deletion/creation of shadow barrier and re-enabled updating position based on mouse position. * Added check for Driver's current vehicle in driving loop in case the Driver exited Vehicle at some point. Adjusted some guard clause logic and log messages. * Removed unused variable * Renamed some methods to improve clarity * Updated version.
441 lines
18 KiB
C#
441 lines
18 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Drawing;
|
|
using System.Linq;
|
|
using System.Windows.Forms;
|
|
using Rage;
|
|
using RAGENativeUI;
|
|
using RAGENativeUI.Elements;
|
|
|
|
namespace SceneManager
|
|
{
|
|
|
|
|
|
static class PathMainMenu
|
|
{
|
|
internal static List<Path> paths = new List<Path>() { };
|
|
private static string[] dismissOptions = new string[] { "From path", "From waypoint", "From world" };
|
|
//private static List<string> dismissOptions = new List<string>() { "From path", "From waypoint", "From world" };
|
|
|
|
internal static UIMenu pathMainMenu = new UIMenu("Scene Manager", "~o~Path Manager Main Menu");
|
|
internal static UIMenuItem createNewPath;
|
|
internal static UIMenuItem deleteAllPaths = new UIMenuItem("Delete All Paths");
|
|
internal static UIMenuNumericScrollerItem<int> editPath;
|
|
internal static UIMenuListScrollerItem<string> directOptions = new UIMenuListScrollerItem<string>("Direct driver to path's", "", new[] { "First waypoint", "Nearest waypoint" });
|
|
internal static UIMenuNumericScrollerItem<int> directDriver;
|
|
internal static UIMenuListScrollerItem<string> dismissDriver = new UIMenuListScrollerItem<string>("Dismiss nearest driver", $"~b~From path: ~w~AI will be released from the path{Environment.NewLine}~b~From waypoint: ~w~AI will skip their current waypoint task{Environment.NewLine}~b~From world: ~w~AI will be removed from the world.", dismissOptions);
|
|
internal static UIMenuCheckboxItem disableAllPaths = new UIMenuCheckboxItem("Disable All Paths", false);
|
|
|
|
internal enum Delete
|
|
{
|
|
Single,
|
|
All
|
|
}
|
|
|
|
internal static void InstantiateMenu()
|
|
{
|
|
pathMainMenu.ParentMenu = MainMenu.mainMenu;
|
|
MenuManager.menuPool.Add(pathMainMenu);
|
|
pathMainMenu.OnItemSelect += PathMenu_OnItemSelected;
|
|
pathMainMenu.OnCheckboxChange += PathMenu_OnCheckboxChange;
|
|
pathMainMenu.OnMenuOpen += PathMenu_OnMenuOpen;
|
|
}
|
|
|
|
internal static void BuildPathMenu()
|
|
{
|
|
MenuManager.menuPool.CloseAllMenus();
|
|
pathMainMenu.Clear();
|
|
|
|
pathMainMenu.AddItem(createNewPath = new UIMenuItem("Create New Path"));
|
|
createNewPath.ForeColor = Color.Gold;
|
|
pathMainMenu.AddItem(editPath = new UIMenuNumericScrollerItem<int>("Edit Path", "", 1, paths.Count, 1));
|
|
editPath.Index = 0;
|
|
editPath.ForeColor = Color.Gold;
|
|
pathMainMenu.AddItem(disableAllPaths);
|
|
disableAllPaths.Enabled = true;
|
|
pathMainMenu.AddItem(deleteAllPaths);
|
|
deleteAllPaths.Enabled = true;
|
|
deleteAllPaths.ForeColor = Color.Gold;
|
|
pathMainMenu.AddItem(directOptions);
|
|
pathMainMenu.AddItem(directDriver = new UIMenuNumericScrollerItem<int>("Direct nearest driver to path", "", 1, paths.Count, 1));
|
|
directDriver.ForeColor = Color.Gold;
|
|
directDriver.Enabled = true;
|
|
pathMainMenu.AddItem(dismissDriver);
|
|
dismissDriver.ForeColor = Color.Gold;
|
|
|
|
if (paths.Count == 8)
|
|
{
|
|
createNewPath.Enabled = false;
|
|
}
|
|
if (paths.Count == 0)
|
|
{
|
|
editPath.Enabled = false;
|
|
deleteAllPaths.Enabled = false;
|
|
disableAllPaths.Enabled = false;
|
|
directDriver.Enabled = false;
|
|
}
|
|
|
|
MenuManager.menuPool.RefreshIndex();
|
|
}
|
|
|
|
private static bool VehicleAndDriverValid(this Vehicle v)
|
|
{
|
|
if (v && v.HasDriver && v.Driver && v.Driver.IsAlive)
|
|
{
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
private static void GoToEditPathMenu()
|
|
{
|
|
pathMainMenu.Visible = false;
|
|
EditPathMenu.editPathMenu.Visible = true;
|
|
}
|
|
|
|
private static void GoToPathCreationMenu()
|
|
{
|
|
if (createNewPath.Text.Contains("Continue"))
|
|
{
|
|
pathMainMenu.Visible = false;
|
|
PathCreationMenu.pathCreationMenu.Visible = true;
|
|
}
|
|
else
|
|
{
|
|
PathCreationMenu.pathCreationMenu.Clear();
|
|
PathCreationMenu.BuildPathCreationMenu();
|
|
pathMainMenu.Visible = false;
|
|
PathCreationMenu.pathCreationMenu.Visible = true;
|
|
|
|
// For each element in paths, determine if the element exists but is not finished yet, or if it doesn't exist, create it.
|
|
for (int i = 0; i <= paths.Count; i++)
|
|
{
|
|
if (paths.ElementAtOrDefault(i) != null && paths[i].State == State.Creating)
|
|
{
|
|
Game.DisplayNotification($"~o~Scene Manager~y~[Creating]\n~w~Resuming path {paths[i].Number}");
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
private static void DisableAllPaths()
|
|
{
|
|
if (disableAllPaths.Checked)
|
|
{
|
|
foreach (Path path in paths)
|
|
{
|
|
path.DisablePath();
|
|
}
|
|
Game.LogTrivial($"All paths disabled.");
|
|
}
|
|
else
|
|
{
|
|
foreach (Path path in paths)
|
|
{
|
|
path.EnablePath();
|
|
}
|
|
Game.LogTrivial($"All paths enabled.");
|
|
}
|
|
}
|
|
|
|
private static void DeleteAllPaths()
|
|
{
|
|
for (int i = 0; i < paths.Count; i++)
|
|
{
|
|
DeletePath(paths[i], Delete.All);
|
|
}
|
|
disableAllPaths.Checked = false;
|
|
paths.Clear();
|
|
BuildPathMenu();
|
|
pathMainMenu.Visible = true;
|
|
Game.LogTrivial($"All paths deleted");
|
|
Game.DisplayNotification($"~o~Scene Manager\n~w~All paths deleted.");
|
|
}
|
|
|
|
internal static void DeletePath(Path path, Delete pathsToDelete)
|
|
{
|
|
//Game.LogTrivial($"Preparing to delete path {path.Number}");
|
|
|
|
RemoveVehiclesFromPath();
|
|
RemoveBlipsAndYieldZones();
|
|
|
|
//Game.LogTrivial($"Clearing path waypoints");
|
|
path.Waypoints.Clear();
|
|
|
|
// Manipulating the menu to reflect specific paths being deleted
|
|
if (pathsToDelete == Delete.Single)
|
|
{
|
|
paths.Remove(path);
|
|
UpdatePathNumbers();
|
|
UpdatePathBlips();
|
|
BuildPathMenu();
|
|
pathMainMenu.Visible = true;
|
|
Game.LogTrivial($"Path {path.Number} deleted successfully.");
|
|
Game.DisplayNotification($"~o~Scene Manager\n~w~Path {path.Number} deleted.");
|
|
}
|
|
|
|
EditPathMenu.editPathMenu.Reset(true, true);
|
|
EditPathMenu.disablePath.Enabled = true;
|
|
|
|
void RemoveVehiclesFromPath()
|
|
{
|
|
//Game.LogTrivial($"Removing all vehicles on the path");
|
|
var pathVehicles = path.CollectedVehicles.Where(cv => cv.Path.Number == path.Number).ToList();
|
|
foreach (CollectedVehicle cv in pathVehicles.Where(cv => cv != null && cv.Vehicle && cv.Driver))
|
|
{
|
|
if (cv.StoppedAtWaypoint)
|
|
{
|
|
Rage.Native.NativeFunction.Natives.x260BE8F09E326A20(cv.Vehicle, 1f, 1, true);
|
|
}
|
|
//cv.StoppedAtWaypoint = false;
|
|
if (cv.Driver.GetAttachedBlip())
|
|
{
|
|
cv.Driver.GetAttachedBlip().Delete();
|
|
}
|
|
cv.Driver.Dismiss();
|
|
cv.Vehicle.IsSirenOn = false;
|
|
cv.Vehicle.IsSirenSilent = true;
|
|
cv.Vehicle.Dismiss();
|
|
|
|
//Game.LogTrivial($"{cv.vehicle.Model.Name} cleared from path {cv.path}");
|
|
path.CollectedVehicles.Remove(cv);
|
|
}
|
|
path.CollectedVehicles.Clear();
|
|
}
|
|
|
|
void RemoveBlipsAndYieldZones()
|
|
{
|
|
//Game.LogTrivial($"Removing waypoint blips and yield zones.");
|
|
foreach (Waypoint waypoint in path.Waypoints)
|
|
{
|
|
if (waypoint.SpeedZone != 0)
|
|
{
|
|
waypoint.RemoveSpeedZone();
|
|
}
|
|
if (waypoint.Blip)
|
|
{
|
|
waypoint.Blip.Delete();
|
|
}
|
|
if (waypoint.CollectorRadiusBlip)
|
|
{
|
|
waypoint.CollectorRadiusBlip.Delete();
|
|
}
|
|
}
|
|
}
|
|
|
|
void UpdatePathBlips()
|
|
{
|
|
foreach (Path p in paths)
|
|
{
|
|
foreach (Waypoint waypoint in p.Waypoints)
|
|
{
|
|
var blipColor = waypoint.Blip.Color;
|
|
waypoint.Blip.Sprite = (BlipSprite)paths.IndexOf(p) + 17;
|
|
waypoint.Blip.Color = blipColor;
|
|
}
|
|
}
|
|
}
|
|
|
|
void UpdatePathNumbers()
|
|
{
|
|
for (int i = 0; i < paths.Count; i++)
|
|
{
|
|
paths[i].Number = i + 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
private static void DirectDriver()
|
|
{
|
|
var nearbyVehicle = Game.LocalPlayer.Character.GetNearbyVehicles(16).Where(v => v != Game.LocalPlayer.Character.CurrentVehicle && v.VehicleAndDriverValid()).FirstOrDefault();
|
|
var path = paths[directDriver.Index];
|
|
var collectedVehicle = path.CollectedVehicles.Where(cv => cv.Vehicle == nearbyVehicle).FirstOrDefault();
|
|
var waypoints = path.Waypoints;
|
|
var firstWaypoint = waypoints.First();
|
|
var nearestWaypoint = waypoints.Where(wp => wp.Position.DistanceTo2D(nearbyVehicle.FrontPosition) < wp.Position.DistanceTo2D(nearbyVehicle.RearPosition)).OrderBy(wp => wp.Position.DistanceTo2D(nearbyVehicle)).FirstOrDefault();
|
|
|
|
if (nearbyVehicle)
|
|
{
|
|
var nearbyVehiclePath = paths.Where(p => p.CollectedVehicles.Any(v => v.Vehicle == nearbyVehicle)).FirstOrDefault();
|
|
if (nearbyVehiclePath != null)
|
|
{
|
|
var nearbyCollectedVehicle = nearbyVehiclePath.CollectedVehicles.Where(v => v.Vehicle == nearbyVehicle).FirstOrDefault();
|
|
if (nearbyCollectedVehicle != null)
|
|
{
|
|
nearbyCollectedVehicle.Dismiss(DismissOption.FromDirected, path);
|
|
if (directOptions.SelectedItem == "First waypoint")
|
|
{
|
|
GameFiber.StartNew(() =>
|
|
{
|
|
nearbyCollectedVehicle.AssignWaypointTasks(path, firstWaypoint);
|
|
//AITasking.AssignWaypointTasks(nearbyCollectedVehicle, path, firstWaypoint);
|
|
});
|
|
}
|
|
else
|
|
{
|
|
if (nearestWaypoint != null)
|
|
{
|
|
GameFiber.StartNew(() =>
|
|
{
|
|
nearbyCollectedVehicle.AssignWaypointTasks(path, nearestWaypoint);
|
|
//AITasking.AssignWaypointTasks(nearbyCollectedVehicle, path, nearestWaypoint);
|
|
});
|
|
}
|
|
}
|
|
return;
|
|
}
|
|
}
|
|
|
|
// The vehicle should only be added to the collection when it's not null AND if the selected item is First Waypoint OR if the selected item is nearestWaypoint AND nearestWaypoint is not null
|
|
if (collectedVehicle == null && directOptions.SelectedItem == "First waypoint" || (directOptions.SelectedItem == "Nearest waypoint" && nearestWaypoint != null))
|
|
{
|
|
Game.LogTrivial($"[Direct Driver] Adding {nearbyVehicle.Model.Name} to collection.");
|
|
path.CollectedVehicles.Add(new CollectedVehicle(nearbyVehicle, path));
|
|
collectedVehicle = path.CollectedVehicles.Where(cv => cv.Vehicle == nearbyVehicle).FirstOrDefault();
|
|
//Logger.Log($"Collected vehicle is {collectedVehicle.Vehicle.Model.Name}");
|
|
}
|
|
|
|
if (collectedVehicle == null)
|
|
{
|
|
return;
|
|
}
|
|
collectedVehicle.Directed = true;
|
|
collectedVehicle.Driver.Tasks.Clear();
|
|
|
|
//Logger.Log($"Collected vehicle properties: Dismissed [{collectedVehicle.Dismissed}], Directed [{collectedVehicle.Directed}], StopppedAtWaypoint [{collectedVehicle.StoppedAtWaypoint}]");
|
|
if (directOptions.SelectedItem == "First waypoint")
|
|
{
|
|
GameFiber.StartNew(() =>
|
|
{
|
|
collectedVehicle.AssignWaypointTasks(path, firstWaypoint);
|
|
//AITasking.AssignWaypointTasks(collectedVehicle, path, firstWaypoint);
|
|
});
|
|
}
|
|
else
|
|
{
|
|
if (nearestWaypoint != null)
|
|
{
|
|
GameFiber.StartNew(() =>
|
|
{
|
|
collectedVehicle.AssignWaypointTasks(path, nearestWaypoint);
|
|
//AITasking.AssignWaypointTasks(collectedVehicle, path, nearestWaypoint);
|
|
});
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
private static void DismissDriver()
|
|
{
|
|
var nearbyVehicle = Game.LocalPlayer.Character.GetNearbyVehicles(16).Where(v => v != Game.LocalPlayer.Character.CurrentVehicle && v.VehicleAndDriverValid()).FirstOrDefault();
|
|
if (nearbyVehicle)
|
|
{
|
|
if (!paths.Any() && dismissDriver.Index == (int)DismissOption.FromWorld)
|
|
{
|
|
Game.LogTrivial($"Dismissed {nearbyVehicle.Model.Name} from the world");
|
|
while (nearbyVehicle && nearbyVehicle.HasOccupants)
|
|
{
|
|
foreach (Ped occupant in nearbyVehicle.Occupants)
|
|
{
|
|
occupant.Delete();
|
|
}
|
|
GameFiber.Yield();
|
|
}
|
|
if (nearbyVehicle)
|
|
{
|
|
nearbyVehicle.Delete();
|
|
}
|
|
return;
|
|
}
|
|
|
|
foreach (Path path in paths)
|
|
{
|
|
var collectedVehicle = path.CollectedVehicles.Where(cv => cv.Vehicle == nearbyVehicle).FirstOrDefault();
|
|
if (collectedVehicle != null)
|
|
{
|
|
collectedVehicle.Dismiss((DismissOption)dismissDriver.Index);
|
|
break;
|
|
}
|
|
else if (dismissDriver.Index == (int)DismissOption.FromWorld)
|
|
{
|
|
Game.LogTrivial($"Dismissed {nearbyVehicle.Model.Name} from the world");
|
|
while (nearbyVehicle && nearbyVehicle.HasOccupants)
|
|
{
|
|
foreach (Ped occupant in nearbyVehicle.Occupants)
|
|
{
|
|
occupant.Delete();
|
|
}
|
|
GameFiber.Yield();
|
|
}
|
|
if (nearbyVehicle)
|
|
{
|
|
nearbyVehicle.Delete();
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
private static void PathMenu_OnItemSelected(UIMenu sender, UIMenuItem selectedItem, int index)
|
|
{
|
|
if (selectedItem == createNewPath)
|
|
{
|
|
GoToPathCreationMenu();
|
|
}
|
|
|
|
if (selectedItem == editPath)
|
|
{
|
|
pathMainMenu.Visible = false;
|
|
EditPathMenu.editPathMenu.Visible = true;
|
|
}
|
|
|
|
if (selectedItem == deleteAllPaths)
|
|
{
|
|
DeleteAllPaths();
|
|
}
|
|
|
|
if (selectedItem == directDriver)
|
|
{
|
|
DirectDriver();
|
|
}
|
|
|
|
if (selectedItem == dismissDriver)
|
|
{
|
|
DismissDriver();
|
|
}
|
|
}
|
|
|
|
private static void PathMenu_OnCheckboxChange(UIMenu sender, UIMenuCheckboxItem checkboxItem, bool @checked)
|
|
{
|
|
if (checkboxItem == disableAllPaths)
|
|
{
|
|
DisableAllPaths();
|
|
}
|
|
}
|
|
|
|
private static void PathMenu_OnMenuOpen(UIMenu menu)
|
|
{
|
|
var scrollerItems = new List<UIMenuScrollerItem> { directOptions, directDriver, dismissDriver, editPath };
|
|
var checkboxItems = new Dictionary<UIMenuCheckboxItem, RNUIMouseInputHandler.Function>()
|
|
{
|
|
{ disableAllPaths, DisableAllPaths }
|
|
};
|
|
|
|
var selectItems = new Dictionary<UIMenuItem, RNUIMouseInputHandler.Function>()
|
|
{
|
|
{ createNewPath, GoToPathCreationMenu },
|
|
{ editPath, GoToEditPathMenu },
|
|
{ deleteAllPaths, DeleteAllPaths },
|
|
{ directDriver, DirectDriver },
|
|
{ dismissDriver, DismissDriver }
|
|
};
|
|
|
|
RNUIMouseInputHandler.Initialize(menu, scrollerItems, checkboxItems, selectItems);
|
|
}
|
|
}
|
|
}
|