Not a member of Pastebin yet?
Sign Up,
it unlocks many cool features!
- --[[DOWNLOAD THIS SCRIPT]]--
- --[[
- local _p = game:WaitForChild("Players")
- local _plr = _p.ChildAdded:Wait()
- if _plr == _p.LocalPlayer then
- _plr.ChildAdded:Connect(function(cccc)
- if c.Name == "PlayerScriptsLoader" then
- c.Disabled = true
- end
- end)
- end
- ]]
- repeat wait()
- a = pcall(function()
- game:WaitForChild("Players").LocalPlayer:WaitForChild("PlayerScripts").ChildAdded:Connect(function(c)
- if c.Name == "PlayerScriptsLoader"then
- c.Disabled = true
- end
- end)
- end)
- if a == true then break end
- until true == false
- game:WaitForChild("Players").LocalPlayer:WaitForChild("PlayerScripts").ChildAdded:Connect(function(c)
- if c.Name == "PlayerScriptsLoader"then
- c.Disabled = true
- end
- end)
- function _CameraUI()
- local Players = game:GetService("Players")
- local TweenService = game:GetService("TweenService")
- local LocalPlayer = Players.LocalPlayer
- if not LocalPlayer then
- Players:GetPropertyChangedSignal("LocalPlayer"):Wait()
- LocalPlayer = Players.LocalPlayer
- end
- local function waitForChildOfClass(parent, class)
- local child = parent:FindFirstChildOfClass(class)
- while not child or child.ClassName ~= class do
- child = parent.ChildAdded:Wait()
- end
- return child
- end
- local PlayerGui = waitForChildOfClass(LocalPlayer, "PlayerGui")
- local TOAST_OPEN_SIZE = UDim2.new(0, 326, 0, 58)
- local TOAST_CLOSED_SIZE = UDim2.new(0, 80, 0, 58)
- local TOAST_BACKGROUND_COLOR = Color3.fromRGB(32, 32, 32)
- local TOAST_BACKGROUND_TRANS = 0.4
- local TOAST_FOREGROUND_COLOR = Color3.fromRGB(200, 200, 200)
- local TOAST_FOREGROUND_TRANS = 0
- -- Convenient syntax for creating a tree of instanes
- local function create(className)
- return function(props)
- local inst = Instance.new(className)
- local parent = props.Parent
- props.Parent = nil
- for name, val in pairs(props) do
- if type(name) == "string" then
- inst[name] = val
- else
- val.Parent = inst
- end
- end
- -- Only set parent after all other properties are initialized
- inst.Parent = parent
- return inst
- end
- end
- local initialized = false
- local uiRoot
- local toast
- local toastIcon
- local toastUpperText
- local toastLowerText
- local function initializeUI()
- assert(not initialized)
- uiRoot = create("ScreenGui"){
- Name = "RbxCameraUI",
- AutoLocalize = false,
- Enabled = true,
- DisplayOrder = -1, -- Appears behind default developer UI
- IgnoreGuiInset = false,
- ResetOnSpawn = false,
- ZIndexBehavior = Enum.ZIndexBehavior.Sibling,
- create("ImageLabel"){
- Name = "Toast",
- Visible = false,
- AnchorPoint = Vector2.new(0.5, 0),
- BackgroundTransparency = 1,
- BorderSizePixel = 0,
- Position = UDim2.new(0.5, 0, 0, 8),
- Size = TOAST_CLOSED_SIZE,
- Image = "rbxasset://textures/ui/Camera/CameraToast9Slice.png",
- ImageColor3 = TOAST_BACKGROUND_COLOR,
- ImageRectSize = Vector2.new(6, 6),
- ImageTransparency = 1,
- ScaleType = Enum.ScaleType.Slice,
- SliceCenter = Rect.new(3, 3, 3, 3),
- ClipsDescendants = true,
- create("Frame"){
- Name = "IconBuffer",
- BackgroundTransparency = 1,
- BorderSizePixel = 0,
- Position = UDim2.new(0, 0, 0, 0),
- Size = UDim2.new(0, 80, 1, 0),
- create("ImageLabel"){
- Name = "Icon",
- AnchorPoint = Vector2.new(0.5, 0.5),
- BackgroundTransparency = 1,
- Position = UDim2.new(0.5, 0, 0.5, 0),
- Size = UDim2.new(0, 48, 0, 48),
- ZIndex = 2,
- Image = "rbxasset://textures/ui/Camera/CameraToastIcon.png",
- ImageColor3 = TOAST_FOREGROUND_COLOR,
- ImageTransparency = 1,
- }
- },
- create("Frame"){
- Name = "TextBuffer",
- BackgroundTransparency = 1,
- BorderSizePixel = 0,
- Position = UDim2.new(0, 80, 0, 0),
- Size = UDim2.new(1, -80, 1, 0),
- ClipsDescendants = true,
- create("TextLabel"){
- Name = "Upper",
- AnchorPoint = Vector2.new(0, 1),
- BackgroundTransparency = 1,
- Position = UDim2.new(0, 0, 0.5, 0),
- Size = UDim2.new(1, 0, 0, 19),
- Font = Enum.Font.GothamSemibold,
- Text = "Camera control enabled",
- TextColor3 = TOAST_FOREGROUND_COLOR,
- TextTransparency = 1,
- TextSize = 19,
- TextXAlignment = Enum.TextXAlignment.Left,
- TextYAlignment = Enum.TextYAlignment.Center,
- },
- create("TextLabel"){
- Name = "Lower",
- AnchorPoint = Vector2.new(0, 0),
- BackgroundTransparency = 1,
- Position = UDim2.new(0, 0, 0.5, 3),
- Size = UDim2.new(1, 0, 0, 15),
- Font = Enum.Font.Gotham,
- Text = "Right mouse button to toggle",
- TextColor3 = TOAST_FOREGROUND_COLOR,
- TextTransparency = 1,
- TextSize = 15,
- TextXAlignment = Enum.TextXAlignment.Left,
- TextYAlignment = Enum.TextYAlignment.Center,
- },
- },
- },
- Parent = PlayerGui,
- }
- toast = uiRoot.Toast
- toastIcon = toast.IconBuffer.Icon
- toastUpperText = toast.TextBuffer.Upper
- toastLowerText = toast.TextBuffer.Lower
- initialized = true
- end
- local CameraUI = {}
- do
- -- Instantaneously disable the toast or enable for opening later on. Used when switching camera modes.
- function CameraUI.setCameraModeToastEnabled(enabled)
- if not enabled and not initialized then
- return
- end
- if not initialized then
- initializeUI()
- end
- toast.Visible = enabled
- if not enabled then
- CameraUI.setCameraModeToastOpen(false)
- end
- end
- local tweenInfo = TweenInfo.new(0.25, Enum.EasingStyle.Quad, Enum.EasingDirection.Out)
- -- Tween the toast in or out. Toast must be enabled with setCameraModeToastEnabled.
- function CameraUI.setCameraModeToastOpen(open)
- assert(initialized)
- TweenService:Create(toast, tweenInfo, {
- Size = open and TOAST_OPEN_SIZE or TOAST_CLOSED_SIZE,
- ImageTransparency = open and TOAST_BACKGROUND_TRANS or 1,
- }):Play()
- TweenService:Create(toastIcon, tweenInfo, {
- ImageTransparency = open and TOAST_FOREGROUND_TRANS or 1,
- }):Play()
- TweenService:Create(toastUpperText, tweenInfo, {
- TextTransparency = open and TOAST_FOREGROUND_TRANS or 1,
- }):Play()
- TweenService:Create(toastLowerText, tweenInfo, {
- TextTransparency = open and TOAST_FOREGROUND_TRANS or 1,
- }):Play()
- end
- end
- return CameraUI
- end
- function _CameraToggleStateController()
- local Players = game:GetService("Players")
- local UserInputService = game:GetService("UserInputService")
- local GameSettings = UserSettings():GetService("UserGameSettings")
- local LocalPlayer = Players.LocalPlayer
- if not LocalPlayer then
- Players:GetPropertyChangedSignal("LocalPlayer"):Wait()
- LocalPlayer = Players.LocalPlayer
- end
- local Mouse = LocalPlayer:GetMouse()
- local Input = _CameraInput()
- local CameraUI = _CameraUI()
- local lastTogglePan = false
- local lastTogglePanChange = tick()
- local CROSS_MOUSE_ICON = "rbxasset://textures/Cursors/CrossMouseIcon.png"
- local lockStateDirty = false
- local wasTogglePanOnTheLastTimeYouWentIntoFirstPerson = false
- local lastFirstPerson = false
- CameraUI.setCameraModeToastEnabled(false)
- return function(isFirstPerson)
- local togglePan = Input.getTogglePan()
- local toastTimeout = 3
- if isFirstPerson and togglePan ~= lastTogglePan then
- lockStateDirty = true
- end
- if lastTogglePan ~= togglePan or tick() - lastTogglePanChange > toastTimeout then
- local doShow = togglePan and tick() - lastTogglePanChange < toastTimeout
- CameraUI.setCameraModeToastOpen(doShow)
- if togglePan then
- lockStateDirty = false
- end
- lastTogglePanChange = tick()
- lastTogglePan = togglePan
- end
- if isFirstPerson ~= lastFirstPerson then
- if isFirstPerson then
- wasTogglePanOnTheLastTimeYouWentIntoFirstPerson = Input.getTogglePan()
- Input.setTogglePan(true)
- elseif not lockStateDirty then
- Input.setTogglePan(wasTogglePanOnTheLastTimeYouWentIntoFirstPerson)
- end
- end
- if isFirstPerson then
- if Input.getTogglePan() then
- Mouse.Icon = CROSS_MOUSE_ICON
- UserInputService.MouseBehavior = Enum.MouseBehavior.LockCenter
- --GameSettings.RotationType = Enum.RotationType.CameraRelative
- else
- Mouse.Icon = ""
- UserInputService.MouseBehavior = Enum.MouseBehavior.Default
- --GameSettings.RotationType = Enum.RotationType.CameraRelative
- end
- elseif Input.getTogglePan() then
- Mouse.Icon = CROSS_MOUSE_ICON
- UserInputService.MouseBehavior = Enum.MouseBehavior.LockCenter
- GameSettings.RotationType = Enum.RotationType.MovementRelative
- elseif Input.getHoldPan() then
- Mouse.Icon = ""
- UserInputService.MouseBehavior = Enum.MouseBehavior.LockCurrentPosition
- GameSettings.RotationType = Enum.RotationType.MovementRelative
- else
- Mouse.Icon = ""
- UserInputService.MouseBehavior = Enum.MouseBehavior.Default
- GameSettings.RotationType = Enum.RotationType.MovementRelative
- end
- lastFirstPerson = isFirstPerson
- end
- end
- function _CameraInput()
- local UserInputService = game:GetService("UserInputService")
- local MB_TAP_LENGTH = 0.3 -- length of time for a short mouse button tap to be registered
- local rmbDown, rmbUp
- do
- local rmbDownBindable = Instance.new("BindableEvent")
- local rmbUpBindable = Instance.new("BindableEvent")
- rmbDown = rmbDownBindable.Event
- rmbUp = rmbUpBindable.Event
- UserInputService.InputBegan:Connect(function(input, gpe)
- if not gpe and input.UserInputType == Enum.UserInputType.MouseButton2 then
- rmbDownBindable:Fire()
- end
- end)
- UserInputService.InputEnded:Connect(function(input, gpe)
- if input.UserInputType == Enum.UserInputType.MouseButton2 then
- rmbUpBindable:Fire()
- end
- end)
- end
- local holdPan = false
- local togglePan = false
- local lastRmbDown = 0 -- tick() timestamp of the last right mouse button down event
- local CameraInput = {}
- function CameraInput.getHoldPan()
- return holdPan
- end
- function CameraInput.getTogglePan()
- return togglePan
- end
- function CameraInput.getPanning()
- return togglePan or holdPan
- end
- function CameraInput.setTogglePan(value)
- togglePan = value
- end
- local cameraToggleInputEnabled = false
- local rmbDownConnection
- local rmbUpConnection
- function CameraInput.enableCameraToggleInput()
- if cameraToggleInputEnabled then
- return
- end
- cameraToggleInputEnabled = true
- holdPan = false
- togglePan = false
- if rmbDownConnection then
- rmbDownConnection:Disconnect()
- end
- if rmbUpConnection then
- rmbUpConnection:Disconnect()
- end
- rmbDownConnection = rmbDown:Connect(function()
- holdPan = true
- lastRmbDown = tick()
- end)
- rmbUpConnection = rmbUp:Connect(function()
- holdPan = false
- if tick() - lastRmbDown < MB_TAP_LENGTH and (togglePan or UserInputService:GetMouseDelta().Magnitude < 2) then
- togglePan = not togglePan
- end
- end)
- end
- function CameraInput.disableCameraToggleInput()
- if not cameraToggleInputEnabled then
- return
- end
- cameraToggleInputEnabled = false
- if rmbDownConnection then
- rmbDownConnection:Disconnect()
- rmbDownConnection = nil
- end
- if rmbUpConnection then
- rmbUpConnection:Disconnect()
- rmbUpConnection = nil
- end
- end
- return CameraInput
- end
- function _BaseCamera()
- --[[
- BaseCamera - Abstract base class for camera control modules
- 2018 Camera Update - AllYourBlox
- --]]
- --[[ Local Constants ]]--
- local UNIT_Z = Vector3.new(0,0,1)
- local X1_Y0_Z1 = Vector3.new(1,0,1) --Note: not a unit vector, used for projecting onto XZ plane
- local THUMBSTICK_DEADZONE = 0.2
- local DEFAULT_DISTANCE = 12.5 -- Studs
- local PORTRAIT_DEFAULT_DISTANCE = 25 -- Studs
- local FIRST_PERSON_DISTANCE_THRESHOLD = 1.0 -- Below this value, snap into first person
- local CAMERA_ACTION_PRIORITY = Enum.ContextActionPriority.Default.Value
- -- Note: DotProduct check in CoordinateFrame::lookAt() prevents using values within about
- -- 8.11 degrees of the +/- Y axis, that's why these limits are currently 80 degrees
- local MIN_Y = math.rad(-80)
- local MAX_Y = math.rad(80)
- local TOUCH_ADJUST_AREA_UP = math.rad(30)
- local TOUCH_ADJUST_AREA_DOWN = math.rad(-15)
- local TOUCH_SENSITIVTY_ADJUST_MAX_Y = 2.1
- local TOUCH_SENSITIVTY_ADJUST_MIN_Y = 0.5
- local VR_ANGLE = math.rad(15)
- local VR_LOW_INTENSITY_ROTATION = Vector2.new(math.rad(15), 0)
- local VR_HIGH_INTENSITY_ROTATION = Vector2.new(math.rad(45), 0)
- local VR_LOW_INTENSITY_REPEAT = 0.1
- local VR_HIGH_INTENSITY_REPEAT = 0.4
- local ZERO_VECTOR2 = Vector2.new(0,0)
- local ZERO_VECTOR3 = Vector3.new(0,0,0)
- local TOUCH_SENSITIVTY = Vector2.new(0.00945 * math.pi, 0.003375 * math.pi)
- local MOUSE_SENSITIVITY = Vector2.new( 0.002 * math.pi, 0.0015 * math.pi )
- local SEAT_OFFSET = Vector3.new(0,5,0)
- local VR_SEAT_OFFSET = Vector3.new(0,4,0)
- local HEAD_OFFSET = Vector3.new(0,1.5,0)
- local R15_HEAD_OFFSET = Vector3.new(0, 1.5, 0)
- local R15_HEAD_OFFSET_NO_SCALING = Vector3.new(0, 2, 0)
- local HUMANOID_ROOT_PART_SIZE = Vector3.new(2, 2, 1)
- local GAMEPAD_ZOOM_STEP_1 = 0
- local GAMEPAD_ZOOM_STEP_2 = 10
- local GAMEPAD_ZOOM_STEP_3 = 20
- local PAN_SENSITIVITY = 20
- local ZOOM_SENSITIVITY_CURVATURE = 0.5
- local abs = math.abs
- local sign = math.sign
- local FFlagUserCameraToggle do
- local success, result = pcall(function()
- return UserSettings():IsUserFeatureEnabled("UserCameraToggle")
- end)
- FFlagUserCameraToggle = success and result
- end
- local FFlagUserDontAdjustSensitvityForPortrait do
- local success, result = pcall(function()
- return UserSettings():IsUserFeatureEnabled("UserDontAdjustSensitvityForPortrait")
- end)
- FFlagUserDontAdjustSensitvityForPortrait = success and result
- end
- local FFlagUserFixZoomInZoomOutDiscrepancy do
- local success, result = pcall(function()
- return UserSettings():IsUserFeatureEnabled("UserFixZoomInZoomOutDiscrepancy")
- end)
- FFlagUserFixZoomInZoomOutDiscrepancy = success and result
- end
- local Util = _CameraUtils()
- local ZoomController = _ZoomController()
- local CameraToggleStateController = _CameraToggleStateController()
- local CameraInput = _CameraInput()
- local CameraUI = _CameraUI()
- --[[ Roblox Services ]]--
- local Players = game:GetService("Players")
- local UserInputService = game:GetService("UserInputService")
- local StarterGui = game:GetService("StarterGui")
- local GuiService = game:GetService("GuiService")
- local ContextActionService = game:GetService("ContextActionService")
- local VRService = game:GetService("VRService")
- local UserGameSettings = UserSettings():GetService("UserGameSettings")
- local player = Players.LocalPlayer
- --[[ The Module ]]--
- local BaseCamera = {}
- BaseCamera.__index = BaseCamera
- function BaseCamera.new()
- local self = setmetatable({}, BaseCamera)
- -- So that derived classes have access to this
- self.FIRST_PERSON_DISTANCE_THRESHOLD = FIRST_PERSON_DISTANCE_THRESHOLD
- self.cameraType = nil
- self.cameraMovementMode = nil
- self.lastCameraTransform = nil
- self.rotateInput = ZERO_VECTOR2
- self.userPanningCamera = false
- self.lastUserPanCamera = tick()
- self.humanoidRootPart = nil
- self.humanoidCache = {}
- -- Subject and position on last update call
- self.lastSubject = nil
- self.lastSubjectPosition = Vector3.new(0,5,0)
- -- These subject distance members refer to the nominal camera-to-subject follow distance that the camera
- -- is trying to maintain, not the actual measured value.
- -- The default is updated when screen orientation or the min/max distances change,
- -- to be sure the default is always in range and appropriate for the orientation.
- self.defaultSubjectDistance = math.clamp(DEFAULT_DISTANCE, player.CameraMinZoomDistance, player.CameraMaxZoomDistance)
- self.currentSubjectDistance = math.clamp(DEFAULT_DISTANCE, player.CameraMinZoomDistance, player.CameraMaxZoomDistance)
- self.inFirstPerson = false
- self.inMouseLockedMode = false
- self.portraitMode = false
- self.isSmallTouchScreen = false
- -- Used by modules which want to reset the camera angle on respawn.
- self.resetCameraAngle = true
- self.enabled = false
- -- Input Event Connections
- self.inputBeganConn = nil
- self.inputChangedConn = nil
- self.inputEndedConn = nil
- self.startPos = nil
- self.lastPos = nil
- self.panBeginLook = nil
- self.panEnabled = true
- self.keyPanEnabled = true
- self.distanceChangeEnabled = true
- self.PlayerGui = nil
- self.cameraChangedConn = nil
- self.viewportSizeChangedConn = nil
- self.boundContextActions = {}
- -- VR Support
- self.shouldUseVRRotation = false
- self.VRRotationIntensityAvailable = false
- self.lastVRRotationIntensityCheckTime = 0
- self.lastVRRotationTime = 0
- self.vrRotateKeyCooldown = {}
- self.cameraTranslationConstraints = Vector3.new(1, 1, 1)
- self.humanoidJumpOrigin = nil
- self.trackingHumanoid = nil
- self.cameraFrozen = false
- self.subjectStateChangedConn = nil
- -- Gamepad support
- self.activeGamepad = nil
- self.gamepadPanningCamera = false
- self.lastThumbstickRotate = nil
- self.numOfSeconds = 0.7
- self.currentSpeed = 0
- self.maxSpeed = 6
- self.vrMaxSpeed = 4
- self.lastThumbstickPos = Vector2.new(0,0)
- self.ySensitivity = 0.65
- self.lastVelocity = nil
- self.gamepadConnectedConn = nil
- self.gamepadDisconnectedConn = nil
- self.currentZoomSpeed = 1.0
- self.L3ButtonDown = false
- self.dpadLeftDown = false
- self.dpadRightDown = false
- -- Touch input support
- self.isDynamicThumbstickEnabled = false
- self.fingerTouches = {}
- self.dynamicTouchInput = nil
- self.numUnsunkTouches = 0
- self.inputStartPositions = {}
- self.inputStartTimes = {}
- self.startingDiff = nil
- self.pinchBeginZoom = nil
- self.userPanningTheCamera = false
- self.touchActivateConn = nil
- -- Mouse locked formerly known as shift lock mode
- self.mouseLockOffset = ZERO_VECTOR3
- -- [[ NOTICE ]] --
- -- Initialization things used to always execute at game load time, but now these camera modules are instantiated
- -- when needed, so the code here may run well after the start of the game
- if player.Character then
- self:OnCharacterAdded(player.Character)
- end
- player.CharacterAdded:Connect(function(char)
- self:OnCharacterAdded(char)
- end)
- if self.cameraChangedConn then self.cameraChangedConn:Disconnect() end
- self.cameraChangedConn = workspace:GetPropertyChangedSignal("CurrentCamera"):Connect(function()
- self:OnCurrentCameraChanged()
- end)
- self:OnCurrentCameraChanged()
- if self.playerCameraModeChangeConn then self.playerCameraModeChangeConn:Disconnect() end
- self.playerCameraModeChangeConn = player:GetPropertyChangedSignal("CameraMode"):Connect(function()
- self:OnPlayerCameraPropertyChange()
- end)
- if self.minDistanceChangeConn then self.minDistanceChangeConn:Disconnect() end
- self.minDistanceChangeConn = player:GetPropertyChangedSignal("CameraMinZoomDistance"):Connect(function()
- self:OnPlayerCameraPropertyChange()
- end)
- if self.maxDistanceChangeConn then self.maxDistanceChangeConn:Disconnect() end
- self.maxDistanceChangeConn = player:GetPropertyChangedSignal("CameraMaxZoomDistance"):Connect(function()
- self:OnPlayerCameraPropertyChange()
- end)
- if self.playerDevTouchMoveModeChangeConn then self.playerDevTouchMoveModeChangeConn:Disconnect() end
- self.playerDevTouchMoveModeChangeConn = player:GetPropertyChangedSignal("DevTouchMovementMode"):Connect(function()
- self:OnDevTouchMovementModeChanged()
- end)
- self:OnDevTouchMovementModeChanged() -- Init
- if self.gameSettingsTouchMoveMoveChangeConn then self.gameSettingsTouchMoveMoveChangeConn:Disconnect() end
- self.gameSettingsTouchMoveMoveChangeConn = UserGameSettings:GetPropertyChangedSignal("TouchMovementMode"):Connect(function()
- self:OnGameSettingsTouchMovementModeChanged()
- end)
- self:OnGameSettingsTouchMovementModeChanged() -- Init
- UserGameSettings:SetCameraYInvertVisible()
- UserGameSettings:SetGamepadCameraSensitivityVisible()
- self.hasGameLoaded = game:IsLoaded()
- if not self.hasGameLoaded then
- self.gameLoadedConn = game.Loaded:Connect(function()
- self.hasGameLoaded = true
- self.gameLoadedConn:Disconnect()
- self.gameLoadedConn = nil
- end)
- end
- self:OnPlayerCameraPropertyChange()
- return self
- end
- function BaseCamera:GetModuleName()
- return "BaseCamera"
- end
- function BaseCamera:OnCharacterAdded(char)
- self.resetCameraAngle = self.resetCameraAngle or self:GetEnabled()
- self.humanoidRootPart = nil
- if UserInputService.TouchEnabled then
- self.PlayerGui = player:WaitForChild("PlayerGui")
- for _, child in ipairs(char:GetChildren()) do
- if child:IsA("Tool") then
- self.isAToolEquipped = true
- end
- end
- char.ChildAdded:Connect(function(child)
- if child:IsA("Tool") then
- self.isAToolEquipped = true
- end
- end)
- char.ChildRemoved:Connect(function(child)
- if child:IsA("Tool") then
- self.isAToolEquipped = false
- end
- end)
- end
- end
- function BaseCamera:GetHumanoidRootPart()
- if not self.humanoidRootPart then
- if player.Character then
- local humanoid = player.Character:FindFirstChildOfClass("Humanoid")
- if humanoid then
- self.humanoidRootPart = humanoid.RootPart
- end
- end
- end
- return self.humanoidRootPart
- end
- function BaseCamera:GetBodyPartToFollow(humanoid, isDead)
- -- If the humanoid is dead, prefer the head part if one still exists as a sibling of the humanoid
- if humanoid:GetState() == Enum.HumanoidStateType.Dead then
- local character = humanoid.Parent
- if character and character:IsA("Model") then
- return character:FindFirstChild("Head") or humanoid.RootPart
- end
- end
- return humanoid.RootPart
- end
- function BaseCamera:GetSubjectPosition()
- local result = self.lastSubjectPosition
- local camera = game.Workspace.CurrentCamera
- local cameraSubject = camera and camera.CameraSubject
- if cameraSubject then
- if cameraSubject:IsA("Humanoid") then
- local humanoid = cameraSubject
- local humanoidIsDead = humanoid:GetState() == Enum.HumanoidStateType.Dead
- if VRService.VREnabled and humanoidIsDead and humanoid == self.lastSubject then
- result = self.lastSubjectPosition
- else
- local bodyPartToFollow = humanoid.RootPart
- -- If the humanoid is dead, prefer their head part as a follow target, if it exists
- if humanoidIsDead then
- if humanoid.Parent and humanoid.Parent:IsA("Model") then
- bodyPartToFollow = humanoid.Parent:FindFirstChild("Head") or bodyPartToFollow
- end
- end
- if bodyPartToFollow and bodyPartToFollow:IsA("BasePart") then
- local heightOffset
- if humanoid.RigType == Enum.HumanoidRigType.R15 then
- if humanoid.AutomaticScalingEnabled then
- heightOffset = R15_HEAD_OFFSET
- if bodyPartToFollow == humanoid.RootPart then
- local rootPartSizeOffset = (humanoid.RootPart.Size.Y/2) - (HUMANOID_ROOT_PART_SIZE.Y/2)
- heightOffset = heightOffset + Vector3.new(0, rootPartSizeOffset, 0)
- end
- else
- heightOffset = R15_HEAD_OFFSET_NO_SCALING
- end
- else
- heightOffset = HEAD_OFFSET
- end
- if humanoidIsDead then
- heightOffset = ZERO_VECTOR3
- end
- result = bodyPartToFollow.CFrame.p + bodyPartToFollow.CFrame:vectorToWorldSpace(heightOffset + humanoid.CameraOffset)
- end
- end
- elseif cameraSubject:IsA("VehicleSeat") then
- local offset = SEAT_OFFSET
- if VRService.VREnabled then
- offset = VR_SEAT_OFFSET
- end
- result = cameraSubject.CFrame.p + cameraSubject.CFrame:vectorToWorldSpace(offset)
- elseif cameraSubject:IsA("SkateboardPlatform") then
- result = cameraSubject.CFrame.p + SEAT_OFFSET
- elseif cameraSubject:IsA("BasePart") then
- result = cameraSubject.CFrame.p
- elseif cameraSubject:IsA("Model") then
- if cameraSubject.PrimaryPart then
- result = cameraSubject:GetPrimaryPartCFrame().p
- else
- result = cameraSubject:GetModelCFrame().p
- end
- end
- else
- -- cameraSubject is nil
- -- Note: Previous RootCamera did not have this else case and let self.lastSubject and self.lastSubjectPosition
- -- both get set to nil in the case of cameraSubject being nil. This function now exits here to preserve the
- -- last set valid values for these, as nil values are not handled cases
- return
- end
- self.lastSubject = cameraSubject
- self.lastSubjectPosition = result
- return result
- end
- function BaseCamera:UpdateDefaultSubjectDistance()
- if self.portraitMode then
- self.defaultSubjectDistance = math.clamp(PORTRAIT_DEFAULT_DISTANCE, player.CameraMinZoomDistance, player.CameraMaxZoomDistance)
- else
- self.defaultSubjectDistance = math.clamp(DEFAULT_DISTANCE, player.CameraMinZoomDistance, player.CameraMaxZoomDistance)
- end
- end
- function BaseCamera:OnViewportSizeChanged()
- local camera = game.Workspace.CurrentCamera
- local size = camera.ViewportSize
- self.portraitMode = size.X < size.Y
- self.isSmallTouchScreen = UserInputService.TouchEnabled and (size.Y < 500 or size.X < 700)
- self:UpdateDefaultSubjectDistance()
- end
- -- Listener for changes to workspace.CurrentCamera
- function BaseCamera:OnCurrentCameraChanged()
- if UserInputService.TouchEnabled then
- if self.viewportSizeChangedConn then
- self.viewportSizeChangedConn:Disconnect()
- self.viewportSizeChangedConn = nil
- end
- local newCamera = game.Workspace.CurrentCamera
- if newCamera then
- self:OnViewportSizeChanged()
- self.viewportSizeChangedConn = newCamera:GetPropertyChangedSignal("ViewportSize"):Connect(function()
- self:OnViewportSizeChanged()
- end)
- end
- end
- -- VR support additions
- if self.cameraSubjectChangedConn then
- self.cameraSubjectChangedConn:Disconnect()
- self.cameraSubjectChangedConn = nil
- end
- local camera = game.Workspace.CurrentCamera
- if camera then
- self.cameraSubjectChangedConn = camera:GetPropertyChangedSignal("CameraSubject"):Connect(function()
- self:OnNewCameraSubject()
- end)
- self:OnNewCameraSubject()
- end
- end
- function BaseCamera:OnDynamicThumbstickEnabled()
- if UserInputService.TouchEnabled then
- self.isDynamicThumbstickEnabled = true
- end
- end
- function BaseCamera:OnDynamicThumbstickDisabled()
- self.isDynamicThumbstickEnabled = false
- end
- function BaseCamera:OnGameSettingsTouchMovementModeChanged()
- if player.DevTouchMovementMode == Enum.DevTouchMovementMode.UserChoice then
- if (UserGameSettings.TouchMovementMode == Enum.TouchMovementMode.DynamicThumbstick
- or UserGameSettings.TouchMovementMode == Enum.TouchMovementMode.Default) then
- self:OnDynamicThumbstickEnabled()
- else
- self:OnDynamicThumbstickDisabled()
- end
- end
- end
- function BaseCamera:OnDevTouchMovementModeChanged()
- if player.DevTouchMovementMode.Name == "DynamicThumbstick" then
- self:OnDynamicThumbstickEnabled()
- else
- self:OnGameSettingsTouchMovementModeChanged()
- end
- end
- function BaseCamera:OnPlayerCameraPropertyChange()
- -- This call forces re-evaluation of player.CameraMode and clamping to min/max distance which may have changed
- self:SetCameraToSubjectDistance(self.currentSubjectDistance)
- end
- function BaseCamera:GetCameraHeight()
- if VRService.VREnabled and not self.inFirstPerson then
- return math.sin(VR_ANGLE) * self.currentSubjectDistance
- end
- return 0
- end
- function BaseCamera:InputTranslationToCameraAngleChange(translationVector, sensitivity)
- if not FFlagUserDontAdjustSensitvityForPortrait then
- local camera = game.Workspace.CurrentCamera
- if camera and camera.ViewportSize.X > 0 and camera.ViewportSize.Y > 0 and (camera.ViewportSize.Y > camera.ViewportSize.X) then
- -- Screen has portrait orientation, swap X and Y sensitivity
- return translationVector * Vector2.new( sensitivity.Y, sensitivity.X)
- end
- end
- return translationVector * sensitivity
- end
- function BaseCamera:Enable(enable)
- if self.enabled ~= enable then
- self.enabled = enable
- if self.enabled then
- self:ConnectInputEvents()
- self:BindContextActions()
- if player.CameraMode == Enum.CameraMode.LockFirstPerson then
- self.currentSubjectDistance = 0.5
- if not self.inFirstPerson then
- self:EnterFirstPerson()
- end
- end
- else
- self:DisconnectInputEvents()
- self:UnbindContextActions()
- -- Clean up additional event listeners and reset a bunch of properties
- self:Cleanup()
- end
- end
- end
- function BaseCamera:GetEnabled()
- return self.enabled
- end
- function BaseCamera:OnInputBegan(input, processed)
- if input.UserInputType == Enum.UserInputType.Touch then
- self:OnTouchBegan(input, processed)
- elseif input.UserInputType == Enum.UserInputType.MouseButton2 then
- self:OnMouse2Down(input, processed)
- elseif input.UserInputType == Enum.UserInputType.MouseButton3 then
- self:OnMouse3Down(input, processed)
- end
- end
- function BaseCamera:OnInputChanged(input, processed)
- if input.UserInputType == Enum.UserInputType.Touch then
- self:OnTouchChanged(input, processed)
- elseif input.UserInputType == Enum.UserInputType.MouseMovement then
- self:OnMouseMoved(input, processed)
- end
- end
- function BaseCamera:OnInputEnded(input, processed)
- if input.UserInputType == Enum.UserInputType.Touch then
- self:OnTouchEnded(input, processed)
- elseif input.UserInputType == Enum.UserInputType.MouseButton2 then
- self:OnMouse2Up(input, processed)
- elseif input.UserInputType == Enum.UserInputType.MouseButton3 then
- self:OnMouse3Up(input, processed)
- end
- end
- function BaseCamera:OnPointerAction(wheel, pan, pinch, processed)
- if processed then
- return
- end
- if pan.Magnitude > 0 then
- local inversionVector = Vector2.new(1, UserGameSettings:GetCameraYInvertValue())
- local rotateDelta = self:InputTranslationToCameraAngleChange(PAN_SENSITIVITY*pan, MOUSE_SENSITIVITY)*inversionVector
- self.rotateInput = self.rotateInput + rotateDelta
- end
- local zoom = self.currentSubjectDistance
- local zoomDelta = -(wheel + pinch)
- if abs(zoomDelta) > 0 then
- local newZoom
- if self.inFirstPerson and zoomDelta > 0 then
- newZoom = FIRST_PERSON_DISTANCE_THRESHOLD
- else
- if FFlagUserFixZoomInZoomOutDiscrepancy then
- if (zoomDelta > 0) then
- newZoom = zoom + zoomDelta*(1 + zoom*ZOOM_SENSITIVITY_CURVATURE)
- else
- newZoom = (zoom + zoomDelta) / (1 - zoomDelta*ZOOM_SENSITIVITY_CURVATURE)
- end
- else
- newZoom = zoom + zoomDelta*(1 + zoom*ZOOM_SENSITIVITY_CURVATURE)
- end
- end
- self:SetCameraToSubjectDistance(newZoom)
- end
- end
- function BaseCamera:ConnectInputEvents()
- self.pointerActionConn = UserInputService.PointerAction:Connect(function(wheel, pan, pinch, processed)
- self:OnPointerAction(wheel, pan, pinch, processed)
- end)
- self.inputBeganConn = UserInputService.InputBegan:Connect(function(input, processed)
- self:OnInputBegan(input, processed)
- end)
- self.inputChangedConn = UserInputService.InputChanged:Connect(function(input, processed)
- self:OnInputChanged(input, processed)
- end)
- self.inputEndedConn = UserInputService.InputEnded:Connect(function(input, processed)
- self:OnInputEnded(input, processed)
- end)
- self.menuOpenedConn = GuiService.MenuOpened:connect(function()
- self:ResetInputStates()
- end)
- self.gamepadConnectedConn = UserInputService.GamepadDisconnected:connect(function(gamepadEnum)
- if self.activeGamepad ~= gamepadEnum then return end
- self.activeGamepad = nil
- self:AssignActivateGamepad()
- end)
- self.gamepadDisconnectedConn = UserInputService.GamepadConnected:connect(function(gamepadEnum)
- if self.activeGamepad == nil then
- self:AssignActivateGamepad()
- end
- end)
- self:AssignActivateGamepad()
- if not FFlagUserCameraToggle then
- self:UpdateMouseBehavior()
- end
- end
- function BaseCamera:BindContextActions()
- self:BindGamepadInputActions()
- self:BindKeyboardInputActions()
- end
- function BaseCamera:AssignActivateGamepad()
- local connectedGamepads = UserInputService:GetConnectedGamepads()
- if #connectedGamepads > 0 then
- for i = 1, #connectedGamepads do
- if self.activeGamepad == nil then
- self.activeGamepad = connectedGamepads[i]
- elseif connectedGamepads[i].Value < self.activeGamepad.Value then
- self.activeGamepad = connectedGamepads[i]
- end
- end
- end
- if self.activeGamepad == nil then -- nothing is connected, at least set up for gamepad1
- self.activeGamepad = Enum.UserInputType.Gamepad1
- end
- end
- function BaseCamera:DisconnectInputEvents()
- if self.inputBeganConn then
- self.inputBeganConn:Disconnect()
- self.inputBeganConn = nil
- end
- if self.inputChangedConn then
- self.inputChangedConn:Disconnect()
- self.inputChangedConn = nil
- end
- if self.inputEndedConn then
- self.inputEndedConn:Disconnect()
- self.inputEndedConn = nil
- end
- end
- function BaseCamera:UnbindContextActions()
- for i = 1, #self.boundContextActions do
- ContextActionService:UnbindAction(self.boundContextActions[i])
- end
- self.boundContextActions = {}
- end
- function BaseCamera:Cleanup()
- if self.pointerActionConn then
- self.pointerActionConn:Disconnect()
- self.pointerActionConn = nil
- end
- if self.menuOpenedConn then
- self.menuOpenedConn:Disconnect()
- self.menuOpenedConn = nil
- end
- if self.mouseLockToggleConn then
- self.mouseLockToggleConn:Disconnect()
- self.mouseLockToggleConn = nil
- end
- if self.gamepadConnectedConn then
- self.gamepadConnectedConn:Disconnect()
- self.gamepadConnectedConn = nil
- end
- if self.gamepadDisconnectedConn then
- self.gamepadDisconnectedConn:Disconnect()
- self.gamepadDisconnectedConn = nil
- end
- if self.subjectStateChangedConn then
- self.subjectStateChangedConn:Disconnect()
- self.subjectStateChangedConn = nil
- end
- if self.viewportSizeChangedConn then
- self.viewportSizeChangedConn:Disconnect()
- self.viewportSizeChangedConn = nil
- end
- if self.touchActivateConn then
- self.touchActivateConn:Disconnect()
- self.touchActivateConn = nil
- end
- self.turningLeft = false
- self.turningRight = false
- self.lastCameraTransform = nil
- self.lastSubjectCFrame = nil
- self.userPanningTheCamera = false
- self.rotateInput = Vector2.new()
- self.gamepadPanningCamera = Vector2.new(0,0)
- -- Reset input states
- self.startPos = nil
- self.lastPos = nil
- self.panBeginLook = nil
- self.isRightMouseDown = false
- self.isMiddleMouseDown = false
- self.fingerTouches = {}
- self.dynamicTouchInput = nil
- self.numUnsunkTouches = 0
- self.startingDiff = nil
- self.pinchBeginZoom = nil
- -- Unlock mouse for example if right mouse button was being held down
- if UserInputService.MouseBehavior ~= Enum.MouseBehavior.LockCenter then
- UserInputService.MouseBehavior = Enum.MouseBehavior.Default
- end
- end
- -- This is called when settings menu is opened
- function BaseCamera:ResetInputStates()
- self.isRightMouseDown = false
- self.isMiddleMouseDown = false
- self:OnMousePanButtonReleased() -- this function doesn't seem to actually need parameters
- if UserInputService.TouchEnabled then
- --[[menu opening was causing serious touch issues
- this should disable all active touch events if
- they're active when menu opens.]]
- for inputObject in pairs(self.fingerTouches) do
- self.fingerTouches[inputObject] = nil
- end
- self.dynamicTouchInput = nil
- self.panBeginLook = nil
- self.startPos = nil
- self.lastPos = nil
- self.userPanningTheCamera = false
- self.startingDiff = nil
- self.pinchBeginZoom = nil
- self.numUnsunkTouches = 0
- end
- end
- function BaseCamera:GetGamepadPan(name, state, input)
- if input.UserInputType == self.activeGamepad and input.KeyCode == Enum.KeyCode.Thumbstick2 then
- -- if self.L3ButtonDown then
- -- -- L3 Thumbstick is depressed, right stick controls dolly in/out
- -- if (input.Position.Y > THUMBSTICK_DEADZONE) then
- -- self.currentZoomSpeed = 0.96
- -- elseif (input.Position.Y < -THUMBSTICK_DEADZONE) then
- -- self.currentZoomSpeed = 1.04
- -- else
- -- self.currentZoomSpeed = 1.00
- -- end
- -- else
- if state == Enum.UserInputState.Cancel then
- self.gamepadPanningCamera = ZERO_VECTOR2
- return
- end
- local inputVector = Vector2.new(input.Position.X, -input.Position.Y)
- if inputVector.magnitude > THUMBSTICK_DEADZONE then
- self.gamepadPanningCamera = Vector2.new(input.Position.X, -input.Position.Y)
- else
- self.gamepadPanningCamera = ZERO_VECTOR2
- end
- --end
- return Enum.ContextActionResult.Sink
- end
- return Enum.ContextActionResult.Pass
- end
- function BaseCamera:DoKeyboardPanTurn(name, state, input)
- if not self.hasGameLoaded and VRService.VREnabled then
- return Enum.ContextActionResult.Pass
- end
- if state == Enum.UserInputState.Cancel then
- self.turningLeft = false
- self.turningRight = false
- return Enum.ContextActionResult.Sink
- end
- if self.panBeginLook == nil and self.keyPanEnabled then
- if input.KeyCode == Enum.KeyCode.Left then
- self.turningLeft = state == Enum.UserInputState.Begin
- elseif input.KeyCode == Enum.KeyCode.Right then
- self.turningRight = state == Enum.UserInputState.Begin
- end
- return Enum.ContextActionResult.Sink
- end
- return Enum.ContextActionResult.Pass
- end
- function BaseCamera:DoPanRotateCamera(rotateAngle)
- local angle = Util.RotateVectorByAngleAndRound(self:GetCameraLookVector() * Vector3.new(1,0,1), rotateAngle, math.pi*0.25)
- if angle ~= 0 then
- self.rotateInput = self.rotateInput + Vector2.new(angle, 0)
- self.lastUserPanCamera = tick()
- self.lastCameraTransform = nil
- end
- end
- function BaseCamera:DoGamepadZoom(name, state, input)
- if input.UserInputType == self.activeGamepad then
- if input.KeyCode == Enum.KeyCode.ButtonR3 then
- if state == Enum.UserInputState.Begin then
- if self.distanceChangeEnabled then
- local dist = self:GetCameraToSubjectDistance()
- if dist > (GAMEPAD_ZOOM_STEP_2 + GAMEPAD_ZOOM_STEP_3)/2 then
- self:SetCameraToSubjectDistance(GAMEPAD_ZOOM_STEP_2)
- elseif dist > (GAMEPAD_ZOOM_STEP_1 + GAMEPAD_ZOOM_STEP_2)/2 then
- self:SetCameraToSubjectDistance(GAMEPAD_ZOOM_STEP_1)
- else
- self:SetCameraToSubjectDistance(GAMEPAD_ZOOM_STEP_3)
- end
- end
- end
- elseif input.KeyCode == Enum.KeyCode.DPadLeft then
- self.dpadLeftDown = (state == Enum.UserInputState.Begin)
- elseif input.KeyCode == Enum.KeyCode.DPadRight then
- self.dpadRightDown = (state == Enum.UserInputState.Begin)
- end
- if self.dpadLeftDown then
- self.currentZoomSpeed = 1.04
- elseif self.dpadRightDown then
- self.currentZoomSpeed = 0.96
- else
- self.currentZoomSpeed = 1.00
- end
- return Enum.ContextActionResult.Sink
- end
- return Enum.ContextActionResult.Pass
- -- elseif input.UserInputType == self.activeGamepad and input.KeyCode == Enum.KeyCode.ButtonL3 then
- -- if (state == Enum.UserInputState.Begin) then
- -- self.L3ButtonDown = true
- -- elseif (state == Enum.UserInputState.End) then
- -- self.L3ButtonDown = false
- -- self.currentZoomSpeed = 1.00
- -- end
- -- end
- end
- function BaseCamera:DoKeyboardZoom(name, state, input)
- if not self.hasGameLoaded and VRService.VREnabled then
- return Enum.ContextActionResult.Pass
- end
- if state ~= Enum.UserInputState.Begin then
- return Enum.ContextActionResult.Pass
- end
- if self.distanceChangeEnabled and player.CameraMode ~= Enum.CameraMode.LockFirstPerson then
- if input.KeyCode == Enum.KeyCode.I then
- self:SetCameraToSubjectDistance( self.currentSubjectDistance - 5 )
- elseif input.KeyCode == Enum.KeyCode.O then
- self:SetCameraToSubjectDistance( self.currentSubjectDistance + 5 )
- end
- return Enum.ContextActionResult.Sink
- end
- return Enum.ContextActionResult.Pass
- end
- function BaseCamera:BindAction(actionName, actionFunc, createTouchButton, ...)
- table.insert(self.boundContextActions, actionName)
- ContextActionService:BindActionAtPriority(actionName, actionFunc, createTouchButton,
- CAMERA_ACTION_PRIORITY, ...)
- end
- function BaseCamera:BindGamepadInputActions()
- self:BindAction("BaseCameraGamepadPan", function(name, state, input) return self:GetGamepadPan(name, state, input) end,
- false, Enum.KeyCode.Thumbstick2)
- self:BindAction("BaseCameraGamepadZoom", function(name, state, input) return self:DoGamepadZoom(name, state, input) end,
- false, Enum.KeyCode.DPadLeft, Enum.KeyCode.DPadRight, Enum.KeyCode.ButtonR3)
- end
- function BaseCamera:BindKeyboardInputActions()
- self:BindAction("BaseCameraKeyboardPanArrowKeys", function(name, state, input) return self:DoKeyboardPanTurn(name, state, input) end,
- false, Enum.KeyCode.Left, Enum.KeyCode.Right)
- self:BindAction("BaseCameraKeyboardZoom", function(name, state, input) return self:DoKeyboardZoom(name, state, input) end,
- false, Enum.KeyCode.I, Enum.KeyCode.O)
- end
- local function isInDynamicThumbstickArea(input)
- local playerGui = player:FindFirstChildOfClass("PlayerGui")
- local touchGui = playerGui and playerGui:FindFirstChild("TouchGui")
- local touchFrame = touchGui and touchGui:FindFirstChild("TouchControlFrame")
- local thumbstickFrame = touchFrame and touchFrame:FindFirstChild("DynamicThumbstickFrame")
- if not thumbstickFrame then
- return false
- end
- local frameCornerTopLeft = thumbstickFrame.AbsolutePosition
- local frameCornerBottomRight = frameCornerTopLeft + thumbstickFrame.AbsoluteSize
- if input.Position.X >= frameCornerTopLeft.X and input.Position.Y >= frameCornerTopLeft.Y then
- if input.Position.X <= frameCornerBottomRight.X and input.Position.Y <= frameCornerBottomRight.Y then
- return true
- end
- end
- return false
- end
- ---Adjusts the camera Y touch Sensitivity when moving away from the center and in the TOUCH_SENSITIVTY_ADJUST_AREA
- function BaseCamera:AdjustTouchSensitivity(delta, sensitivity)
- local cameraCFrame = game.Workspace.CurrentCamera and game.Workspace.CurrentCamera.CFrame
- if not cameraCFrame then
- return sensitivity
- end
- local currPitchAngle = cameraCFrame:ToEulerAnglesYXZ()
- local multiplierY = TOUCH_SENSITIVTY_ADJUST_MAX_Y
- if currPitchAngle > TOUCH_ADJUST_AREA_UP and delta.Y < 0 then
- local fractionAdjust = (currPitchAngle - TOUCH_ADJUST_AREA_UP)/(MAX_Y - TOUCH_ADJUST_AREA_UP)
- fractionAdjust = 1 - (1 - fractionAdjust)^3
- multiplierY = TOUCH_SENSITIVTY_ADJUST_MAX_Y - fractionAdjust * (
- TOUCH_SENSITIVTY_ADJUST_MAX_Y - TOUCH_SENSITIVTY_ADJUST_MIN_Y)
- elseif currPitchAngle < TOUCH_ADJUST_AREA_DOWN and delta.Y > 0 then
- local fractionAdjust = (currPitchAngle - TOUCH_ADJUST_AREA_DOWN)/(MIN_Y - TOUCH_ADJUST_AREA_DOWN)
- fractionAdjust = 1 - (1 - fractionAdjust)^3
- multiplierY = TOUCH_SENSITIVTY_ADJUST_MAX_Y - fractionAdjust * (
- TOUCH_SENSITIVTY_ADJUST_MAX_Y - TOUCH_SENSITIVTY_ADJUST_MIN_Y)
- end
- return Vector2.new(
- sensitivity.X,
- sensitivity.Y * multiplierY
- )
- end
- function BaseCamera:OnTouchBegan(input, processed)
- local canUseDynamicTouch = self.isDynamicThumbstickEnabled and not processed
- if canUseDynamicTouch then
- if self.dynamicTouchInput == nil and isInDynamicThumbstickArea(input) then
- -- First input in the dynamic thumbstick area should always be ignored for camera purposes
- -- Even if the dynamic thumbstick does not process it immediately
- self.dynamicTouchInput = input
- return
- end
- self.fingerTouches[input] = processed
- self.inputStartPositions[input] = input.Position
- self.inputStartTimes[input] = tick()
- self.numUnsunkTouches = self.numUnsunkTouches + 1
- end
- end
- function BaseCamera:OnTouchChanged(input, processed)
- if self.fingerTouches[input] == nil then
- if self.isDynamicThumbstickEnabled then
- return
- end
- self.fingerTouches[input] = processed
- if not processed then
- self.numUnsunkTouches = self.numUnsunkTouches + 1
- end
- end
- if self.numUnsunkTouches == 1 then
- if self.fingerTouches[input] == false then
- self.panBeginLook = self.panBeginLook or self:GetCameraLookVector()
- self.startPos = self.startPos or input.Position
- self.lastPos = self.lastPos or self.startPos
- self.userPanningTheCamera = true
- local delta = input.Position - self.lastPos
- delta = Vector2.new(delta.X, delta.Y * UserGameSettings:GetCameraYInvertValue())
- if self.panEnabled then
- local adjustedTouchSensitivity = TOUCH_SENSITIVTY
- self:AdjustTouchSensitivity(delta, TOUCH_SENSITIVTY)
- local desiredXYVector = self:InputTranslationToCameraAngleChange(delta, adjustedTouchSensitivity)
- self.rotateInput = self.rotateInput + desiredXYVector
- end
- self.lastPos = input.Position
- end
- else
- self.panBeginLook = nil
- self.startPos = nil
- self.lastPos = nil
- self.userPanningTheCamera = false
- end
- if self.numUnsunkTouches == 2 then
- local unsunkTouches = {}
- for touch, wasSunk in pairs(self.fingerTouches) do
- if not wasSunk then
- table.insert(unsunkTouches, touch)
- end
- end
- if #unsunkTouches == 2 then
- local difference = (unsunkTouches[1].Position - unsunkTouches[2].Position).magnitude
- if self.startingDiff and self.pinchBeginZoom then
- local scale = difference / math.max(0.01, self.startingDiff)
- local clampedScale = math.clamp(scale, 0.1, 10)
- if self.distanceChangeEnabled then
- self:SetCameraToSubjectDistance(self.pinchBeginZoom / clampedScale)
- end
- else
- self.startingDiff = difference
- self.pinchBeginZoom = self:GetCameraToSubjectDistance()
- end
- end
- else
- self.startingDiff = nil
- self.pinchBeginZoom = nil
- end
- end
- function BaseCamera:OnTouchEnded(input, processed)
- if input == self.dynamicTouchInput then
- self.dynamicTouchInput = nil
- return
- end
- if self.fingerTouches[input] == false then
- if self.numUnsunkTouches == 1 then
- self.panBeginLook = nil
- self.startPos = nil
- self.lastPos = nil
- self.userPanningTheCamera = false
- elseif self.numUnsunkTouches == 2 then
- self.startingDiff = nil
- self.pinchBeginZoom = nil
- end
- end
- if self.fingerTouches[input] ~= nil and self.fingerTouches[input] == false then
- self.numUnsunkTouches = self.numUnsunkTouches - 1
- end
- self.fingerTouches[input] = nil
- self.inputStartPositions[input] = nil
- self.inputStartTimes[input] = nil
- end
- function BaseCamera:OnMouse2Down(input, processed)
- if processed then return end
- self.isRightMouseDown = true
- self:OnMousePanButtonPressed(input, processed)
- end
- function BaseCamera:OnMouse2Up(input, processed)
- self.isRightMouseDown = false
- self:OnMousePanButtonReleased(input, processed)
- end
- function BaseCamera:OnMouse3Down(input, processed)
- if processed then return end
- self.isMiddleMouseDown = true
- self:OnMousePanButtonPressed(input, processed)
- end
- function BaseCamera:OnMouse3Up(input, processed)
- self.isMiddleMouseDown = false
- self:OnMousePanButtonReleased(input, processed)
- end
- function BaseCamera:OnMouseMoved(input, processed)
- if not self.hasGameLoaded and VRService.VREnabled then
- return
- end
- local inputDelta = input.Delta
- inputDelta = Vector2.new(inputDelta.X, inputDelta.Y * UserGameSettings:GetCameraYInvertValue())
- local isInputPanning = FFlagUserCameraToggle and CameraInput.getPanning()
- local isBeginLook = self.startPos and self.lastPos and self.panBeginLook
- local isPanning = isBeginLook or self.inFirstPerson or self.inMouseLockedMode or isInputPanning
- if self.panEnabled and isPanning then
- local desiredXYVector = self:InputTranslationToCameraAngleChange(inputDelta, MOUSE_SENSITIVITY)
- self.rotateInput = self.rotateInput + desiredXYVector
- end
- if self.startPos and self.lastPos and self.panBeginLook then
- self.lastPos = self.lastPos + input.Delta
- end
- end
- function BaseCamera:OnMousePanButtonPressed(input, processed)
- if processed then return end
- if not FFlagUserCameraToggle then
- self:UpdateMouseBehavior()
- end
- self.panBeginLook = self.panBeginLook or self:GetCameraLookVector()
- self.startPos = self.startPos or input.Position
- self.lastPos = self.lastPos or self.startPos
- self.userPanningTheCamera = true
- end
- function BaseCamera:OnMousePanButtonReleased(input, processed)
- if not FFlagUserCameraToggle then
- self:UpdateMouseBehavior()
- end
- if not (self.isRightMouseDown or self.isMiddleMouseDown) then
- self.panBeginLook = nil
- self.startPos = nil
- self.lastPos = nil
- self.userPanningTheCamera = false
- end
- end
- function BaseCamera:UpdateMouseBehavior()
- if FFlagUserCameraToggle and self.isCameraToggle then
- CameraUI.setCameraModeToastEnabled(true)
- CameraInput.enableCameraToggleInput()
- CameraToggleStateController(self.inFirstPerson)
- else
- if FFlagUserCameraToggle then
- CameraUI.setCameraModeToastEnabled(false)
- CameraInput.disableCameraToggleInput()
- end
- -- first time transition to first person mode or mouse-locked third person
- if self.inFirstPerson or self.inMouseLockedMode then
- --UserGameSettings.RotationType = Enum.RotationType.CameraRelative
- UserInputService.MouseBehavior = Enum.MouseBehavior.LockCenter
- else
- UserGameSettings.RotationType = Enum.RotationType.MovementRelative
- if self.isRightMouseDown or self.isMiddleMouseDown then
- UserInputService.MouseBehavior = Enum.MouseBehavior.LockCurrentPosition
- else
- UserInputService.MouseBehavior = Enum.MouseBehavior.Default
- end
- end
- end
- end
- function BaseCamera:UpdateForDistancePropertyChange()
- -- Calling this setter with the current value will force checking that it is still
- -- in range after a change to the min/max distance limits
- self:SetCameraToSubjectDistance(self.currentSubjectDistance)
- end
- function BaseCamera:SetCameraToSubjectDistance(desiredSubjectDistance)
- local lastSubjectDistance = self.currentSubjectDistance
- -- By default, camera modules will respect LockFirstPerson and override the currentSubjectDistance with 0
- -- regardless of what Player.CameraMinZoomDistance is set to, so that first person can be made
- -- available by the developer without needing to allow players to mousewheel dolly into first person.
- -- Some modules will override this function to remove or change first-person capability.
- if player.CameraMode == Enum.CameraMode.LockFirstPerson then
- self.currentSubjectDistance = 0.5
- if not self.inFirstPerson then
- self:EnterFirstPerson()
- end
- else
- local newSubjectDistance = math.clamp(desiredSubjectDistance, player.CameraMinZoomDistance, player.CameraMaxZoomDistance)
- if newSubjectDistance < FIRST_PERSON_DISTANCE_THRESHOLD then
- self.currentSubjectDistance = 0.5
- if not self.inFirstPerson then
- self:EnterFirstPerson()
- end
- else
- self.currentSubjectDistance = newSubjectDistance
- if self.inFirstPerson then
- self:LeaveFirstPerson()
- end
- end
- end
- -- Pass target distance and zoom direction to the zoom controller
- ZoomController.SetZoomParameters(self.currentSubjectDistance, math.sign(desiredSubjectDistance - lastSubjectDistance))
- -- Returned only for convenience to the caller to know the outcome
- return self.currentSubjectDistance
- end
- function BaseCamera:SetCameraType( cameraType )
- --Used by derived classes
- self.cameraType = cameraType
- end
- function BaseCamera:GetCameraType()
- return self.cameraType
- end
- -- Movement mode standardized to Enum.ComputerCameraMovementMode values
- function BaseCamera:SetCameraMovementMode( cameraMovementMode )
- self.cameraMovementMode = cameraMovementMode
- end
- function BaseCamera:GetCameraMovementMode()
- return self.cameraMovementMode
- end
- function BaseCamera:SetIsMouseLocked(mouseLocked)
- self.inMouseLockedMode = mouseLocked
- if not FFlagUserCameraToggle then
- self:UpdateMouseBehavior()
- end
- end
- function BaseCamera:GetIsMouseLocked()
- return self.inMouseLockedMode
- end
- function BaseCamera:SetMouseLockOffset(offsetVector)
- self.mouseLockOffset = offsetVector
- end
- function BaseCamera:GetMouseLockOffset()
- return self.mouseLockOffset
- end
- function BaseCamera:InFirstPerson()
- return self.inFirstPerson
- end
- function BaseCamera:EnterFirstPerson()
- -- Overridden in ClassicCamera, the only module which supports FirstPerson
- end
- function BaseCamera:LeaveFirstPerson()
- -- Overridden in ClassicCamera, the only module which supports FirstPerson
- end
- -- Nominal distance, set by dollying in and out with the mouse wheel or equivalent, not measured distance
- function BaseCamera:GetCameraToSubjectDistance()
- return self.currentSubjectDistance
- end
- -- Actual measured distance to the camera Focus point, which may be needed in special circumstances, but should
- -- never be used as the starting point for updating the nominal camera-to-subject distance (self.currentSubjectDistance)
- -- since that is a desired target value set only by mouse wheel (or equivalent) input, PopperCam, and clamped to min max camera distance
- function BaseCamera:GetMeasuredDistanceToFocus()
- local camera = game.Workspace.CurrentCamera
- if camera then
- return (camera.CoordinateFrame.p - camera.Focus.p).magnitude
- end
- return nil
- end
- function BaseCamera:GetCameraLookVector()
- return game.Workspace.CurrentCamera and game.Workspace.CurrentCamera.CFrame.lookVector or UNIT_Z
- end
- -- Replacements for RootCamera:RotateCamera() which did not actually rotate the camera
- -- suppliedLookVector is not normally passed in, it's used only by Watch camera
- function BaseCamera:CalculateNewLookCFrame(suppliedLookVector)
- local currLookVector = suppliedLookVector or self:GetCameraLookVector()
- local currPitchAngle = math.asin(currLookVector.y)
- local yTheta = math.clamp(self.rotateInput.y, -MAX_Y + currPitchAngle, -MIN_Y + currPitchAngle)
- local constrainedRotateInput = Vector2.new(self.rotateInput.x, yTheta)
- local startCFrame = CFrame.new(ZERO_VECTOR3, currLookVector)
- local newLookCFrame = CFrame.Angles(0, -constrainedRotateInput.x, 0) * startCFrame * CFrame.Angles(-constrainedRotateInput.y,0,0)
- return newLookCFrame
- end
- function BaseCamera:CalculateNewLookVector(suppliedLookVector)
- local newLookCFrame = self:CalculateNewLookCFrame(suppliedLookVector)
- return newLookCFrame.lookVector
- end
- function BaseCamera:CalculateNewLookVectorVR()
- local subjectPosition = self:GetSubjectPosition()
- local vecToSubject = (subjectPosition - game.Workspace.CurrentCamera.CFrame.p)
- local currLookVector = (vecToSubject * X1_Y0_Z1).unit
- local vrRotateInput = Vector2.new(self.rotateInput.x, 0)
- local startCFrame = CFrame.new(ZERO_VECTOR3, currLookVector)
- local yawRotatedVector = (CFrame.Angles(0, -vrRotateInput.x, 0) * startCFrame * CFrame.Angles(-vrRotateInput.y,0,0)).lookVector
- return (yawRotatedVector * X1_Y0_Z1).unit
- end
- function BaseCamera:GetHumanoid()
- local character = player and player.Character
- if character then
- local resultHumanoid = self.humanoidCache[player]
- if resultHumanoid and resultHumanoid.Parent == character then
- return resultHumanoid
- else
- self.humanoidCache[player] = nil -- Bust Old Cache
- local humanoid = character:FindFirstChildOfClass("Humanoid")
- if humanoid then
- self.humanoidCache[player] = humanoid
- end
- return humanoid
- end
- end
- return nil
- end
- function BaseCamera:GetHumanoidPartToFollow(humanoid, humanoidStateType)
- if humanoidStateType == Enum.HumanoidStateType.Dead then
- local character = humanoid.Parent
- if character then
- return character:FindFirstChild("Head") or humanoid.Torso
- else
- return humanoid.Torso
- end
- else
- return humanoid.Torso
- end
- end
- function BaseCamera:UpdateGamepad()
- local gamepadPan = self.gamepadPanningCamera
- if gamepadPan and (self.hasGameLoaded or not VRService.VREnabled) then
- gamepadPan = Util.GamepadLinearToCurve(gamepadPan)
- local currentTime = tick()
- if gamepadPan.X ~= 0 or gamepadPan.Y ~= 0 then
- self.userPanningTheCamera = true
- elseif gamepadPan == ZERO_VECTOR2 then
- self.lastThumbstickRotate = nil
- if self.lastThumbstickPos == ZERO_VECTOR2 then
- self.currentSpeed = 0
- end
- end
- local finalConstant = 0
- if self.lastThumbstickRotate then
- if VRService.VREnabled then
- self.currentSpeed = self.vrMaxSpeed
- else
- local elapsedTime = (currentTime - self.lastThumbstickRotate) * 10
- self.currentSpeed = self.currentSpeed + (self.maxSpeed * ((elapsedTime*elapsedTime)/self.numOfSeconds))
- if self.currentSpeed > self.maxSpeed then self.currentSpeed = self.maxSpeed end
- if self.lastVelocity then
- local velocity = (gamepadPan - self.lastThumbstickPos)/(currentTime - self.lastThumbstickRotate)
- local velocityDeltaMag = (velocity - self.lastVelocity).magnitude
- if velocityDeltaMag > 12 then
- self.currentSpeed = self.currentSpeed * (20/velocityDeltaMag)
- if self.currentSpeed > self.maxSpeed then self.currentSpeed = self.maxSpeed end
- end
- end
- end
- finalConstant = UserGameSettings.GamepadCameraSensitivity * self.currentSpeed
- self.lastVelocity = (gamepadPan - self.lastThumbstickPos)/(currentTime - self.lastThumbstickRotate)
- end
- self.lastThumbstickPos = gamepadPan
- self.lastThumbstickRotate = currentTime
- return Vector2.new( gamepadPan.X * finalConstant, gamepadPan.Y * finalConstant * self.ySensitivity * UserGameSettings:GetCameraYInvertValue())
- end
- return ZERO_VECTOR2
- end
- -- [[ VR Support Section ]] --
- function BaseCamera:ApplyVRTransform()
- if not VRService.VREnabled then
- return
- end
- --we only want this to happen in first person VR
- local rootJoint = self.humanoidRootPart and self.humanoidRootPart:FindFirstChild("RootJoint")
- if not rootJoint then
- return
- end
- local cameraSubject = game.Workspace.CurrentCamera.CameraSubject
- local isInVehicle = cameraSubject and cameraSubject:IsA("VehicleSeat")
- if self.inFirstPerson and not isInVehicle then
- local vrFrame = VRService:GetUserCFrame(Enum.UserCFrame.Head)
- local vrRotation = vrFrame - vrFrame.p
- rootJoint.C0 = CFrame.new(vrRotation:vectorToObjectSpace(vrFrame.p)) * CFrame.new(0, 0, 0, -1, 0, 0, 0, 0, 1, 0, 1, 0)
- else
- rootJoint.C0 = CFrame.new(0, 0, 0, -1, 0, 0, 0, 0, 1, 0, 1, 0)
- end
- end
- function BaseCamera:IsInFirstPerson()
- return self.inFirstPerson
- end
- function BaseCamera:ShouldUseVRRotation()
- if not VRService.VREnabled then
- return false
- end
- if not self.VRRotationIntensityAvailable and tick() - self.lastVRRotationIntensityCheckTime < 1 then
- return false
- end
- local success, vrRotationIntensity = pcall(function() return StarterGui:GetCore("VRRotationIntensity") end)
- self.VRRotationIntensityAvailable = success and vrRotationIntensity ~= nil
- self.lastVRRotationIntensityCheckTime = tick()
- self.shouldUseVRRotation = success and vrRotationIntensity ~= nil and vrRotationIntensity ~= "Smooth"
- return self.shouldUseVRRotation
- end
- function BaseCamera:GetVRRotationInput()
- local vrRotateSum = ZERO_VECTOR2
- local success, vrRotationIntensity = pcall(function() return StarterGui:GetCore("VRRotationIntensity") end)
- if not success then
- return
- end
- local vrGamepadRotation = self.GamepadPanningCamera or ZERO_VECTOR2
- local delayExpired = (tick() - self.lastVRRotationTime) >= self:GetRepeatDelayValue(vrRotationIntensity)
- if math.abs(vrGamepadRotation.x) >= self:GetActivateValue() then
- if (delayExpired or not self.vrRotateKeyCooldown[Enum.KeyCode.Thumbstick2]) then
- local sign = 1
- if vrGamepadRotation.x < 0 then
- sign = -1
- end
- vrRotateSum = vrRotateSum + self:GetRotateAmountValue(vrRotationIntensity) * sign
- self.vrRotateKeyCooldown[Enum.KeyCode.Thumbstick2] = true
- end
- elseif math.abs(vrGamepadRotation.x) < self:GetActivateValue() - 0.1 then
- self.vrRotateKeyCooldown[Enum.KeyCode.Thumbstick2] = nil
- end
- if self.turningLeft then
- if delayExpired or not self.vrRotateKeyCooldown[Enum.KeyCode.Left] then
- vrRotateSum = vrRotateSum - self:GetRotateAmountValue(vrRotationIntensity)
- self.vrRotateKeyCooldown[Enum.KeyCode.Left] = true
- end
- else
- self.vrRotateKeyCooldown[Enum.KeyCode.Left] = nil
- end
- if self.turningRight then
- if (delayExpired or not self.vrRotateKeyCooldown[Enum.KeyCode.Right]) then
- vrRotateSum = vrRotateSum + self:GetRotateAmountValue(vrRotationIntensity)
- self.vrRotateKeyCooldown[Enum.KeyCode.Right] = true
- end
- else
- self.vrRotateKeyCooldown[Enum.KeyCode.Right] = nil
- end
- if vrRotateSum ~= ZERO_VECTOR2 then
- self.lastVRRotationTime = tick()
- end
- return vrRotateSum
- end
- function BaseCamera:CancelCameraFreeze(keepConstraints)
- if not keepConstraints then
- self.cameraTranslationConstraints = Vector3.new(self.cameraTranslationConstraints.x, 1, self.cameraTranslationConstraints.z)
- end
- if self.cameraFrozen then
- self.trackingHumanoid = nil
- self.cameraFrozen = false
- end
- end
- function BaseCamera:StartCameraFreeze(subjectPosition, humanoidToTrack)
- if not self.cameraFrozen then
- self.humanoidJumpOrigin = subjectPosition
- self.trackingHumanoid = humanoidToTrack
- self.cameraTranslationConstraints = Vector3.new(self.cameraTranslationConstraints.x, 0, self.cameraTranslationConstraints.z)
- self.cameraFrozen = true
- end
- end
- function BaseCamera:OnNewCameraSubject()
- if self.subjectStateChangedConn then
- self.subjectStateChangedConn:Disconnect()
- self.subjectStateChangedConn = nil
- end
- local humanoid = workspace.CurrentCamera and workspace.CurrentCamera.CameraSubject
- if self.trackingHumanoid ~= humanoid then
- self:CancelCameraFreeze()
- end
- if humanoid and humanoid:IsA("Humanoid") then
- self.subjectStateChangedConn = humanoid.StateChanged:Connect(function(oldState, newState)
- if VRService.VREnabled and newState == Enum.HumanoidStateType.Jumping and not self.inFirstPerson then
- self:StartCameraFreeze(self:GetSubjectPosition(), humanoid)
- elseif newState ~= Enum.HumanoidStateType.Jumping and newState ~= Enum.HumanoidStateType.Freefall then
- self:CancelCameraFreeze(true)
- end
- end)
- end
- end
- function BaseCamera:GetVRFocus(subjectPosition, timeDelta)
- local lastFocus = self.LastCameraFocus or subjectPosition
- if not self.cameraFrozen then
- self.cameraTranslationConstraints = Vector3.new(self.cameraTranslationConstraints.x, math.min(1, self.cameraTranslationConstraints.y + 0.42 * timeDelta), self.cameraTranslationConstraints.z)
- end
- local newFocus
- if self.cameraFrozen and self.humanoidJumpOrigin and self.humanoidJumpOrigin.y > lastFocus.y then
- newFocus = CFrame.new(Vector3.new(subjectPosition.x, math.min(self.humanoidJumpOrigin.y, lastFocus.y + 5 * timeDelta), subjectPosition.z))
- else
- newFocus = CFrame.new(Vector3.new(subjectPosition.x, lastFocus.y, subjectPosition.z):lerp(subjectPosition, self.cameraTranslationConstraints.y))
- end
- if self.cameraFrozen then
- -- No longer in 3rd person
- if self.inFirstPerson then -- not VRService.VREnabled
- self:CancelCameraFreeze()
- end
- -- This case you jumped off a cliff and want to keep your character in view
- -- 0.5 is to fix floating point error when not jumping off cliffs
- if self.humanoidJumpOrigin and subjectPosition.y < (self.humanoidJumpOrigin.y - 0.5) then
- self:CancelCameraFreeze()
- end
- end
- return newFocus
- end
- function BaseCamera:GetRotateAmountValue(vrRotationIntensity)
- vrRotationIntensity = vrRotationIntensity or StarterGui:GetCore("VRRotationIntensity")
- if vrRotationIntensity then
- if vrRotationIntensity == "Low" then
- return VR_LOW_INTENSITY_ROTATION
- elseif vrRotationIntensity == "High" then
- return VR_HIGH_INTENSITY_ROTATION
- end
- end
- return ZERO_VECTOR2
- end
- function BaseCamera:GetRepeatDelayValue(vrRotationIntensity)
- vrRotationIntensity = vrRotationIntensity or StarterGui:GetCore("VRRotationIntensity")
- if vrRotationIntensity then
- if vrRotationIntensity == "Low" then
- return VR_LOW_INTENSITY_REPEAT
- elseif vrRotationIntensity == "High" then
- return VR_HIGH_INTENSITY_REPEAT
- end
- end
- return 0
- end
- function BaseCamera:Update(dt)
- error("BaseCamera:Update() This is a virtual function that should never be getting called.", 2)
- end
- BaseCamera.UpCFrame = CFrame.new()
- function BaseCamera:UpdateUpCFrame(cf)
- self.UpCFrame = cf
- end
- local ZERO = Vector3.new(0, 0, 0)
- function BaseCamera:CalculateNewLookCFrame(suppliedLookVector)
- local currLookVector = suppliedLookVector or self:GetCameraLookVector()
- currLookVector = self.UpCFrame:VectorToObjectSpace(currLookVector)
- local currPitchAngle = math.asin(currLookVector.y)
- local yTheta = math.clamp(self.rotateInput.y, -MAX_Y + currPitchAngle, -MIN_Y + currPitchAngle)
- local constrainedRotateInput = Vector2.new(self.rotateInput.x, yTheta)
- local startCFrame = CFrame.new(ZERO, currLookVector)
- local newLookCFrame = CFrame.Angles(0, -constrainedRotateInput.x, 0) * startCFrame * CFrame.Angles(-constrainedRotateInput.y,0,0)
- return newLookCFrame
- end
- return BaseCamera
- end
- function _BaseOcclusion()
- --[[ The Module ]]--
- local BaseOcclusion = {}
- BaseOcclusion.__index = BaseOcclusion
- setmetatable(BaseOcclusion, {
- __call = function(_, ...)
- return BaseOcclusion.new(...)
- end
- })
- function BaseOcclusion.new()
- local self = setmetatable({}, BaseOcclusion)
- return self
- end
- -- Called when character is added
- function BaseOcclusion:CharacterAdded(char, player)
- end
- -- Called when character is about to be removed
- function BaseOcclusion:CharacterRemoving(char, player)
- end
- function BaseOcclusion:OnCameraSubjectChanged(newSubject)
- end
- --[[ Derived classes are required to override and implement all of the following functions ]]--
- function BaseOcclusion:GetOcclusionMode()
- -- Must be overridden in derived classes to return an Enum.DevCameraOcclusionMode value
- warn("BaseOcclusion GetOcclusionMode must be overridden by derived classes")
- return nil
- end
- function BaseOcclusion:Enable(enabled)
- warn("BaseOcclusion Enable must be overridden by derived classes")
- end
- function BaseOcclusion:Update(dt, desiredCameraCFrame, desiredCameraFocus)
- warn("BaseOcclusion Update must be overridden by derived classes")
- return desiredCameraCFrame, desiredCameraFocus
- end
- return BaseOcclusion
- end
- function _Popper()
- local Players = game:GetService("Players")
- local camera = game.Workspace.CurrentCamera
- local min = math.min
- local tan = math.tan
- local rad = math.rad
- local inf = math.huge
- local ray = Ray.new
- local function getTotalTransparency(part)
- return 1 - (1 - part.Transparency)*(1 - part.LocalTransparencyModifier)
- end
- local function eraseFromEnd(t, toSize)
- for i = #t, toSize + 1, -1 do
- t[i] = nil
- end
- end
- local nearPlaneZ, projX, projY do
- local function updateProjection()
- local fov = rad(camera.FieldOfView)
- local view = camera.ViewportSize
- local ar = view.X/view.Y
- projY = 2*tan(fov/2)
- projX = ar*projY
- end
- camera:GetPropertyChangedSignal("FieldOfView"):Connect(updateProjection)
- camera:GetPropertyChangedSignal("ViewportSize"):Connect(updateProjection)
- updateProjection()
- nearPlaneZ = camera.NearPlaneZ
- camera:GetPropertyChangedSignal("NearPlaneZ"):Connect(function()
- nearPlaneZ = camera.NearPlaneZ
- end)
- end
- local blacklist = {} do
- local charMap = {}
- local function refreshIgnoreList()
- local n = 1
- blacklist = {}
- for _, character in pairs(charMap) do
- blacklist[n] = character
- n = n + 1
- end
- end
- local function playerAdded(player)
- local function characterAdded(character)
- charMap[player] = character
- refreshIgnoreList()
- end
- local function characterRemoving()
- charMap[player] = nil
- refreshIgnoreList()
- end
- player.CharacterAdded:Connect(characterAdded)
- player.CharacterRemoving:Connect(characterRemoving)
- if player.Character then
- characterAdded(player.Character)
- end
- end
- local function playerRemoving(player)
- charMap[player] = nil
- refreshIgnoreList()
- end
- Players.PlayerAdded:Connect(playerAdded)
- Players.PlayerRemoving:Connect(playerRemoving)
- for _, player in ipairs(Players:GetPlayers()) do
- playerAdded(player)
- end
- refreshIgnoreList()
- end
- --------------------------------------------------------------------------------------------
- -- Popper uses the level geometry find an upper bound on subject-to-camera distance.
- --
- -- Hard limits are applied immediately and unconditionally. They are generally caused
- -- when level geometry intersects with the near plane (with exceptions, see below).
- --
- -- Soft limits are only applied under certain conditions.
- -- They are caused when level geometry occludes the subject without actually intersecting
- -- with the near plane at the target distance.
- --
- -- Soft limits can be promoted to hard limits and hard limits can be demoted to soft limits.
- -- We usually don"t want the latter to happen.
- --
- -- A soft limit will be promoted to a hard limit if an obstruction
- -- lies between the current and target camera positions.
- --------------------------------------------------------------------------------------------
- local subjectRoot
- local subjectPart
- camera:GetPropertyChangedSignal("CameraSubject"):Connect(function()
- local subject = camera.CameraSubject
- if subject:IsA("Humanoid") then
- subjectPart = subject.RootPart
- elseif subject:IsA("BasePart") then
- subjectPart = subject
- else
- subjectPart = nil
- end
- end)
- local function canOcclude(part)
- -- Occluders must be:
- -- 1. Opaque
- -- 2. Interactable
- -- 3. Not in the same assembly as the subject
- return
- getTotalTransparency(part) < 0.25 and
- part.CanCollide and
- subjectRoot ~= (part:GetRootPart() or part) and
- not part:IsA("TrussPart")
- end
- -- Offsets for the volume visibility test
- local SCAN_SAMPLE_OFFSETS = {
- Vector2.new( 0.4, 0.0),
- Vector2.new(-0.4, 0.0),
- Vector2.new( 0.0,-0.4),
- Vector2.new( 0.0, 0.4),
- Vector2.new( 0.0, 0.2),
- }
- --------------------------------------------------------------------------------
- -- Piercing raycasts
- local function getCollisionPoint(origin, dir)
- local originalSize = #blacklist
- repeat
- local hitPart, hitPoint = workspace:FindPartOnRayWithIgnoreList(
- ray(origin, dir), blacklist, false, true
- )
- if hitPart then
- if hitPart.CanCollide then
- eraseFromEnd(blacklist, originalSize)
- return hitPoint, true
- end
- blacklist[#blacklist + 1] = hitPart
- end
- until not hitPart
- eraseFromEnd(blacklist, originalSize)
- return origin + dir, false
- end
- --------------------------------------------------------------------------------
- local function queryPoint(origin, unitDir, dist, lastPos)
- debug.profilebegin("queryPoint")
- local originalSize = #blacklist
- dist = dist + nearPlaneZ
- local target = origin + unitDir*dist
- local softLimit = inf
- local hardLimit = inf
- local movingOrigin = origin
- repeat
- local entryPart, entryPos = workspace:FindPartOnRayWithIgnoreList(ray(movingOrigin, target - movingOrigin), blacklist, false, true)
- if entryPart then
- if canOcclude(entryPart) then
- local wl = {entryPart}
- local exitPart = workspace:FindPartOnRayWithWhitelist(ray(target, entryPos - target), wl, true)
- local lim = (entryPos - origin).Magnitude
- if exitPart then
- local promote = false
- if lastPos then
- promote =
- workspace:FindPartOnRayWithWhitelist(ray(lastPos, target - lastPos), wl, true) or
- workspace:FindPartOnRayWithWhitelist(ray(target, lastPos - target), wl, true)
- end
- if promote then
- -- Ostensibly a soft limit, but the camera has passed through it in the last frame, so promote to a hard limit.
- hardLimit = lim
- elseif dist < softLimit then
- -- Trivial soft limit
- softLimit = lim
- end
- else
- -- Trivial hard limit
- hardLimit = lim
- end
- end
- blacklist[#blacklist + 1] = entryPart
- movingOrigin = entryPos - unitDir*1e-3
- end
- until hardLimit < inf or not entryPart
- eraseFromEnd(blacklist, originalSize)
- debug.profileend()
- return softLimit - nearPlaneZ, hardLimit - nearPlaneZ
- end
- local function queryViewport(focus, dist)
- debug.profilebegin("queryViewport")
- local fP = focus.p
- local fX = focus.rightVector
- local fY = focus.upVector
- local fZ = -focus.lookVector
- local viewport = camera.ViewportSize
- local hardBoxLimit = inf
- local softBoxLimit = inf
- -- Center the viewport on the PoI, sweep points on the edge towards the target, and take the minimum limits
- for viewX = 0, 1 do
- local worldX = fX*((viewX - 0.5)*projX)
- for viewY = 0, 1 do
- local worldY = fY*((viewY - 0.5)*projY)
- local origin = fP + nearPlaneZ*(worldX + worldY)
- local lastPos = camera:ViewportPointToRay(
- viewport.x*viewX,
- viewport.y*viewY
- ).Origin
- local softPointLimit, hardPointLimit = queryPoint(origin, fZ, dist, lastPos)
- if hardPointLimit < hardBoxLimit then
- hardBoxLimit = hardPointLimit
- end
- if softPointLimit < softBoxLimit then
- softBoxLimit = softPointLimit
- end
- end
- end
- debug.profileend()
- return softBoxLimit, hardBoxLimit
- end
- local function testPromotion(focus, dist, focusExtrapolation)
- debug.profilebegin("testPromotion")
- local fP = focus.p
- local fX = focus.rightVector
- local fY = focus.upVector
- local fZ = -focus.lookVector
- do
- -- Dead reckoning the camera rotation and focus
- debug.profilebegin("extrapolate")
- local SAMPLE_DT = 0.0625
- local SAMPLE_MAX_T = 1.25
- local maxDist = (getCollisionPoint(fP, focusExtrapolation.posVelocity*SAMPLE_MAX_T) - fP).Magnitude
- -- Metric that decides how many samples to take
- local combinedSpeed = focusExtrapolation.posVelocity.magnitude
- for dt = 0, min(SAMPLE_MAX_T, focusExtrapolation.rotVelocity.magnitude + maxDist/combinedSpeed), SAMPLE_DT do
- local cfDt = focusExtrapolation.extrapolate(dt) -- Extrapolated CFrame at time dt
- if queryPoint(cfDt.p, -cfDt.lookVector, dist) >= dist then
- return false
- end
- end
- debug.profileend()
- end
- do
- -- Test screen-space offsets from the focus for the presence of soft limits
- debug.profilebegin("testOffsets")
- for _, offset in ipairs(SCAN_SAMPLE_OFFSETS) do
- local scaledOffset = offset
- local pos = getCollisionPoint(fP, fX*scaledOffset.x + fY*scaledOffset.y)
- if queryPoint(pos, (fP + fZ*dist - pos).Unit, dist) == inf then
- return false
- end
- end
- debug.profileend()
- end
- debug.profileend()
- return true
- end
- local function Popper(focus, targetDist, focusExtrapolation)
- debug.profilebegin("popper")
- subjectRoot = subjectPart and subjectPart:GetRootPart() or subjectPart
- local dist = targetDist
- local soft, hard = queryViewport(focus, targetDist)
- if hard < dist then
- dist = hard
- end
- if soft < dist and testPromotion(focus, targetDist, focusExtrapolation) then
- dist = soft
- end
- subjectRoot = nil
- debug.profileend()
- return dist
- end
- return Popper
- end
- function _ZoomController()
- local ZOOM_STIFFNESS = 4.5
- local ZOOM_DEFAULT = 12.5
- local ZOOM_ACCELERATION = 0.0375
- local MIN_FOCUS_DIST = 0.5
- local DIST_OPAQUE = 1
- local Popper = _Popper()
- local clamp = math.clamp
- local exp = math.exp
- local min = math.min
- local max = math.max
- local pi = math.pi
- local cameraMinZoomDistance, cameraMaxZoomDistance do
- local Player = game:GetService("Players").LocalPlayer
- local function updateBounds()
- cameraMinZoomDistance = Player.CameraMinZoomDistance
- cameraMaxZoomDistance = Player.CameraMaxZoomDistance
- end
- updateBounds()
- Player:GetPropertyChangedSignal("CameraMinZoomDistance"):Connect(updateBounds)
- Player:GetPropertyChangedSignal("CameraMaxZoomDistance"):Connect(updateBounds)
- end
- local ConstrainedSpring = {} do
- ConstrainedSpring.__index = ConstrainedSpring
- function ConstrainedSpring.new(freq, x, minValue, maxValue)
- x = clamp(x, minValue, maxValue)
- return setmetatable({
- freq = freq, -- Undamped frequency (Hz)
- x = x, -- Current position
- v = 0, -- Current velocity
- minValue = minValue, -- Minimum bound
- maxValue = maxValue, -- Maximum bound
- goal = x, -- Goal position
- }, ConstrainedSpring)
- end
- function ConstrainedSpring:Step(dt)
- local freq = self.freq*2*pi -- Convert from Hz to rad/s
- local x = self.x
- local v = self.v
- local minValue = self.minValue
- local maxValue = self.maxValue
- local goal = self.goal
- -- Solve the spring ODE for position and velocity after time t, assuming critical damping:
- -- 2*f*x'[t] + x''[t] = f^2*(g - x[t])
- -- Knowns are x[0] and x'[0].
- -- Solve for x[t] and x'[t].
- local offset = goal - x
- local step = freq*dt
- local decay = exp(-step)
- local x1 = goal + (v*dt - offset*(step + 1))*decay
- local v1 = ((offset*freq - v)*step + v)*decay
- -- Constrain
- if x1 < minValue then
- x1 = minValue
- v1 = 0
- elseif x1 > maxValue then
- x1 = maxValue
- v1 = 0
- end
- self.x = x1
- self.v = v1
- return x1
- end
- end
- local zoomSpring = ConstrainedSpring.new(ZOOM_STIFFNESS, ZOOM_DEFAULT, MIN_FOCUS_DIST, cameraMaxZoomDistance)
- local function stepTargetZoom(z, dz, zoomMin, zoomMax)
- z = clamp(z + dz*(1 + z*ZOOM_ACCELERATION), zoomMin, zoomMax)
- if z < DIST_OPAQUE then
- z = dz <= 0 and zoomMin or DIST_OPAQUE
- end
- return z
- end
- local zoomDelta = 0
- local Zoom = {} do
- function Zoom.Update(renderDt, focus, extrapolation)
- local poppedZoom = math.huge
- if zoomSpring.goal > DIST_OPAQUE then
- -- Make a pessimistic estimate of zoom distance for this step without accounting for poppercam
- local maxPossibleZoom = max(
- zoomSpring.x,
- stepTargetZoom(zoomSpring.goal, zoomDelta, cameraMinZoomDistance, cameraMaxZoomDistance)
- )
- -- Run the Popper algorithm on the feasible zoom range, [MIN_FOCUS_DIST, maxPossibleZoom]
- poppedZoom = Popper(
- focus*CFrame.new(0, 0, MIN_FOCUS_DIST),
- maxPossibleZoom - MIN_FOCUS_DIST,
- extrapolation
- ) + MIN_FOCUS_DIST
- end
- zoomSpring.minValue = MIN_FOCUS_DIST
- zoomSpring.maxValue = min(cameraMaxZoomDistance, poppedZoom)
- return zoomSpring:Step(renderDt)
- end
- function Zoom.SetZoomParameters(targetZoom, newZoomDelta)
- zoomSpring.goal = targetZoom
- zoomDelta = newZoomDelta
- end
- end
- return Zoom
- end
- function _MouseLockController()
- --[[ Constants ]]--
- local DEFAULT_MOUSE_LOCK_CURSOR = "rbxasset://textures/MouseLockedCursor.png"
- local CONTEXT_ACTION_NAME = "MouseLockSwitchAction"
- local MOUSELOCK_ACTION_PRIORITY = Enum.ContextActionPriority.Default.Value
- --[[ Services ]]--
- local PlayersService = game:GetService("Players")
- local ContextActionService = game:GetService("ContextActionService")
- local Settings = UserSettings() -- ignore warning
- local GameSettings = Settings.GameSettings
- local Mouse = PlayersService.LocalPlayer:GetMouse()
- --[[ The Module ]]--
- local MouseLockController = {}
- MouseLockController.__index = MouseLockController
- function MouseLockController.new()
- local self = setmetatable({}, MouseLockController)
- self.isMouseLocked = false
- self.savedMouseCursor = nil
- self.boundKeys = {Enum.KeyCode.LeftShift, Enum.KeyCode.RightShift} -- defaults
- self.mouseLockToggledEvent = Instance.new("BindableEvent")
- local boundKeysObj = script:FindFirstChild("BoundKeys")
- if (not boundKeysObj) or (not boundKeysObj:IsA("StringValue")) then
- -- If object with correct name was found, but it's not a StringValue, destroy and replace
- if boundKeysObj then
- boundKeysObj:Destroy()
- end
- boundKeysObj = Instance.new("StringValue")
- boundKeysObj.Name = "BoundKeys"
- boundKeysObj.Value = "LeftShift,RightShift"
- boundKeysObj.Parent = script
- end
- if boundKeysObj then
- boundKeysObj.Changed:Connect(function(value)
- self:OnBoundKeysObjectChanged(value)
- end)
- self:OnBoundKeysObjectChanged(boundKeysObj.Value) -- Initial setup call
- end
- -- Watch for changes to user's ControlMode and ComputerMovementMode settings and update the feature availability accordingly
- GameSettings.Changed:Connect(function(property)
- if property == "ControlMode" or property == "ComputerMovementMode" then
- self:UpdateMouseLockAvailability()
- end
- end)
- -- Watch for changes to DevEnableMouseLock and update the feature availability accordingly
- PlayersService.LocalPlayer:GetPropertyChangedSignal("DevEnableMouseLock"):Connect(function()
- self:UpdateMouseLockAvailability()
- end)
- -- Watch for changes to DevEnableMouseLock and update the feature availability accordingly
- PlayersService.LocalPlayer:GetPropertyChangedSignal("DevComputerMovementMode"):Connect(function()
- self:UpdateMouseLockAvailability()
- end)
- self:UpdateMouseLockAvailability()
- return self
- end
- function MouseLockController:GetIsMouseLocked()
- return self.isMouseLocked
- end
- function MouseLockController:GetBindableToggleEvent()
- return self.mouseLockToggledEvent.Event
- end
- function MouseLockController:GetMouseLockOffset()
- local offsetValueObj = script:FindFirstChild("CameraOffset")
- if offsetValueObj and offsetValueObj:IsA("Vector3Value") then
- return offsetValueObj.Value
- else
- -- If CameraOffset object was found but not correct type, destroy
- if offsetValueObj then
- offsetValueObj:Destroy()
- end
- offsetValueObj = Instance.new("Vector3Value")
- offsetValueObj.Name = "CameraOffset"
- offsetValueObj.Value = Vector3.new(1.75,0,0) -- Legacy Default Value
- offsetValueObj.Parent = script
- end
- if offsetValueObj and offsetValueObj.Value then
- return offsetValueObj.Value
- end
- return Vector3.new(1.75,0,0)
- end
- function MouseLockController:UpdateMouseLockAvailability()
- local devAllowsMouseLock = PlayersService.LocalPlayer.DevEnableMouseLock
- local devMovementModeIsScriptable = PlayersService.LocalPlayer.DevComputerMovementMode == Enum.DevComputerMovementMode.Scriptable
- local userHasMouseLockModeEnabled = GameSettings.ControlMode == Enum.ControlMode.MouseLockSwitch
- local userHasClickToMoveEnabled = GameSettings.ComputerMovementMode == Enum.ComputerMovementMode.ClickToMove
- local MouseLockAvailable = devAllowsMouseLock and userHasMouseLockModeEnabled and not userHasClickToMoveEnabled and not devMovementModeIsScriptable
- if MouseLockAvailable~=self.enabled then
- self:EnableMouseLock(MouseLockAvailable)
- end
- end
- function MouseLockController:OnBoundKeysObjectChanged(newValue)
- self.boundKeys = {} -- Overriding defaults, note: possibly with nothing at all if boundKeysObj.Value is "" or contains invalid values
- for token in string.gmatch(newValue,"[^%s,]+") do
- for _, keyEnum in pairs(Enum.KeyCode:GetEnumItems()) do
- if token == keyEnum.Name then
- self.boundKeys[#self.boundKeys+1] = keyEnum
- break
- end
- end
- end
- self:UnbindContextActions()
- self:BindContextActions()
- end
- --[[ Local Functions ]]--
- function MouseLockController:OnMouseLockToggled()
- self.isMouseLocked = not self.isMouseLocked
- if self.isMouseLocked then
- local cursorImageValueObj = script:FindFirstChild("CursorImage")
- if cursorImageValueObj and cursorImageValueObj:IsA("StringValue") and cursorImageValueObj.Value then
- self.savedMouseCursor = Mouse.Icon
- Mouse.Icon = cursorImageValueObj.Value
- else
- if cursorImageValueObj then
- cursorImageValueObj:Destroy()
- end
- cursorImageValueObj = Instance.new("StringValue")
- cursorImageValueObj.Name = "CursorImage"
- cursorImageValueObj.Value = DEFAULT_MOUSE_LOCK_CURSOR
- cursorImageValueObj.Parent = script
- self.savedMouseCursor = Mouse.Icon
- Mouse.Icon = DEFAULT_MOUSE_LOCK_CURSOR
- end
- else
- if self.savedMouseCursor then
- Mouse.Icon = self.savedMouseCursor
- self.savedMouseCursor = nil
- end
- end
- self.mouseLockToggledEvent:Fire()
- end
- function MouseLockController:DoMouseLockSwitch(name, state, input)
- if state == Enum.UserInputState.Begin then
- self:OnMouseLockToggled()
- return Enum.ContextActionResult.Sink
- end
- return Enum.ContextActionResult.Pass
- end
- function MouseLockController:BindContextActions()
- ContextActionService:BindActionAtPriority(CONTEXT_ACTION_NAME, function(name, state, input)
- return self:DoMouseLockSwitch(name, state, input)
- end, false, MOUSELOCK_ACTION_PRIORITY, unpack(self.boundKeys))
- end
- function MouseLockController:UnbindContextActions()
- ContextActionService:UnbindAction(CONTEXT_ACTION_NAME)
- end
- function MouseLockController:IsMouseLocked()
- return self.enabled and self.isMouseLocked
- end
- function MouseLockController:EnableMouseLock(enable)
- if enable ~= self.enabled then
- self.enabled = enable
- if self.enabled then
- -- Enabling the mode
- self:BindContextActions()
- else
- -- Disabling
- -- Restore mouse cursor
- if Mouse.Icon~="" then
- Mouse.Icon = ""
- end
- self:UnbindContextActions()
- -- If the mode is disabled while being used, fire the event to toggle it off
- if self.isMouseLocked then
- self.mouseLockToggledEvent:Fire()
- end
- self.isMouseLocked = false
- end
- end
- end
- return MouseLockController
- end
- function _TransparencyController()
- local MAX_TWEEN_RATE = 2.8 -- per second
- local Util = _CameraUtils()
- --[[ The Module ]]--
- local TransparencyController = {}
- TransparencyController.__index = TransparencyController
- function TransparencyController.new()
- local self = setmetatable({}, TransparencyController)
- self.lastUpdate = tick()
- self.transparencyDirty = false
- self.enabled = false
- self.lastTransparency = nil
- self.descendantAddedConn, self.descendantRemovingConn = nil, nil
- self.toolDescendantAddedConns = {}
- self.toolDescendantRemovingConns = {}
- self.cachedParts = {}
- return self
- end
- function TransparencyController:HasToolAncestor(object)
- if object.Parent == nil then return false end
- return object.Parent:IsA('Tool') or self:HasToolAncestor(object.Parent)
- end
- function TransparencyController:IsValidPartToModify(part)
- if part:IsA('BasePart') or part:IsA('Decal') then
- return not self:HasToolAncestor(part)
- end
- return false
- end
- function TransparencyController:CachePartsRecursive(object)
- if object then
- if self:IsValidPartToModify(object) then
- self.cachedParts[object] = true
- self.transparencyDirty = true
- end
- for _, child in pairs(object:GetChildren()) do
- self:CachePartsRecursive(child)
- end
- end
- end
- function TransparencyController:TeardownTransparency()
- for child, _ in pairs(self.cachedParts) do
- child.LocalTransparencyModifier = 0
- end
- self.cachedParts = {}
- self.transparencyDirty = true
- self.lastTransparency = nil
- if self.descendantAddedConn then
- self.descendantAddedConn:disconnect()
- self.descendantAddedConn = nil
- end
- if self.descendantRemovingConn then
- self.descendantRemovingConn:disconnect()
- self.descendantRemovingConn = nil
- end
- for object, conn in pairs(self.toolDescendantAddedConns) do
- conn:Disconnect()
- self.toolDescendantAddedConns[object] = nil
- end
- for object, conn in pairs(self.toolDescendantRemovingConns) do
- conn:Disconnect()
- self.toolDescendantRemovingConns[object] = nil
- end
- end
- function TransparencyController:SetupTransparency(character)
- self:TeardownTransparency()
- if self.descendantAddedConn then self.descendantAddedConn:disconnect() end
- self.descendantAddedConn = character.DescendantAdded:Connect(function(object)
- -- This is a part we want to invisify
- if self:IsValidPartToModify(object) then
- self.cachedParts[object] = true
- self.transparencyDirty = true
- -- There is now a tool under the character
- elseif object:IsA('Tool') then
- if self.toolDescendantAddedConns[object] then self.toolDescendantAddedConns[object]:Disconnect() end
- self.toolDescendantAddedConns[object] = object.DescendantAdded:Connect(function(toolChild)
- self.cachedParts[toolChild] = nil
- if toolChild:IsA('BasePart') or toolChild:IsA('Decal') then
- -- Reset the transparency
- toolChild.LocalTransparencyModifier = 0
- end
- end)
- if self.toolDescendantRemovingConns[object] then self.toolDescendantRemovingConns[object]:disconnect() end
- self.toolDescendantRemovingConns[object] = object.DescendantRemoving:Connect(function(formerToolChild)
- wait() -- wait for new parent
- if character and formerToolChild and formerToolChild:IsDescendantOf(character) then
- if self:IsValidPartToModify(formerToolChild) then
- self.cachedParts[formerToolChild] = true
- self.transparencyDirty = true
- end
- end
- end)
- end
- end)
- if self.descendantRemovingConn then self.descendantRemovingConn:disconnect() end
- self.descendantRemovingConn = character.DescendantRemoving:connect(function(object)
- if self.cachedParts[object] then
- self.cachedParts[object] = nil
- -- Reset the transparency
- object.LocalTransparencyModifier = 0
- end
- end)
- self:CachePartsRecursive(character)
- end
- function TransparencyController:Enable(enable)
- if self.enabled ~= enable then
- self.enabled = enable
- self:Update()
- end
- end
- function TransparencyController:SetSubject(subject)
- local character = nil
- if subject and subject:IsA("Humanoid") then
- character = subject.Parent
- end
- if subject and subject:IsA("VehicleSeat") and subject.Occupant then
- character = subject.Occupant.Parent
- end
- if character then
- self:SetupTransparency(character)
- else
- self:TeardownTransparency()
- end
- end
- function TransparencyController:Update()
- local instant = false
- local now = tick()
- local currentCamera = workspace.CurrentCamera
- if currentCamera then
- local transparency = 0
- if not self.enabled then
- instant = true
- else
- local distance = (currentCamera.Focus.p - currentCamera.CoordinateFrame.p).magnitude
- transparency = (distance<2) and (1.0-(distance-0.5)/1.5) or 0 --(7 - distance) / 5
- if transparency < 0.5 then
- transparency = 0
- end
- if self.lastTransparency then
- local deltaTransparency = transparency - self.lastTransparency
- -- Don't tween transparency if it is instant or your character was fully invisible last frame
- if not instant and transparency < 1 and self.lastTransparency < 0.95 then
- local maxDelta = MAX_TWEEN_RATE * (now - self.lastUpdate)
- deltaTransparency = math.clamp(deltaTransparency, -maxDelta, maxDelta)
- end
- transparency = self.lastTransparency + deltaTransparency
- else
- self.transparencyDirty = true
- end
- transparency = math.clamp(Util.Round(transparency, 2), 0, 1)
- end
- if self.transparencyDirty or self.lastTransparency ~= transparency then
- for child, _ in pairs(self.cachedParts) do
- child.LocalTransparencyModifier = transparency
- end
- self.transparencyDirty = false
- self.lastTransparency = transparency
- end
- end
- self.lastUpdate = now
- end
- return TransparencyController
- end
- function _Poppercam()
- local ZoomController = _ZoomController()
- local TransformExtrapolator = {} do
- TransformExtrapolator.__index = TransformExtrapolator
- local CF_IDENTITY = CFrame.new()
- local function cframeToAxis(cframe)
- local axis, angle = cframe:toAxisAngle()
- return axis*angle
- end
- local function axisToCFrame(axis)
- local angle = axis.magnitude
- if angle > 1e-5 then
- return CFrame.fromAxisAngle(axis, angle)
- end
- return CF_IDENTITY
- end
- local function extractRotation(cf)
- local _, _, _, xx, yx, zx, xy, yy, zy, xz, yz, zz = cf:components()
- return CFrame.new(0, 0, 0, xx, yx, zx, xy, yy, zy, xz, yz, zz)
- end
- function TransformExtrapolator.new()
- return setmetatable({
- lastCFrame = nil,
- }, TransformExtrapolator)
- end
- function TransformExtrapolator:Step(dt, currentCFrame)
- local lastCFrame = self.lastCFrame or currentCFrame
- self.lastCFrame = currentCFrame
- local currentPos = currentCFrame.p
- local currentRot = extractRotation(currentCFrame)
- local lastPos = lastCFrame.p
- local lastRot = extractRotation(lastCFrame)
- -- Estimate velocities from the delta between now and the last frame
- -- This estimation can be a little noisy.
- local dp = (currentPos - lastPos)/dt
- local dr = cframeToAxis(currentRot*lastRot:inverse())/dt
- local function extrapolate(t)
- local p = dp*t + currentPos
- local r = axisToCFrame(dr*t)*currentRot
- return r + p
- end
- return {
- extrapolate = extrapolate,
- posVelocity = dp,
- rotVelocity = dr,
- }
- end
- function TransformExtrapolator:Reset()
- self.lastCFrame = nil
- end
- end
- --[[ The Module ]]--
- local BaseOcclusion = _BaseOcclusion()
- local Poppercam = setmetatable({}, BaseOcclusion)
- Poppercam.__index = Poppercam
- function Poppercam.new()
- local self = setmetatable(BaseOcclusion.new(), Poppercam)
- self.focusExtrapolator = TransformExtrapolator.new()
- return self
- end
- function Poppercam:GetOcclusionMode()
- return Enum.DevCameraOcclusionMode.Zoom
- end
- function Poppercam:Enable(enable)
- self.focusExtrapolator:Reset()
- end
- function Poppercam:Update(renderDt, desiredCameraCFrame, desiredCameraFocus, cameraController)
- local rotatedFocus = CFrame.new(desiredCameraFocus.p, desiredCameraCFrame.p)*CFrame.new(
- 0, 0, 0,
- -1, 0, 0,
- 0, 1, 0,
- 0, 0, -1
- )
- local extrapolation = self.focusExtrapolator:Step(renderDt, rotatedFocus)
- local zoom = ZoomController.Update(renderDt, rotatedFocus, extrapolation)
- return rotatedFocus*CFrame.new(0, 0, zoom), desiredCameraFocus
- end
- -- Called when character is added
- function Poppercam:CharacterAdded(character, player)
- end
- -- Called when character is about to be removed
- function Poppercam:CharacterRemoving(character, player)
- end
- function Poppercam:OnCameraSubjectChanged(newSubject)
- end
- local ZoomController = _ZoomController()
- function Poppercam:Update(renderDt, desiredCameraCFrame, desiredCameraFocus, cameraController)
- local rotatedFocus = desiredCameraFocus * (desiredCameraCFrame - desiredCameraCFrame.p)
- local extrapolation = self.focusExtrapolator:Step(renderDt, rotatedFocus)
- local zoom = ZoomController.Update(renderDt, rotatedFocus, extrapolation)
- return rotatedFocus*CFrame.new(0, 0, zoom), desiredCameraFocus
- end
- return Poppercam
- end
- function _Invisicam()
- --[[ Top Level Roblox Services ]]--
- local PlayersService = game:GetService("Players")
- --[[ Constants ]]--
- local ZERO_VECTOR3 = Vector3.new(0,0,0)
- local USE_STACKING_TRANSPARENCY = true -- Multiple items between the subject and camera get transparency values that add up to TARGET_TRANSPARENCY
- local TARGET_TRANSPARENCY = 0.75 -- Classic Invisicam's Value, also used by new invisicam for parts hit by head and torso rays
- local TARGET_TRANSPARENCY_PERIPHERAL = 0.5 -- Used by new SMART_CIRCLE mode for items not hit by head and torso rays
- local MODE = {
- --CUSTOM = 1, -- Retired, unused
- LIMBS = 2, -- Track limbs
- MOVEMENT = 3, -- Track movement
- CORNERS = 4, -- Char model corners
- CIRCLE1 = 5, -- Circle of casts around character
- CIRCLE2 = 6, -- Circle of casts around character, camera relative
- LIMBMOVE = 7, -- LIMBS mode + MOVEMENT mode
- SMART_CIRCLE = 8, -- More sample points on and around character
- CHAR_OUTLINE = 9, -- Dynamic outline around the character
- }
- local LIMB_TRACKING_SET = {
- -- Body parts common to R15 and R6
- ['Head'] = true,
- -- Body parts unique to R6
- ['Left Arm'] = true,
- ['Right Arm'] = true,
- ['Left Leg'] = true,
- ['Right Leg'] = true,
- -- Body parts unique to R15
- ['LeftLowerArm'] = true,
- ['RightLowerArm'] = true,
- ['LeftUpperLeg'] = true,
- ['RightUpperLeg'] = true
- }
- local CORNER_FACTORS = {
- Vector3.new(1,1,-1),
- Vector3.new(1,-1,-1),
- Vector3.new(-1,-1,-1),
- Vector3.new(-1,1,-1)
- }
- local CIRCLE_CASTS = 10
- local MOVE_CASTS = 3
- local SMART_CIRCLE_CASTS = 24
- local SMART_CIRCLE_INCREMENT = 2.0 * math.pi / SMART_CIRCLE_CASTS
- local CHAR_OUTLINE_CASTS = 24
- -- Used to sanitize user-supplied functions
- local function AssertTypes(param, ...)
- local allowedTypes = {}
- local typeString = ''
- for _, typeName in pairs({...}) do
- allowedTypes[typeName] = true
- typeString = typeString .. (typeString == '' and '' or ' or ') .. typeName
- end
- local theType = type(param)
- assert(allowedTypes[theType], typeString .. " type expected, got: " .. theType)
- end
- -- Helper function for Determinant of 3x3, not in CameraUtils for performance reasons
- local function Det3x3(a,b,c,d,e,f,g,h,i)
- return (a*(e*i-f*h)-b*(d*i-f*g)+c*(d*h-e*g))
- end
- -- Smart Circle mode needs the intersection of 2 rays that are known to be in the same plane
- -- because they are generated from cross products with a common vector. This function is computing
- -- that intersection, but it's actually the general solution for the point halfway between where
- -- two skew lines come nearest to each other, which is more forgiving.
- local function RayIntersection(p0, v0, p1, v1)
- local v2 = v0:Cross(v1)
- local d1 = p1.x - p0.x
- local d2 = p1.y - p0.y
- local d3 = p1.z - p0.z
- local denom = Det3x3(v0.x,-v1.x,v2.x,v0.y,-v1.y,v2.y,v0.z,-v1.z,v2.z)
- if (denom == 0) then
- return ZERO_VECTOR3 -- No solution (rays are parallel)
- end
- local t0 = Det3x3(d1,-v1.x,v2.x,d2,-v1.y,v2.y,d3,-v1.z,v2.z) / denom
- local t1 = Det3x3(v0.x,d1,v2.x,v0.y,d2,v2.y,v0.z,d3,v2.z) / denom
- local s0 = p0 + t0 * v0
- local s1 = p1 + t1 * v1
- local s = s0 + 0.5 * ( s1 - s0 )
- -- 0.25 studs is a threshold for deciding if the rays are
- -- close enough to be considered intersecting, found through testing
- if (s1-s0).Magnitude < 0.25 then
- return s
- else
- return ZERO_VECTOR3
- end
- end
- --[[ The Module ]]--
- local BaseOcclusion = _BaseOcclusion()
- local Invisicam = setmetatable({}, BaseOcclusion)
- Invisicam.__index = Invisicam
- function Invisicam.new()
- local self = setmetatable(BaseOcclusion.new(), Invisicam)
- self.char = nil
- self.humanoidRootPart = nil
- self.torsoPart = nil
- self.headPart = nil
- self.childAddedConn = nil
- self.childRemovedConn = nil
- self.behaviors = {} -- Map of modes to behavior fns
- self.behaviors[MODE.LIMBS] = self.LimbBehavior
- self.behaviors[MODE.MOVEMENT] = self.MoveBehavior
- self.behaviors[MODE.CORNERS] = self.CornerBehavior
- self.behaviors[MODE.CIRCLE1] = self.CircleBehavior
- self.behaviors[MODE.CIRCLE2] = self.CircleBehavior
- self.behaviors[MODE.LIMBMOVE] = self.LimbMoveBehavior
- self.behaviors[MODE.SMART_CIRCLE] = self.SmartCircleBehavior
- self.behaviors[MODE.CHAR_OUTLINE] = self.CharacterOutlineBehavior
- self.mode = MODE.SMART_CIRCLE
- self.behaviorFunction = self.SmartCircleBehavior
- self.savedHits = {} -- Objects currently being faded in/out
- self.trackedLimbs = {} -- Used in limb-tracking casting modes
- self.camera = game.Workspace.CurrentCamera
- self.enabled = false
- return self
- end
- function Invisicam:Enable(enable)
- self.enabled = enable
- if not enable then
- self:Cleanup()
- end
- end
- function Invisicam:GetOcclusionMode()
- return Enum.DevCameraOcclusionMode.Invisicam
- end
- --[[ Module functions ]]--
- function Invisicam:LimbBehavior(castPoints)
- for limb, _ in pairs(self.trackedLimbs) do
- castPoints[#castPoints + 1] = limb.Position
- end
- end
- function Invisicam:MoveBehavior(castPoints)
- for i = 1, MOVE_CASTS do
- local position, velocity = self.humanoidRootPart.Position, self.humanoidRootPart.Velocity
- local horizontalSpeed = Vector3.new(velocity.X, 0, velocity.Z).Magnitude / 2
- local offsetVector = (i - 1) * self.humanoidRootPart.CFrame.lookVector * horizontalSpeed
- castPoints[#castPoints + 1] = position + offsetVector
- end
- end
- function Invisicam:CornerBehavior(castPoints)
- local cframe = self.humanoidRootPart.CFrame
- local centerPoint = cframe.p
- local rotation = cframe - centerPoint
- local halfSize = self.char:GetExtentsSize() / 2 --NOTE: Doesn't update w/ limb animations
- castPoints[#castPoints + 1] = centerPoint
- for i = 1, #CORNER_FACTORS do
- castPoints[#castPoints + 1] = centerPoint + (rotation * (halfSize * CORNER_FACTORS[i]))
- end
- end
- function Invisicam:CircleBehavior(castPoints)
- local cframe
- if self.mode == MODE.CIRCLE1 then
- cframe = self.humanoidRootPart.CFrame
- else
- local camCFrame = self.camera.CoordinateFrame
- cframe = camCFrame - camCFrame.p + self.humanoidRootPart.Position
- end
- castPoints[#castPoints + 1] = cframe.p
- for i = 0, CIRCLE_CASTS - 1 do
- local angle = (2 * math.pi / CIRCLE_CASTS) * i
- local offset = 3 * Vector3.new(math.cos(angle), math.sin(angle), 0)
- castPoints[#castPoints + 1] = cframe * offset
- end
- end
- function Invisicam:LimbMoveBehavior(castPoints)
- self:LimbBehavior(castPoints)
- self:MoveBehavior(castPoints)
- end
- function Invisicam:CharacterOutlineBehavior(castPoints)
- local torsoUp = self.torsoPart.CFrame.upVector.unit
- local torsoRight = self.torsoPart.CFrame.rightVector.unit
- -- Torso cross of points for interior coverage
- castPoints[#castPoints + 1] = self.torsoPart.CFrame.p
- castPoints[#castPoints + 1] = self.torsoPart.CFrame.p + torsoUp
- castPoints[#castPoints + 1] = self.torsoPart.CFrame.p - torsoUp
- castPoints[#castPoints + 1] = self.torsoPart.CFrame.p + torsoRight
- castPoints[#castPoints + 1] = self.torsoPart.CFrame.p - torsoRight
- if self.headPart then
- castPoints[#castPoints + 1] = self.headPart.CFrame.p
- end
- local cframe = CFrame.new(ZERO_VECTOR3,Vector3.new(self.camera.CoordinateFrame.lookVector.X,0,self.camera.CoordinateFrame.lookVector.Z))
- local centerPoint = (self.torsoPart and self.torsoPart.Position or self.humanoidRootPart.Position)
- local partsWhitelist = {self.torsoPart}
- if self.headPart then
- partsWhitelist[#partsWhitelist + 1] = self.headPart
- end
- for i = 1, CHAR_OUTLINE_CASTS do
- local angle = (2 * math.pi * i / CHAR_OUTLINE_CASTS)
- local offset = cframe * (3 * Vector3.new(math.cos(angle), math.sin(angle), 0))
- offset = Vector3.new(offset.X, math.max(offset.Y, -2.25), offset.Z)
- local ray = Ray.new(centerPoint + offset, -3 * offset)
- local hit, hitPoint = game.Workspace:FindPartOnRayWithWhitelist(ray, partsWhitelist, false, false)
- if hit then
- -- Use hit point as the cast point, but nudge it slightly inside the character so that bumping up against
- -- walls is less likely to cause a transparency glitch
- castPoints[#castPoints + 1] = hitPoint + 0.2 * (centerPoint - hitPoint).unit
- end
- end
- end
- function Invisicam:SmartCircleBehavior(castPoints)
- local torsoUp = self.torsoPart.CFrame.upVector.unit
- local torsoRight = self.torsoPart.CFrame.rightVector.unit
- -- SMART_CIRCLE mode includes rays to head and 5 to the torso.
- -- Hands, arms, legs and feet are not included since they
- -- are not canCollide and can therefore go inside of parts
- castPoints[#castPoints + 1] = self.torsoPart.CFrame.p
- castPoints[#castPoints + 1] = self.torsoPart.CFrame.p + torsoUp
- castPoints[#castPoints + 1] = self.torsoPart.CFrame.p - torsoUp
- castPoints[#castPoints + 1] = self.torsoPart.CFrame.p + torsoRight
- castPoints[#castPoints + 1] = self.torsoPart.CFrame.p - torsoRight
- if self.headPart then
- castPoints[#castPoints + 1] = self.headPart.CFrame.p
- end
- local cameraOrientation = self.camera.CFrame - self.camera.CFrame.p
- local torsoPoint = Vector3.new(0,0.5,0) + (self.torsoPart and self.torsoPart.Position or self.humanoidRootPart.Position)
- local radius = 2.5
- -- This loop first calculates points in a circle of radius 2.5 around the torso of the character, in the
- -- plane orthogonal to the camera's lookVector. Each point is then raycast to, to determine if it is within
- -- the free space surrounding the player (not inside anything). Two iterations are done to adjust points that
- -- are inside parts, to try to move them to valid locations that are still on their camera ray, so that the
- -- circle remains circular from the camera's perspective, but does not cast rays into walls or parts that are
- -- behind, below or beside the character and not really obstructing view of the character. This minimizes
- -- the undesirable situation where the character walks up to an exterior wall and it is made invisible even
- -- though it is behind the character.
- for i = 1, SMART_CIRCLE_CASTS do
- local angle = SMART_CIRCLE_INCREMENT * i - 0.5 * math.pi
- local offset = radius * Vector3.new(math.cos(angle), math.sin(angle), 0)
- local circlePoint = torsoPoint + cameraOrientation * offset
- -- Vector from camera to point on the circle being tested
- local vp = circlePoint - self.camera.CFrame.p
- local ray = Ray.new(torsoPoint, circlePoint - torsoPoint)
- local hit, hp, hitNormal = game.Workspace:FindPartOnRayWithIgnoreList(ray, {self.char}, false, false )
- local castPoint = circlePoint
- if hit then
- local hprime = hp + 0.1 * hitNormal.unit -- Slightly offset hit point from the hit surface
- local v0 = hprime - torsoPoint -- Vector from torso to offset hit point
- local perp = (v0:Cross(vp)).unit
- -- Vector from the offset hit point, along the hit surface
- local v1 = (perp:Cross(hitNormal)).unit
- -- Vector from camera to offset hit
- local vprime = (hprime - self.camera.CFrame.p).unit
- -- This dot product checks to see if the vector along the hit surface would hit the correct
- -- side of the invisicam cone, or if it would cross the camera look vector and hit the wrong side
- if ( v0.unit:Dot(-v1) < v0.unit:Dot(vprime)) then
- castPoint = RayIntersection(hprime, v1, circlePoint, vp)
- if castPoint.Magnitude > 0 then
- local ray = Ray.new(hprime, castPoint - hprime)
- local hit, hitPoint, hitNormal = game.Workspace:FindPartOnRayWithIgnoreList(ray, {self.char}, false, false )
- if hit then
- local hprime2 = hitPoint + 0.1 * hitNormal.unit
- castPoint = hprime2
- end
- else
- castPoint = hprime
- end
- else
- castPoint = hprime
- end
- local ray = Ray.new(torsoPoint, (castPoint - torsoPoint))
- local hit, hitPoint, hitNormal = game.Workspace:FindPartOnRayWithIgnoreList(ray, {self.char}, false, false )
- if hit then
- local castPoint2 = hitPoint - 0.1 * (castPoint - torsoPoint).unit
- castPoint = castPoint2
- end
- end
- castPoints[#castPoints + 1] = castPoint
- end
- end
- function Invisicam:CheckTorsoReference()
- if self.char then
- self.torsoPart = self.char:FindFirstChild("Torso")
- if not self.torsoPart then
- self.torsoPart = self.char:FindFirstChild("UpperTorso")
- if not self.torsoPart then
- self.torsoPart = self.char:FindFirstChild("HumanoidRootPart")
- end
- end
- self.headPart = self.char:FindFirstChild("Head")
- end
- end
- function Invisicam:CharacterAdded(char, player)
- -- We only want the LocalPlayer's character
- if player~=PlayersService.LocalPlayer then return end
- if self.childAddedConn then
- self.childAddedConn:Disconnect()
- self.childAddedConn = nil
- end
- if self.childRemovedConn then
- self.childRemovedConn:Disconnect()
- self.childRemovedConn = nil
- end
- self.char = char
- self.trackedLimbs = {}
- local function childAdded(child)
- if child:IsA("BasePart") then
- if LIMB_TRACKING_SET[child.Name] then
- self.trackedLimbs[child] = true
- end
- if child.Name == "Torso" or child.Name == "UpperTorso" then
- self.torsoPart = child
- end
- if child.Name == "Head" then
- self.headPart = child
- end
- end
- end
- local function childRemoved(child)
- self.trackedLimbs[child] = nil
- -- If removed/replaced part is 'Torso' or 'UpperTorso' double check that we still have a TorsoPart to use
- self:CheckTorsoReference()
- end
- self.childAddedConn = char.ChildAdded:Connect(childAdded)
- self.childRemovedConn = char.ChildRemoved:Connect(childRemoved)
- for _, child in pairs(self.char:GetChildren()) do
- childAdded(child)
- end
- end
- function Invisicam:SetMode(newMode)
- AssertTypes(newMode, 'number')
- for _, modeNum in pairs(MODE) do
- if modeNum == newMode then
- self.mode = newMode
- self.behaviorFunction = self.behaviors[self.mode]
- return
- end
- end
- error("Invalid mode number")
- end
- function Invisicam:GetObscuredParts()
- return self.savedHits
- end
- -- Want to turn off Invisicam? Be sure to call this after.
- function Invisicam:Cleanup()
- for hit, originalFade in pairs(self.savedHits) do
- hit.LocalTransparencyModifier = originalFade
- end
- end
- function Invisicam:Update(dt, desiredCameraCFrame, desiredCameraFocus)
- -- Bail if there is no Character
- if not self.enabled or not self.char then
- return desiredCameraCFrame, desiredCameraFocus
- end
- self.camera = game.Workspace.CurrentCamera
- -- TODO: Move this to a GetHumanoidRootPart helper, probably combine with CheckTorsoReference
- -- Make sure we still have a HumanoidRootPart
- if not self.humanoidRootPart then
- local humanoid = self.char:FindFirstChildOfClass("Humanoid")
- if humanoid and humanoid.RootPart then
- self.humanoidRootPart = humanoid.RootPart
- else
- -- Not set up with Humanoid? Try and see if there's one in the Character at all:
- self.humanoidRootPart = self.char:FindFirstChild("HumanoidRootPart")
- if not self.humanoidRootPart then
- -- Bail out, since we're relying on HumanoidRootPart existing
- return desiredCameraCFrame, desiredCameraFocus
- end
- end
- -- TODO: Replace this with something more sensible
- local ancestryChangedConn
- ancestryChangedConn = self.humanoidRootPart.AncestryChanged:Connect(function(child, parent)
- if child == self.humanoidRootPart and not parent then
- self.humanoidRootPart = nil
- if ancestryChangedConn and ancestryChangedConn.Connected then
- ancestryChangedConn:Disconnect()
- ancestryChangedConn = nil
- end
- end
- end)
- end
- if not self.torsoPart then
- self:CheckTorsoReference()
- if not self.torsoPart then
- -- Bail out, since we're relying on Torso existing, should never happen since we fall back to using HumanoidRootPart as torso
- return desiredCameraCFrame, desiredCameraFocus
- end
- end
- -- Make a list of world points to raycast to
- local castPoints = {}
- self.behaviorFunction(self, castPoints)
- -- Cast to get a list of objects between the camera and the cast points
- local currentHits = {}
- local ignoreList = {self.char}
- local function add(hit)
- currentHits[hit] = true
- if not self.savedHits[hit] then
- self.savedHits[hit] = hit.LocalTransparencyModifier
- end
- end
- local hitParts
- local hitPartCount = 0
- -- Hash table to treat head-ray-hit parts differently than the rest of the hit parts hit by other rays
- -- head/torso ray hit parts will be more transparent than peripheral parts when USE_STACKING_TRANSPARENCY is enabled
- local headTorsoRayHitParts = {}
- local perPartTransparencyHeadTorsoHits = TARGET_TRANSPARENCY
- local perPartTransparencyOtherHits = TARGET_TRANSPARENCY
- if USE_STACKING_TRANSPARENCY then
- -- This first call uses head and torso rays to find out how many parts are stacked up
- -- for the purpose of calculating required per-part transparency
- local headPoint = self.headPart and self.headPart.CFrame.p or castPoints[1]
- local torsoPoint = self.torsoPart and self.torsoPart.CFrame.p or castPoints[2]
- hitParts = self.camera:GetPartsObscuringTarget({headPoint, torsoPoint}, ignoreList)
- -- Count how many things the sample rays passed through, including decals. This should only
- -- count decals facing the camera, but GetPartsObscuringTarget does not return surface normals,
- -- so my compromise for now is to just let any decal increase the part count by 1. Only one
- -- decal per part will be considered.
- for i = 1, #hitParts do
- local hitPart = hitParts[i]
- hitPartCount = hitPartCount + 1 -- count the part itself
- headTorsoRayHitParts[hitPart] = true
- for _, child in pairs(hitPart:GetChildren()) do
- if child:IsA('Decal') or child:IsA('Texture') then
- hitPartCount = hitPartCount + 1 -- count first decal hit, then break
- break
- end
- end
- end
- if (hitPartCount > 0) then
- perPartTransparencyHeadTorsoHits = math.pow( ((0.5 * TARGET_TRANSPARENCY) + (0.5 * TARGET_TRANSPARENCY / hitPartCount)), 1 / hitPartCount )
- perPartTransparencyOtherHits = math.pow( ((0.5 * TARGET_TRANSPARENCY_PERIPHERAL) + (0.5 * TARGET_TRANSPARENCY_PERIPHERAL / hitPartCount)), 1 / hitPartCount )
- end
- end
- -- Now get all the parts hit by all the rays
- hitParts = self.camera:GetPartsObscuringTarget(castPoints, ignoreList)
- local partTargetTransparency = {}
- -- Include decals and textures
- for i = 1, #hitParts do
- local hitPart = hitParts[i]
- partTargetTransparency[hitPart] =headTorsoRayHitParts[hitPart] and perPartTransparencyHeadTorsoHits or perPartTransparencyOtherHits
- -- If the part is not already as transparent or more transparent than what invisicam requires, add it to the list of
- -- parts to be modified by invisicam
- if hitPart.Transparency < partTargetTransparency[hitPart] then
- add(hitPart)
- end
- -- Check all decals and textures on the part
- for _, child in pairs(hitPart:GetChildren()) do
- if child:IsA('Decal') or child:IsA('Texture') then
- if (child.Transparency < partTargetTransparency[hitPart]) then
- partTargetTransparency[child] = partTargetTransparency[hitPart]
- add(child)
- end
- end
- end
- end
- -- Invisibilize objects that are in the way, restore those that aren't anymore
- for hitPart, originalLTM in pairs(self.savedHits) do
- if currentHits[hitPart] then
- -- LocalTransparencyModifier gets whatever value is required to print the part's total transparency to equal perPartTransparency
- hitPart.LocalTransparencyModifier = (hitPart.Transparency < 1) and ((partTargetTransparency[hitPart] - hitPart.Transparency) / (1.0 - hitPart.Transparency)) or 0
- else -- Restore original pre-invisicam value of LTM
- hitPart.LocalTransparencyModifier = originalLTM
- self.savedHits[hitPart] = nil
- end
- end
- -- Invisicam does not change the camera values
- return desiredCameraCFrame, desiredCameraFocus
- end
- return Invisicam
- end
- function _LegacyCamera()
- local ZERO_VECTOR2 = Vector2.new(0,0)
- local Util = _CameraUtils()
- --[[ Services ]]--
- local PlayersService = game:GetService('Players')
- --[[ The Module ]]--
- local BaseCamera = _BaseCamera()
- local LegacyCamera = setmetatable({}, BaseCamera)
- LegacyCamera.__index = LegacyCamera
- function LegacyCamera.new()
- local self = setmetatable(BaseCamera.new(), LegacyCamera)
- self.cameraType = Enum.CameraType.Fixed
- self.lastUpdate = tick()
- self.lastDistanceToSubject = nil
- return self
- end
- function LegacyCamera:GetModuleName()
- return "LegacyCamera"
- end
- --[[ Functions overridden from BaseCamera ]]--
- function LegacyCamera:SetCameraToSubjectDistance(desiredSubjectDistance)
- return BaseCamera.SetCameraToSubjectDistance(self,desiredSubjectDistance)
- end
- function LegacyCamera:Update(dt)
- -- Cannot update until cameraType has been set
- if not self.cameraType then return end
- local now = tick()
- local timeDelta = (now - self.lastUpdate)
- local camera = workspace.CurrentCamera
- local newCameraCFrame = camera.CFrame
- local newCameraFocus = camera.Focus
- local player = PlayersService.LocalPlayer
- if self.lastUpdate == nil or timeDelta > 1 then
- self.lastDistanceToSubject = nil
- end
- local subjectPosition = self:GetSubjectPosition()
- if self.cameraType == Enum.CameraType.Fixed then
- if self.lastUpdate then
- -- Cap out the delta to 0.1 so we don't get some crazy things when we re-resume from
- local delta = math.min(0.1, now - self.lastUpdate)
- local gamepadRotation = self:UpdateGamepad()
- self.rotateInput = self.rotateInput + (gamepadRotation * delta)
- end
- if subjectPosition and player and camera then
- local distanceToSubject = self:GetCameraToSubjectDistance()
- local newLookVector = self:CalculateNewLookVector()
- self.rotateInput = ZERO_VECTOR2
- newCameraFocus = camera.Focus -- Fixed camera does not change focus
- newCameraCFrame = CFrame.new(camera.CFrame.p, camera.CFrame.p + (distanceToSubject * newLookVector))
- end
- elseif self.cameraType == Enum.CameraType.Attach then
- if subjectPosition and camera then
- local distanceToSubject = self:GetCameraToSubjectDistance()
- local humanoid = self:GetHumanoid()
- if self.lastUpdate and humanoid and humanoid.RootPart then
- -- Cap out the delta to 0.1 so we don't get some crazy things when we re-resume from
- local delta = math.min(0.1, now - self.lastUpdate)
- local gamepadRotation = self:UpdateGamepad()
- self.rotateInput = self.rotateInput + (gamepadRotation * delta)
- local forwardVector = humanoid.RootPart.CFrame.lookVector
- local y = Util.GetAngleBetweenXZVectors(forwardVector, self:GetCameraLookVector())
- if Util.IsFinite(y) then
- -- Preserve vertical rotation from user input
- self.rotateInput = Vector2.new(y, self.rotateInput.Y)
- end
- end
- local newLookVector = self:CalculateNewLookVector()
- self.rotateInput = ZERO_VECTOR2
- newCameraFocus = CFrame.new(subjectPosition)
- newCameraCFrame = CFrame.new(subjectPosition - (distanceToSubject * newLookVector), subjectPosition)
- end
- elseif self.cameraType == Enum.CameraType.Watch then
- if subjectPosition and player and camera then
- local cameraLook = nil
- local humanoid = self:GetHumanoid()
- if humanoid and humanoid.RootPart then
- local diffVector = subjectPosition - camera.CFrame.p
- cameraLook = diffVector.unit
- if self.lastDistanceToSubject and self.lastDistanceToSubject == self:GetCameraToSubjectDistance() then
- -- Don't clobber the zoom if they zoomed the camera
- local newDistanceToSubject = diffVector.magnitude
- self:SetCameraToSubjectDistance(newDistanceToSubject)
- end
- end
- local distanceToSubject = self:GetCameraToSubjectDistance()
- local newLookVector = self:CalculateNewLookVector(cameraLook)
- self.rotateInput = ZERO_VECTOR2
- newCameraFocus = CFrame.new(subjectPosition)
- newCameraCFrame = CFrame.new(subjectPosition - (distanceToSubject * newLookVector), subjectPosition)
- self.lastDistanceToSubject = distanceToSubject
- end
- else
- -- Unsupported type, return current values unchanged
- return camera.CFrame, camera.Focus
- end
- self.lastUpdate = now
- return newCameraCFrame, newCameraFocus
- end
- return LegacyCamera
- end
- function _OrbitalCamera()
- -- Local private variables and constants
- local UNIT_Z = Vector3.new(0,0,1)
- local X1_Y0_Z1 = Vector3.new(1,0,1) --Note: not a unit vector, used for projecting onto XZ plane
- local ZERO_VECTOR3 = Vector3.new(0,0,0)
- local ZERO_VECTOR2 = Vector2.new(0,0)
- local TAU = 2 * math.pi
- --[[ Gamepad Support ]]--
- local THUMBSTICK_DEADZONE = 0.2
- -- Do not edit these values, they are not the developer-set limits, they are limits
- -- to the values the camera system equations can correctly handle
- local MIN_ALLOWED_ELEVATION_DEG = -80
- local MAX_ALLOWED_ELEVATION_DEG = 80
- local externalProperties = {}
- externalProperties["InitialDistance"] = 25
- externalProperties["MinDistance"] = 10
- externalProperties["MaxDistance"] = 100
- externalProperties["InitialElevation"] = 35
- externalProperties["MinElevation"] = 35
- externalProperties["MaxElevation"] = 35
- externalProperties["ReferenceAzimuth"] = -45 -- Angle around the Y axis where the camera starts. -45 offsets the camera in the -X and +Z directions equally
- externalProperties["CWAzimuthTravel"] = 90 -- How many degrees the camera is allowed to rotate from the reference position, CW as seen from above
- externalProperties["CCWAzimuthTravel"] = 90 -- How many degrees the camera is allowed to rotate from the reference position, CCW as seen from above
- externalProperties["UseAzimuthLimits"] = false -- Full rotation around Y axis available by default
- local Util = _CameraUtils()
- --[[ Services ]]--
- local PlayersService = game:GetService('Players')
- local VRService = game:GetService("VRService")
- --[[ The Module ]]--
- local BaseCamera = _BaseCamera()
- local OrbitalCamera = setmetatable({}, BaseCamera)
- OrbitalCamera.__index = OrbitalCamera
- function OrbitalCamera.new()
- local self = setmetatable(BaseCamera.new(), OrbitalCamera)
- self.lastUpdate = tick()
- -- OrbitalCamera-specific members
- self.changedSignalConnections = {}
- self.refAzimuthRad = nil
- self.curAzimuthRad = nil
- self.minAzimuthAbsoluteRad = nil
- self.maxAzimuthAbsoluteRad = nil
- self.useAzimuthLimits = nil
- self.curElevationRad = nil
- self.minElevationRad = nil
- self.maxElevationRad = nil
- self.curDistance = nil
- self.minDistance = nil
- self.maxDistance = nil
- -- Gamepad
- self.r3ButtonDown = false
- self.l3ButtonDown = false
- self.gamepadDollySpeedMultiplier = 1
- self.lastUserPanCamera = tick()
- self.externalProperties = {}
- self.externalProperties["InitialDistance"] = 25
- self.externalProperties["MinDistance"] = 10
- self.externalProperties["MaxDistance"] = 100
- self.externalProperties["InitialElevation"] = 35
- self.externalProperties["MinElevation"] = 35
- self.externalProperties["MaxElevation"] = 35
- self.externalProperties["ReferenceAzimuth"] = -45 -- Angle around the Y axis where the camera starts. -45 offsets the camera in the -X and +Z directions equally
- self.externalProperties["CWAzimuthTravel"] = 90 -- How many degrees the camera is allowed to rotate from the reference position, CW as seen from above
- self.externalProperties["CCWAzimuthTravel"] = 90 -- How many degrees the camera is allowed to rotate from the reference position, CCW as seen from above
- self.externalProperties["UseAzimuthLimits"] = false -- Full rotation around Y axis available by default
- self:LoadNumberValueParameters()
- return self
- end
- function OrbitalCamera:LoadOrCreateNumberValueParameter(name, valueType, updateFunction)
- local valueObj = script:FindFirstChild(name)
- if valueObj and valueObj:isA(valueType) then
- -- Value object exists and is the correct type, use its value
- self.externalProperties[name] = valueObj.Value
- elseif self.externalProperties[name] ~= nil then
- -- Create missing (or replace incorrectly-typed) valueObject with default value
- valueObj = Instance.new(valueType)
- valueObj.Name = name
- valueObj.Parent = script
- valueObj.Value = self.externalProperties[name]
- else
- print("externalProperties table has no entry for ",name)
- return
- end
- if updateFunction then
- if self.changedSignalConnections[name] then
- self.changedSignalConnections[name]:Disconnect()
- end
- self.changedSignalConnections[name] = valueObj.Changed:Connect(function(newValue)
- self.externalProperties[name] = newValue
- updateFunction(self)
- end)
- end
- end
- function OrbitalCamera:SetAndBoundsCheckAzimuthValues()
- self.minAzimuthAbsoluteRad = math.rad(self.externalProperties["ReferenceAzimuth"]) - math.abs(math.rad(self.externalProperties["CWAzimuthTravel"]))
- self.maxAzimuthAbsoluteRad = math.rad(self.externalProperties["ReferenceAzimuth"]) + math.abs(math.rad(self.externalProperties["CCWAzimuthTravel"]))
- self.useAzimuthLimits = self.externalProperties["UseAzimuthLimits"]
- if self.useAzimuthLimits then
- self.curAzimuthRad = math.max(self.curAzimuthRad, self.minAzimuthAbsoluteRad)
- self.curAzimuthRad = math.min(self.curAzimuthRad, self.maxAzimuthAbsoluteRad)
- end
- end
- function OrbitalCamera:SetAndBoundsCheckElevationValues()
- -- These degree values are the direct user input values. It is deliberate that they are
- -- ranged checked only against the extremes, and not against each other. Any time one
- -- is changed, both of the internal values in radians are recalculated. This allows for
- -- A developer to change the values in any order and for the end results to be that the
- -- internal values adjust to match intent as best as possible.
- local minElevationDeg = math.max(self.externalProperties["MinElevation"], MIN_ALLOWED_ELEVATION_DEG)
- local maxElevationDeg = math.min(self.externalProperties["MaxElevation"], MAX_ALLOWED_ELEVATION_DEG)
- -- Set internal values in radians
- self.minElevationRad = math.rad(math.min(minElevationDeg, maxElevationDeg))
- self.maxElevationRad = math.rad(math.max(minElevationDeg, maxElevationDeg))
- self.curElevationRad = math.max(self.curElevationRad, self.minElevationRad)
- self.curElevationRad = math.min(self.curElevationRad, self.maxElevationRad)
- end
- function OrbitalCamera:SetAndBoundsCheckDistanceValues()
- self.minDistance = self.externalProperties["MinDistance"]
- self.maxDistance = self.externalProperties["MaxDistance"]
- self.curDistance = math.max(self.curDistance, self.minDistance)
- self.curDistance = math.min(self.curDistance, self.maxDistance)
- end
- -- This loads from, or lazily creates, NumberValue objects for exposed parameters
- function OrbitalCamera:LoadNumberValueParameters()
- -- These initial values do not require change listeners since they are read only once
- self:LoadOrCreateNumberValueParameter("InitialElevation", "NumberValue", nil)
- self:LoadOrCreateNumberValueParameter("InitialDistance", "NumberValue", nil)
- -- Note: ReferenceAzimuth is also used as an initial value, but needs a change listener because it is used in the calculation of the limits
- self:LoadOrCreateNumberValueParameter("ReferenceAzimuth", "NumberValue", self.SetAndBoundsCheckAzimuthValue)
- self:LoadOrCreateNumberValueParameter("CWAzimuthTravel", "NumberValue", self.SetAndBoundsCheckAzimuthValues)
- self:LoadOrCreateNumberValueParameter("CCWAzimuthTravel", "NumberValue", self.SetAndBoundsCheckAzimuthValues)
- self:LoadOrCreateNumberValueParameter("MinElevation", "NumberValue", self.SetAndBoundsCheckElevationValues)
- self:LoadOrCreateNumberValueParameter("MaxElevation", "NumberValue", self.SetAndBoundsCheckElevationValues)
- self:LoadOrCreateNumberValueParameter("MinDistance", "NumberValue", self.SetAndBoundsCheckDistanceValues)
- self:LoadOrCreateNumberValueParameter("MaxDistance", "NumberValue", self.SetAndBoundsCheckDistanceValues)
- self:LoadOrCreateNumberValueParameter("UseAzimuthLimits", "BoolValue", self.SetAndBoundsCheckAzimuthValues)
- -- Internal values set (in radians, from degrees), plus sanitization
- self.curAzimuthRad = math.rad(self.externalProperties["ReferenceAzimuth"])
- self.curElevationRad = math.rad(self.externalProperties["InitialElevation"])
- self.curDistance = self.externalProperties["InitialDistance"]
- self:SetAndBoundsCheckAzimuthValues()
- self:SetAndBoundsCheckElevationValues()
- self:SetAndBoundsCheckDistanceValues()
- end
- function OrbitalCamera:GetModuleName()
- return "OrbitalCamera"
- end
- function OrbitalCamera:SetInitialOrientation(humanoid)
- if not humanoid or not humanoid.RootPart then
- warn("OrbitalCamera could not set initial orientation due to missing humanoid")
- return
- end
- local newDesiredLook = (humanoid.RootPart.CFrame.lookVector - Vector3.new(0,0.23,0)).unit
- local horizontalShift = Util.GetAngleBetweenXZVectors(newDesiredLook, self:GetCameraLookVector())
- local vertShift = math.asin(self:GetCameraLookVector().y) - math.asin(newDesiredLook.y)
- if not Util.IsFinite(horizontalShift) then
- horizontalShift = 0
- end
- if not Util.IsFinite(vertShift) then
- vertShift = 0
- end
- self.rotateInput = Vector2.new(horizontalShift, vertShift)
- end
- --[[ Functions of BaseCamera that are overridden by OrbitalCamera ]]--
- function OrbitalCamera:GetCameraToSubjectDistance()
- return self.curDistance
- end
- function OrbitalCamera:SetCameraToSubjectDistance(desiredSubjectDistance)
- print("OrbitalCamera SetCameraToSubjectDistance ",desiredSubjectDistance)
- local player = PlayersService.LocalPlayer
- if player then
- self.currentSubjectDistance = math.clamp(desiredSubjectDistance, self.minDistance, self.maxDistance)
- -- OrbitalCamera is not allowed to go into the first-person range
- self.currentSubjectDistance = math.max(self.currentSubjectDistance, self.FIRST_PERSON_DISTANCE_THRESHOLD)
- end
- self.inFirstPerson = false
- self:UpdateMouseBehavior()
- return self.currentSubjectDistance
- end
- function OrbitalCamera:CalculateNewLookVector(suppliedLookVector, xyRotateVector)
- local currLookVector = suppliedLookVector or self:GetCameraLookVector()
- local currPitchAngle = math.asin(currLookVector.y)
- local yTheta = math.clamp(xyRotateVector.y, currPitchAngle - math.rad(MAX_ALLOWED_ELEVATION_DEG), currPitchAngle - math.rad(MIN_ALLOWED_ELEVATION_DEG))
- local constrainedRotateInput = Vector2.new(xyRotateVector.x, yTheta)
- local startCFrame = CFrame.new(ZERO_VECTOR3, currLookVector)
- local newLookVector = (CFrame.Angles(0, -constrainedRotateInput.x, 0) * startCFrame * CFrame.Angles(-constrainedRotateInput.y,0,0)).lookVector
- return newLookVector
- end
- function OrbitalCamera:GetGamepadPan(name, state, input)
- if input.UserInputType == self.activeGamepad and input.KeyCode == Enum.KeyCode.Thumbstick2 then
- if self.r3ButtonDown or self.l3ButtonDown then
- -- R3 or L3 Thumbstick is depressed, right stick controls dolly in/out
- if (input.Position.Y > THUMBSTICK_DEADZONE) then
- self.gamepadDollySpeedMultiplier = 0.96
- elseif (input.Position.Y < -THUMBSTICK_DEADZONE) then
- self.gamepadDollySpeedMultiplier = 1.04
- else
- self.gamepadDollySpeedMultiplier = 1.00
- end
- else
- if state == Enum.UserInputState.Cancel then
- self.gamepadPanningCamera = ZERO_VECTOR2
- return
- end
- local inputVector = Vector2.new(input.Position.X, -input.Position.Y)
- if inputVector.magnitude > THUMBSTICK_DEADZONE then
- self.gamepadPanningCamera = Vector2.new(input.Position.X, -input.Position.Y)
- else
- self.gamepadPanningCamera = ZERO_VECTOR2
- end
- end
- return Enum.ContextActionResult.Sink
- end
- return Enum.ContextActionResult.Pass
- end
- function OrbitalCamera:DoGamepadZoom(name, state, input)
- if input.UserInputType == self.activeGamepad and (input.KeyCode == Enum.KeyCode.ButtonR3 or input.KeyCode == Enum.KeyCode.ButtonL3) then
- if (state == Enum.UserInputState.Begin) then
- self.r3ButtonDown = input.KeyCode == Enum.KeyCode.ButtonR3
- self.l3ButtonDown = input.KeyCode == Enum.KeyCode.ButtonL3
- elseif (state == Enum.UserInputState.End) then
- if (input.KeyCode == Enum.KeyCode.ButtonR3) then
- self.r3ButtonDown = false
- elseif (input.KeyCode == Enum.KeyCode.ButtonL3) then
- self.l3ButtonDown = false
- end
- if (not self.r3ButtonDown) and (not self.l3ButtonDown) then
- self.gamepadDollySpeedMultiplier = 1.00
- end
- end
- return Enum.ContextActionResult.Sink
- end
- return Enum.ContextActionResult.Pass
- end
- function OrbitalCamera:BindGamepadInputActions()
- self:BindAction("OrbitalCamGamepadPan", function(name, state, input) return self:GetGamepadPan(name, state, input) end,
- false, Enum.KeyCode.Thumbstick2)
- self:BindAction("OrbitalCamGamepadZoom", function(name, state, input) return self:DoGamepadZoom(name, state, input) end,
- false, Enum.KeyCode.ButtonR3, Enum.KeyCode.ButtonL3)
- end
- -- [[ Update ]]--
- function OrbitalCamera:Update(dt)
- local now = tick()
- local timeDelta = (now - self.lastUpdate)
- local userPanningTheCamera = (self.UserPanningTheCamera == true)
- local camera = workspace.CurrentCamera
- local newCameraCFrame = camera.CFrame
- local newCameraFocus = camera.Focus
- local player = PlayersService.LocalPlayer
- local cameraSubject = camera and camera.CameraSubject
- local isInVehicle = cameraSubject and cameraSubject:IsA('VehicleSeat')
- local isOnASkateboard = cameraSubject and cameraSubject:IsA('SkateboardPlatform')
- if self.lastUpdate == nil or timeDelta > 1 then
- self.lastCameraTransform = nil
- end
- if self.lastUpdate then
- local gamepadRotation = self:UpdateGamepad()
- if self:ShouldUseVRRotation() then
- self.RotateInput = self.RotateInput + self:GetVRRotationInput()
- else
- -- Cap out the delta to 0.1 so we don't get some crazy things when we re-resume from
- local delta = math.min(0.1, timeDelta)
- if gamepadRotation ~= ZERO_VECTOR2 then
- userPanningTheCamera = true
- self.rotateInput = self.rotateInput + (gamepadRotation * delta)
- end
- local angle = 0
- if not (isInVehicle or isOnASkateboard) then
- angle = angle + (self.TurningLeft and -120 or 0)
- angle = angle + (self.TurningRight and 120 or 0)
- end
- if angle ~= 0 then
- self.rotateInput = self.rotateInput + Vector2.new(math.rad(angle * delta), 0)
- userPanningTheCamera = true
- end
- end
- end
- -- Reset tween speed if user is panning
- if userPanningTheCamera then
- self.lastUserPanCamera = tick()
- end
- local subjectPosition = self:GetSubjectPosition()
- if subjectPosition and player and camera then
- -- Process any dollying being done by gamepad
- -- TODO: Move this
- if self.gamepadDollySpeedMultiplier ~= 1 then
- self:SetCameraToSubjectDistance(self.currentSubjectDistance * self.gamepadDollySpeedMultiplier)
- end
- local VREnabled = VRService.VREnabled
- newCameraFocus = VREnabled and self:GetVRFocus(subjectPosition, timeDelta) or CFrame.new(subjectPosition)
- local cameraFocusP = newCameraFocus.p
- if VREnabled and not self:IsInFirstPerson() then
- local cameraHeight = self:GetCameraHeight()
- local vecToSubject = (subjectPosition - camera.CFrame.p)
- local distToSubject = vecToSubject.magnitude
- -- Only move the camera if it exceeded a maximum distance to the subject in VR
- if distToSubject > self.currentSubjectDistance or self.rotateInput.x ~= 0 then
- local desiredDist = math.min(distToSubject, self.currentSubjectDistance)
- -- Note that CalculateNewLookVector is overridden from BaseCamera
- vecToSubject = self:CalculateNewLookVector(vecToSubject.unit * X1_Y0_Z1, Vector2.new(self.rotateInput.x, 0)) * desiredDist
- local newPos = cameraFocusP - vecToSubject
- local desiredLookDir = camera.CFrame.lookVector
- if self.rotateInput.x ~= 0 then
- desiredLookDir = vecToSubject
- end
- local lookAt = Vector3.new(newPos.x + desiredLookDir.x, newPos.y, newPos.z + desiredLookDir.z)
- self.RotateInput = ZERO_VECTOR2
- newCameraCFrame = CFrame.new(newPos, lookAt) + Vector3.new(0, cameraHeight, 0)
- end
- else
- -- self.RotateInput is a Vector2 of mouse movement deltas since last update
- self.curAzimuthRad = self.curAzimuthRad - self.rotateInput.x
- if self.useAzimuthLimits then
- self.curAzimuthRad = math.clamp(self.curAzimuthRad, self.minAzimuthAbsoluteRad, self.maxAzimuthAbsoluteRad)
- else
- self.curAzimuthRad = (self.curAzimuthRad ~= 0) and (math.sign(self.curAzimuthRad) * (math.abs(self.curAzimuthRad) % TAU)) or 0
- end
- self.curElevationRad = math.clamp(self.curElevationRad + self.rotateInput.y, self.minElevationRad, self.maxElevationRad)
- local cameraPosVector = self.currentSubjectDistance * ( CFrame.fromEulerAnglesYXZ( -self.curElevationRad, self.curAzimuthRad, 0 ) * UNIT_Z )
- local camPos = subjectPosition + cameraPosVector
- newCameraCFrame = CFrame.new(camPos, subjectPosition)
- self.rotateInput = ZERO_VECTOR2
- end
- self.lastCameraTransform = newCameraCFrame
- self.lastCameraFocus = newCameraFocus
- if (isInVehicle or isOnASkateboard) and cameraSubject:IsA('BasePart') then
- self.lastSubjectCFrame = cameraSubject.CFrame
- else
- self.lastSubjectCFrame = nil
- end
- end
- self.lastUpdate = now
- return newCameraCFrame, newCameraFocus
- end
- return OrbitalCamera
- end
- function _ClassicCamera()
- -- Local private variables and constants
- local ZERO_VECTOR2 = Vector2.new(0,0)
- local tweenAcceleration = math.rad(220) --Radians/Second^2
- local tweenSpeed = math.rad(0) --Radians/Second
- local tweenMaxSpeed = math.rad(250) --Radians/Second
- local TIME_BEFORE_AUTO_ROTATE = 2.0 --Seconds, used when auto-aligning camera with vehicles
- local INITIAL_CAMERA_ANGLE = CFrame.fromOrientation(math.rad(-15), 0, 0)
- local FFlagUserCameraToggle do
- local success, result = pcall(function()
- return UserSettings():IsUserFeatureEnabled("UserCameraToggle")
- end)
- FFlagUserCameraToggle = success and result
- end
- --[[ Services ]]--
- local PlayersService = game:GetService('Players')
- local VRService = game:GetService("VRService")
- local CameraInput = _CameraInput()
- local Util = _CameraUtils()
- --[[ The Module ]]--
- local BaseCamera = _BaseCamera()
- local ClassicCamera = setmetatable({}, BaseCamera)
- ClassicCamera.__index = ClassicCamera
- function ClassicCamera.new()
- local self = setmetatable(BaseCamera.new(), ClassicCamera)
- self.isFollowCamera = false
- self.isCameraToggle = false
- self.lastUpdate = tick()
- self.cameraToggleSpring = Util.Spring.new(5, 0)
- return self
- end
- function ClassicCamera:GetCameraToggleOffset(dt)
- assert(FFlagUserCameraToggle)
- if self.isCameraToggle then
- local zoom = self.currentSubjectDistance
- if CameraInput.getTogglePan() then
- self.cameraToggleSpring.goal = math.clamp(Util.map(zoom, 0.5, self.FIRST_PERSON_DISTANCE_THRESHOLD, 0, 1), 0, 1)
- else
- self.cameraToggleSpring.goal = 0
- end
- local distanceOffset = math.clamp(Util.map(zoom, 0.5, 64, 0, 1), 0, 1) + 1
- return Vector3.new(0, self.cameraToggleSpring:step(dt)*distanceOffset, 0)
- end
- return Vector3.new()
- end
- -- Movement mode standardized to Enum.ComputerCameraMovementMode values
- function ClassicCamera:SetCameraMovementMode(cameraMovementMode)
- BaseCamera.SetCameraMovementMode(self, cameraMovementMode)
- self.isFollowCamera = cameraMovementMode == Enum.ComputerCameraMovementMode.Follow
- self.isCameraToggle = cameraMovementMode == Enum.ComputerCameraMovementMode.CameraToggle
- end
- function ClassicCamera:Update()
- local now = tick()
- local timeDelta = now - self.lastUpdate
- local camera = workspace.CurrentCamera
- local newCameraCFrame = camera.CFrame
- local newCameraFocus = camera.Focus
- local overrideCameraLookVector = nil
- if self.resetCameraAngle then
- local rootPart = self:GetHumanoidRootPart()
- if rootPart then
- overrideCameraLookVector = (rootPart.CFrame * INITIAL_CAMERA_ANGLE).lookVector
- else
- overrideCameraLookVector = INITIAL_CAMERA_ANGLE.lookVector
- end
- self.resetCameraAngle = false
- end
- local player = PlayersService.LocalPlayer
- local humanoid = self:GetHumanoid()
- local cameraSubject = camera.CameraSubject
- local isInVehicle = cameraSubject and cameraSubject:IsA('VehicleSeat')
- local isOnASkateboard = cameraSubject and cameraSubject:IsA('SkateboardPlatform')
- local isClimbing = humanoid and humanoid:GetState() == Enum.HumanoidStateType.Climbing
- if self.lastUpdate == nil or timeDelta > 1 then
- self.lastCameraTransform = nil
- end
- if self.lastUpdate then
- local gamepadRotation = self:UpdateGamepad()
- if self:ShouldUseVRRotation() then
- self.rotateInput = self.rotateInput + self:GetVRRotationInput()
- else
- -- Cap out the delta to 0.1 so we don't get some crazy things when we re-resume from
- local delta = math.min(0.1, timeDelta)
- if gamepadRotation ~= ZERO_VECTOR2 then
- self.rotateInput = self.rotateInput + (gamepadRotation * delta)
- end
- local angle = 0
- if not (isInVehicle or isOnASkateboard) then
- angle = angle + (self.turningLeft and -120 or 0)
- angle = angle + (self.turningRight and 120 or 0)
- end
- if angle ~= 0 then
- self.rotateInput = self.rotateInput + Vector2.new(math.rad(angle * delta), 0)
- end
- end
- end
- local cameraHeight = self:GetCameraHeight()
- -- Reset tween speed if user is panning
- if self.userPanningTheCamera then
- tweenSpeed = 0
- self.lastUserPanCamera = tick()
- end
- local userRecentlyPannedCamera = now - self.lastUserPanCamera < TIME_BEFORE_AUTO_ROTATE
- local subjectPosition = self:GetSubjectPosition()
- if subjectPosition and player and camera then
- local zoom = self:GetCameraToSubjectDistance()
- if zoom < 0.5 then
- zoom = 0.5
- end
- if self:GetIsMouseLocked() and not self:IsInFirstPerson() then
- -- We need to use the right vector of the camera after rotation, not before
- local newLookCFrame = self:CalculateNewLookCFrame(overrideCameraLookVector)
- local offset = self:GetMouseLockOffset()
- local cameraRelativeOffset = offset.X * newLookCFrame.rightVector + offset.Y * newLookCFrame.upVector + offset.Z * newLookCFrame.lookVector
- --offset can be NAN, NAN, NAN if newLookVector has only y component
- if Util.IsFiniteVector3(cameraRelativeOffset) then
- subjectPosition = subjectPosition + cameraRelativeOffset
- end
- else
- if not self.userPanningTheCamera and self.lastCameraTransform then
- local isInFirstPerson = self:IsInFirstPerson()
- if (isInVehicle or isOnASkateboard or (self.isFollowCamera and isClimbing)) and self.lastUpdate and humanoid and humanoid.Torso then
- if isInFirstPerson then
- if self.lastSubjectCFrame and (isInVehicle or isOnASkateboard) and cameraSubject:IsA('BasePart') then
- local y = -Util.GetAngleBetweenXZVectors(self.lastSubjectCFrame.lookVector, cameraSubject.CFrame.lookVector)
- if Util.IsFinite(y) then
- self.rotateInput = self.rotateInput + Vector2.new(y, 0)
- end
- tweenSpeed = 0
- end
- elseif not userRecentlyPannedCamera then
- local forwardVector = humanoid.Torso.CFrame.lookVector
- if isOnASkateboard then
- forwardVector = cameraSubject.CFrame.lookVector
- end
- tweenSpeed = math.clamp(tweenSpeed + tweenAcceleration * timeDelta, 0, tweenMaxSpeed)
- local percent = math.clamp(tweenSpeed * timeDelta, 0, 1)
- if self:IsInFirstPerson() and not (self.isFollowCamera and self.isClimbing) then
- percent = 1
- end
- local y = Util.GetAngleBetweenXZVectors(forwardVector, self:GetCameraLookVector())
- if Util.IsFinite(y) and math.abs(y) > 0.0001 then
- self.rotateInput = self.rotateInput + Vector2.new(y * percent, 0)
- end
- end
- elseif self.isFollowCamera and (not (isInFirstPerson or userRecentlyPannedCamera) and not VRService.VREnabled) then
- -- Logic that was unique to the old FollowCamera module
- local lastVec = -(self.lastCameraTransform.p - subjectPosition)
- local y = Util.GetAngleBetweenXZVectors(lastVec, self:GetCameraLookVector())
- -- This cutoff is to decide if the humanoid's angle of movement,
- -- relative to the camera's look vector, is enough that
- -- we want the camera to be following them. The point is to provide
- -- a sizable dead zone to allow more precise forward movements.
- local thetaCutoff = 0.4
- -- Check for NaNs
- if Util.IsFinite(y) and math.abs(y) > 0.0001 and math.abs(y) > thetaCutoff * timeDelta then
- self.rotateInput = self.rotateInput + Vector2.new(y, 0)
- end
- end
- end
- end
- if not self.isFollowCamera then
- local VREnabled = VRService.VREnabled
- if VREnabled then
- newCameraFocus = self:GetVRFocus(subjectPosition, timeDelta)
- else
- newCameraFocus = CFrame.new(subjectPosition)
- end
- local cameraFocusP = newCameraFocus.p
- if VREnabled and not self:IsInFirstPerson() then
- local vecToSubject = (subjectPosition - camera.CFrame.p)
- local distToSubject = vecToSubject.magnitude
- -- Only move the camera if it exceeded a maximum distance to the subject in VR
- if distToSubject > zoom or self.rotateInput.x ~= 0 then
- local desiredDist = math.min(distToSubject, zoom)
- vecToSubject = self:CalculateNewLookVectorVR() * desiredDist
- local newPos = cameraFocusP - vecToSubject
- local desiredLookDir = camera.CFrame.lookVector
- if self.rotateInput.x ~= 0 then
- desiredLookDir = vecToSubject
- end
- local lookAt = Vector3.new(newPos.x + desiredLookDir.x, newPos.y, newPos.z + desiredLookDir.z)
- self.rotateInput = ZERO_VECTOR2
- newCameraCFrame = CFrame.new(newPos, lookAt) + Vector3.new(0, cameraHeight, 0)
- end
- else
- local newLookVector = self:CalculateNewLookVector(overrideCameraLookVector)
- self.rotateInput = ZERO_VECTOR2
- newCameraCFrame = CFrame.new(cameraFocusP - (zoom * newLookVector), cameraFocusP)
- end
- else -- is FollowCamera
- local newLookVector = self:CalculateNewLookVector(overrideCameraLookVector)
- self.rotateInput = ZERO_VECTOR2
- if VRService.VREnabled then
- newCameraFocus = self:GetVRFocus(subjectPosition, timeDelta)
- else
- newCameraFocus = CFrame.new(subjectPosition)
- end
- newCameraCFrame = CFrame.new(newCameraFocus.p - (zoom * newLookVector), newCameraFocus.p) + Vector3.new(0, cameraHeight, 0)
- end
- if FFlagUserCameraToggle then
- local toggleOffset = self:GetCameraToggleOffset(timeDelta)
- newCameraFocus = newCameraFocus + toggleOffset
- newCameraCFrame = newCameraCFrame + toggleOffset
- end
- self.lastCameraTransform = newCameraCFrame
- self.lastCameraFocus = newCameraFocus
- if (isInVehicle or isOnASkateboard) and cameraSubject:IsA('BasePart') then
- self.lastSubjectCFrame = cameraSubject.CFrame
- else
- self.lastSubjectCFrame = nil
- end
- end
- self.lastUpdate = now
- return newCameraCFrame, newCameraFocus
- end
- function ClassicCamera:EnterFirstPerson()
- self.inFirstPerson = true
- self:UpdateMouseBehavior()
- end
- function ClassicCamera:LeaveFirstPerson()
- self.inFirstPerson = false
- self:UpdateMouseBehavior()
- end
- return ClassicCamera
- end
- function _CameraUtils()
- local CameraUtils = {}
- local FFlagUserCameraToggle do
- local success, result = pcall(function()
- return UserSettings():IsUserFeatureEnabled("UserCameraToggle")
- end)
- FFlagUserCameraToggle = success and result
- end
- local function round(num)
- return math.floor(num + 0.5)
- end
- -- Critically damped spring class for fluid motion effects
- local Spring = {} do
- Spring.__index = Spring
- -- Initialize to a given undamped frequency and default position
- function Spring.new(freq, pos)
- return setmetatable({
- freq = freq,
- goal = pos,
- pos = pos,
- vel = 0,
- }, Spring)
- end
- -- Advance the spring simulation by `dt` seconds
- function Spring:step(dt)
- local f = self.freq*2*math.pi
- local g = self.goal
- local p0 = self.pos
- local v0 = self.vel
- local offset = p0 - g
- local decay = math.exp(-f*dt)
- local p1 = (offset*(1 + f*dt) + v0*dt)*decay + g
- local v1 = (v0*(1 - f*dt) - offset*(f*f*dt))*decay
- self.pos = p1
- self.vel = v1
- return p1
- end
- end
- CameraUtils.Spring = Spring
- -- map a value from one range to another
- function CameraUtils.map(x, inMin, inMax, outMin, outMax)
- return (x - inMin)*(outMax - outMin)/(inMax - inMin) + outMin
- end
- -- From TransparencyController
- function CameraUtils.Round(num, places)
- local decimalPivot = 10^places
- return math.floor(num * decimalPivot + 0.5) / decimalPivot
- end
- function CameraUtils.IsFinite(val)
- return val == val and val ~= math.huge and val ~= -math.huge
- end
- function CameraUtils.IsFiniteVector3(vec3)
- return CameraUtils.IsFinite(vec3.X) and CameraUtils.IsFinite(vec3.Y) and CameraUtils.IsFinite(vec3.Z)
- end
- -- Legacy implementation renamed
- function CameraUtils.GetAngleBetweenXZVectors(v1, v2)
- return math.atan2(v2.X*v1.Z-v2.Z*v1.X, v2.X*v1.X+v2.Z*v1.Z)
- end
- function CameraUtils.RotateVectorByAngleAndRound(camLook, rotateAngle, roundAmount)
- if camLook.Magnitude > 0 then
- camLook = camLook.unit
- local currAngle = math.atan2(camLook.z, camLook.x)
- local newAngle = round((math.atan2(camLook.z, camLook.x) + rotateAngle) / roundAmount) * roundAmount
- return newAngle - currAngle
- end
- return 0
- end
- -- K is a tunable parameter that changes the shape of the S-curve
- -- the larger K is the more straight/linear the curve gets
- local k = 0.35
- local lowerK = 0.8
- local function SCurveTranform(t)
- t = math.clamp(t, -1, 1)
- if t >= 0 then
- return (k*t) / (k - t + 1)
- end
- return -((lowerK*-t) / (lowerK + t + 1))
- end
- local DEADZONE = 0.1
- local function toSCurveSpace(t)
- return (1 + DEADZONE) * (2*math.abs(t) - 1) - DEADZONE
- end
- local function fromSCurveSpace(t)
- return t/2 + 0.5
- end
- function CameraUtils.GamepadLinearToCurve(thumbstickPosition)
- local function onAxis(axisValue)
- local sign = 1
- if axisValue < 0 then
- sign = -1
- end
- local point = fromSCurveSpace(SCurveTranform(toSCurveSpace(math.abs(axisValue))))
- point = point * sign
- return math.clamp(point, -1, 1)
- end
- return Vector2.new(onAxis(thumbstickPosition.x), onAxis(thumbstickPosition.y))
- end
- -- This function converts 4 different, redundant enumeration types to one standard so the values can be compared
- function CameraUtils.ConvertCameraModeEnumToStandard(enumValue)
- if enumValue == Enum.TouchCameraMovementMode.Default then
- return Enum.ComputerCameraMovementMode.Follow
- end
- if enumValue == Enum.ComputerCameraMovementMode.Default then
- return Enum.ComputerCameraMovementMode.Classic
- end
- if enumValue == Enum.TouchCameraMovementMode.Classic or
- enumValue == Enum.DevTouchCameraMovementMode.Classic or
- enumValue == Enum.DevComputerCameraMovementMode.Classic or
- enumValue == Enum.ComputerCameraMovementMode.Classic then
- return Enum.ComputerCameraMovementMode.Classic
- end
- if enumValue == Enum.TouchCameraMovementMode.Follow or
- enumValue == Enum.DevTouchCameraMovementMode.Follow or
- enumValue == Enum.DevComputerCameraMovementMode.Follow or
- enumValue == Enum.ComputerCameraMovementMode.Follow then
- return Enum.ComputerCameraMovementMode.Follow
- end
- if enumValue == Enum.TouchCameraMovementMode.Orbital or
- enumValue == Enum.DevTouchCameraMovementMode.Orbital or
- enumValue == Enum.DevComputerCameraMovementMode.Orbital or
- enumValue == Enum.ComputerCameraMovementMode.Orbital then
- return Enum.ComputerCameraMovementMode.Orbital
- end
- if FFlagUserCameraToggle then
- if enumValue == Enum.ComputerCameraMovementMode.CameraToggle or
- enumValue == Enum.DevComputerCameraMovementMode.CameraToggle then
- return Enum.ComputerCameraMovementMode.CameraToggle
- end
- end
- -- Note: Only the Dev versions of the Enums have UserChoice as an option
- if enumValue == Enum.DevTouchCameraMovementMode.UserChoice or
- enumValue == Enum.DevComputerCameraMovementMode.UserChoice then
- return Enum.DevComputerCameraMovementMode.UserChoice
- end
- -- For any unmapped options return Classic camera
- return Enum.ComputerCameraMovementMode.Classic
- end
- return CameraUtils
- end
- function _CameraModule()
- local CameraModule = {}
- CameraModule.__index = CameraModule
- local FFlagUserCameraToggle do
- local success, result = pcall(function()
- return UserSettings():IsUserFeatureEnabled("UserCameraToggle")
- end)
- FFlagUserCameraToggle = success and result
- end
- local FFlagUserRemoveTheCameraApi do
- local success, result = pcall(function()
- return UserSettings():IsUserFeatureEnabled("UserRemoveTheCameraApi")
- end)
- FFlagUserRemoveTheCameraApi = success and result
- end
- -- NOTICE: Player property names do not all match their StarterPlayer equivalents,
- -- with the differences noted in the comments on the right
- local PLAYER_CAMERA_PROPERTIES =
- {
- "CameraMinZoomDistance",
- "CameraMaxZoomDistance",
- "CameraMode",
- "DevCameraOcclusionMode",
- "DevComputerCameraMode", -- Corresponds to StarterPlayer.DevComputerCameraMovementMode
- "DevTouchCameraMode", -- Corresponds to StarterPlayer.DevTouchCameraMovementMode
- -- Character movement mode
- "DevComputerMovementMode",
- "DevTouchMovementMode",
- "DevEnableMouseLock", -- Corresponds to StarterPlayer.EnableMouseLockOption
- }
- local USER_GAME_SETTINGS_PROPERTIES =
- {
- "ComputerCameraMovementMode",
- "ComputerMovementMode",
- "ControlMode",
- "GamepadCameraSensitivity",
- "MouseSensitivity",
- "RotationType",
- "TouchCameraMovementMode",
- "TouchMovementMode",
- }
- --[[ Roblox Services ]]--
- local Players = game:GetService("Players")
- local RunService = game:GetService("RunService")
- local UserInputService = game:GetService("UserInputService")
- local UserGameSettings = UserSettings():GetService("UserGameSettings")
- -- Camera math utility library
- local CameraUtils = _CameraUtils()
- -- Load Roblox Camera Controller Modules
- local ClassicCamera = _ClassicCamera()
- local OrbitalCamera = _OrbitalCamera()
- local LegacyCamera = _LegacyCamera()
- -- Load Roblox Occlusion Modules
- local Invisicam = _Invisicam()
- local Poppercam = _Poppercam()
- -- Load the near-field character transparency controller and the mouse lock "shift lock" controller
- local TransparencyController = _TransparencyController()
- local MouseLockController = _MouseLockController()
- -- Table of camera controllers that have been instantiated. They are instantiated as they are used.
- local instantiatedCameraControllers = {}
- local instantiatedOcclusionModules = {}
- -- Management of which options appear on the Roblox User Settings screen
- do
- local PlayerScripts = Players.LocalPlayer:WaitForChild("PlayerScripts")
- PlayerScripts:RegisterTouchCameraMovementMode(Enum.TouchCameraMovementMode.Default)
- PlayerScripts:RegisterTouchCameraMovementMode(Enum.TouchCameraMovementMode.Follow)
- PlayerScripts:RegisterTouchCameraMovementMode(Enum.TouchCameraMovementMode.Classic)
- PlayerScripts:RegisterComputerCameraMovementMode(Enum.ComputerCameraMovementMode.Default)
- PlayerScripts:RegisterComputerCameraMovementMode(Enum.ComputerCameraMovementMode.Follow)
- PlayerScripts:RegisterComputerCameraMovementMode(Enum.ComputerCameraMovementMode.Classic)
- if FFlagUserCameraToggle then
- PlayerScripts:RegisterComputerCameraMovementMode(Enum.ComputerCameraMovementMode.CameraToggle)
- end
- end
- CameraModule.FFlagUserCameraToggle = FFlagUserCameraToggle
- function CameraModule.new()
- local self = setmetatable({},CameraModule)
- -- Current active controller instances
- self.activeCameraController = nil
- self.activeOcclusionModule = nil
- self.activeTransparencyController = nil
- self.activeMouseLockController = nil
- self.currentComputerCameraMovementMode = nil
- -- Connections to events
- self.cameraSubjectChangedConn = nil
- self.cameraTypeChangedConn = nil
- -- Adds CharacterAdded and CharacterRemoving event handlers for all current players
- for _,player in pairs(Players:GetPlayers()) do
- self:OnPlayerAdded(player)
- end
- -- Adds CharacterAdded and CharacterRemoving event handlers for all players who join in the future
- Players.PlayerAdded:Connect(function(player)
- self:OnPlayerAdded(player)
- end)
- self.activeTransparencyController = TransparencyController.new()
- self.activeTransparencyController:Enable(true)
- if not UserInputService.TouchEnabled then
- self.activeMouseLockController = MouseLockController.new()
- local toggleEvent = self.activeMouseLockController:GetBindableToggleEvent()
- if toggleEvent then
- toggleEvent:Connect(function()
- self:OnMouseLockToggled()
- end)
- end
- end
- self:ActivateCameraController(self:GetCameraControlChoice())
- self:ActivateOcclusionModule(Players.LocalPlayer.DevCameraOcclusionMode)
- self:OnCurrentCameraChanged() -- Does initializations and makes first camera controller
- RunService:BindToRenderStep("cameraRenderUpdate", Enum.RenderPriority.Camera.Value, function(dt) self:Update(dt) end)
- -- Connect listeners to camera-related properties
- for _, propertyName in pairs(PLAYER_CAMERA_PROPERTIES) do
- Players.LocalPlayer:GetPropertyChangedSignal(propertyName):Connect(function()
- self:OnLocalPlayerCameraPropertyChanged(propertyName)
- end)
- end
- for _, propertyName in pairs(USER_GAME_SETTINGS_PROPERTIES) do
- UserGameSettings:GetPropertyChangedSignal(propertyName):Connect(function()
- self:OnUserGameSettingsPropertyChanged(propertyName)
- end)
- end
- game.Workspace:GetPropertyChangedSignal("CurrentCamera"):Connect(function()
- self:OnCurrentCameraChanged()
- end)
- self.lastInputType = UserInputService:GetLastInputType()
- UserInputService.LastInputTypeChanged:Connect(function(newLastInputType)
- self.lastInputType = newLastInputType
- end)
- return self
- end
- function CameraModule:GetCameraMovementModeFromSettings()
- local cameraMode = Players.LocalPlayer.CameraMode
- -- Lock First Person trumps all other settings and forces ClassicCamera
- if cameraMode == Enum.CameraMode.LockFirstPerson then
- return CameraUtils.ConvertCameraModeEnumToStandard(Enum.ComputerCameraMovementMode.Classic)
- end
- local devMode, userMode
- if UserInputService.TouchEnabled then
- devMode = CameraUtils.ConvertCameraModeEnumToStandard(Players.LocalPlayer.DevTouchCameraMode)
- userMode = CameraUtils.ConvertCameraModeEnumToStandard(UserGameSettings.TouchCameraMovementMode)
- else
- devMode = CameraUtils.ConvertCameraModeEnumToStandard(Players.LocalPlayer.DevComputerCameraMode)
- userMode = CameraUtils.ConvertCameraModeEnumToStandard(UserGameSettings.ComputerCameraMovementMode)
- end
- if devMode == Enum.DevComputerCameraMovementMode.UserChoice then
- -- Developer is allowing user choice, so user setting is respected
- return userMode
- end
- return devMode
- end
- function CameraModule:ActivateOcclusionModule( occlusionMode )
- local newModuleCreator
- if occlusionMode == Enum.DevCameraOcclusionMode.Zoom then
- newModuleCreator = Poppercam
- elseif occlusionMode == Enum.DevCameraOcclusionMode.Invisicam then
- newModuleCreator = Invisicam
- else
- warn("CameraScript ActivateOcclusionModule called with unsupported mode")
- return
- end
- -- First check to see if there is actually a change. If the module being requested is already
- -- the currently-active solution then just make sure it's enabled and exit early
- if self.activeOcclusionModule and self.activeOcclusionModule:GetOcclusionMode() == occlusionMode then
- if not self.activeOcclusionModule:GetEnabled() then
- self.activeOcclusionModule:Enable(true)
- end
- return
- end
- -- Save a reference to the current active module (may be nil) so that we can disable it if
- -- we are successful in activating its replacement
- local prevOcclusionModule = self.activeOcclusionModule
- -- If there is no active module, see if the one we need has already been instantiated
- self.activeOcclusionModule = instantiatedOcclusionModules[newModuleCreator]
- -- If the module was not already instantiated and selected above, instantiate it
- if not self.activeOcclusionModule then
- self.activeOcclusionModule = newModuleCreator.new()
- if self.activeOcclusionModule then
- instantiatedOcclusionModules[newModuleCreator] = self.activeOcclusionModule
- end
- end
- -- If we were successful in either selecting or instantiating the module,
- -- enable it if it's not already the currently-active enabled module
- if self.activeOcclusionModule then
- local newModuleOcclusionMode = self.activeOcclusionModule:GetOcclusionMode()
- -- Sanity check that the module we selected or instantiated actually supports the desired occlusionMode
- if newModuleOcclusionMode ~= occlusionMode then
- warn("CameraScript ActivateOcclusionModule mismatch: ",self.activeOcclusionModule:GetOcclusionMode(),"~=",occlusionMode)
- end
- -- Deactivate current module if there is one
- if prevOcclusionModule then
- -- Sanity check that current module is not being replaced by itself (that should have been handled above)
- if prevOcclusionModule ~= self.activeOcclusionModule then
- prevOcclusionModule:Enable(false)
- else
- warn("CameraScript ActivateOcclusionModule failure to detect already running correct module")
- end
- end
- -- Occlusion modules need to be initialized with information about characters and cameraSubject
- -- Invisicam needs the LocalPlayer's character
- -- Poppercam needs all player characters and the camera subject
- if occlusionMode == Enum.DevCameraOcclusionMode.Invisicam then
- -- Optimization to only send Invisicam what we know it needs
- if Players.LocalPlayer.Character then
- self.activeOcclusionModule:CharacterAdded(Players.LocalPlayer.Character, Players.LocalPlayer )
- end
- else
- -- When Poppercam is enabled, we send it all existing player characters for its raycast ignore list
- for _, player in pairs(Players:GetPlayers()) do
- if player and player.Character then
- self.activeOcclusionModule:CharacterAdded(player.Character, player)
- end
- end
- self.activeOcclusionModule:OnCameraSubjectChanged(game.Workspace.CurrentCamera.CameraSubject)
- end
- -- Activate new choice
- self.activeOcclusionModule:Enable(true)
- end
- end
- -- When supplied, legacyCameraType is used and cameraMovementMode is ignored (should be nil anyways)
- -- Next, if userCameraCreator is passed in, that is used as the cameraCreator
- function CameraModule:ActivateCameraController(cameraMovementMode, legacyCameraType)
- local newCameraCreator = nil
- if legacyCameraType~=nil then
- --[[
- This function has been passed a CameraType enum value. Some of these map to the use of
- the LegacyCamera module, the value "Custom" will be translated to a movementMode enum
- value based on Dev and User settings, and "Scriptable" will disable the camera controller.
- --]]
- if legacyCameraType == Enum.CameraType.Scriptable then
- if self.activeCameraController then
- self.activeCameraController:Enable(false)
- self.activeCameraController = nil
- return
- end
- elseif legacyCameraType == Enum.CameraType.Custom then
- cameraMovementMode = self:GetCameraMovementModeFromSettings()
- elseif legacyCameraType == Enum.CameraType.Track then
- -- Note: The TrackCamera module was basically an older, less fully-featured
- -- version of ClassicCamera, no longer actively maintained, but it is re-implemented in
- -- case a game was dependent on its lack of ClassicCamera's extra functionality.
- cameraMovementMode = Enum.ComputerCameraMovementMode.Classic
- elseif legacyCameraType == Enum.CameraType.Follow then
- cameraMovementMode = Enum.ComputerCameraMovementMode.Follow
- elseif legacyCameraType == Enum.CameraType.Orbital then
- cameraMovementMode = Enum.ComputerCameraMovementMode.Orbital
- elseif legacyCameraType == Enum.CameraType.Attach or
- legacyCameraType == Enum.CameraType.Watch or
- legacyCameraType == Enum.CameraType.Fixed then
- newCameraCreator = LegacyCamera
- else
- warn("CameraScript encountered an unhandled Camera.CameraType value: ",legacyCameraType)
- end
- end
- if not newCameraCreator then
- if cameraMovementMode == Enum.ComputerCameraMovementMode.Classic or
- cameraMovementMode == Enum.ComputerCameraMovementMode.Follow or
- cameraMovementMode == Enum.ComputerCameraMovementMode.Default or
- (FFlagUserCameraToggle and cameraMovementMode == Enum.ComputerCameraMovementMode.CameraToggle) then
- newCameraCreator = ClassicCamera
- elseif cameraMovementMode == Enum.ComputerCameraMovementMode.Orbital then
- newCameraCreator = OrbitalCamera
- else
- warn("ActivateCameraController did not select a module.")
- return
- end
- end
- -- Create the camera control module we need if it does not already exist in instantiatedCameraControllers
- local newCameraController
- if not instantiatedCameraControllers[newCameraCreator] then
- newCameraController = newCameraCreator.new()
- instantiatedCameraControllers[newCameraCreator] = newCameraController
- else
- newCameraController = instantiatedCameraControllers[newCameraCreator]
- end
- -- If there is a controller active and it's not the one we need, disable it,
- -- if it is the one we need, make sure it's enabled
- if self.activeCameraController then
- if self.activeCameraController ~= newCameraController then
- self.activeCameraController:Enable(false)
- self.activeCameraController = newCameraController
- self.activeCameraController:Enable(true)
- elseif not self.activeCameraController:GetEnabled() then
- self.activeCameraController:Enable(true)
- end
- elseif newCameraController ~= nil then
- self.activeCameraController = newCameraController
- self.activeCameraController:Enable(true)
- end
- if self.activeCameraController then
- if cameraMovementMode~=nil then
- self.activeCameraController:SetCameraMovementMode(cameraMovementMode)
- elseif legacyCameraType~=nil then
- -- Note that this is only called when legacyCameraType is not a type that
- -- was convertible to a ComputerCameraMovementMode value, i.e. really only applies to LegacyCamera
- self.activeCameraController:SetCameraType(legacyCameraType)
- end
- end
- end
- -- Note: The active transparency controller could be made to listen for this event itself.
- function CameraModule:OnCameraSubjectChanged()
- if self.activeTransparencyController then
- self.activeTransparencyController:SetSubject(game.Workspace.CurrentCamera.CameraSubject)
- end
- if self.activeOcclusionModule then
- self.activeOcclusionModule:OnCameraSubjectChanged(game.Workspace.CurrentCamera.CameraSubject)
- end
- end
- function CameraModule:OnCameraTypeChanged(newCameraType)
- if newCameraType == Enum.CameraType.Scriptable then
- if UserInputService.MouseBehavior == Enum.MouseBehavior.LockCenter then
- UserInputService.MouseBehavior = Enum.MouseBehavior.Default
- end
- end
- -- Forward the change to ActivateCameraController to handle
- self:ActivateCameraController(nil, newCameraType)
- end
- -- Note: Called whenever workspace.CurrentCamera changes, but also on initialization of this script
- function CameraModule:OnCurrentCameraChanged()
- local currentCamera = game.Workspace.CurrentCamera
- if not currentCamera then return end
- if self.cameraSubjectChangedConn then
- self.cameraSubjectChangedConn:Disconnect()
- end
- if self.cameraTypeChangedConn then
- self.cameraTypeChangedConn:Disconnect()
- end
- self.cameraSubjectChangedConn = currentCamera:GetPropertyChangedSignal("CameraSubject"):Connect(function()
- self:OnCameraSubjectChanged(currentCamera.CameraSubject)
- end)
- self.cameraTypeChangedConn = currentCamera:GetPropertyChangedSignal("CameraType"):Connect(function()
- self:OnCameraTypeChanged(currentCamera.CameraType)
- end)
- self:OnCameraSubjectChanged(currentCamera.CameraSubject)
- self:OnCameraTypeChanged(currentCamera.CameraType)
- end
- function CameraModule:OnLocalPlayerCameraPropertyChanged(propertyName)
- if propertyName == "CameraMode" then
- -- CameraMode is only used to turn on/off forcing the player into first person view. The
- -- Note: The case "Classic" is used for all other views and does not correspond only to the ClassicCamera module
- if Players.LocalPlayer.CameraMode == Enum.CameraMode.LockFirstPerson then
- -- Locked in first person, use ClassicCamera which supports this
- if not self.activeCameraController or self.activeCameraController:GetModuleName() ~= "ClassicCamera" then
- self:ActivateCameraController(CameraUtils.ConvertCameraModeEnumToStandard(Enum.DevComputerCameraMovementMode.Classic))
- end
- if self.activeCameraController then
- self.activeCameraController:UpdateForDistancePropertyChange()
- end
- elseif Players.LocalPlayer.CameraMode == Enum.CameraMode.Classic then
- -- Not locked in first person view
- local cameraMovementMode =self: GetCameraMovementModeFromSettings()
- self:ActivateCameraController(CameraUtils.ConvertCameraModeEnumToStandard(cameraMovementMode))
- else
- warn("Unhandled value for property player.CameraMode: ",Players.LocalPlayer.CameraMode)
- end
- elseif propertyName == "DevComputerCameraMode" or
- propertyName == "DevTouchCameraMode" then
- local cameraMovementMode = self:GetCameraMovementModeFromSettings()
- self:ActivateCameraController(CameraUtils.ConvertCameraModeEnumToStandard(cameraMovementMode))
- elseif propertyName == "DevCameraOcclusionMode" then
- self:ActivateOcclusionModule(Players.LocalPlayer.DevCameraOcclusionMode)
- elseif propertyName == "CameraMinZoomDistance" or propertyName == "CameraMaxZoomDistance" then
- if self.activeCameraController then
- self.activeCameraController:UpdateForDistancePropertyChange()
- end
- elseif propertyName == "DevTouchMovementMode" then
- elseif propertyName == "DevComputerMovementMode" then
- elseif propertyName == "DevEnableMouseLock" then
- -- This is the enabling/disabling of "Shift Lock" mode, not LockFirstPerson (which is a CameraMode)
- -- Note: Enabling and disabling of MouseLock mode is normally only a publish-time choice made via
- -- the corresponding EnableMouseLockOption checkbox of StarterPlayer, and this script does not have
- -- support for changing the availability of MouseLock at runtime (this would require listening to
- -- Player.DevEnableMouseLock changes)
- end
- end
- function CameraModule:OnUserGameSettingsPropertyChanged(propertyName)
- if propertyName == "ComputerCameraMovementMode" then
- local cameraMovementMode = self:GetCameraMovementModeFromSettings()
- self:ActivateCameraController(CameraUtils.ConvertCameraModeEnumToStandard(cameraMovementMode))
- end
- end
- --[[
- Main RenderStep Update. The camera controller and occlusion module both have opportunities
- to set and modify (respectively) the CFrame and Focus before it is set once on CurrentCamera.
- The camera and occlusion modules should only return CFrames, not set the CFrame property of
- CurrentCamera directly.
- --]]
- function CameraModule:Update(dt)
- if self.activeCameraController then
- if FFlagUserCameraToggle then
- self.activeCameraController:UpdateMouseBehavior()
- end
- local newCameraCFrame, newCameraFocus = self.activeCameraController:Update(dt)
- self.activeCameraController:ApplyVRTransform()
- if self.activeOcclusionModule then
- newCameraCFrame, newCameraFocus = self.activeOcclusionModule:Update(dt, newCameraCFrame, newCameraFocus)
- end
- -- Here is where the new CFrame and Focus are set for this render frame
- game.Workspace.CurrentCamera.CFrame = newCameraCFrame
- game.Workspace.CurrentCamera.Focus = newCameraFocus
- -- Update to character local transparency as needed based on camera-to-subject distance
- if self.activeTransparencyController then
- self.activeTransparencyController:Update()
- end
- end
- end
- -- Formerly getCurrentCameraMode, this function resolves developer and user camera control settings to
- -- decide which camera control module should be instantiated. The old method of converting redundant enum types
- function CameraModule:GetCameraControlChoice()
- local player = Players.LocalPlayer
- if player then
- if self.lastInputType == Enum.UserInputType.Touch or UserInputService.TouchEnabled then
- -- Touch
- if player.DevTouchCameraMode == Enum.DevTouchCameraMovementMode.UserChoice then
- return CameraUtils.ConvertCameraModeEnumToStandard( UserGameSettings.TouchCameraMovementMode )
- else
- return CameraUtils.ConvertCameraModeEnumToStandard( player.DevTouchCameraMode )
- end
- else
- -- Computer
- if player.DevComputerCameraMode == Enum.DevComputerCameraMovementMode.UserChoice then
- local computerMovementMode = CameraUtils.ConvertCameraModeEnumToStandard(UserGameSettings.ComputerCameraMovementMode)
- return CameraUtils.ConvertCameraModeEnumToStandard(computerMovementMode)
- else
- return CameraUtils.ConvertCameraModeEnumToStandard(player.DevComputerCameraMode)
- end
- end
- end
- end
- function CameraModule:OnCharacterAdded(char, player)
- if self.activeOcclusionModule then
- self.activeOcclusionModule:CharacterAdded(char, player)
- end
- end
- function CameraModule:OnCharacterRemoving(char, player)
- if self.activeOcclusionModule then
- self.activeOcclusionModule:CharacterRemoving(char, player)
- end
- end
- function CameraModule:OnPlayerAdded(player)
- player.CharacterAdded:Connect(function(char)
- self:OnCharacterAdded(char, player)
- end)
- player.CharacterRemoving:Connect(function(char)
- self:OnCharacterRemoving(char, player)
- end)
- end
- function CameraModule:OnMouseLockToggled()
- if self.activeMouseLockController then
- local mouseLocked = self.activeMouseLockController:GetIsMouseLocked()
- local mouseLockOffset = self.activeMouseLockController:GetMouseLockOffset()
- if self.activeCameraController then
- self.activeCameraController:SetIsMouseLocked(mouseLocked)
- self.activeCameraController:SetMouseLockOffset(mouseLockOffset)
- end
- end
- end
- --begin edit
- local Camera = CameraModule
- local IDENTITYCF = CFrame.new()
- local lastUpCFrame = IDENTITYCF
- Camera.UpVector = Vector3.new(0, 1, 0)
- Camera.TransitionRate = 0.15
- Camera.UpCFrame = IDENTITYCF
- function Camera:GetUpVector(oldUpVector)
- return oldUpVector
- end
- local function getRotationBetween(u, v, axis)
- local dot, uxv = u:Dot(v), u:Cross(v)
- if (dot < -0.99999) then return CFrame.fromAxisAngle(axis, math.pi) end
- return CFrame.new(0, 0, 0, uxv.x, uxv.y, uxv.z, 1 + dot)
- end
- function Camera:CalculateUpCFrame()
- local oldUpVector = self.UpVector
- local newUpVector = self:GetUpVector(oldUpVector)
- local backup = game.Workspace.CurrentCamera.CFrame.RightVector
- local transitionCF = getRotationBetween(oldUpVector, newUpVector, backup)
- local vecSlerpCF = IDENTITYCF:Lerp(transitionCF, self.TransitionRate)
- self.UpVector = vecSlerpCF * oldUpVector
- self.UpCFrame = vecSlerpCF * self.UpCFrame
- lastUpCFrame = self.UpCFrame
- end
- function Camera:Update(dt)
- if self.activeCameraController then
- if Camera.FFlagUserCameraToggle then
- self.activeCameraController:UpdateMouseBehavior()
- end
- local newCameraCFrame, newCameraFocus = self.activeCameraController:Update(dt)
- self.activeCameraController:ApplyVRTransform()
- self:CalculateUpCFrame()
- self.activeCameraController:UpdateUpCFrame(self.UpCFrame)
- -- undo shift-lock offset
- local lockOffset = Vector3.new(0, 0, 0)
- if (self.activeMouseLockController and self.activeMouseLockController:GetIsMouseLocked()) then
- lockOffset = self.activeMouseLockController:GetMouseLockOffset()
- end
- local offset = newCameraFocus:ToObjectSpace(newCameraCFrame)
- local camRotation = self.UpCFrame * offset
- newCameraFocus = newCameraFocus - newCameraCFrame:VectorToWorldSpace(lockOffset) + camRotation:VectorToWorldSpace(lockOffset)
- newCameraCFrame = newCameraFocus * camRotation
- --local offset = newCameraFocus:Inverse() * newCameraCFrame
- --newCameraCFrame = newCameraFocus * self.UpCFrame * offset
- if (self.activeCameraController.lastCameraTransform) then
- self.activeCameraController.lastCameraTransform = newCameraCFrame
- self.activeCameraController.lastCameraFocus = newCameraFocus
- end
- if self.activeOcclusionModule then
- newCameraCFrame, newCameraFocus = self.activeOcclusionModule:Update(dt, newCameraCFrame, newCameraFocus)
- end
- game.Workspace.CurrentCamera.CFrame = newCameraCFrame
- game.Workspace.CurrentCamera.Focus = newCameraFocus
- if self.activeTransparencyController then
- self.activeTransparencyController:Update()
- end
- end
- end
- function Camera:IsFirstPerson()
- if self.activeCameraController then
- return self.activeCameraController:InFirstPerson()
- end
- return false
- end
- function Camera:IsMouseLocked()
- if self.activeCameraController then
- return self.activeCameraController:GetIsMouseLocked()
- end
- return false
- end
- function Camera:IsToggleMode()
- if self.activeCameraController then
- return self.activeCameraController.isCameraToggle
- end
- return false
- end
- function Camera:IsCamRelative()
- return self:IsMouseLocked() or self:IsFirstPerson()
- --return self:IsToggleMode(), self:IsMouseLocked(), self:IsFirstPerson()
- end
- --
- local Utils = _CameraUtils()
- function Utils.GetAngleBetweenXZVectors(v1, v2)
- local upCFrame = lastUpCFrame
- v1 = upCFrame:VectorToObjectSpace(v1)
- v2 = upCFrame:VectorToObjectSpace(v2)
- return math.atan2(v2.X*v1.Z-v2.Z*v1.X, v2.X*v1.X+v2.Z*v1.Z)
- end
- --end edit
- local cameraModuleObject = CameraModule.new()
- local cameraApi = {}
- return cameraModuleObject
- end
- function _ClickToMoveDisplay()
- local ClickToMoveDisplay = {}
- local FAILURE_ANIMATION_ID = "rbxassetid://2874840706"
- local TrailDotIcon = "rbxasset://textures/ui/traildot.png"
- local EndWaypointIcon = "rbxasset://textures/ui/waypoint.png"
- local WaypointsAlwaysOnTop = false
- local WAYPOINT_INCLUDE_FACTOR = 2
- local LAST_DOT_DISTANCE = 3
- local WAYPOINT_BILLBOARD_SIZE = UDim2.new(0, 1.68 * 25, 0, 2 * 25)
- local ENDWAYPOINT_SIZE_OFFSET_MIN = Vector2.new(0, 0.5)
- local ENDWAYPOINT_SIZE_OFFSET_MAX = Vector2.new(0, 1)
- local FAIL_WAYPOINT_SIZE_OFFSET_CENTER = Vector2.new(0, 0.5)
- local FAIL_WAYPOINT_SIZE_OFFSET_LEFT = Vector2.new(0.1, 0.5)
- local FAIL_WAYPOINT_SIZE_OFFSET_RIGHT = Vector2.new(-0.1, 0.5)
- local FAILURE_TWEEN_LENGTH = 0.125
- local FAILURE_TWEEN_COUNT = 4
- local TWEEN_WAYPOINT_THRESHOLD = 5
- local TRAIL_DOT_PARENT_NAME = "ClickToMoveDisplay"
- local TrailDotSize = Vector2.new(1.5, 1.5)
- local TRAIL_DOT_MIN_SCALE = 1
- local TRAIL_DOT_MIN_DISTANCE = 10
- local TRAIL_DOT_MAX_SCALE = 2.5
- local TRAIL_DOT_MAX_DISTANCE = 100
- local PlayersService = game:GetService("Players")
- local TweenService = game:GetService("TweenService")
- local RunService = game:GetService("RunService")
- local Workspace = game:GetService("Workspace")
- local LocalPlayer = PlayersService.LocalPlayer
- local function CreateWaypointTemplates()
- local TrailDotTemplate = Instance.new("Part")
- TrailDotTemplate.Size = Vector3.new(1, 1, 1)
- TrailDotTemplate.Anchored = true
- TrailDotTemplate.CanCollide = false
- TrailDotTemplate.Name = "TrailDot"
- TrailDotTemplate.Transparency = 1
- local TrailDotImage = Instance.new("ImageHandleAdornment")
- TrailDotImage.Name = "TrailDotImage"
- TrailDotImage.Size = TrailDotSize
- TrailDotImage.SizeRelativeOffset = Vector3.new(0, 0, -0.1)
- TrailDotImage.AlwaysOnTop = WaypointsAlwaysOnTop
- TrailDotImage.Image = TrailDotIcon
- TrailDotImage.Adornee = TrailDotTemplate
- TrailDotImage.Parent = TrailDotTemplate
- local EndWaypointTemplate = Instance.new("Part")
- EndWaypointTemplate.Size = Vector3.new(2, 2, 2)
- EndWaypointTemplate.Anchored = true
- EndWaypointTemplate.CanCollide = false
- EndWaypointTemplate.Name = "EndWaypoint"
- EndWaypointTemplate.Transparency = 1
- local EndWaypointImage = Instance.new("ImageHandleAdornment")
- EndWaypointImage.Name = "TrailDotImage"
- EndWaypointImage.Size = TrailDotSize
- EndWaypointImage.SizeRelativeOffset = Vector3.new(0, 0, -0.1)
- EndWaypointImage.AlwaysOnTop = WaypointsAlwaysOnTop
- EndWaypointImage.Image = TrailDotIcon
- EndWaypointImage.Adornee = EndWaypointTemplate
- EndWaypointImage.Parent = EndWaypointTemplate
- local EndWaypointBillboard = Instance.new("BillboardGui")
- EndWaypointBillboard.Name = "EndWaypointBillboard"
- EndWaypointBillboard.Size = WAYPOINT_BILLBOARD_SIZE
- EndWaypointBillboard.LightInfluence = 0
- EndWaypointBillboard.SizeOffset = ENDWAYPOINT_SIZE_OFFSET_MIN
- EndWaypointBillboard.AlwaysOnTop = true
- EndWaypointBillboard.Adornee = EndWaypointTemplate
- EndWaypointBillboard.Parent = EndWaypointTemplate
- local EndWaypointImageLabel = Instance.new("ImageLabel")
- EndWaypointImageLabel.Image = EndWaypointIcon
- EndWaypointImageLabel.BackgroundTransparency = 1
- EndWaypointImageLabel.Size = UDim2.new(1, 0, 1, 0)
- EndWaypointImageLabel.Parent = EndWaypointBillboard
- local FailureWaypointTemplate = Instance.new("Part")
- FailureWaypointTemplate.Size = Vector3.new(2, 2, 2)
- FailureWaypointTemplate.Anchored = true
- FailureWaypointTemplate.CanCollide = false
- FailureWaypointTemplate.Name = "FailureWaypoint"
- FailureWaypointTemplate.Transparency = 1
- local FailureWaypointImage = Instance.new("ImageHandleAdornment")
- FailureWaypointImage.Name = "TrailDotImage"
- FailureWaypointImage.Size = TrailDotSize
- FailureWaypointImage.SizeRelativeOffset = Vector3.new(0, 0, -0.1)
- FailureWaypointImage.AlwaysOnTop = WaypointsAlwaysOnTop
- FailureWaypointImage.Image = TrailDotIcon
- FailureWaypointImage.Adornee = FailureWaypointTemplate
- FailureWaypointImage.Parent = FailureWaypointTemplate
- local FailureWaypointBillboard = Instance.new("BillboardGui")
- FailureWaypointBillboard.Name = "FailureWaypointBillboard"
- FailureWaypointBillboard.Size = WAYPOINT_BILLBOARD_SIZE
- FailureWaypointBillboard.LightInfluence = 0
- FailureWaypointBillboard.SizeOffset = FAIL_WAYPOINT_SIZE_OFFSET_CENTER
- FailureWaypointBillboard.AlwaysOnTop = true
- FailureWaypointBillboard.Adornee = FailureWaypointTemplate
- FailureWaypointBillboard.Parent = FailureWaypointTemplate
- local FailureWaypointFrame = Instance.new("Frame")
- FailureWaypointFrame.BackgroundTransparency = 1
- FailureWaypointFrame.Size = UDim2.new(0, 0, 0, 0)
- FailureWaypointFrame.Position = UDim2.new(0.5, 0, 1, 0)
- FailureWaypointFrame.Parent = FailureWaypointBillboard
- local FailureWaypointImageLabel = Instance.new("ImageLabel")
- FailureWaypointImageLabel.Image = EndWaypointIcon
- FailureWaypointImageLabel.BackgroundTransparency = 1
- FailureWaypointImageLabel.Position = UDim2.new(
- 0, -WAYPOINT_BILLBOARD_SIZE.X.Offset/2, 0, -WAYPOINT_BILLBOARD_SIZE.Y.Offset
- )
- FailureWaypointImageLabel.Size = WAYPOINT_BILLBOARD_SIZE
- FailureWaypointImageLabel.Parent = FailureWaypointFrame
- return TrailDotTemplate, EndWaypointTemplate, FailureWaypointTemplate
- end
- local TrailDotTemplate, EndWaypointTemplate, FailureWaypointTemplate = CreateWaypointTemplates()
- local function getTrailDotParent()
- local camera = Workspace.CurrentCamera
- local trailParent = camera:FindFirstChild(TRAIL_DOT_PARENT_NAME)
- if not trailParent then
- trailParent = Instance.new("Model")
- trailParent.Name = TRAIL_DOT_PARENT_NAME
- trailParent.Parent = camera
- end
- return trailParent
- end
- local function placePathWaypoint(waypointModel, position)
- local ray = Ray.new(position + Vector3.new(0, 2.5, 0), Vector3.new(0, -10, 0))
- local hitPart, hitPoint, hitNormal = Workspace:FindPartOnRayWithIgnoreList(
- ray,
- { Workspace.CurrentCamera, LocalPlayer.Character }
- )
- if hitPart then
- waypointModel.CFrame = CFrame.new(hitPoint, hitPoint + hitNormal)
- waypointModel.Parent = getTrailDotParent()
- end
- end
- local TrailDot = {}
- TrailDot.__index = TrailDot
- function TrailDot:Destroy()
- self.DisplayModel:Destroy()
- end
- function TrailDot:NewDisplayModel(position)
- local newDisplayModel = TrailDotTemplate:Clone()
- placePathWaypoint(newDisplayModel, position)
- return newDisplayModel
- end
- function TrailDot.new(position, closestWaypoint)
- local self = setmetatable({}, TrailDot)
- self.DisplayModel = self:NewDisplayModel(position)
- self.ClosestWayPoint = closestWaypoint
- return self
- end
- local EndWaypoint = {}
- EndWaypoint.__index = EndWaypoint
- function EndWaypoint:Destroy()
- self.Destroyed = true
- self.Tween:Cancel()
- self.DisplayModel:Destroy()
- end
- function EndWaypoint:NewDisplayModel(position)
- local newDisplayModel = EndWaypointTemplate:Clone()
- placePathWaypoint(newDisplayModel, position)
- return newDisplayModel
- end
- function EndWaypoint:CreateTween()
- local tweenInfo = TweenInfo.new(0.5, Enum.EasingStyle.Sine, Enum.EasingDirection.Out, -1, true)
- local tween = TweenService:Create(
- self.DisplayModel.EndWaypointBillboard,
- tweenInfo,
- { SizeOffset = ENDWAYPOINT_SIZE_OFFSET_MAX }
- )
- tween:Play()
- return tween
- end
- function EndWaypoint:TweenInFrom(originalPosition)
- local currentPositon = self.DisplayModel.Position
- local studsOffset = originalPosition - currentPositon
- self.DisplayModel.EndWaypointBillboard.StudsOffset = Vector3.new(0, studsOffset.Y, 0)
- local tweenInfo = TweenInfo.new(1, Enum.EasingStyle.Sine, Enum.EasingDirection.Out)
- local tween = TweenService:Create(
- self.DisplayModel.EndWaypointBillboard,
- tweenInfo,
- { StudsOffset = Vector3.new(0, 0, 0) }
- )
- tween:Play()
- return tween
- end
- function EndWaypoint.new(position, closestWaypoint, originalPosition)
- local self = setmetatable({}, EndWaypoint)
- self.DisplayModel = self:NewDisplayModel(position)
- self.Destroyed = false
- if originalPosition and (originalPosition - position).magnitude > TWEEN_WAYPOINT_THRESHOLD then
- self.Tween = self:TweenInFrom(originalPosition)
- coroutine.wrap(function()
- self.Tween.Completed:Wait()
- if not self.Destroyed then
- self.Tween = self:CreateTween()
- end
- end)()
- else
- self.Tween = self:CreateTween()
- end
- self.ClosestWayPoint = closestWaypoint
- return self
- end
- local FailureWaypoint = {}
- FailureWaypoint.__index = FailureWaypoint
- function FailureWaypoint:Hide()
- self.DisplayModel.Parent = nil
- end
- function FailureWaypoint:Destroy()
- self.DisplayModel:Destroy()
- end
- function FailureWaypoint:NewDisplayModel(position)
- local newDisplayModel = FailureWaypointTemplate:Clone()
- placePathWaypoint(newDisplayModel, position)
- local ray = Ray.new(position + Vector3.new(0, 2.5, 0), Vector3.new(0, -10, 0))
- local hitPart, hitPoint, hitNormal = Workspace:FindPartOnRayWithIgnoreList(
- ray, { Workspace.CurrentCamera, LocalPlayer.Character }
- )
- if hitPart then
- newDisplayModel.CFrame = CFrame.new(hitPoint, hitPoint + hitNormal)
- newDisplayModel.Parent = getTrailDotParent()
- end
- return newDisplayModel
- end
- function FailureWaypoint:RunFailureTween()
- wait(FAILURE_TWEEN_LENGTH) -- Delay one tween length betfore starting tweening
- -- Tween out from center
- local tweenInfo = TweenInfo.new(FAILURE_TWEEN_LENGTH/2, Enum.EasingStyle.Sine, Enum.EasingDirection.Out)
- local tweenLeft = TweenService:Create(self.DisplayModel.FailureWaypointBillboard, tweenInfo,
- { SizeOffset = FAIL_WAYPOINT_SIZE_OFFSET_LEFT })
- tweenLeft:Play()
- local tweenLeftRoation = TweenService:Create(self.DisplayModel.FailureWaypointBillboard.Frame, tweenInfo,
- { Rotation = 10 })
- tweenLeftRoation:Play()
- tweenLeft.Completed:wait()
- -- Tween back and forth
- tweenInfo = TweenInfo.new(FAILURE_TWEEN_LENGTH, Enum.EasingStyle.Sine, Enum.EasingDirection.Out,
- FAILURE_TWEEN_COUNT - 1, true)
- local tweenSideToSide = TweenService:Create(self.DisplayModel.FailureWaypointBillboard, tweenInfo,
- { SizeOffset = FAIL_WAYPOINT_SIZE_OFFSET_RIGHT})
- tweenSideToSide:Play()
- -- Tween flash dark and roate left and right
- tweenInfo = TweenInfo.new(FAILURE_TWEEN_LENGTH, Enum.EasingStyle.Sine, Enum.EasingDirection.Out,
- FAILURE_TWEEN_COUNT - 1, true)
- local tweenFlash = TweenService:Create(self.DisplayModel.FailureWaypointBillboard.Frame.ImageLabel, tweenInfo,
- { ImageColor3 = Color3.new(0.75, 0.75, 0.75)})
- tweenFlash:Play()
- local tweenRotate = TweenService:Create(self.DisplayModel.FailureWaypointBillboard.Frame, tweenInfo,
- { Rotation = -10 })
- tweenRotate:Play()
- tweenSideToSide.Completed:wait()
- -- Tween back to center
- tweenInfo = TweenInfo.new(FAILURE_TWEEN_LENGTH/2, Enum.EasingStyle.Sine, Enum.EasingDirection.Out)
- local tweenCenter = TweenService:Create(self.DisplayModel.FailureWaypointBillboard, tweenInfo,
- { SizeOffset = FAIL_WAYPOINT_SIZE_OFFSET_CENTER })
- tweenCenter:Play()
- local tweenRoation = TweenService:Create(self.DisplayModel.FailureWaypointBillboard.Frame, tweenInfo,
- { Rotation = 0 })
- tweenRoation:Play()
- tweenCenter.Completed:wait()
- wait(FAILURE_TWEEN_LENGTH) -- Delay one tween length betfore removing
- end
- function FailureWaypoint.new(position)
- local self = setmetatable({}, FailureWaypoint)
- self.DisplayModel = self:NewDisplayModel(position)
- return self
- end
- local failureAnimation = Instance.new("Animation")
- failureAnimation.AnimationId = FAILURE_ANIMATION_ID
- local lastHumanoid = nil
- local lastFailureAnimationTrack = nil
- local function getFailureAnimationTrack(myHumanoid)
- if myHumanoid == lastHumanoid then
- return lastFailureAnimationTrack
- end
- lastFailureAnimationTrack = myHumanoid:LoadAnimation(failureAnimation)
- lastFailureAnimationTrack.Priority = Enum.AnimationPriority.Action
- lastFailureAnimationTrack.Looped = false
- return lastFailureAnimationTrack
- end
- local function findPlayerHumanoid()
- local character = LocalPlayer.Character
- if character then
- return character:FindFirstChildOfClass("Humanoid")
- end
- end
- local function createTrailDots(wayPoints, originalEndWaypoint)
- local newTrailDots = {}
- local count = 1
- for i = 1, #wayPoints - 1 do
- local closeToEnd = (wayPoints[i].Position - wayPoints[#wayPoints].Position).magnitude < LAST_DOT_DISTANCE
- local includeWaypoint = i % WAYPOINT_INCLUDE_FACTOR == 0 and not closeToEnd
- if includeWaypoint then
- local trailDot = TrailDot.new(wayPoints[i].Position, i)
- newTrailDots[count] = trailDot
- count = count + 1
- end
- end
- local newEndWaypoint = EndWaypoint.new(wayPoints[#wayPoints].Position, #wayPoints, originalEndWaypoint)
- table.insert(newTrailDots, newEndWaypoint)
- local reversedTrailDots = {}
- count = 1
- for i = #newTrailDots, 1, -1 do
- reversedTrailDots[count] = newTrailDots[i]
- count = count + 1
- end
- return reversedTrailDots
- end
- local function getTrailDotScale(distanceToCamera, defaultSize)
- local rangeLength = TRAIL_DOT_MAX_DISTANCE - TRAIL_DOT_MIN_DISTANCE
- local inRangePoint = math.clamp(distanceToCamera - TRAIL_DOT_MIN_DISTANCE, 0, rangeLength)/rangeLength
- local scale = TRAIL_DOT_MIN_SCALE + (TRAIL_DOT_MAX_SCALE - TRAIL_DOT_MIN_SCALE)*inRangePoint
- return defaultSize * scale
- end
- local createPathCount = 0
- -- originalEndWaypoint is optional, causes the waypoint to tween from that position.
- function ClickToMoveDisplay.CreatePathDisplay(wayPoints, originalEndWaypoint)
- createPathCount = createPathCount + 1
- local trailDots = createTrailDots(wayPoints, originalEndWaypoint)
- local function removePathBeforePoint(wayPointNumber)
- -- kill all trailDots before and at wayPointNumber
- for i = #trailDots, 1, -1 do
- local trailDot = trailDots[i]
- if trailDot.ClosestWayPoint <= wayPointNumber then
- trailDot:Destroy()
- trailDots[i] = nil
- else
- break
- end
- end
- end
- local reiszeTrailDotsUpdateName = "ClickToMoveResizeTrail" ..createPathCount
- local function resizeTrailDots()
- if #trailDots == 0 then
- RunService:UnbindFromRenderStep(reiszeTrailDotsUpdateName)
- return
- end
- local cameraPos = Workspace.CurrentCamera.CFrame.p
- for i = 1, #trailDots do
- local trailDotImage = trailDots[i].DisplayModel:FindFirstChild("TrailDotImage")
- if trailDotImage then
- local distanceToCamera = (trailDots[i].DisplayModel.Position - cameraPos).magnitude
- trailDotImage.Size = getTrailDotScale(distanceToCamera, TrailDotSize)
- end
- end
- end
- RunService:BindToRenderStep(reiszeTrailDotsUpdateName, Enum.RenderPriority.Camera.Value - 1, resizeTrailDots)
- local function removePath()
- removePathBeforePoint(#wayPoints)
- end
- return removePath, removePathBeforePoint
- end
- local lastFailureWaypoint = nil
- function ClickToMoveDisplay.DisplayFailureWaypoint(position)
- if lastFailureWaypoint then
- lastFailureWaypoint:Hide()
- end
- local failureWaypoint = FailureWaypoint.new(position)
- lastFailureWaypoint = failureWaypoint
- coroutine.wrap(function()
- failureWaypoint:RunFailureTween()
- failureWaypoint:Destroy()
- failureWaypoint = nil
- end)()
- end
- function ClickToMoveDisplay.CreateEndWaypoint(position)
- return EndWaypoint.new(position)
- end
- function ClickToMoveDisplay.PlayFailureAnimation()
- local myHumanoid = findPlayerHumanoid()
- if myHumanoid then
- local animationTrack = getFailureAnimationTrack(myHumanoid)
- animationTrack:Play()
- end
- end
- function ClickToMoveDisplay.CancelFailureAnimation()
- if lastFailureAnimationTrack ~= nil and lastFailureAnimationTrack.IsPlaying then
- lastFailureAnimationTrack:Stop()
- end
- end
- function ClickToMoveDisplay.SetWaypointTexture(texture)
- TrailDotIcon = texture
- TrailDotTemplate, EndWaypointTemplate, FailureWaypointTemplate = CreateWaypointTemplates()
- end
- function ClickToMoveDisplay.GetWaypointTexture()
- return TrailDotIcon
- end
- function ClickToMoveDisplay.SetWaypointRadius(radius)
- TrailDotSize = Vector2.new(radius, radius)
- TrailDotTemplate, EndWaypointTemplate, FailureWaypointTemplate = CreateWaypointTemplates()
- end
- function ClickToMoveDisplay.GetWaypointRadius()
- return TrailDotSize.X
- end
- function ClickToMoveDisplay.SetEndWaypointTexture(texture)
- EndWaypointIcon = texture
- TrailDotTemplate, EndWaypointTemplate, FailureWaypointTemplate = CreateWaypointTemplates()
- end
- function ClickToMoveDisplay.GetEndWaypointTexture()
- return EndWaypointIcon
- end
- function ClickToMoveDisplay.SetWaypointsAlwaysOnTop(alwaysOnTop)
- WaypointsAlwaysOnTop = alwaysOnTop
- TrailDotTemplate, EndWaypointTemplate, FailureWaypointTemplate = CreateWaypointTemplates()
- end
- function ClickToMoveDisplay.GetWaypointsAlwaysOnTop()
- return WaypointsAlwaysOnTop
- end
- return ClickToMoveDisplay
- end
- function _BaseCharacterController()
- local ZERO_VECTOR3 = Vector3.new(0,0,0)
- --[[ The Module ]]--
- local BaseCharacterController = {}
- BaseCharacterController.__index = BaseCharacterController
- function BaseCharacterController.new()
- local self = setmetatable({}, BaseCharacterController)
- self.enabled = false
- self.moveVector = ZERO_VECTOR3
- self.moveVectorIsCameraRelative = true
- self.isJumping = false
- return self
- end
- function BaseCharacterController:OnRenderStepped(dt)
- -- By default, nothing to do
- end
- function BaseCharacterController:GetMoveVector()
- return self.moveVector
- end
- function BaseCharacterController:IsMoveVectorCameraRelative()
- return self.moveVectorIsCameraRelative
- end
- function BaseCharacterController:GetIsJumping()
- return self.isJumping
- end
- -- Override in derived classes to set self.enabled and return boolean indicating
- -- whether Enable/Disable was successful. Return true if controller is already in the requested state.
- function BaseCharacterController:Enable(enable)
- error("BaseCharacterController:Enable must be overridden in derived classes and should not be called.")
- return false
- end
- return BaseCharacterController
- end
- function _VehicleController()
- local ContextActionService = game:GetService("ContextActionService")
- --[[ Constants ]]--
- -- Set this to true if you want to instead use the triggers for the throttle
- local useTriggersForThrottle = true
- -- Also set this to true if you want the thumbstick to not affect throttle, only triggers when a gamepad is conected
- local onlyTriggersForThrottle = false
- local ZERO_VECTOR3 = Vector3.new(0,0,0)
- local AUTO_PILOT_DEFAULT_MAX_STEERING_ANGLE = 35
- -- Note that VehicleController does not derive from BaseCharacterController, it is a special case
- local VehicleController = {}
- VehicleController.__index = VehicleController
- function VehicleController.new(CONTROL_ACTION_PRIORITY)
- local self = setmetatable({}, VehicleController)
- self.CONTROL_ACTION_PRIORITY = CONTROL_ACTION_PRIORITY
- self.enabled = false
- self.vehicleSeat = nil
- self.throttle = 0
- self.steer = 0
- self.acceleration = 0
- self.decceleration = 0
- self.turningRight = 0
- self.turningLeft = 0
- self.vehicleMoveVector = ZERO_VECTOR3
- self.autoPilot = {}
- self.autoPilot.MaxSpeed = 0
- self.autoPilot.MaxSteeringAngle = 0
- return self
- end
- function VehicleController:BindContextActions()
- if useTriggersForThrottle then
- ContextActionService:BindActionAtPriority("throttleAccel", (function(actionName, inputState, inputObject)
- self:OnThrottleAccel(actionName, inputState, inputObject)
- return Enum.ContextActionResult.Pass
- end), false, self.CONTROL_ACTION_PRIORITY, Enum.KeyCode.ButtonR2)
- ContextActionService:BindActionAtPriority("throttleDeccel", (function(actionName, inputState, inputObject)
- self:OnThrottleDeccel(actionName, inputState, inputObject)
- return Enum.ContextActionResult.Pass
- end), false, self.CONTROL_ACTION_PRIORITY, Enum.KeyCode.ButtonL2)
- end
- ContextActionService:BindActionAtPriority("arrowSteerRight", (function(actionName, inputState, inputObject)
- self:OnSteerRight(actionName, inputState, inputObject)
- return Enum.ContextActionResult.Pass
- end), false, self.CONTROL_ACTION_PRIORITY, Enum.KeyCode.Right)
- ContextActionService:BindActionAtPriority("arrowSteerLeft", (function(actionName, inputState, inputObject)
- self:OnSteerLeft(actionName, inputState, inputObject)
- return Enum.ContextActionResult.Pass
- end), false, self.CONTROL_ACTION_PRIORITY, Enum.KeyCode.Left)
- end
- function VehicleController:Enable(enable, vehicleSeat)
- if enable == self.enabled and vehicleSeat == self.vehicleSeat then
- return
- end
- self.enabled = enable
- self.vehicleMoveVector = ZERO_VECTOR3
- if enable then
- if vehicleSeat then
- self.vehicleSeat = vehicleSeat
- self:SetupAutoPilot()
- self:BindContextActions()
- end
- else
- if useTriggersForThrottle then
- ContextActionService:UnbindAction("throttleAccel")
- ContextActionService:UnbindAction("throttleDeccel")
- end
- ContextActionService:UnbindAction("arrowSteerRight")
- ContextActionService:UnbindAction("arrowSteerLeft")
- self.vehicleSeat = nil
- end
- end
- function VehicleController:OnThrottleAccel(actionName, inputState, inputObject)
- if inputState == Enum.UserInputState.End or inputState == Enum.UserInputState.Cancel then
- self.acceleration = 0
- else
- self.acceleration = -1
- end
- self.throttle = self.acceleration + self.decceleration
- end
- function VehicleController:OnThrottleDeccel(actionName, inputState, inputObject)
- if inputState == Enum.UserInputState.End or inputState == Enum.UserInputState.Cancel then
- self.decceleration = 0
- else
- self.decceleration = 1
- end
- self.throttle = self.acceleration + self.decceleration
- end
- function VehicleController:OnSteerRight(actionName, inputState, inputObject)
- if inputState == Enum.UserInputState.End or inputState == Enum.UserInputState.Cancel then
- self.turningRight = 0
- else
- self.turningRight = 1
- end
- self.steer = self.turningRight + self.turningLeft
- end
- function VehicleController:OnSteerLeft(actionName, inputState, inputObject)
- if inputState == Enum.UserInputState.End or inputState == Enum.UserInputState.Cancel then
- self.turningLeft = 0
- else
- self.turningLeft = -1
- end
- self.steer = self.turningRight + self.turningLeft
- end
- -- Call this from a function bound to Renderstep with Input Priority
- function VehicleController:Update(moveVector, cameraRelative, usingGamepad)
- if self.vehicleSeat then
- if cameraRelative then
- -- This is the default steering mode
- moveVector = moveVector + Vector3.new(self.steer, 0, self.throttle)
- if usingGamepad and onlyTriggersForThrottle and useTriggersForThrottle then
- self.vehicleSeat.ThrottleFloat = -self.throttle
- else
- self.vehicleSeat.ThrottleFloat = -moveVector.Z
- end
- self.vehicleSeat.SteerFloat = moveVector.X
- return moveVector, true
- else
- -- This is the path following mode
- local localMoveVector = self.vehicleSeat.Occupant.RootPart.CFrame:VectorToObjectSpace(moveVector)
- self.vehicleSeat.ThrottleFloat = self:ComputeThrottle(localMoveVector)
- self.vehicleSeat.SteerFloat = self:ComputeSteer(localMoveVector)
- return ZERO_VECTOR3, true
- end
- end
- return moveVector, false
- end
- function VehicleController:ComputeThrottle(localMoveVector)
- if localMoveVector ~= ZERO_VECTOR3 then
- local throttle = -localMoveVector.Z
- return throttle
- else
- return 0.0
- end
- end
- function VehicleController:ComputeSteer(localMoveVector)
- if localMoveVector ~= ZERO_VECTOR3 then
- local steerAngle = -math.atan2(-localMoveVector.x, -localMoveVector.z) * (180 / math.pi)
- return steerAngle / self.autoPilot.MaxSteeringAngle
- else
- return 0.0
- end
- end
- function VehicleController:SetupAutoPilot()
- -- Setup default
- self.autoPilot.MaxSpeed = self.vehicleSeat.MaxSpeed
- self.autoPilot.MaxSteeringAngle = AUTO_PILOT_DEFAULT_MAX_STEERING_ANGLE
- -- VehicleSeat should have a MaxSteeringAngle as well.
- -- Or we could look for a child "AutoPilotConfigModule" to find these values
- -- Or allow developer to set them through the API as like the CLickToMove customization API
- end
- return VehicleController
- end
- function _TouchJump()
- local Players = game:GetService("Players")
- local GuiService = game:GetService("GuiService")
- --[[ Constants ]]--
- local TOUCH_CONTROL_SHEET = "rbxasset://textures/ui/Input/TouchControlsSheetV2.png"
- --[[ The Module ]]--
- local BaseCharacterController = _BaseCharacterController()
- local TouchJump = setmetatable({}, BaseCharacterController)
- TouchJump.__index = TouchJump
- function TouchJump.new()
- local self = setmetatable(BaseCharacterController.new(), TouchJump)
- self.parentUIFrame = nil
- self.jumpButton = nil
- self.characterAddedConn = nil
- self.humanoidStateEnabledChangedConn = nil
- self.humanoidJumpPowerConn = nil
- self.humanoidParentConn = nil
- self.externallyEnabled = false
- self.jumpPower = 0
- self.jumpStateEnabled = true
- self.isJumping = false
- self.humanoid = nil -- saved reference because property change connections are made using it
- return self
- end
- function TouchJump:EnableButton(enable)
- if enable then
- if not self.jumpButton then
- self:Create()
- end
- local humanoid = Players.LocalPlayer.Character and Players.LocalPlayer.Character:FindFirstChildOfClass("Humanoid")
- if humanoid and self.externallyEnabled then
- if self.externallyEnabled then
- if humanoid.JumpPower > 0 then
- self.jumpButton.Visible = true
- end
- end
- end
- else
- self.jumpButton.Visible = false
- self.isJumping = false
- self.jumpButton.ImageRectOffset = Vector2.new(1, 146)
- end
- end
- function TouchJump:UpdateEnabled()
- if self.jumpPower > 0 and self.jumpStateEnabled then
- self:EnableButton(true)
- else
- self:EnableButton(false)
- end
- end
- function TouchJump:HumanoidChanged(prop)
- local humanoid = Players.LocalPlayer.Character and Players.LocalPlayer.Character:FindFirstChildOfClass("Humanoid")
- if humanoid then
- if prop == "JumpPower" then
- self.jumpPower = humanoid.JumpPower
- self:UpdateEnabled()
- elseif prop == "Parent" then
- if not humanoid.Parent then
- self.humanoidChangeConn:Disconnect()
- end
- end
- end
- end
- function TouchJump:HumanoidStateEnabledChanged(state, isEnabled)
- if state == Enum.HumanoidStateType.Jumping then
- self.jumpStateEnabled = isEnabled
- self:UpdateEnabled()
- end
- end
- function TouchJump:CharacterAdded(char)
- if self.humanoidChangeConn then
- self.humanoidChangeConn:Disconnect()
- self.humanoidChangeConn = nil
- end
- self.humanoid = char:FindFirstChildOfClass("Humanoid")
- while not self.humanoid do
- char.ChildAdded:wait()
- self.humanoid = char:FindFirstChildOfClass("Humanoid")
- end
- self.humanoidJumpPowerConn = self.humanoid:GetPropertyChangedSignal("JumpPower"):Connect(function()
- self.jumpPower = self.humanoid.JumpPower
- self:UpdateEnabled()
- end)
- self.humanoidParentConn = self.humanoid:GetPropertyChangedSignal("Parent"):Connect(function()
- if not self.humanoid.Parent then
- self.humanoidJumpPowerConn:Disconnect()
- self.humanoidJumpPowerConn = nil
- self.humanoidParentConn:Disconnect()
- self.humanoidParentConn = nil
- end
- end)
- self.humanoidStateEnabledChangedConn = self.humanoid.StateEnabledChanged:Connect(function(state, enabled)
- self:HumanoidStateEnabledChanged(state, enabled)
- end)
- self.jumpPower = self.humanoid.JumpPower
- self.jumpStateEnabled = self.humanoid:GetStateEnabled(Enum.HumanoidStateType.Jumping)
- self:UpdateEnabled()
- end
- function TouchJump:SetupCharacterAddedFunction()
- self.characterAddedConn = Players.LocalPlayer.CharacterAdded:Connect(function(char)
- self:CharacterAdded(char)
- end)
- if Players.LocalPlayer.Character then
- self:CharacterAdded(Players.LocalPlayer.Character)
- end
- end
- function TouchJump:Enable(enable, parentFrame)
- if parentFrame then
- self.parentUIFrame = parentFrame
- end
- self.externallyEnabled = enable
- self:EnableButton(enable)
- end
- function TouchJump:Create()
- if not self.parentUIFrame then
- return
- end
- if self.jumpButton then
- self.jumpButton:Destroy()
- self.jumpButton = nil
- end
- local minAxis = math.min(self.parentUIFrame.AbsoluteSize.x, self.parentUIFrame.AbsoluteSize.y)
- local isSmallScreen = minAxis <= 500
- local jumpButtonSize = isSmallScreen and 70 or 120
- self.jumpButton = Instance.new("ImageButton")
- self.jumpButton.Name = "JumpButton"
- self.jumpButton.Visible = false
- self.jumpButton.BackgroundTransparency = 1
- self.jumpButton.Image = TOUCH_CONTROL_SHEET
- self.jumpButton.ImageRectOffset = Vector2.new(1, 146)
- self.jumpButton.ImageRectSize = Vector2.new(144, 144)
- self.jumpButton.Size = UDim2.new(0, jumpButtonSize, 0, jumpButtonSize)
- self.jumpButton.Position = isSmallScreen and UDim2.new(1, -(jumpButtonSize*1.5-10), 1, -jumpButtonSize - 20) or
- UDim2.new(1, -(jumpButtonSize*1.5-10), 1, -jumpButtonSize * 1.75)
- local touchObject = nil
- self.jumpButton.InputBegan:connect(function(inputObject)
- --A touch that starts elsewhere on the screen will be sent to a frame's InputBegan event
- --if it moves over the frame. So we check that this is actually a new touch (inputObject.UserInputState ~= Enum.UserInputState.Begin)
- if touchObject or inputObject.UserInputType ~= Enum.UserInputType.Touch
- or inputObject.UserInputState ~= Enum.UserInputState.Begin then
- return
- end
- touchObject = inputObject
- self.jumpButton.ImageRectOffset = Vector2.new(146, 146)
- self.isJumping = true
- end)
- local OnInputEnded = function()
- touchObject = nil
- self.isJumping = false
- self.jumpButton.ImageRectOffset = Vector2.new(1, 146)
- end
- self.jumpButton.InputEnded:connect(function(inputObject)
- if inputObject == touchObject then
- OnInputEnded()
- end
- end)
- GuiService.MenuOpened:connect(function()
- if touchObject then
- OnInputEnded()
- end
- end)
- if not self.characterAddedConn then
- self:SetupCharacterAddedFunction()
- end
- self.jumpButton.Parent = self.parentUIFrame
- end
- return TouchJump
- end
- function _ClickToMoveController()
- --[[ Roblox Services ]]--
- local UserInputService = game:GetService("UserInputService")
- local PathfindingService = game:GetService("PathfindingService")
- local Players = game:GetService("Players")
- local DebrisService = game:GetService('Debris')
- local StarterGui = game:GetService("StarterGui")
- local Workspace = game:GetService("Workspace")
- local CollectionService = game:GetService("CollectionService")
- local GuiService = game:GetService("GuiService")
- --[[ Configuration ]]
- local ShowPath = true
- local PlayFailureAnimation = true
- local UseDirectPath = false
- local UseDirectPathForVehicle = true
- local AgentSizeIncreaseFactor = 1.0
- local UnreachableWaypointTimeout = 8
- --[[ Constants ]]--
- local movementKeys = {
- [Enum.KeyCode.W] = true;
- [Enum.KeyCode.A] = true;
- [Enum.KeyCode.S] = true;
- [Enum.KeyCode.D] = true;
- [Enum.KeyCode.Up] = true;
- [Enum.KeyCode.Down] = true;
- }
- local FFlagUserNavigationClickToMoveSkipPassedWaypointsSuccess, FFlagUserNavigationClickToMoveSkipPassedWaypointsResult = pcall(function() return UserSettings():IsUserFeatureEnabled("UserNavigationClickToMoveSkipPassedWaypoints") end)
- local FFlagUserNavigationClickToMoveSkipPassedWaypoints = FFlagUserNavigationClickToMoveSkipPassedWaypointsSuccess and FFlagUserNavigationClickToMoveSkipPassedWaypointsResult
- local Player = Players.LocalPlayer
- local ClickToMoveDisplay = _ClickToMoveDisplay()
- local ZERO_VECTOR3 = Vector3.new(0,0,0)
- local ALMOST_ZERO = 0.000001
- --------------------------UTIL LIBRARY-------------------------------
- local Utility = {}
- do
- local function FindCharacterAncestor(part)
- if part then
- local humanoid = part:FindFirstChildOfClass("Humanoid")
- if humanoid then
- return part, humanoid
- else
- return FindCharacterAncestor(part.Parent)
- end
- end
- end
- Utility.FindCharacterAncestor = FindCharacterAncestor
- local function Raycast(ray, ignoreNonCollidable, ignoreList)
- ignoreList = ignoreList or {}
- local hitPart, hitPos, hitNorm, hitMat = Workspace:FindPartOnRayWithIgnoreList(ray, ignoreList)
- if hitPart then
- if ignoreNonCollidable and hitPart.CanCollide == false then
- -- We always include character parts so a user can click on another character
- -- to walk to them.
- local _, humanoid = FindCharacterAncestor(hitPart)
- if humanoid == nil then
- table.insert(ignoreList, hitPart)
- return Raycast(ray, ignoreNonCollidable, ignoreList)
- end
- end
- return hitPart, hitPos, hitNorm, hitMat
- end
- return nil, nil
- end
- Utility.Raycast = Raycast
- end
- local humanoidCache = {}
- local function findPlayerHumanoid(player)
- local character = player and player.Character
- if character then
- local resultHumanoid = humanoidCache[player]
- if resultHumanoid and resultHumanoid.Parent == character then
- return resultHumanoid
- else
- humanoidCache[player] = nil -- Bust Old Cache
- local humanoid = character:FindFirstChildOfClass("Humanoid")
- if humanoid then
- humanoidCache[player] = humanoid
- end
- return humanoid
- end
- end
- end
- --------------------------CHARACTER CONTROL-------------------------------
- local CurrentIgnoreList
- local CurrentIgnoreTag = nil
- local TaggedInstanceAddedConnection = nil
- local TaggedInstanceRemovedConnection = nil
- local function GetCharacter()
- return Player and Player.Character
- end
- local function UpdateIgnoreTag(newIgnoreTag)
- if newIgnoreTag == CurrentIgnoreTag then
- return
- end
- if TaggedInstanceAddedConnection then
- TaggedInstanceAddedConnection:Disconnect()
- TaggedInstanceAddedConnection = nil
- end
- if TaggedInstanceRemovedConnection then
- TaggedInstanceRemovedConnection:Disconnect()
- TaggedInstanceRemovedConnection = nil
- end
- CurrentIgnoreTag = newIgnoreTag
- CurrentIgnoreList = {GetCharacter()}
- if CurrentIgnoreTag ~= nil then
- local ignoreParts = CollectionService:GetTagged(CurrentIgnoreTag)
- for _, ignorePart in ipairs(ignoreParts) do
- table.insert(CurrentIgnoreList, ignorePart)
- end
- TaggedInstanceAddedConnection = CollectionService:GetInstanceAddedSignal(
- CurrentIgnoreTag):Connect(function(ignorePart)
- table.insert(CurrentIgnoreList, ignorePart)
- end)
- TaggedInstanceRemovedConnection = CollectionService:GetInstanceRemovedSignal(
- CurrentIgnoreTag):Connect(function(ignorePart)
- for i = 1, #CurrentIgnoreList do
- if CurrentIgnoreList[i] == ignorePart then
- CurrentIgnoreList[i] = CurrentIgnoreList[#CurrentIgnoreList]
- table.remove(CurrentIgnoreList)
- break
- end
- end
- end)
- end
- end
- local function getIgnoreList()
- if CurrentIgnoreList then
- return CurrentIgnoreList
- end
- CurrentIgnoreList = {}
- table.insert(CurrentIgnoreList, GetCharacter())
- return CurrentIgnoreList
- end
- -----------------------------------PATHER--------------------------------------
- local function Pather(endPoint, surfaceNormal, overrideUseDirectPath)
- local this = {}
- local directPathForHumanoid
- local directPathForVehicle
- if overrideUseDirectPath ~= nil then
- directPathForHumanoid = overrideUseDirectPath
- directPathForVehicle = overrideUseDirectPath
- else
- directPathForHumanoid = UseDirectPath
- directPathForVehicle = UseDirectPathForVehicle
- end
- this.Cancelled = false
- this.Started = false
- this.Finished = Instance.new("BindableEvent")
- this.PathFailed = Instance.new("BindableEvent")
- this.PathComputing = false
- this.PathComputed = false
- this.OriginalTargetPoint = endPoint
- this.TargetPoint = endPoint
- this.TargetSurfaceNormal = surfaceNormal
- this.DiedConn = nil
- this.SeatedConn = nil
- this.BlockedConn = nil
- this.TeleportedConn = nil
- this.CurrentPoint = 0
- this.HumanoidOffsetFromPath = ZERO_VECTOR3
- this.CurrentWaypointPosition = nil
- this.CurrentWaypointPlaneNormal = ZERO_VECTOR3
- this.CurrentWaypointPlaneDistance = 0
- this.CurrentWaypointNeedsJump = false;
- this.CurrentHumanoidPosition = ZERO_VECTOR3
- this.CurrentHumanoidVelocity = 0
- this.NextActionMoveDirection = ZERO_VECTOR3
- this.NextActionJump = false
- this.Timeout = 0
- this.Humanoid = findPlayerHumanoid(Player)
- this.OriginPoint = nil
- this.AgentCanFollowPath = false
- this.DirectPath = false
- this.DirectPathRiseFirst = false
- local rootPart = this.Humanoid and this.Humanoid.RootPart
- if rootPart then
- -- Setup origin
- this.OriginPoint = rootPart.CFrame.p
- -- Setup agent
- local agentRadius = 2
- local agentHeight = 5
- local agentCanJump = true
- local seat = this.Humanoid.SeatPart
- if seat and seat:IsA("VehicleSeat") then
- -- Humanoid is seated on a vehicle
- local vehicle = seat:FindFirstAncestorOfClass("Model")
- if vehicle then
- -- Make sure the PrimaryPart is set to the vehicle seat while we compute the extends.
- local tempPrimaryPart = vehicle.PrimaryPart
- vehicle.PrimaryPart = seat
- -- For now, only direct path
- if directPathForVehicle then
- local extents = vehicle:GetExtentsSize()
- agentRadius = AgentSizeIncreaseFactor * 0.5 * math.sqrt(extents.X * extents.X + extents.Z * extents.Z)
- agentHeight = AgentSizeIncreaseFactor * extents.Y
- agentCanJump = false
- this.AgentCanFollowPath = true
- this.DirectPath = directPathForVehicle
- end
- -- Reset PrimaryPart
- vehicle.PrimaryPart = tempPrimaryPart
- end
- else
- local extents = GetCharacter():GetExtentsSize()
- agentRadius = AgentSizeIncreaseFactor * 0.5 * math.sqrt(extents.X * extents.X + extents.Z * extents.Z)
- agentHeight = AgentSizeIncreaseFactor * extents.Y
- agentCanJump = (this.Humanoid.JumpPower > 0)
- this.AgentCanFollowPath = true
- this.DirectPath = directPathForHumanoid
- this.DirectPathRiseFirst = this.Humanoid.Sit
- end
- -- Build path object
- this.pathResult = PathfindingService:CreatePath({AgentRadius = agentRadius, AgentHeight = agentHeight, AgentCanJump = agentCanJump})
- end
- function this:Cleanup()
- if this.stopTraverseFunc then
- this.stopTraverseFunc()
- this.stopTraverseFunc = nil
- end
- if this.MoveToConn then
- this.MoveToConn:Disconnect()
- this.MoveToConn = nil
- end
- if this.BlockedConn then
- this.BlockedConn:Disconnect()
- this.BlockedConn = nil
- end
- if this.DiedConn then
- this.DiedConn:Disconnect()
- this.DiedConn = nil
- end
- if this.SeatedConn then
- this.SeatedConn:Disconnect()
- this.SeatedConn = nil
- end
- if this.TeleportedConn then
- this.TeleportedConn:Disconnect()
- this.TeleportedConn = nil
- end
- this.Started = false
- end
- function this:Cancel()
- this.Cancelled = true
- this:Cleanup()
- end
- function this:IsActive()
- return this.AgentCanFollowPath and this.Started and not this.Cancelled
- end
- function this:OnPathInterrupted()
- -- Stop moving
- this.Cancelled = true
- this:OnPointReached(false)
- end
- function this:ComputePath()
- if this.OriginPoint then
- if this.PathComputed or this.PathComputing then return end
- this.PathComputing = true
- if this.AgentCanFollowPath then
- if this.DirectPath then
- this.pointList = {
- PathWaypoint.new(this.OriginPoint, Enum.PathWaypointAction.Walk),
- PathWaypoint.new(this.TargetPoint, this.DirectPathRiseFirst and Enum.PathWaypointAction.Jump or Enum.PathWaypointAction.Walk)
- }
- this.PathComputed = true
- else
- this.pathResult:ComputeAsync(this.OriginPoint, this.TargetPoint)
- this.pointList = this.pathResult:GetWaypoints()
- this.BlockedConn = this.pathResult.Blocked:Connect(function(blockedIdx) this:OnPathBlocked(blockedIdx) end)
- this.PathComputed = this.pathResult.Status == Enum.PathStatus.Success
- end
- end
- this.PathComputing = false
- end
- end
- function this:IsValidPath()
- this:ComputePath()
- return this.PathComputed and this.AgentCanFollowPath
- end
- this.Recomputing = false
- function this:OnPathBlocked(blockedWaypointIdx)
- local pathBlocked = blockedWaypointIdx >= this.CurrentPoint
- if not pathBlocked or this.Recomputing then
- return
- end
- this.Recomputing = true
- if this.stopTraverseFunc then
- this.stopTraverseFunc()
- this.stopTraverseFunc = nil
- end
- this.OriginPoint = this.Humanoid.RootPart.CFrame.p
- this.pathResult:ComputeAsync(this.OriginPoint, this.TargetPoint)
- this.pointList = this.pathResult:GetWaypoints()
- if #this.pointList > 0 then
- this.HumanoidOffsetFromPath = this.pointList[1].Position - this.OriginPoint
- end
- this.PathComputed = this.pathResult.Status == Enum.PathStatus.Success
- if ShowPath then
- this.stopTraverseFunc, this.setPointFunc = ClickToMoveDisplay.CreatePathDisplay(this.pointList)
- end
- if this.PathComputed then
- this.CurrentPoint = 1 -- The first waypoint is always the start location. Skip it.
- this:OnPointReached(true) -- Move to first point
- else
- this.PathFailed:Fire()
- this:Cleanup()
- end
- this.Recomputing = false
- end
- function this:OnRenderStepped(dt)
- if this.Started and not this.Cancelled then
- -- Check for Timeout (if a waypoint is not reached within the delay, we fail)
- this.Timeout = this.Timeout + dt
- if this.Timeout > UnreachableWaypointTimeout then
- this:OnPointReached(false)
- return
- end
- -- Get Humanoid position and velocity
- this.CurrentHumanoidPosition = this.Humanoid.RootPart.Position + this.HumanoidOffsetFromPath
- this.CurrentHumanoidVelocity = this.Humanoid.RootPart.Velocity
- -- Check if it has reached some waypoints
- while this.Started and this:IsCurrentWaypointReached() do
- this:OnPointReached(true)
- end
- -- If still started, update actions
- if this.Started then
- -- Move action
- this.NextActionMoveDirection = this.CurrentWaypointPosition - this.CurrentHumanoidPosition
- if this.NextActionMoveDirection.Magnitude > ALMOST_ZERO then
- this.NextActionMoveDirection = this.NextActionMoveDirection.Unit
- else
- this.NextActionMoveDirection = ZERO_VECTOR3
- end
- -- Jump action
- if this.CurrentWaypointNeedsJump then
- this.NextActionJump = true
- this.CurrentWaypointNeedsJump = false -- Request jump only once
- else
- this.NextActionJump = false
- end
- end
- end
- end
- function this:IsCurrentWaypointReached()
- local reached = false
- -- Check we do have a plane, if not, we consider the waypoint reached
- if this.CurrentWaypointPlaneNormal ~= ZERO_VECTOR3 then
- -- Compute distance of Humanoid from destination plane
- local dist = this.CurrentWaypointPlaneNormal:Dot(this.CurrentHumanoidPosition) - this.CurrentWaypointPlaneDistance
- -- Compute the component of the Humanoid velocity that is towards the plane
- local velocity = -this.CurrentWaypointPlaneNormal:Dot(this.CurrentHumanoidVelocity)
- -- Compute the threshold from the destination plane based on Humanoid velocity
- local threshold = math.max(1.0, 0.0625 * velocity)
- -- If we are less then threshold in front of the plane (between 0 and threshold) or if we are behing the plane (less then 0), we consider we reached it
- reached = dist < threshold
- else
- reached = true
- end
- if reached then
- this.CurrentWaypointPosition = nil
- this.CurrentWaypointPlaneNormal = ZERO_VECTOR3
- this.CurrentWaypointPlaneDistance = 0
- end
- return reached
- end
- function this:OnPointReached(reached)
- if reached and not this.Cancelled then
- -- First, destroyed the current displayed waypoint
- if this.setPointFunc then
- this.setPointFunc(this.CurrentPoint)
- end
- local nextWaypointIdx = this.CurrentPoint + 1
- if nextWaypointIdx > #this.pointList then
- -- End of path reached
- if this.stopTraverseFunc then
- this.stopTraverseFunc()
- end
- this.Finished:Fire()
- this:Cleanup()
- else
- local currentWaypoint = this.pointList[this.CurrentPoint]
- local nextWaypoint = this.pointList[nextWaypointIdx]
- -- If airborne, only allow to keep moving
- -- if nextWaypoint.Action ~= Jump, or path mantains a direction
- -- Otherwise, wait until the humanoid gets to the ground
- local currentState = this.Humanoid:GetState()
- local isInAir = currentState == Enum.HumanoidStateType.FallingDown
- or currentState == Enum.HumanoidStateType.Freefall
- or currentState == Enum.HumanoidStateType.Jumping
- if isInAir then
- local shouldWaitForGround = nextWaypoint.Action == Enum.PathWaypointAction.Jump
- if not shouldWaitForGround and this.CurrentPoint > 1 then
- local prevWaypoint = this.pointList[this.CurrentPoint - 1]
- local prevDir = currentWaypoint.Position - prevWaypoint.Position
- local currDir = nextWaypoint.Position - currentWaypoint.Position
- local prevDirXZ = Vector2.new(prevDir.x, prevDir.z).Unit
- local currDirXZ = Vector2.new(currDir.x, currDir.z).Unit
- local THRESHOLD_COS = 0.996 -- ~cos(5 degrees)
- shouldWaitForGround = prevDirXZ:Dot(currDirXZ) < THRESHOLD_COS
- end
- if shouldWaitForGround then
- this.Humanoid.FreeFalling:Wait()
- -- Give time to the humanoid's state to change
- -- Otherwise, the jump flag in Humanoid
- -- will be reset by the state change
- wait(0.1)
- end
- end
- -- Move to the next point
- if FFlagUserNavigationClickToMoveSkipPassedWaypoints then
- this:MoveToNextWayPoint(currentWaypoint, nextWaypoint, nextWaypointIdx)
- else
- if this.setPointFunc then
- this.setPointFunc(nextWaypointIdx)
- end
- if nextWaypoint.Action == Enum.PathWaypointAction.Jump then
- this.Humanoid.Jump = true
- end
- this.Humanoid:MoveTo(nextWaypoint.Position)
- this.CurrentPoint = nextWaypointIdx
- end
- end
- else
- this.PathFailed:Fire()
- this:Cleanup()
- end
- end
- function this:MoveToNextWayPoint(currentWaypoint, nextWaypoint, nextWaypointIdx)
- -- Build next destination plane
- -- (plane normal is perpendicular to the y plane and is from next waypoint towards current one (provided the two waypoints are not at the same location))
- -- (plane location is at next waypoint)
- this.CurrentWaypointPlaneNormal = currentWaypoint.Position - nextWaypoint.Position
- this.CurrentWaypointPlaneNormal = Vector3.new(this.CurrentWaypointPlaneNormal.X, 0, this.CurrentWaypointPlaneNormal.Z)
- if this.CurrentWaypointPlaneNormal.Magnitude > ALMOST_ZERO then
- this.CurrentWaypointPlaneNormal = this.CurrentWaypointPlaneNormal.Unit
- this.CurrentWaypointPlaneDistance = this.CurrentWaypointPlaneNormal:Dot(nextWaypoint.Position)
- else
- -- Next waypoint is the same as current waypoint so no plane
- this.CurrentWaypointPlaneNormal = ZERO_VECTOR3
- this.CurrentWaypointPlaneDistance = 0
- end
- -- Should we jump
- this.CurrentWaypointNeedsJump = nextWaypoint.Action == Enum.PathWaypointAction.Jump;
- -- Remember next waypoint position
- this.CurrentWaypointPosition = nextWaypoint.Position
- -- Move to next point
- this.CurrentPoint = nextWaypointIdx
- -- Finally reset Timeout
- this.Timeout = 0
- end
- function this:Start(overrideShowPath)
- if not this.AgentCanFollowPath then
- this.PathFailed:Fire()
- return
- end
- if this.Started then return end
- this.Started = true
- ClickToMoveDisplay.CancelFailureAnimation()
- if ShowPath then
- if overrideShowPath == nil or overrideShowPath then
- this.stopTraverseFunc, this.setPointFunc = ClickToMoveDisplay.CreatePathDisplay(this.pointList, this.OriginalTargetPoint)
- end
- end
- if #this.pointList > 0 then
- -- Determine the humanoid offset from the path's first point
- -- Offset of the first waypoint from the path's origin point
- this.HumanoidOffsetFromPath = Vector3.new(0, this.pointList[1].Position.Y - this.OriginPoint.Y, 0)
- -- As well as its current position and velocity
- this.CurrentHumanoidPosition = this.Humanoid.RootPart.Position + this.HumanoidOffsetFromPath
- this.CurrentHumanoidVelocity = this.Humanoid.RootPart.Velocity
- -- Connect to events
- this.SeatedConn = this.Humanoid.Seated:Connect(function(isSeated, seat) this:OnPathInterrupted() end)
- this.DiedConn = this.Humanoid.Died:Connect(function() this:OnPathInterrupted() end)
- this.TeleportedConn = this.Humanoid.RootPart:GetPropertyChangedSignal("CFrame"):Connect(function() this:OnPathInterrupted() end)
- -- Actually start
- this.CurrentPoint = 1 -- The first waypoint is always the start location. Skip it.
- this:OnPointReached(true) -- Move to first point
- else
- this.PathFailed:Fire()
- if this.stopTraverseFunc then
- this.stopTraverseFunc()
- end
- end
- end
- --We always raycast to the ground in the case that the user clicked a wall.
- local offsetPoint = this.TargetPoint + this.TargetSurfaceNormal*1.5
- local ray = Ray.new(offsetPoint, Vector3.new(0,-1,0)*50)
- local newHitPart, newHitPos = Workspace:FindPartOnRayWithIgnoreList(ray, getIgnoreList())
- if newHitPart then
- this.TargetPoint = newHitPos
- end
- this:ComputePath()
- return this
- end
- -------------------------------------------------------------------------
- local function CheckAlive()
- local humanoid = findPlayerHumanoid(Player)
- return humanoid ~= nil and humanoid.Health > 0
- end
- local function GetEquippedTool(character)
- if character ~= nil then
- for _, child in pairs(character:GetChildren()) do
- if child:IsA('Tool') then
- return child
- end
- end
- end
- end
- local ExistingPather = nil
- local ExistingIndicator = nil
- local PathCompleteListener = nil
- local PathFailedListener = nil
- local function CleanupPath()
- if ExistingPather then
- ExistingPather:Cancel()
- ExistingPather = nil
- end
- if PathCompleteListener then
- PathCompleteListener:Disconnect()
- PathCompleteListener = nil
- end
- if PathFailedListener then
- PathFailedListener:Disconnect()
- PathFailedListener = nil
- end
- if ExistingIndicator then
- ExistingIndicator:Destroy()
- end
- end
- local function HandleMoveTo(thisPather, hitPt, hitChar, character, overrideShowPath)
- if ExistingPather then
- CleanupPath()
- end
- ExistingPather = thisPather
- thisPather:Start(overrideShowPath)
- PathCompleteListener = thisPather.Finished.Event:Connect(function()
- CleanupPath()
- if hitChar then
- local currentWeapon = GetEquippedTool(character)
- if currentWeapon then
- currentWeapon:Activate()
- end
- end
- end)
- PathFailedListener = thisPather.PathFailed.Event:Connect(function()
- CleanupPath()
- if overrideShowPath == nil or overrideShowPath then
- local shouldPlayFailureAnim = PlayFailureAnimation and not (ExistingPather and ExistingPather:IsActive())
- if shouldPlayFailureAnim then
- ClickToMoveDisplay.PlayFailureAnimation()
- end
- ClickToMoveDisplay.DisplayFailureWaypoint(hitPt)
- end
- end)
- end
- local function ShowPathFailedFeedback(hitPt)
- if ExistingPather and ExistingPather:IsActive() then
- ExistingPather:Cancel()
- end
- if PlayFailureAnimation then
- ClickToMoveDisplay.PlayFailureAnimation()
- end
- ClickToMoveDisplay.DisplayFailureWaypoint(hitPt)
- end
- function OnTap(tapPositions, goToPoint, wasTouchTap)
- -- Good to remember if this is the latest tap event
- local camera = Workspace.CurrentCamera
- local character = Player.Character
- if not CheckAlive() then return end
- -- This is a path tap position
- if #tapPositions == 1 or goToPoint then
- if camera then
- local unitRay = camera:ScreenPointToRay(tapPositions[1].x, tapPositions[1].y)
- local ray = Ray.new(unitRay.Origin, unitRay.Direction*1000)
- local myHumanoid = findPlayerHumanoid(Player)
- local hitPart, hitPt, hitNormal = Utility.Raycast(ray, true, getIgnoreList())
- local hitChar, hitHumanoid = Utility.FindCharacterAncestor(hitPart)
- if wasTouchTap and hitHumanoid and StarterGui:GetCore("AvatarContextMenuEnabled") then
- local clickedPlayer = Players:GetPlayerFromCharacter(hitHumanoid.Parent)
- if clickedPlayer then
- CleanupPath()
- return
- end
- end
- if goToPoint then
- hitPt = goToPoint
- hitChar = nil
- end
- if hitPt and character then
- -- Clean up current path
- CleanupPath()
- local thisPather = Pather(hitPt, hitNormal)
- if thisPather:IsValidPath() then
- HandleMoveTo(thisPather, hitPt, hitChar, character)
- else
- -- Clean up
- thisPather:Cleanup()
- -- Feedback here for when we don't have a good path
- ShowPathFailedFeedback(hitPt)
- end
- end
- end
- elseif #tapPositions >= 2 then
- if camera then
- -- Do shoot
- local currentWeapon = GetEquippedTool(character)
- if currentWeapon then
- currentWeapon:Activate()
- end
- end
- end
- end
- local function DisconnectEvent(event)
- if event then
- event:Disconnect()
- end
- end
- --[[ The ClickToMove Controller Class ]]--
- local KeyboardController = _Keyboard()
- local ClickToMove = setmetatable({}, KeyboardController)
- ClickToMove.__index = ClickToMove
- function ClickToMove.new(CONTROL_ACTION_PRIORITY)
- local self = setmetatable(KeyboardController.new(CONTROL_ACTION_PRIORITY), ClickToMove)
- self.fingerTouches = {}
- self.numUnsunkTouches = 0
- -- PC simulation
- self.mouse1Down = tick()
- self.mouse1DownPos = Vector2.new()
- self.mouse2DownTime = tick()
- self.mouse2DownPos = Vector2.new()
- self.mouse2UpTime = tick()
- self.keyboardMoveVector = ZERO_VECTOR3
- self.tapConn = nil
- self.inputBeganConn = nil
- self.inputChangedConn = nil
- self.inputEndedConn = nil
- self.humanoidDiedConn = nil
- self.characterChildAddedConn = nil
- self.onCharacterAddedConn = nil
- self.characterChildRemovedConn = nil
- self.renderSteppedConn = nil
- self.menuOpenedConnection = nil
- self.running = false
- self.wasdEnabled = false
- return self
- end
- function ClickToMove:DisconnectEvents()
- DisconnectEvent(self.tapConn)
- DisconnectEvent(self.inputBeganConn)
- DisconnectEvent(self.inputChangedConn)
- DisconnectEvent(self.inputEndedConn)
- DisconnectEvent(self.humanoidDiedConn)
- DisconnectEvent(self.characterChildAddedConn)
- DisconnectEvent(self.onCharacterAddedConn)
- DisconnectEvent(self.renderSteppedConn)
- DisconnectEvent(self.characterChildRemovedConn)
- DisconnectEvent(self.menuOpenedConnection)
- end
- function ClickToMove:OnTouchBegan(input, processed)
- if self.fingerTouches[input] == nil and not processed then
- self.numUnsunkTouches = self.numUnsunkTouches + 1
- end
- self.fingerTouches[input] = processed
- end
- function ClickToMove:OnTouchChanged(input, processed)
- if self.fingerTouches[input] == nil then
- self.fingerTouches[input] = processed
- if not processed then
- self.numUnsunkTouches = self.numUnsunkTouches + 1
- end
- end
- end
- function ClickToMove:OnTouchEnded(input, processed)
- if self.fingerTouches[input] ~= nil and self.fingerTouches[input] == false then
- self.numUnsunkTouches = self.numUnsunkTouches - 1
- end
- self.fingerTouches[input] = nil
- end
- function ClickToMove:OnCharacterAdded(character)
- self:DisconnectEvents()
- self.inputBeganConn = UserInputService.InputBegan:Connect(function(input, processed)
- if input.UserInputType == Enum.UserInputType.Touch then
- self:OnTouchBegan(input, processed)
- end
- -- Cancel path when you use the keyboard controls if wasd is enabled.
- if self.wasdEnabled and processed == false and input.UserInputType == Enum.UserInputType.Keyboard
- and movementKeys[input.KeyCode] then
- CleanupPath()
- ClickToMoveDisplay.CancelFailureAnimation()
- end
- if input.UserInputType == Enum.UserInputType.MouseButton1 then
- self.mouse1DownTime = tick()
- self.mouse1DownPos = input.Position
- end
- if input.UserInputType == Enum.UserInputType.MouseButton2 then
- self.mouse2DownTime = tick()
- self.mouse2DownPos = input.Position
- end
- end)
- self.inputChangedConn = UserInputService.InputChanged:Connect(function(input, processed)
- if input.UserInputType == Enum.UserInputType.Touch then
- self:OnTouchChanged(input, processed)
- end
- end)
- self.inputEndedConn = UserInputService.InputEnded:Connect(function(input, processed)
- if input.UserInputType == Enum.UserInputType.Touch then
- self:OnTouchEnded(input, processed)
- end
- if input.UserInputType == Enum.UserInputType.MouseButton2 then
- self.mouse2UpTime = tick()
- local currPos = input.Position
- -- We allow click to move during path following or if there is no keyboard movement
- local allowed = ExistingPather or self.keyboardMoveVector.Magnitude <= 0
- if self.mouse2UpTime - self.mouse2DownTime < 0.25 and (currPos - self.mouse2DownPos).magnitude < 5 and allowed then
- local positions = {currPos}
- OnTap(positions)
- end
- end
- end)
- self.tapConn = UserInputService.TouchTap:Connect(function(touchPositions, processed)
- if not processed then
- OnTap(touchPositions, nil, true)
- end
- end)
- self.menuOpenedConnection = GuiService.MenuOpened:Connect(function()
- CleanupPath()
- end)
- local function OnCharacterChildAdded(child)
- if UserInputService.TouchEnabled then
- if child:IsA('Tool') then
- child.ManualActivationOnly = true
- end
- end
- if child:IsA('Humanoid') then
- DisconnectEvent(self.humanoidDiedConn)
- self.humanoidDiedConn = child.Died:Connect(function()
- if ExistingIndicator then
- DebrisService:AddItem(ExistingIndicator.Model, 1)
- end
- end)
- end
- end
- self.characterChildAddedConn = character.ChildAdded:Connect(function(child)
- OnCharacterChildAdded(child)
- end)
- self.characterChildRemovedConn = character.ChildRemoved:Connect(function(child)
- if UserInputService.TouchEnabled then
- if child:IsA('Tool') then
- child.ManualActivationOnly = false
- end
- end
- end)
- for _, child in pairs(character:GetChildren()) do
- OnCharacterChildAdded(child)
- end
- end
- function ClickToMove:Start()
- self:Enable(true)
- end
- function ClickToMove:Stop()
- self:Enable(false)
- end
- function ClickToMove:CleanupPath()
- CleanupPath()
- end
- function ClickToMove:Enable(enable, enableWASD, touchJumpController)
- if enable then
- if not self.running then
- if Player.Character then -- retro-listen
- self:OnCharacterAdded(Player.Character)
- end
- self.onCharacterAddedConn = Player.CharacterAdded:Connect(function(char)
- self:OnCharacterAdded(char)
- end)
- self.running = true
- end
- self.touchJumpController = touchJumpController
- if self.touchJumpController then
- self.touchJumpController:Enable(self.jumpEnabled)
- end
- else
- if self.running then
- self:DisconnectEvents()
- CleanupPath()
- -- Restore tool activation on shutdown
- if UserInputService.TouchEnabled then
- local character = Player.Character
- if character then
- for _, child in pairs(character:GetChildren()) do
- if child:IsA('Tool') then
- child.ManualActivationOnly = false
- end
- end
- end
- end
- self.running = false
- end
- if self.touchJumpController and not self.jumpEnabled then
- self.touchJumpController:Enable(true)
- end
- self.touchJumpController = nil
- end
- -- Extension for initializing Keyboard input as this class now derives from Keyboard
- if UserInputService.KeyboardEnabled and enable ~= self.enabled then
- self.forwardValue = 0
- self.backwardValue = 0
- self.leftValue = 0
- self.rightValue = 0
- self.moveVector = ZERO_VECTOR3
- if enable then
- self:BindContextActions()
- self:ConnectFocusEventListeners()
- else
- self:UnbindContextActions()
- self:DisconnectFocusEventListeners()
- end
- end
- self.wasdEnabled = enable and enableWASD or false
- self.enabled = enable
- end
- function ClickToMove:OnRenderStepped(dt)
- -- Reset jump
- self.isJumping = false
- -- Handle Pather
- if ExistingPather then
- -- Let the Pather update
- ExistingPather:OnRenderStepped(dt)
- -- If we still have a Pather, set the resulting actions
- if ExistingPather then
- -- Setup move (NOT relative to camera)
- self.moveVector = ExistingPather.NextActionMoveDirection
- self.moveVectorIsCameraRelative = false
- -- Setup jump (but do NOT prevent the base Keayboard class from requesting jumps as well)
- if ExistingPather.NextActionJump then
- self.isJumping = true
- end
- else
- self.moveVector = self.keyboardMoveVector
- self.moveVectorIsCameraRelative = true
- end
- else
- self.moveVector = self.keyboardMoveVector
- self.moveVectorIsCameraRelative = true
- end
- -- Handle Keyboard's jump
- if self.jumpRequested then
- self.isJumping = true
- end
- end
- -- Overrides Keyboard:UpdateMovement(inputState) to conditionally consider self.wasdEnabled and let OnRenderStepped handle the movement
- function ClickToMove:UpdateMovement(inputState)
- if inputState == Enum.UserInputState.Cancel then
- self.keyboardMoveVector = ZERO_VECTOR3
- elseif self.wasdEnabled then
- self.keyboardMoveVector = Vector3.new(self.leftValue + self.rightValue, 0, self.forwardValue + self.backwardValue)
- end
- end
- -- Overrides Keyboard:UpdateJump() because jump is handled in OnRenderStepped
- function ClickToMove:UpdateJump()
- -- Nothing to do (handled in OnRenderStepped)
- end
- --Public developer facing functions
- function ClickToMove:SetShowPath(value)
- ShowPath = value
- end
- function ClickToMove:GetShowPath()
- return ShowPath
- end
- function ClickToMove:SetWaypointTexture(texture)
- ClickToMoveDisplay.SetWaypointTexture(texture)
- end
- function ClickToMove:GetWaypointTexture()
- return ClickToMoveDisplay.GetWaypointTexture()
- end
- function ClickToMove:SetWaypointRadius(radius)
- ClickToMoveDisplay.SetWaypointRadius(radius)
- end
- function ClickToMove:GetWaypointRadius()
- return ClickToMoveDisplay.GetWaypointRadius()
- end
- function ClickToMove:SetEndWaypointTexture(texture)
- ClickToMoveDisplay.SetEndWaypointTexture(texture)
- end
- function ClickToMove:GetEndWaypointTexture()
- return ClickToMoveDisplay.GetEndWaypointTexture()
- end
- function ClickToMove:SetWaypointsAlwaysOnTop(alwaysOnTop)
- ClickToMoveDisplay.SetWaypointsAlwaysOnTop(alwaysOnTop)
- end
- function ClickToMove:GetWaypointsAlwaysOnTop()
- return ClickToMoveDisplay.GetWaypointsAlwaysOnTop()
- end
- function ClickToMove:SetFailureAnimationEnabled(enabled)
- PlayFailureAnimation = enabled
- end
- function ClickToMove:GetFailureAnimationEnabled()
- return PlayFailureAnimation
- end
- function ClickToMove:SetIgnoredPartsTag(tag)
- UpdateIgnoreTag(tag)
- end
- function ClickToMove:GetIgnoredPartsTag()
- return CurrentIgnoreTag
- end
- function ClickToMove:SetUseDirectPath(directPath)
- UseDirectPath = directPath
- end
- function ClickToMove:GetUseDirectPath()
- return UseDirectPath
- end
- function ClickToMove:SetAgentSizeIncreaseFactor(increaseFactorPercent)
- AgentSizeIncreaseFactor = 1.0 + (increaseFactorPercent / 100.0)
- end
- function ClickToMove:GetAgentSizeIncreaseFactor()
- return (AgentSizeIncreaseFactor - 1.0) * 100.0
- end
- function ClickToMove:SetUnreachableWaypointTimeout(timeoutInSec)
- UnreachableWaypointTimeout = timeoutInSec
- end
- function ClickToMove:GetUnreachableWaypointTimeout()
- return UnreachableWaypointTimeout
- end
- function ClickToMove:SetUserJumpEnabled(jumpEnabled)
- self.jumpEnabled = jumpEnabled
- if self.touchJumpController then
- self.touchJumpController:Enable(jumpEnabled)
- end
- end
- function ClickToMove:GetUserJumpEnabled()
- return self.jumpEnabled
- end
- function ClickToMove:MoveTo(position, showPath, useDirectPath)
- local character = Player.Character
- if character == nil then
- return false
- end
- local thisPather = Pather(position, Vector3.new(0, 1, 0), useDirectPath)
- if thisPather and thisPather:IsValidPath() then
- HandleMoveTo(thisPather, position, nil, character, showPath)
- return true
- end
- return false
- end
- return ClickToMove
- end
- function _TouchThumbstick()
- local Players = game:GetService("Players")
- local GuiService = game:GetService("GuiService")
- local UserInputService = game:GetService("UserInputService")
- --[[ Constants ]]--
- local ZERO_VECTOR3 = Vector3.new(0,0,0)
- local TOUCH_CONTROL_SHEET = "rbxasset://textures/ui/TouchControlsSheet.png"
- --[[ The Module ]]--
- local BaseCharacterController = _BaseCharacterController()
- local TouchThumbstick = setmetatable({}, BaseCharacterController)
- TouchThumbstick.__index = TouchThumbstick
- function TouchThumbstick.new()
- local self = setmetatable(BaseCharacterController.new(), TouchThumbstick)
- self.isFollowStick = false
- self.thumbstickFrame = nil
- self.moveTouchObject = nil
- self.onTouchMovedConn = nil
- self.onTouchEndedConn = nil
- self.screenPos = nil
- self.stickImage = nil
- self.thumbstickSize = nil -- Float
- return self
- end
- function TouchThumbstick:Enable(enable, uiParentFrame)
- if enable == nil then return false end -- If nil, return false (invalid argument)
- enable = enable and true or false -- Force anything non-nil to boolean before comparison
- if self.enabled == enable then return true end -- If no state change, return true indicating already in requested state
- self.moveVector = ZERO_VECTOR3
- self.isJumping = false
- if enable then
- -- Enable
- if not self.thumbstickFrame then
- self:Create(uiParentFrame)
- end
- self.thumbstickFrame.Visible = true
- else
- -- Disable
- self.thumbstickFrame.Visible = false
- self:OnInputEnded()
- end
- self.enabled = enable
- end
- function TouchThumbstick:OnInputEnded()
- self.thumbstickFrame.Position = self.screenPos
- self.stickImage.Position = UDim2.new(0, self.thumbstickFrame.Size.X.Offset/2 - self.thumbstickSize/4, 0, self.thumbstickFrame.Size.Y.Offset/2 - self.thumbstickSize/4)
- self.moveVector = ZERO_VECTOR3
- self.isJumping = false
- self.thumbstickFrame.Position = self.screenPos
- self.moveTouchObject = nil
- end
- function TouchThumbstick:Create(parentFrame)
- if self.thumbstickFrame then
- self.thumbstickFrame:Destroy()
- self.thumbstickFrame = nil
- if self.onTouchMovedConn then
- self.onTouchMovedConn:Disconnect()
- self.onTouchMovedConn = nil
- end
- if self.onTouchEndedConn then
- self.onTouchEndedConn:Disconnect()
- self.onTouchEndedConn = nil
- end
- end
- local minAxis = math.min(parentFrame.AbsoluteSize.x, parentFrame.AbsoluteSize.y)
- local isSmallScreen = minAxis <= 500
- self.thumbstickSize = isSmallScreen and 70 or 120
- self.screenPos = isSmallScreen and UDim2.new(0, (self.thumbstickSize/2) - 10, 1, -self.thumbstickSize - 20) or
- UDim2.new(0, self.thumbstickSize/2, 1, -self.thumbstickSize * 1.75)
- self.thumbstickFrame = Instance.new("Frame")
- self.thumbstickFrame.Name = "ThumbstickFrame"
- self.thumbstickFrame.Active = true
- self.thumbstickFrame.Visible = false
- self.thumbstickFrame.Size = UDim2.new(0, self.thumbstickSize, 0, self.thumbstickSize)
- self.thumbstickFrame.Position = self.screenPos
- self.thumbstickFrame.BackgroundTransparency = 1
- local outerImage = Instance.new("ImageLabel")
- outerImage.Name = "OuterImage"
- outerImage.Image = TOUCH_CONTROL_SHEET
- outerImage.ImageRectOffset = Vector2.new()
- outerImage.ImageRectSize = Vector2.new(220, 220)
- outerImage.BackgroundTransparency = 1
- outerImage.Size = UDim2.new(0, self.thumbstickSize, 0, self.thumbstickSize)
- outerImage.Position = UDim2.new(0, 0, 0, 0)
- outerImage.Parent = self.thumbstickFrame
- self.stickImage = Instance.new("ImageLabel")
- self.stickImage.Name = "StickImage"
- self.stickImage.Image = TOUCH_CONTROL_SHEET
- self.stickImage.ImageRectOffset = Vector2.new(220, 0)
- self.stickImage.ImageRectSize = Vector2.new(111, 111)
- self.stickImage.BackgroundTransparency = 1
- self.stickImage.Size = UDim2.new(0, self.thumbstickSize/2, 0, self.thumbstickSize/2)
- self.stickImage.Position = UDim2.new(0, self.thumbstickSize/2 - self.thumbstickSize/4, 0, self.thumbstickSize/2 - self.thumbstickSize/4)
- self.stickImage.ZIndex = 2
- self.stickImage.Parent = self.thumbstickFrame
- local centerPosition = nil
- local deadZone = 0.05
- local function DoMove(direction)
- local currentMoveVector = direction / (self.thumbstickSize/2)
- -- Scaled Radial Dead Zone
- local inputAxisMagnitude = currentMoveVector.magnitude
- if inputAxisMagnitude < deadZone then
- currentMoveVector = Vector3.new()
- else
- currentMoveVector = currentMoveVector.unit * ((inputAxisMagnitude - deadZone) / (1 - deadZone))
- -- NOTE: Making currentMoveVector a unit vector will cause the player to instantly go max speed
- -- must check for zero length vector is using unit
- currentMoveVector = Vector3.new(currentMoveVector.x, 0, currentMoveVector.y)
- end
- self.moveVector = currentMoveVector
- end
- local function MoveStick(pos)
- local relativePosition = Vector2.new(pos.x - centerPosition.x, pos.y - centerPosition.y)
- local length = relativePosition.magnitude
- local maxLength = self.thumbstickFrame.AbsoluteSize.x/2
- if self.isFollowStick and length > maxLength then
- local offset = relativePosition.unit * maxLength
- self.thumbstickFrame.Position = UDim2.new(
- 0, pos.x - self.thumbstickFrame.AbsoluteSize.x/2 - offset.x,
- 0, pos.y - self.thumbstickFrame.AbsoluteSize.y/2 - offset.y)
- else
- length = math.min(length, maxLength)
- relativePosition = relativePosition.unit * length
- end
- self.stickImage.Position = UDim2.new(0, relativePosition.x + self.stickImage.AbsoluteSize.x/2, 0, relativePosition.y + self.stickImage.AbsoluteSize.y/2)
- end
- -- input connections
- self.thumbstickFrame.InputBegan:Connect(function(inputObject)
- --A touch that starts elsewhere on the screen will be sent to a frame's InputBegan event
- --if it moves over the frame. So we check that this is actually a new touch (inputObject.UserInputState ~= Enum.UserInputState.Begin)
- if self.moveTouchObject or inputObject.UserInputType ~= Enum.UserInputType.Touch
- or inputObject.UserInputState ~= Enum.UserInputState.Begin then
- return
- end
- self.moveTouchObject = inputObject
- self.thumbstickFrame.Position = UDim2.new(0, inputObject.Position.x - self.thumbstickFrame.Size.X.Offset/2, 0, inputObject.Position.y - self.thumbstickFrame.Size.Y.Offset/2)
- centerPosition = Vector2.new(self.thumbstickFrame.AbsolutePosition.x + self.thumbstickFrame.AbsoluteSize.x/2,
- self.thumbstickFrame.AbsolutePosition.y + self.thumbstickFrame.AbsoluteSize.y/2)
- local direction = Vector2.new(inputObject.Position.x - centerPosition.x, inputObject.Position.y - centerPosition.y)
- end)
- self.onTouchMovedConn = UserInputService.TouchMoved:Connect(function(inputObject, isProcessed)
- if inputObject == self.moveTouchObject then
- centerPosition = Vector2.new(self.thumbstickFrame.AbsolutePosition.x + self.thumbstickFrame.AbsoluteSize.x/2,
- self.thumbstickFrame.AbsolutePosition.y + self.thumbstickFrame.AbsoluteSize.y/2)
- local direction = Vector2.new(inputObject.Position.x - centerPosition.x, inputObject.Position.y - centerPosition.y)
- DoMove(direction)
- MoveStick(inputObject.Position)
- end
- end)
- self.onTouchEndedConn = UserInputService.TouchEnded:Connect(function(inputObject, isProcessed)
- if inputObject == self.moveTouchObject then
- self:OnInputEnded()
- end
- end)
- GuiService.MenuOpened:Connect(function()
- if self.moveTouchObject then
- self:OnInputEnded()
- end
- end)
- self.thumbstickFrame.Parent = parentFrame
- end
- return TouchThumbstick
- end
- function _DynamicThumbstick()
- local ZERO_VECTOR3 = Vector3.new(0,0,0)
- local TOUCH_CONTROLS_SHEET = "rbxasset://textures/ui/Input/TouchControlsSheetV2.png"
- local DYNAMIC_THUMBSTICK_ACTION_NAME = "DynamicThumbstickAction"
- local DYNAMIC_THUMBSTICK_ACTION_PRIORITY = Enum.ContextActionPriority.High.Value
- local MIDDLE_TRANSPARENCIES = {
- 1 - 0.89,
- 1 - 0.70,
- 1 - 0.60,
- 1 - 0.50,
- 1 - 0.40,
- 1 - 0.30,
- 1 - 0.25
- }
- local NUM_MIDDLE_IMAGES = #MIDDLE_TRANSPARENCIES
- local FADE_IN_OUT_BACKGROUND = true
- local FADE_IN_OUT_MAX_ALPHA = 0.35
- local FADE_IN_OUT_HALF_DURATION_DEFAULT = 0.3
- local FADE_IN_OUT_BALANCE_DEFAULT = 0.5
- local ThumbstickFadeTweenInfo = TweenInfo.new(0.15, Enum.EasingStyle.Quad, Enum.EasingDirection.InOut)
- local Players = game:GetService("Players")
- local GuiService = game:GetService("GuiService")
- local UserInputService = game:GetService("UserInputService")
- local ContextActionService = game:GetService("ContextActionService")
- local RunService = game:GetService("RunService")
- local TweenService = game:GetService("TweenService")
- local LocalPlayer = Players.LocalPlayer
- if not LocalPlayer then
- Players:GetPropertyChangedSignal("LocalPlayer"):Wait()
- LocalPlayer = Players.LocalPlayer
- end
- --[[ The Module ]]--
- local BaseCharacterController = _BaseCharacterController()
- local DynamicThumbstick = setmetatable({}, BaseCharacterController)
- DynamicThumbstick.__index = DynamicThumbstick
- function DynamicThumbstick.new()
- local self = setmetatable(BaseCharacterController.new(), DynamicThumbstick)
- self.moveTouchObject = nil
- self.moveTouchLockedIn = false
- self.moveTouchFirstChanged = false
- self.moveTouchStartPosition = nil
- self.startImage = nil
- self.endImage = nil
- self.middleImages = {}
- self.startImageFadeTween = nil
- self.endImageFadeTween = nil
- self.middleImageFadeTweens = {}
- self.isFirstTouch = true
- self.thumbstickFrame = nil
- self.onRenderSteppedConn = nil
- self.fadeInAndOutBalance = FADE_IN_OUT_BALANCE_DEFAULT
- self.fadeInAndOutHalfDuration = FADE_IN_OUT_HALF_DURATION_DEFAULT
- self.hasFadedBackgroundInPortrait = false
- self.hasFadedBackgroundInLandscape = false
- self.tweenInAlphaStart = nil
- self.tweenOutAlphaStart = nil
- return self
- end
- -- Note: Overrides base class GetIsJumping with get-and-clear behavior to do a single jump
- -- rather than sustained jumping. This is only to preserve the current behavior through the refactor.
- function DynamicThumbstick:GetIsJumping()
- local wasJumping = self.isJumping
- self.isJumping = false
- return wasJumping
- end
- function DynamicThumbstick:Enable(enable, uiParentFrame)
- if enable == nil then return false end -- If nil, return false (invalid argument)
- enable = enable and true or false -- Force anything non-nil to boolean before comparison
- if self.enabled == enable then return true end -- If no state change, return true indicating already in requested state
- if enable then
- -- Enable
- if not self.thumbstickFrame then
- self:Create(uiParentFrame)
- end
- self:BindContextActions()
- else
- ContextActionService:UnbindAction(DYNAMIC_THUMBSTICK_ACTION_NAME)
- -- Disable
- self:OnInputEnded() -- Cleanup
- end
- self.enabled = enable
- self.thumbstickFrame.Visible = enable
- end
- -- Was called OnMoveTouchEnded in previous version
- function DynamicThumbstick:OnInputEnded()
- self.moveTouchObject = nil
- self.moveVector = ZERO_VECTOR3
- self:FadeThumbstick(false)
- end
- function DynamicThumbstick:FadeThumbstick(visible)
- if not visible and self.moveTouchObject then
- return
- end
- if self.isFirstTouch then return end
- if self.startImageFadeTween then
- self.startImageFadeTween:Cancel()
- end
- if self.endImageFadeTween then
- self.endImageFadeTween:Cancel()
- end
- for i = 1, #self.middleImages do
- if self.middleImageFadeTweens[i] then
- self.middleImageFadeTweens[i]:Cancel()
- end
- end
- if visible then
- self.startImageFadeTween = TweenService:Create(self.startImage, ThumbstickFadeTweenInfo, { ImageTransparency = 0 })
- self.startImageFadeTween:Play()
- self.endImageFadeTween = TweenService:Create(self.endImage, ThumbstickFadeTweenInfo, { ImageTransparency = 0.2 })
- self.endImageFadeTween:Play()
- for i = 1, #self.middleImages do
- self.middleImageFadeTweens[i] = TweenService:Create(self.middleImages[i], ThumbstickFadeTweenInfo, { ImageTransparency = MIDDLE_TRANSPARENCIES[i] })
- self.middleImageFadeTweens[i]:Play()
- end
- else
- self.startImageFadeTween = TweenService:Create(self.startImage, ThumbstickFadeTweenInfo, { ImageTransparency = 1 })
- self.startImageFadeTween:Play()
- self.endImageFadeTween = TweenService:Create(self.endImage, ThumbstickFadeTweenInfo, { ImageTransparency = 1 })
- self.endImageFadeTween:Play()
- for i = 1, #self.middleImages do
- self.middleImageFadeTweens[i] = TweenService:Create(self.middleImages[i], ThumbstickFadeTweenInfo, { ImageTransparency = 1 })
- self.middleImageFadeTweens[i]:Play()
- end
- end
- end
- function DynamicThumbstick:FadeThumbstickFrame(fadeDuration, fadeRatio)
- self.fadeInAndOutHalfDuration = fadeDuration * 0.5
- self.fadeInAndOutBalance = fadeRatio
- self.tweenInAlphaStart = tick()
- end
- function DynamicThumbstick:InputInFrame(inputObject)
- local frameCornerTopLeft = self.thumbstickFrame.AbsolutePosition
- local frameCornerBottomRight = frameCornerTopLeft + self.thumbstickFrame.AbsoluteSize
- local inputPosition = inputObject.Position
- if inputPosition.X >= frameCornerTopLeft.X and inputPosition.Y >= frameCornerTopLeft.Y then
- if inputPosition.X <= frameCornerBottomRight.X and inputPosition.Y <= frameCornerBottomRight.Y then
- return true
- end
- end
- return false
- end
- function DynamicThumbstick:DoFadeInBackground()
- local playerGui = LocalPlayer:FindFirstChildOfClass("PlayerGui")
- local hasFadedBackgroundInOrientation = false
- -- only fade in/out the background once per orientation
- if playerGui then
- if playerGui.CurrentScreenOrientation == Enum.ScreenOrientation.LandscapeLeft or
- playerGui.CurrentScreenOrientation == Enum.ScreenOrientation.LandscapeRight then
- hasFadedBackgroundInOrientation = self.hasFadedBackgroundInLandscape
- self.hasFadedBackgroundInLandscape = true
- elseif playerGui.CurrentScreenOrientation == Enum.ScreenOrientation.Portrait then
- hasFadedBackgroundInOrientation = self.hasFadedBackgroundInPortrait
- self.hasFadedBackgroundInPortrait = true
- end
- end
- if not hasFadedBackgroundInOrientation then
- self.fadeInAndOutHalfDuration = FADE_IN_OUT_HALF_DURATION_DEFAULT
- self.fadeInAndOutBalance = FADE_IN_OUT_BALANCE_DEFAULT
- self.tweenInAlphaStart = tick()
- end
- end
- function DynamicThumbstick:DoMove(direction)
- local currentMoveVector = direction
- -- Scaled Radial Dead Zone
- local inputAxisMagnitude = currentMoveVector.magnitude
- if inputAxisMagnitude < self.radiusOfDeadZone then
- currentMoveVector = ZERO_VECTOR3
- else
- currentMoveVector = currentMoveVector.unit*(
- 1 - math.max(0, (self.radiusOfMaxSpeed - currentMoveVector.magnitude)/self.radiusOfMaxSpeed)
- )
- currentMoveVector = Vector3.new(currentMoveVector.x, 0, currentMoveVector.y)
- end
- self.moveVector = currentMoveVector
- end
- function DynamicThumbstick:LayoutMiddleImages(startPos, endPos)
- local startDist = (self.thumbstickSize / 2) + self.middleSize
- local vector = endPos - startPos
- local distAvailable = vector.magnitude - (self.thumbstickRingSize / 2) - self.middleSize
- local direction = vector.unit
- local distNeeded = self.middleSpacing * NUM_MIDDLE_IMAGES
- local spacing = self.middleSpacing
- if distNeeded < distAvailable then
- spacing = distAvailable / NUM_MIDDLE_IMAGES
- end
- for i = 1, NUM_MIDDLE_IMAGES do
- local image = self.middleImages[i]
- local distWithout = startDist + (spacing * (i - 2))
- local currentDist = startDist + (spacing * (i - 1))
- if distWithout < distAvailable then
- local pos = endPos - direction * currentDist
- local exposedFraction = math.clamp(1 - ((currentDist - distAvailable) / spacing), 0, 1)
- image.Visible = true
- image.Position = UDim2.new(0, pos.X, 0, pos.Y)
- image.Size = UDim2.new(0, self.middleSize * exposedFraction, 0, self.middleSize * exposedFraction)
- else
- image.Visible = false
- end
- end
- end
- function DynamicThumbstick:MoveStick(pos)
- local vector2StartPosition = Vector2.new(self.moveTouchStartPosition.X, self.moveTouchStartPosition.Y)
- local startPos = vector2StartPosition - self.thumbstickFrame.AbsolutePosition
- local endPos = Vector2.new(pos.X, pos.Y) - self.thumbstickFrame.AbsolutePosition
- self.endImage.Position = UDim2.new(0, endPos.X, 0, endPos.Y)
- self:LayoutMiddleImages(startPos, endPos)
- end
- function DynamicThumbstick:BindContextActions()
- local function inputBegan(inputObject)
- if self.moveTouchObject then
- return Enum.ContextActionResult.Pass
- end
- if not self:InputInFrame(inputObject) then
- return Enum.ContextActionResult.Pass
- end
- if self.isFirstTouch then
- self.isFirstTouch = false
- local tweenInfo = TweenInfo.new(0.5, Enum.EasingStyle.Quad, Enum.EasingDirection.Out,0,false,0)
- TweenService:Create(self.startImage, tweenInfo, {Size = UDim2.new(0, 0, 0, 0)}):Play()
- TweenService:Create(
- self.endImage,
- tweenInfo,
- {Size = UDim2.new(0, self.thumbstickSize, 0, self.thumbstickSize), ImageColor3 = Color3.new(0,0,0)}
- ):Play()
- end
- self.moveTouchLockedIn = false
- self.moveTouchObject = inputObject
- self.moveTouchStartPosition = inputObject.Position
- self.moveTouchFirstChanged = true
- if FADE_IN_OUT_BACKGROUND then
- self:DoFadeInBackground()
- end
- return Enum.ContextActionResult.Pass
- end
- local function inputChanged(inputObject)
- if inputObject == self.moveTouchObject then
- if self.moveTouchFirstChanged then
- self.moveTouchFirstChanged = false
- local startPosVec2 = Vector2.new(
- inputObject.Position.X - self.thumbstickFrame.AbsolutePosition.X,
- inputObject.Position.Y - self.thumbstickFrame.AbsolutePosition.Y
- )
- self.startImage.Visible = true
- self.startImage.Position = UDim2.new(0, startPosVec2.X, 0, startPosVec2.Y)
- self.endImage.Visible = true
- self.endImage.Position = self.startImage.Position
- self:FadeThumbstick(true)
- self:MoveStick(inputObject.Position)
- end
- self.moveTouchLockedIn = true
- local direction = Vector2.new(
- inputObject.Position.x - self.moveTouchStartPosition.x,
- inputObject.Position.y - self.moveTouchStartPosition.y
- )
- if math.abs(direction.x) > 0 or math.abs(direction.y) > 0 then
- self:DoMove(direction)
- self:MoveStick(inputObject.Position)
- end
- return Enum.ContextActionResult.Sink
- end
- return Enum.ContextActionResult.Pass
- end
- local function inputEnded(inputObject)
- if inputObject == self.moveTouchObject then
- self:OnInputEnded()
- if self.moveTouchLockedIn then
- return Enum.ContextActionResult.Sink
- end
- end
- return Enum.ContextActionResult.Pass
- end
- local function handleInput(actionName, inputState, inputObject)
- if inputState == Enum.UserInputState.Begin then
- return inputBegan(inputObject)
- elseif inputState == Enum.UserInputState.Change then
- return inputChanged(inputObject)
- elseif inputState == Enum.UserInputState.End then
- return inputEnded(inputObject)
- elseif inputState == Enum.UserInputState.Cancel then
- self:OnInputEnded()
- end
- end
- ContextActionService:BindActionAtPriority(
- DYNAMIC_THUMBSTICK_ACTION_NAME,
- handleInput,
- false,
- DYNAMIC_THUMBSTICK_ACTION_PRIORITY,
- Enum.UserInputType.Touch)
- end
- function DynamicThumbstick:Create(parentFrame)
- if self.thumbstickFrame then
- self.thumbstickFrame:Destroy()
- self.thumbstickFrame = nil
- if self.onRenderSteppedConn then
- self.onRenderSteppedConn:Disconnect()
- self.onRenderSteppedConn = nil
- end
- end
- self.thumbstickSize = 45
- self.thumbstickRingSize = 20
- self.middleSize = 10
- self.middleSpacing = self.middleSize + 4
- self.radiusOfDeadZone = 2
- self.radiusOfMaxSpeed = 20
- local screenSize = parentFrame.AbsoluteSize
- local isBigScreen = math.min(screenSize.x, screenSize.y) > 500
- if isBigScreen then
- self.thumbstickSize = self.thumbstickSize * 2
- self.thumbstickRingSize = self.thumbstickRingSize * 2
- self.middleSize = self.middleSize * 2
- self.middleSpacing = self.middleSpacing * 2
- self.radiusOfDeadZone = self.radiusOfDeadZone * 2
- self.radiusOfMaxSpeed = self.radiusOfMaxSpeed * 2
- end
- local function layoutThumbstickFrame(portraitMode)
- if portraitMode then
- self.thumbstickFrame.Size = UDim2.new(1, 0, 0.4, 0)
- self.thumbstickFrame.Position = UDim2.new(0, 0, 0.6, 0)
- else
- self.thumbstickFrame.Size = UDim2.new(0.4, 0, 2/3, 0)
- self.thumbstickFrame.Position = UDim2.new(0, 0, 1/3, 0)
- end
- end
- self.thumbstickFrame = Instance.new("Frame")
- self.thumbstickFrame.BorderSizePixel = 0
- self.thumbstickFrame.Name = "DynamicThumbstickFrame"
- self.thumbstickFrame.Visible = false
- self.thumbstickFrame.BackgroundTransparency = 1.0
- self.thumbstickFrame.BackgroundColor3 = Color3.fromRGB(0, 0, 0)
- self.thumbstickFrame.Active = false
- layoutThumbstickFrame(false)
- self.startImage = Instance.new("ImageLabel")
- self.startImage.Name = "ThumbstickStart"
- self.startImage.Visible = true
- self.startImage.BackgroundTransparency = 1
- self.startImage.Image = TOUCH_CONTROLS_SHEET
- self.startImage.ImageRectOffset = Vector2.new(1,1)
- self.startImage.ImageRectSize = Vector2.new(144, 144)
- self.startImage.ImageColor3 = Color3.new(0, 0, 0)
- self.startImage.AnchorPoint = Vector2.new(0.5, 0.5)
- self.startImage.Position = UDim2.new(0, self.thumbstickRingSize * 3.3, 1, -self.thumbstickRingSize * 2.8)
- self.startImage.Size = UDim2.new(0, self.thumbstickRingSize * 3.7, 0, self.thumbstickRingSize * 3.7)
- self.startImage.ZIndex = 10
- self.startImage.Parent = self.thumbstickFrame
- self.endImage = Instance.new("ImageLabel")
- self.endImage.Name = "ThumbstickEnd"
- self.endImage.Visible = true
- self.endImage.BackgroundTransparency = 1
- self.endImage.Image = TOUCH_CONTROLS_SHEET
- self.endImage.ImageRectOffset = Vector2.new(1,1)
- self.endImage.ImageRectSize = Vector2.new(144, 144)
- self.endImage.AnchorPoint = Vector2.new(0.5, 0.5)
- self.endImage.Position = self.startImage.Position
- self.endImage.Size = UDim2.new(0, self.thumbstickSize * 0.8, 0, self.thumbstickSize * 0.8)
- self.endImage.ZIndex = 10
- self.endImage.Parent = self.thumbstickFrame
- for i = 1, NUM_MIDDLE_IMAGES do
- self.middleImages[i] = Instance.new("ImageLabel")
- self.middleImages[i].Name = "ThumbstickMiddle"
- self.middleImages[i].Visible = false
- self.middleImages[i].BackgroundTransparency = 1
- self.middleImages[i].Image = TOUCH_CONTROLS_SHEET
- self.middleImages[i].ImageRectOffset = Vector2.new(1,1)
- self.middleImages[i].ImageRectSize = Vector2.new(144, 144)
- self.middleImages[i].ImageTransparency = MIDDLE_TRANSPARENCIES[i]
- self.middleImages[i].AnchorPoint = Vector2.new(0.5, 0.5)
- self.middleImages[i].ZIndex = 9
- self.middleImages[i].Parent = self.thumbstickFrame
- end
- local CameraChangedConn = nil
- local function onCurrentCameraChanged()
- if CameraChangedConn then
- CameraChangedConn:Disconnect()
- CameraChangedConn = nil
- end
- local newCamera = workspace.CurrentCamera
- if newCamera then
- local function onViewportSizeChanged()
- local size = newCamera.ViewportSize
- local portraitMode = size.X < size.Y
- layoutThumbstickFrame(portraitMode)
- end
- CameraChangedConn = newCamera:GetPropertyChangedSignal("ViewportSize"):Connect(onViewportSizeChanged)
- onViewportSizeChanged()
- end
- end
- workspace:GetPropertyChangedSignal("CurrentCamera"):Connect(onCurrentCameraChanged)
- if workspace.CurrentCamera then
- onCurrentCameraChanged()
- end
- self.moveTouchStartPosition = nil
- self.startImageFadeTween = nil
- self.endImageFadeTween = nil
- self.middleImageFadeTweens = {}
- self.onRenderSteppedConn = RunService.RenderStepped:Connect(function()
- if self.tweenInAlphaStart ~= nil then
- local delta = tick() - self.tweenInAlphaStart
- local fadeInTime = (self.fadeInAndOutHalfDuration * 2 * self.fadeInAndOutBalance)
- self.thumbstickFrame.BackgroundTransparency = 1 - FADE_IN_OUT_MAX_ALPHA*math.min(delta/fadeInTime, 1)
- if delta > fadeInTime then
- self.tweenOutAlphaStart = tick()
- self.tweenInAlphaStart = nil
- end
- elseif self.tweenOutAlphaStart ~= nil then
- local delta = tick() - self.tweenOutAlphaStart
- local fadeOutTime = (self.fadeInAndOutHalfDuration * 2) - (self.fadeInAndOutHalfDuration * 2 * self.fadeInAndOutBalance)
- self.thumbstickFrame.BackgroundTransparency = 1 - FADE_IN_OUT_MAX_ALPHA + FADE_IN_OUT_MAX_ALPHA*math.min(delta/fadeOutTime, 1)
- if delta > fadeOutTime then
- self.tweenOutAlphaStart = nil
- end
- end
- end)
- self.onTouchEndedConn = UserInputService.TouchEnded:connect(function(inputObject)
- if inputObject == self.moveTouchObject then
- self:OnInputEnded()
- end
- end)
- GuiService.MenuOpened:connect(function()
- if self.moveTouchObject then
- self:OnInputEnded()
- end
- end)
- local playerGui = LocalPlayer:FindFirstChildOfClass("PlayerGui")
- while not playerGui do
- LocalPlayer.ChildAdded:wait()
- playerGui = LocalPlayer:FindFirstChildOfClass("PlayerGui")
- end
- local playerGuiChangedConn = nil
- local originalScreenOrientationWasLandscape = playerGui.CurrentScreenOrientation == Enum.ScreenOrientation.LandscapeLeft or
- playerGui.CurrentScreenOrientation == Enum.ScreenOrientation.LandscapeRight
- local function longShowBackground()
- self.fadeInAndOutHalfDuration = 2.5
- self.fadeInAndOutBalance = 0.05
- self.tweenInAlphaStart = tick()
- end
- playerGuiChangedConn = playerGui:GetPropertyChangedSignal("CurrentScreenOrientation"):Connect(function()
- if (originalScreenOrientationWasLandscape and playerGui.CurrentScreenOrientation == Enum.ScreenOrientation.Portrait) or
- (not originalScreenOrientationWasLandscape and playerGui.CurrentScreenOrientation ~= Enum.ScreenOrientation.Portrait) then
- playerGuiChangedConn:disconnect()
- longShowBackground()
- if originalScreenOrientationWasLandscape then
- self.hasFadedBackgroundInPortrait = true
- else
- self.hasFadedBackgroundInLandscape = true
- end
- end
- end)
- self.thumbstickFrame.Parent = parentFrame
- if game:IsLoaded() then
- longShowBackground()
- else
- coroutine.wrap(function()
- game.Loaded:Wait()
- longShowBackground()
- end)()
- end
- end
- return DynamicThumbstick
- end
- function _Gamepad()
- local UserInputService = game:GetService("UserInputService")
- local ContextActionService = game:GetService("ContextActionService")
- --[[ Constants ]]--
- local ZERO_VECTOR3 = Vector3.new(0,0,0)
- local NONE = Enum.UserInputType.None
- local thumbstickDeadzone = 0.2
- --[[ The Module ]]--
- local BaseCharacterController = _BaseCharacterController()
- local Gamepad = setmetatable({}, BaseCharacterController)
- Gamepad.__index = Gamepad
- function Gamepad.new(CONTROL_ACTION_PRIORITY)
- local self = setmetatable(BaseCharacterController.new(), Gamepad)
- self.CONTROL_ACTION_PRIORITY = CONTROL_ACTION_PRIORITY
- self.forwardValue = 0
- self.backwardValue = 0
- self.leftValue = 0
- self.rightValue = 0
- self.activeGamepad = NONE -- Enum.UserInputType.Gamepad1, 2, 3...
- self.gamepadConnectedConn = nil
- self.gamepadDisconnectedConn = nil
- return self
- end
- function Gamepad:Enable(enable)
- if not UserInputService.GamepadEnabled then
- return false
- end
- if enable == self.enabled then
- -- Module is already in the state being requested. True is returned here since the module will be in the state
- -- expected by the code that follows the Enable() call. This makes more sense than returning false to indicate
- -- no action was necessary. False indicates failure to be in requested/expected state.
- return true
- end
- self.forwardValue = 0
- self.backwardValue = 0
- self.leftValue = 0
- self.rightValue = 0
- self.moveVector = ZERO_VECTOR3
- self.isJumping = false
- if enable then
- self.activeGamepad = self:GetHighestPriorityGamepad()
- if self.activeGamepad ~= NONE then
- self:BindContextActions()
- self:ConnectGamepadConnectionListeners()
- else
- -- No connected gamepads, failure to enable
- return false
- end
- else
- self:UnbindContextActions()
- self:DisconnectGamepadConnectionListeners()
- self.activeGamepad = NONE
- end
- self.enabled = enable
- return true
- end
- -- This function selects the lowest number gamepad from the currently-connected gamepad
- -- and sets it as the active gamepad
- function Gamepad:GetHighestPriorityGamepad()
- local connectedGamepads = UserInputService:GetConnectedGamepads()
- local bestGamepad = NONE -- Note that this value is higher than all valid gamepad values
- for _, gamepad in pairs(connectedGamepads) do
- if gamepad.Value < bestGamepad.Value then
- bestGamepad = gamepad
- end
- end
- return bestGamepad
- end
- function Gamepad:BindContextActions()
- if self.activeGamepad == NONE then
- -- There must be an active gamepad to set up bindings
- return false
- end
- local handleJumpAction = function(actionName, inputState, inputObject)
- self.isJumping = (inputState == Enum.UserInputState.Begin)
- return Enum.ContextActionResult.Sink
- end
- local handleThumbstickInput = function(actionName, inputState, inputObject)
- if inputState == Enum.UserInputState.Cancel then
- self.moveVector = ZERO_VECTOR3
- return Enum.ContextActionResult.Sink
- end
- if self.activeGamepad ~= inputObject.UserInputType then
- return Enum.ContextActionResult.Pass
- end
- if inputObject.KeyCode ~= Enum.KeyCode.Thumbstick1 then return end
- if inputObject.Position.magnitude > thumbstickDeadzone then
- self.moveVector = Vector3.new(inputObject.Position.X, 0, -inputObject.Position.Y)
- else
- self.moveVector = ZERO_VECTOR3
- end
- return Enum.ContextActionResult.Sink
- end
- ContextActionService:BindActivate(self.activeGamepad, Enum.KeyCode.ButtonR2)
- ContextActionService:BindActionAtPriority("jumpAction", handleJumpAction, false,
- self.CONTROL_ACTION_PRIORITY, Enum.KeyCode.ButtonA)
- ContextActionService:BindActionAtPriority("moveThumbstick", handleThumbstickInput, false,
- self.CONTROL_ACTION_PRIORITY, Enum.KeyCode.Thumbstick1)
- return true
- end
- function Gamepad:UnbindContextActions()
- if self.activeGamepad ~= NONE then
- ContextActionService:UnbindActivate(self.activeGamepad, Enum.KeyCode.ButtonR2)
- end
- ContextActionService:UnbindAction("moveThumbstick")
- ContextActionService:UnbindAction("jumpAction")
- end
- function Gamepad:OnNewGamepadConnected()
- -- A new gamepad has been connected.
- local bestGamepad = self:GetHighestPriorityGamepad()
- if bestGamepad == self.activeGamepad then
- -- A new gamepad was connected, but our active gamepad is not changing
- return
- end
- if bestGamepad == NONE then
- -- There should be an active gamepad when GamepadConnected fires, so this should not
- -- normally be hit. If there is no active gamepad, unbind actions but leave
- -- the module enabled and continue to listen for a new gamepad connection.
- warn("Gamepad:OnNewGamepadConnected found no connected gamepads")
- self:UnbindContextActions()
- return
- end
- if self.activeGamepad ~= NONE then
- -- Switching from one active gamepad to another
- self:UnbindContextActions()
- end
- self.activeGamepad = bestGamepad
- self:BindContextActions()
- end
- function Gamepad:OnCurrentGamepadDisconnected()
- if self.activeGamepad ~= NONE then
- ContextActionService:UnbindActivate(self.activeGamepad, Enum.KeyCode.ButtonR2)
- end
- local bestGamepad = self:GetHighestPriorityGamepad()
- if self.activeGamepad ~= NONE and bestGamepad == self.activeGamepad then
- warn("Gamepad:OnCurrentGamepadDisconnected found the supposedly disconnected gamepad in connectedGamepads.")
- self:UnbindContextActions()
- self.activeGamepad = NONE
- return
- end
- if bestGamepad == NONE then
- -- No active gamepad, unbinding actions but leaving gamepad connection listener active
- self:UnbindContextActions()
- self.activeGamepad = NONE
- else
- -- Set new gamepad as active and bind to tool activation
- self.activeGamepad = bestGamepad
- ContextActionService:BindActivate(self.activeGamepad, Enum.KeyCode.ButtonR2)
- end
- end
- function Gamepad:ConnectGamepadConnectionListeners()
- self.gamepadConnectedConn = UserInputService.GamepadConnected:Connect(function(gamepadEnum)
- self:OnNewGamepadConnected()
- end)
- self.gamepadDisconnectedConn = UserInputService.GamepadDisconnected:Connect(function(gamepadEnum)
- if self.activeGamepad == gamepadEnum then
- self:OnCurrentGamepadDisconnected()
- end
- end)
- end
- function Gamepad:DisconnectGamepadConnectionListeners()
- if self.gamepadConnectedConn then
- self.gamepadConnectedConn:Disconnect()
- self.gamepadConnectedConn = nil
- end
- if self.gamepadDisconnectedConn then
- self.gamepadDisconnectedConn:Disconnect()
- self.gamepadDisconnectedConn = nil
- end
- end
- return Gamepad
- end
- function _Keyboard()
- --[[ Roblox Services ]]--
- local UserInputService = game:GetService("UserInputService")
- local ContextActionService = game:GetService("ContextActionService")
- --[[ Constants ]]--
- local ZERO_VECTOR3 = Vector3.new(0,0,0)
- --[[ The Module ]]--
- local BaseCharacterController = _BaseCharacterController()
- local Keyboard = setmetatable({}, BaseCharacterController)
- Keyboard.__index = Keyboard
- function Keyboard.new(CONTROL_ACTION_PRIORITY)
- local self = setmetatable(BaseCharacterController.new(), Keyboard)
- self.CONTROL_ACTION_PRIORITY = CONTROL_ACTION_PRIORITY
- self.textFocusReleasedConn = nil
- self.textFocusGainedConn = nil
- self.windowFocusReleasedConn = nil
- self.forwardValue = 0
- self.backwardValue = 0
- self.leftValue = 0
- self.rightValue = 0
- self.jumpEnabled = true
- return self
- end
- function Keyboard:Enable(enable)
- if not UserInputService.KeyboardEnabled then
- return false
- end
- if enable == self.enabled then
- -- Module is already in the state being requested. True is returned here since the module will be in the state
- -- expected by the code that follows the Enable() call. This makes more sense than returning false to indicate
- -- no action was necessary. False indicates failure to be in requested/expected state.
- return true
- end
- self.forwardValue = 0
- self.backwardValue = 0
- self.leftValue = 0
- self.rightValue = 0
- self.moveVector = ZERO_VECTOR3
- self.jumpRequested = false
- self:UpdateJump()
- if enable then
- self:BindContextActions()
- self:ConnectFocusEventListeners()
- else
- self:UnbindContextActions()
- self:DisconnectFocusEventListeners()
- end
- self.enabled = enable
- return true
- end
- function Keyboard:UpdateMovement(inputState)
- if inputState == Enum.UserInputState.Cancel then
- self.moveVector = ZERO_VECTOR3
- else
- self.moveVector = Vector3.new(self.leftValue + self.rightValue, 0, self.forwardValue + self.backwardValue)
- end
- end
- function Keyboard:UpdateJump()
- self.isJumping = self.jumpRequested
- end
- function Keyboard:BindContextActions()
- -- Note: In the previous version of this code, the movement values were not zeroed-out on UserInputState. Cancel, now they are,
- -- which fixes them from getting stuck on.
- -- We return ContextActionResult.Pass here for legacy reasons.
- -- Many games rely on gameProcessedEvent being false on UserInputService.InputBegan for these control actions.
- local handleMoveForward = function(actionName, inputState, inputObject)
- self.forwardValue = (inputState == Enum.UserInputState.Begin) and -1 or 0
- self:UpdateMovement(inputState)
- return Enum.ContextActionResult.Pass
- end
- local handleMoveBackward = function(actionName, inputState, inputObject)
- self.backwardValue = (inputState == Enum.UserInputState.Begin) and 1 or 0
- self:UpdateMovement(inputState)
- return Enum.ContextActionResult.Pass
- end
- local handleMoveLeft = function(actionName, inputState, inputObject)
- self.leftValue = (inputState == Enum.UserInputState.Begin) and -1 or 0
- self:UpdateMovement(inputState)
- return Enum.ContextActionResult.Pass
- end
- local handleMoveRight = function(actionName, inputState, inputObject)
- self.rightValue = (inputState == Enum.UserInputState.Begin) and 1 or 0
- self:UpdateMovement(inputState)
- return Enum.ContextActionResult.Pass
- end
- local handleJumpAction = function(actionName, inputState, inputObject)
- self.jumpRequested = self.jumpEnabled and (inputState == Enum.UserInputState.Begin)
- self:UpdateJump()
- return Enum.ContextActionResult.Pass
- end
- -- TODO: Revert to KeyCode bindings so that in the future the abstraction layer from actual keys to
- -- movement direction is done in Lua
- ContextActionService:BindActionAtPriority("moveForwardAction", handleMoveForward, false,
- self.CONTROL_ACTION_PRIORITY, Enum.PlayerActions.CharacterForward)
- ContextActionService:BindActionAtPriority("moveBackwardAction", handleMoveBackward, false,
- self.CONTROL_ACTION_PRIORITY, Enum.PlayerActions.CharacterBackward)
- ContextActionService:BindActionAtPriority("moveLeftAction", handleMoveLeft, false,
- self.CONTROL_ACTION_PRIORITY, Enum.PlayerActions.CharacterLeft)
- ContextActionService:BindActionAtPriority("moveRightAction", handleMoveRight, false,
- self.CONTROL_ACTION_PRIORITY, Enum.PlayerActions.CharacterRight)
- ContextActionService:BindActionAtPriority("jumpAction", handleJumpAction, false,
- self.CONTROL_ACTION_PRIORITY, Enum.PlayerActions.CharacterJump)
- end
- function Keyboard:UnbindContextActions()
- ContextActionService:UnbindAction("moveForwardAction")
- ContextActionService:UnbindAction("moveBackwardAction")
- ContextActionService:UnbindAction("moveLeftAction")
- ContextActionService:UnbindAction("moveRightAction")
- ContextActionService:UnbindAction("jumpAction")
- end
- function Keyboard:ConnectFocusEventListeners()
- local function onFocusReleased()
- self.moveVector = ZERO_VECTOR3
- self.forwardValue = 0
- self.backwardValue = 0
- self.leftValue = 0
- self.rightValue = 0
- self.jumpRequested = false
- self:UpdateJump()
- end
- local function onTextFocusGained(textboxFocused)
- self.jumpRequested = false
- self:UpdateJump()
- end
- self.textFocusReleasedConn = UserInputService.TextBoxFocusReleased:Connect(onFocusReleased)
- self.textFocusGainedConn = UserInputService.TextBoxFocused:Connect(onTextFocusGained)
- self.windowFocusReleasedConn = UserInputService.WindowFocused:Connect(onFocusReleased)
- end
- function Keyboard:DisconnectFocusEventListeners()
- if self.textFocusReleasedCon then
- self.textFocusReleasedCon:Disconnect()
- self.textFocusReleasedCon = nil
- end
- if self.textFocusGainedConn then
- self.textFocusGainedConn:Disconnect()
- self.textFocusGainedConn = nil
- end
- if self.windowFocusReleasedConn then
- self.windowFocusReleasedConn:Disconnect()
- self.windowFocusReleasedConn = nil
- end
- end
- return Keyboard
- end
- function _ControlModule()
- local ControlModule = {}
- ControlModule.__index = ControlModule
- --[[ Roblox Services ]]--
- local Players = game:GetService("Players")
- local RunService = game:GetService("RunService")
- local UserInputService = game:GetService("UserInputService")
- local Workspace = game:GetService("Workspace")
- local UserGameSettings = UserSettings():GetService("UserGameSettings")
- -- Roblox User Input Control Modules - each returns a new() constructor function used to create controllers as needed
- local Keyboard = _Keyboard()
- local Gamepad = _Gamepad()
- local DynamicThumbstick = _DynamicThumbstick()
- local FFlagUserMakeThumbstickDynamic do
- local success, value = pcall(function()
- return UserSettings():IsUserFeatureEnabled("UserMakeThumbstickDynamic")
- end)
- FFlagUserMakeThumbstickDynamic = success and value
- end
- local TouchThumbstick = FFlagUserMakeThumbstickDynamic and DynamicThumbstick or _TouchThumbstick()
- -- These controllers handle only walk/run movement, jumping is handled by the
- -- TouchJump controller if any of these are active
- local ClickToMove = _ClickToMoveController()
- local TouchJump = _TouchJump()
- local VehicleController = _VehicleController()
- local CONTROL_ACTION_PRIORITY = Enum.ContextActionPriority.Default.Value
- -- Mapping from movement mode and lastInputType enum values to control modules to avoid huge if elseif switching
- local movementEnumToModuleMap = {
- [Enum.TouchMovementMode.DPad] = DynamicThumbstick,
- [Enum.DevTouchMovementMode.DPad] = DynamicThumbstick,
- [Enum.TouchMovementMode.Thumbpad] = DynamicThumbstick,
- [Enum.DevTouchMovementMode.Thumbpad] = DynamicThumbstick,
- [Enum.TouchMovementMode.Thumbstick] = TouchThumbstick,
- [Enum.DevTouchMovementMode.Thumbstick] = TouchThumbstick,
- [Enum.TouchMovementMode.DynamicThumbstick] = DynamicThumbstick,
- [Enum.DevTouchMovementMode.DynamicThumbstick] = DynamicThumbstick,
- [Enum.TouchMovementMode.ClickToMove] = ClickToMove,
- [Enum.DevTouchMovementMode.ClickToMove] = ClickToMove,
- -- Current default
- [Enum.TouchMovementMode.Default] = DynamicThumbstick,
- [Enum.ComputerMovementMode.Default] = Keyboard,
- [Enum.ComputerMovementMode.KeyboardMouse] = Keyboard,
- [Enum.DevComputerMovementMode.KeyboardMouse] = Keyboard,
- [Enum.DevComputerMovementMode.Scriptable] = nil,
- [Enum.ComputerMovementMode.ClickToMove] = ClickToMove,
- [Enum.DevComputerMovementMode.ClickToMove] = ClickToMove,
- }
- -- Keyboard controller is really keyboard and mouse controller
- local computerInputTypeToModuleMap = {
- [Enum.UserInputType.Keyboard] = Keyboard,
- [Enum.UserInputType.MouseButton1] = Keyboard,
- [Enum.UserInputType.MouseButton2] = Keyboard,
- [Enum.UserInputType.MouseButton3] = Keyboard,
- [Enum.UserInputType.MouseWheel] = Keyboard,
- [Enum.UserInputType.MouseMovement] = Keyboard,
- [Enum.UserInputType.Gamepad1] = Gamepad,
- [Enum.UserInputType.Gamepad2] = Gamepad,
- [Enum.UserInputType.Gamepad3] = Gamepad,
- [Enum.UserInputType.Gamepad4] = Gamepad,
- }
- local lastInputType
- function ControlModule.new()
- local self = setmetatable({},ControlModule)
- -- The Modules above are used to construct controller instances as-needed, and this
- -- table is a map from Module to the instance created from it
- self.controllers = {}
- self.activeControlModule = nil -- Used to prevent unnecessarily expensive checks on each input event
- self.activeController = nil
- self.touchJumpController = nil
- self.moveFunction = Players.LocalPlayer.Move
- self.humanoid = nil
- self.lastInputType = Enum.UserInputType.None
- -- For Roblox self.vehicleController
- self.humanoidSeatedConn = nil
- self.vehicleController = nil
- self.touchControlFrame = nil
- self.vehicleController = VehicleController.new(CONTROL_ACTION_PRIORITY)
- Players.LocalPlayer.CharacterAdded:Connect(function(char) self:OnCharacterAdded(char) end)
- Players.LocalPlayer.CharacterRemoving:Connect(function(char) self:OnCharacterRemoving(char) end)
- if Players.LocalPlayer.Character then
- self:OnCharacterAdded(Players.LocalPlayer.Character)
- end
- RunService:BindToRenderStep("ControlScriptRenderstep", Enum.RenderPriority.Input.Value, function(dt)
- self:OnRenderStepped(dt)
- end)
- UserInputService.LastInputTypeChanged:Connect(function(newLastInputType)
- self:OnLastInputTypeChanged(newLastInputType)
- end)
- UserGameSettings:GetPropertyChangedSignal("TouchMovementMode"):Connect(function()
- self:OnTouchMovementModeChange()
- end)
- Players.LocalPlayer:GetPropertyChangedSignal("DevTouchMovementMode"):Connect(function()
- self:OnTouchMovementModeChange()
- end)
- UserGameSettings:GetPropertyChangedSignal("ComputerMovementMode"):Connect(function()
- self:OnComputerMovementModeChange()
- end)
- Players.LocalPlayer:GetPropertyChangedSignal("DevComputerMovementMode"):Connect(function()
- self:OnComputerMovementModeChange()
- end)
- --[[ Touch Device UI ]]--
- self.playerGui = nil
- self.touchGui = nil
- self.playerGuiAddedConn = nil
- if UserInputService.TouchEnabled then
- self.playerGui = Players.LocalPlayer:FindFirstChildOfClass("PlayerGui")
- if self.playerGui then
- self:CreateTouchGuiContainer()
- self:OnLastInputTypeChanged(UserInputService:GetLastInputType())
- else
- self.playerGuiAddedConn = Players.LocalPlayer.ChildAdded:Connect(function(child)
- if child:IsA("PlayerGui") then
- self.playerGui = child
- self:CreateTouchGuiContainer()
- self.playerGuiAddedConn:Disconnect()
- self.playerGuiAddedConn = nil
- self:OnLastInputTypeChanged(UserInputService:GetLastInputType())
- end
- end)
- end
- else
- self:OnLastInputTypeChanged(UserInputService:GetLastInputType())
- end
- return self
- end
- -- Convenience function so that calling code does not have to first get the activeController
- -- and then call GetMoveVector on it. When there is no active controller, this function returns
- -- nil so that this case can be distinguished from no current movement (which returns zero vector).
- function ControlModule:GetMoveVector()
- if self.activeController then
- return self.activeController:GetMoveVector()
- end
- return Vector3.new(0,0,0)
- end
- function ControlModule:GetActiveController()
- return self.activeController
- end
- function ControlModule:EnableActiveControlModule()
- if self.activeControlModule == ClickToMove then
- -- For ClickToMove, when it is the player's choice, we also enable the full keyboard controls.
- -- When the developer is forcing click to move, the most keyboard controls (WASD) are not available, only jump.
- self.activeController:Enable(
- true,
- Players.LocalPlayer.DevComputerMovementMode == Enum.DevComputerMovementMode.UserChoice,
- self.touchJumpController
- )
- elseif self.touchControlFrame then
- self.activeController:Enable(true, self.touchControlFrame)
- else
- self.activeController:Enable(true)
- end
- end
- function ControlModule:Enable(enable)
- if not self.activeController then
- return
- end
- if enable == nil then
- enable = true
- end
- if enable then
- self:EnableActiveControlModule()
- else
- self:Disable()
- end
- end
- -- For those who prefer distinct functions
- function ControlModule:Disable()
- if self.activeController then
- self.activeController:Enable(false)
- if self.moveFunction then
- self.moveFunction(Players.LocalPlayer, Vector3.new(0,0,0), true)
- end
- end
- end
- -- Returns module (possibly nil) and success code to differentiate returning nil due to error vs Scriptable
- function ControlModule:SelectComputerMovementModule()
- if not (UserInputService.KeyboardEnabled or UserInputService.GamepadEnabled) then
- return nil, false
- end
- local computerModule
- local DevMovementMode = Players.LocalPlayer.DevComputerMovementMode
- if DevMovementMode == Enum.DevComputerMovementMode.UserChoice then
- computerModule = computerInputTypeToModuleMap[lastInputType]
- if UserGameSettings.ComputerMovementMode == Enum.ComputerMovementMode.ClickToMove and computerModule == Keyboard then
- -- User has ClickToMove set in Settings, prefer ClickToMove controller for keyboard and mouse lastInputTypes
- computerModule = ClickToMove
- end
- else
- -- Developer has selected a mode that must be used.
- computerModule = movementEnumToModuleMap[DevMovementMode]
- -- computerModule is expected to be nil here only when developer has selected Scriptable
- if (not computerModule) and DevMovementMode ~= Enum.DevComputerMovementMode.Scriptable then
- warn("No character control module is associated with DevComputerMovementMode ", DevMovementMode)
- end
- end
- if computerModule then
- return computerModule, true
- elseif DevMovementMode == Enum.DevComputerMovementMode.Scriptable then
- -- Special case where nil is returned and we actually want to set self.activeController to nil for Scriptable
- return nil, true
- else
- -- This case is for when computerModule is nil because of an error and no suitable control module could
- -- be found.
- return nil, false
- end
- end
- -- Choose current Touch control module based on settings (user, dev)
- -- Returns module (possibly nil) and success code to differentiate returning nil due to error vs Scriptable
- function ControlModule:SelectTouchModule()
- if not UserInputService.TouchEnabled then
- return nil, false
- end
- local touchModule
- local DevMovementMode = Players.LocalPlayer.DevTouchMovementMode
- if DevMovementMode == Enum.DevTouchMovementMode.UserChoice then
- touchModule = movementEnumToModuleMap[UserGameSettings.TouchMovementMode]
- elseif DevMovementMode == Enum.DevTouchMovementMode.Scriptable then
- return nil, true
- else
- touchModule = movementEnumToModuleMap[DevMovementMode]
- end
- return touchModule, true
- end
- local function calculateRawMoveVector(humanoid, cameraRelativeMoveVector)
- local camera = Workspace.CurrentCamera
- if not camera then
- return cameraRelativeMoveVector
- end
- if humanoid:GetState() == Enum.HumanoidStateType.Swimming then
- return camera.CFrame:VectorToWorldSpace(cameraRelativeMoveVector)
- end
- local c, s
- local _, _, _, R00, R01, R02, _, _, R12, _, _, R22 = camera.CFrame:GetComponents()
- if R12 < 1 and R12 > -1 then
- -- X and Z components from back vector.
- c = R22
- s = R02
- else
- -- In this case the camera is looking straight up or straight down.
- -- Use X components from right and up vectors.
- c = R00
- s = -R01*math.sign(R12)
- end
- local norm = math.sqrt(c*c + s*s)
- return Vector3.new(
- (c*cameraRelativeMoveVector.x + s*cameraRelativeMoveVector.z)/norm,
- 0,
- (c*cameraRelativeMoveVector.z - s*cameraRelativeMoveVector.x)/norm
- )
- end
- function ControlModule:OnRenderStepped(dt)
- if self.activeController and self.activeController.enabled and self.humanoid then
- -- Give the controller a chance to adjust its state
- self.activeController:OnRenderStepped(dt)
- -- Now retrieve info from the controller
- local moveVector = self.activeController:GetMoveVector()
- local cameraRelative = self.activeController:IsMoveVectorCameraRelative()
- local clickToMoveController = self:GetClickToMoveController()
- if self.activeController ~= clickToMoveController then
- if moveVector.magnitude > 0 then
- -- Clean up any developer started MoveTo path
- clickToMoveController:CleanupPath()
- else
- -- Get move vector for developer started MoveTo
- clickToMoveController:OnRenderStepped(dt)
- moveVector = clickToMoveController:GetMoveVector()
- cameraRelative = clickToMoveController:IsMoveVectorCameraRelative()
- end
- end
- -- Are we driving a vehicle ?
- local vehicleConsumedInput = false
- if self.vehicleController then
- moveVector, vehicleConsumedInput = self.vehicleController:Update(moveVector, cameraRelative, self.activeControlModule==Gamepad)
- end
- -- If not, move the player
- -- Verification of vehicleConsumedInput is commented out to preserve legacy behavior,
- -- in case some game relies on Humanoid.MoveDirection still being set while in a VehicleSeat
- --if not vehicleConsumedInput then
- if cameraRelative then
- moveVector = calculateRawMoveVector(self.humanoid, moveVector)
- end
- self.moveFunction(Players.LocalPlayer, moveVector, false)
- --end
- -- And make them jump if needed
- self.humanoid.Jump = self.activeController:GetIsJumping() or (self.touchJumpController and self.touchJumpController:GetIsJumping())
- end
- end
- function ControlModule:OnHumanoidSeated(active, currentSeatPart)
- if active then
- if currentSeatPart and currentSeatPart:IsA("VehicleSeat") then
- if not self.vehicleController then
- self.vehicleController = self.vehicleController.new(CONTROL_ACTION_PRIORITY)
- end
- self.vehicleController:Enable(true, currentSeatPart)
- end
- else
- if self.vehicleController then
- self.vehicleController:Enable(false, currentSeatPart)
- end
- end
- end
- function ControlModule:OnCharacterAdded(char)
- self.humanoid = char:FindFirstChildOfClass("Humanoid")
- while not self.humanoid do
- char.ChildAdded:wait()
- self.humanoid = char:FindFirstChildOfClass("Humanoid")
- end
- if self.touchGui then
- self.touchGui.Enabled = true
- end
- if self.humanoidSeatedConn then
- self.humanoidSeatedConn:Disconnect()
- self.humanoidSeatedConn = nil
- end
- self.humanoidSeatedConn = self.humanoid.Seated:Connect(function(active, currentSeatPart)
- self:OnHumanoidSeated(active, currentSeatPart)
- end)
- end
- function ControlModule:OnCharacterRemoving(char)
- self.humanoid = nil
- if self.touchGui then
- self.touchGui.Enabled = false
- end
- end
- -- Helper function to lazily instantiate a controller if it does not yet exist,
- -- disable the active controller if it is different from the on being switched to,
- -- and then enable the requested controller. The argument to this function must be
- -- a reference to one of the control modules, i.e. Keyboard, Gamepad, etc.
- function ControlModule:SwitchToController(controlModule)
- if not controlModule then
- if self.activeController then
- self.activeController:Enable(false)
- end
- self.activeController = nil
- self.activeControlModule = nil
- else
- if not self.controllers[controlModule] then
- self.controllers[controlModule] = controlModule.new(CONTROL_ACTION_PRIORITY)
- end
- if self.activeController ~= self.controllers[controlModule] then
- if self.activeController then
- self.activeController:Enable(false)
- end
- self.activeController = self.controllers[controlModule]
- self.activeControlModule = controlModule -- Only used to check if controller switch is necessary
- if self.touchControlFrame and (self.activeControlModule == ClickToMove
- or self.activeControlModule == TouchThumbstick
- or self.activeControlModule == DynamicThumbstick) then
- if not self.controllers[TouchJump] then
- self.controllers[TouchJump] = TouchJump.new()
- end
- self.touchJumpController = self.controllers[TouchJump]
- self.touchJumpController:Enable(true, self.touchControlFrame)
- else
- if self.touchJumpController then
- self.touchJumpController:Enable(false)
- end
- end
- self:EnableActiveControlModule()
- end
- end
- end
- function ControlModule:OnLastInputTypeChanged(newLastInputType)
- if lastInputType == newLastInputType then
- warn("LastInputType Change listener called with current type.")
- end
- lastInputType = newLastInputType
- if lastInputType == Enum.UserInputType.Touch then
- -- TODO: Check if touch module already active
- local touchModule, success = self:SelectTouchModule()
- if success then
- while not self.touchControlFrame do
- wait()
- end
- self:SwitchToController(touchModule)
- end
- elseif computerInputTypeToModuleMap[lastInputType] ~= nil then
- local computerModule = self:SelectComputerMovementModule()
- if computerModule then
- self:SwitchToController(computerModule)
- end
- end
- end
- -- Called when any relevant values of GameSettings or LocalPlayer change, forcing re-evalulation of
- -- current control scheme
- function ControlModule:OnComputerMovementModeChange()
- local controlModule, success = self:SelectComputerMovementModule()
- if success then
- self:SwitchToController(controlModule)
- end
- end
- function ControlModule:OnTouchMovementModeChange()
- local touchModule, success = self:SelectTouchModule()
- if success then
- while not self.touchControlFrame do
- wait()
- end
- self:SwitchToController(touchModule)
- end
- end
- function ControlModule:CreateTouchGuiContainer()
- if self.touchGui then self.touchGui:Destroy() end
- -- Container for all touch device guis
- self.touchGui = Instance.new("ScreenGui")
- self.touchGui.Name = "TouchGui"
- self.touchGui.ResetOnSpawn = false
- self.touchGui.ZIndexBehavior = Enum.ZIndexBehavior.Sibling
- self.touchGui.Enabled = self.humanoid ~= nil
- self.touchControlFrame = Instance.new("Frame")
- self.touchControlFrame.Name = "TouchControlFrame"
- self.touchControlFrame.Size = UDim2.new(1, 0, 1, 0)
- self.touchControlFrame.BackgroundTransparency = 1
- self.touchControlFrame.Parent = self.touchGui
- self.touchGui.Parent = self.playerGui
- end
- function ControlModule:GetClickToMoveController()
- if not self.controllers[ClickToMove] then
- self.controllers[ClickToMove] = ClickToMove.new(CONTROL_ACTION_PRIORITY)
- end
- return self.controllers[ClickToMove]
- end
- function ControlModule:IsJumping()
- if self.activeController then
- return self.activeController:GetIsJumping() or (self.touchJumpController and self.touchJumpController:GetIsJumping())
- end
- return false
- end
- return ControlModule.new()
- end
- function _PlayerModule()
- local PlayerModule = {}
- PlayerModule.__index = PlayerModule
- function PlayerModule.new()
- local self = setmetatable({},PlayerModule)
- self.cameras = _CameraModule()
- self.controls = _ControlModule()
- return self
- end
- function PlayerModule:GetCameras()
- return self.cameras
- end
- function PlayerModule:GetControls()
- return self.controls
- end
- function PlayerModule:GetClickToMoveController()
- return self.controls:GetClickToMoveController()
- end
- return PlayerModule.new()
- end
- function _sounds()
- local SetState = Instance.new("BindableEvent",script)
- local Players = game:GetService("Players")
- local RunService = game:GetService("RunService")
- local SOUND_DATA = {
- Climbing = {
- SoundId = "rbxasset://sounds/action_footsteps_plastic.mp3",
- Looped = true,
- },
- Died = {
- SoundId = "rbxasset://sounds/uuhhh.mp3",
- },
- FreeFalling = {
- SoundId = "rbxasset://sounds/action_falling.mp3",
- Looped = true,
- },
- GettingUp = {
- SoundId = "rbxasset://sounds/action_get_up.mp3",
- },
- Jumping = {
- SoundId = "rbxasset://sounds/action_jump.mp3",
- },
- Landing = {
- SoundId = "rbxasset://sounds/action_jump_land.mp3",
- },
- Running = {
- SoundId = "rbxasset://sounds/action_footsteps_plastic.mp3",
- Looped = true,
- Pitch = 1.85,
- },
- Splash = {
- SoundId = "rbxasset://sounds/impact_water.mp3",
- },
- Swimming = {
- SoundId = "rbxasset://sounds/action_swim.mp3",
- Looped = true,
- Pitch = 1.6,
- },
- }
- -- wait for the first of the passed signals to fire
- local function waitForFirst(...)
- local shunt = Instance.new("BindableEvent")
- local slots = {...}
- local function fire(...)
- for i = 1, #slots do
- slots[i]:Disconnect()
- end
- return shunt:Fire(...)
- end
- for i = 1, #slots do
- slots[i] = slots[i]:Connect(fire)
- end
- return shunt.Event:Wait()
- end
- -- map a value from one range to another
- local function map(x, inMin, inMax, outMin, outMax)
- return (x - inMin)*(outMax - outMin)/(inMax - inMin) + outMin
- end
- local function playSound(sound)
- sound.TimePosition = 0
- sound.Playing = true
- end
- local function stopSound(sound)
- sound.Playing = false
- sound.TimePosition = 0
- end
- local function shallowCopy(t)
- local out = {}
- for k, v in pairs(t) do
- out[k] = v
- end
- return out
- end
- local function initializeSoundSystem(player, humanoid, rootPart)
- local sounds = {}
- -- initialize sounds
- for name, props in pairs(SOUND_DATA) do
- local sound = Instance.new("Sound")
- sound.Name = name
- -- set default values
- sound.Archivable = false
- sound.EmitterSize = 5
- sound.MaxDistance = 150
- sound.Volume = 0.65
- for propName, propValue in pairs(props) do
- sound[propName] = propValue
- end
- sound.Parent = rootPart
- sounds[name] = sound
- end
- local playingLoopedSounds = {}
- local function stopPlayingLoopedSounds(except)
- for sound in pairs(shallowCopy(playingLoopedSounds)) do
- if sound ~= except then
- sound.Playing = false
- playingLoopedSounds[sound] = nil
- end
- end
- end
- -- state transition callbacks
- local stateTransitions = {
- [Enum.HumanoidStateType.FallingDown] = function()
- stopPlayingLoopedSounds()
- end,
- [Enum.HumanoidStateType.GettingUp] = function()
- stopPlayingLoopedSounds()
- playSound(sounds.GettingUp)
- end,
- [Enum.HumanoidStateType.Jumping] = function()
- stopPlayingLoopedSounds()
- playSound(sounds.Jumping)
- end,
- [Enum.HumanoidStateType.Swimming] = function()
- local verticalSpeed = math.abs(rootPart.Velocity.Y)
- if verticalSpeed > 0.1 then
- sounds.Splash.Volume = math.clamp(map(verticalSpeed, 100, 350, 0.28, 1), 0, 1)
- playSound(sounds.Splash)
- end
- stopPlayingLoopedSounds(sounds.Swimming)
- sounds.Swimming.Playing = true
- playingLoopedSounds[sounds.Swimming] = true
- end,
- [Enum.HumanoidStateType.Freefall] = function()
- sounds.FreeFalling.Volume = 0
- stopPlayingLoopedSounds(sounds.FreeFalling)
- playingLoopedSounds[sounds.FreeFalling] = true
- end,
- [Enum.HumanoidStateType.Landed] = function()
- stopPlayingLoopedSounds()
- local verticalSpeed = math.abs(rootPart.Velocity.Y)
- if verticalSpeed > 75 then
- sounds.Landing.Volume = math.clamp(map(verticalSpeed, 50, 100, 0, 1), 0, 1)
- playSound(sounds.Landing)
- end
- end,
- [Enum.HumanoidStateType.Running] = function()
- stopPlayingLoopedSounds(sounds.Running)
- sounds.Running.Playing = true
- playingLoopedSounds[sounds.Running] = true
- end,
- [Enum.HumanoidStateType.Climbing] = function()
- local sound = sounds.Climbing
- if math.abs(rootPart.Velocity.Y) > 0.1 then
- sound.Playing = true
- stopPlayingLoopedSounds(sound)
- else
- stopPlayingLoopedSounds()
- end
- playingLoopedSounds[sound] = true
- end,
- [Enum.HumanoidStateType.Seated] = function()
- stopPlayingLoopedSounds()
- end,
- [Enum.HumanoidStateType.Dead] = function()
- stopPlayingLoopedSounds()
- playSound(sounds.Died)
- end,
- }
- -- updaters for looped sounds
- local loopedSoundUpdaters = {
- [sounds.Climbing] = function(dt, sound, vel)
- sound.Playing = vel.Magnitude > 0.1
- end,
- [sounds.FreeFalling] = function(dt, sound, vel)
- if vel.Magnitude > 75 then
- sound.Volume = math.clamp(sound.Volume + 0.9*dt, 0, 1)
- else
- sound.Volume = 0
- end
- end,
- [sounds.Running] = function(dt, sound, vel)
- sound.Playing = vel.Magnitude > 0.5 and humanoid.MoveDirection.Magnitude > 0.5
- end,
- }
- -- state substitutions to avoid duplicating entries in the state table
- local stateRemap = {
- [Enum.HumanoidStateType.RunningNoPhysics] = Enum.HumanoidStateType.Running,
- }
- local activeState = stateRemap[humanoid:GetState()] or humanoid:GetState()
- local activeConnections = {}
- local stateChangedConn = humanoid.StateChanged:Connect(function(_, state)
- state = stateRemap[state] or state
- if state ~= activeState then
- local transitionFunc = stateTransitions[state]
- if transitionFunc then
- transitionFunc()
- end
- activeState = state
- end
- end)
- local customStateChangedConn = SetState.Event:Connect(function(state)
- state = stateRemap[state] or state
- if state ~= activeState then
- local transitionFunc = stateTransitions[state]
- if transitionFunc then
- transitionFunc()
- end
- activeState = state
- end
- end)
- local steppedConn = RunService.Stepped:Connect(function(_, worldDt)
- -- update looped sounds on stepped
- for sound in pairs(playingLoopedSounds) do
- local updater = loopedSoundUpdaters[sound]
- if updater then
- updater(worldDt, sound, rootPart.Velocity)
- end
- end
- end)
- local humanoidAncestryChangedConn
- local rootPartAncestryChangedConn
- local characterAddedConn
- local function terminate()
- stateChangedConn:Disconnect()
- customStateChangedConn:Disconnect()
- steppedConn:Disconnect()
- humanoidAncestryChangedConn:Disconnect()
- rootPartAncestryChangedConn:Disconnect()
- characterAddedConn:Disconnect()
- end
- humanoidAncestryChangedConn = humanoid.AncestryChanged:Connect(function(_, parent)
- if not parent then
- terminate()
- end
- end)
- rootPartAncestryChangedConn = rootPart.AncestryChanged:Connect(function(_, parent)
- if not parent then
- terminate()
- end
- end)
- characterAddedConn = player.CharacterAdded:Connect(terminate)
- end
- local function playerAdded(player)
- local function characterAdded(character)
- -- Avoiding memory leaks in the face of Character/Humanoid/RootPart lifetime has a few complications:
- -- * character deparenting is a Remove instead of a Destroy, so signals are not cleaned up automatically.
- -- ** must use a waitForFirst on everything and listen for hierarchy changes.
- -- * the character might not be in the dm by the time CharacterAdded fires
- -- ** constantly check consistency with player.Character and abort if CharacterAdded is fired again
- -- * Humanoid may not exist immediately, and by the time it's inserted the character might be deparented.
- -- * RootPart probably won't exist immediately.
- -- ** by the time RootPart is inserted and Humanoid.RootPart is set, the character or the humanoid might be deparented.
- if not character.Parent then
- waitForFirst(character.AncestryChanged, player.CharacterAdded)
- end
- if player.Character ~= character or not character.Parent then
- return
- end
- local humanoid = character:FindFirstChildOfClass("Humanoid")
- while character:IsDescendantOf(game) and not humanoid do
- waitForFirst(character.ChildAdded, character.AncestryChanged, player.CharacterAdded)
- humanoid = character:FindFirstChildOfClass("Humanoid")
- end
- if player.Character ~= character or not character:IsDescendantOf(game) then
- return
- end
- -- must rely on HumanoidRootPart naming because Humanoid.RootPart does not fire changed signals
- local rootPart = character:FindFirstChild("HumanoidRootPart")
- while character:IsDescendantOf(game) and not rootPart do
- waitForFirst(character.ChildAdded, character.AncestryChanged, humanoid.AncestryChanged, player.CharacterAdded)
- rootPart = character:FindFirstChild("HumanoidRootPart")
- end
- if rootPart and humanoid:IsDescendantOf(game) and character:IsDescendantOf(game) and player.Character == character then
- initializeSoundSystem(player, humanoid, rootPart)
- end
- end
- if player.Character then
- characterAdded(player.Character)
- end
- player.CharacterAdded:Connect(characterAdded)
- end
- Players.PlayerAdded:Connect(playerAdded)
- for _, player in ipairs(Players:GetPlayers()) do
- playerAdded(player)
- end
- return SetState
- end
- function _StateTracker()
- local EPSILON = 0.1
- local SPEED = {
- ["onRunning"] = true,
- ["onClimbing"] = true
- }
- local INAIR = {
- ["onFreeFall"] = true,
- ["onJumping"] = true
- }
- local STATEMAP = {
- ["onRunning"] = Enum.HumanoidStateType.Running,
- ["onJumping"] = Enum.HumanoidStateType.Jumping,
- ["onFreeFall"] = Enum.HumanoidStateType.Freefall
- }
- local StateTracker = {}
- StateTracker.__index = StateTracker
- function StateTracker.new(humanoid, soundState)
- local self = setmetatable({}, StateTracker)
- self.Humanoid = humanoid
- self.HRP = humanoid.RootPart
- self.Speed = 0
- self.State = "onRunning"
- self.Jumped = false
- self.JumpTick = tick()
- self.SoundState = soundState
- self._ChangedEvent = Instance.new("BindableEvent")
- self.Changed = self._ChangedEvent.Event
- return self
- end
- function StateTracker:Destroy()
- self._ChangedEvent:Destroy()
- end
- function StateTracker:RequestedJump()
- self.Jumped = true
- self.JumpTick = tick()
- end
- function StateTracker:OnStep(gravityUp, grounded, isMoving)
- local cVelocity = self.HRP.Velocity
- local gVelocity = cVelocity:Dot(gravityUp)
- local oldState, oldSpeed = self.State, self.Speed
- local newState
- local newSpeed = cVelocity.Magnitude
- if (not grounded) then
- if (gVelocity > 0) then
- if (self.Jumped) then
- newState = "onJumping"
- else
- newState = "onFreeFall"
- end
- else
- if (self.Jumped) then
- self.Jumped = false
- end
- newState = "onFreeFall"
- end
- else
- if (self.Jumped and tick() - self.JumpTick > 0.1) then
- self.Jumped = false
- end
- newSpeed = (cVelocity - gVelocity*gravityUp).Magnitude
- newState = "onRunning"
- end
- newSpeed = isMoving and newSpeed or 0
- if (oldState ~= newState or (SPEED[newState] and math.abs(oldSpeed - newSpeed) > EPSILON)) then
- self.State = newState
- self.Speed = newSpeed
- self.SoundState:Fire(STATEMAP[newState])
- self._ChangedEvent:Fire(self.State, self.Speed)
- end
- end
- return StateTracker
- end
- function _InitObjects()
- local model = workspace:FindFirstChild("objects") or game:GetObjects("rbxassetid://5045408489")[1]
- local SPHERE = model:WaitForChild("Sphere")
- local FLOOR = model:WaitForChild("Floor")
- local VFORCE = model:WaitForChild("VectorForce")
- local BGYRO = model:WaitForChild("BodyGyro")
- local function initObjects(self)
- local hrp = self.HRP
- local humanoid = self.Humanoid
- local sphere = SPHERE:Clone()
- sphere.Parent = self.Character
- local floor = FLOOR:Clone()
- floor.Parent = self.Character
- local isR15 = (humanoid.RigType == Enum.HumanoidRigType.R15)
- local height = isR15 and (humanoid.HipHeight + 0.05) or 2
- local weld = Instance.new("Weld")
- weld.C0 = CFrame.new(0, -height, 0.1)
- weld.Part0 = hrp
- weld.Part1 = sphere
- weld.Parent = sphere
- local weld2 = Instance.new("Weld")
- weld2.C0 = CFrame.new(0, -(height + 1.5), 0)
- weld2.Part0 = hrp
- weld2.Part1 = floor
- weld2.Parent = floor
- local gyro = BGYRO:Clone()
- gyro.CFrame = hrp.CFrame
- gyro.Parent = hrp
- local vForce = VFORCE:Clone()
- vForce.Attachment0 = isR15 and hrp:WaitForChild("RootRigAttachment") or hrp:WaitForChild("RootAttachment")
- vForce.Parent = hrp
- return sphere, gyro, vForce, floor
- end
- return initObjects
- end
- local plr = game.Players.LocalPlayer
- local ms = plr:GetMouse()
- local char
- plr.CharacterAdded:Connect(function(c)
- char = c
- end)
- function _R6()
- function r6()
- local Figure = char
- local Torso = Figure:WaitForChild("Torso")
- local RightShoulder = Torso:WaitForChild("Right Shoulder")
- local LeftShoulder = Torso:WaitForChild("Left Shoulder")
- local RightHip = Torso:WaitForChild("Right Hip")
- local LeftHip = Torso:WaitForChild("Left Hip")
- local Neck = Torso:WaitForChild("Neck")
- local Humanoid = Figure:WaitForChild("Humanoid")
- local pose = "Standing"
- local currentAnim = ""
- local currentAnimInstance = nil
- local currentAnimTrack = nil
- local currentAnimKeyframeHandler = nil
- local currentAnimSpeed = 1.0
- local animTable = {}
- local animNames = {
- idle = {
- { id = "http://www.roblox.com/asset/?id=180435571", weight = 9 },
- { id = "http://www.roblox.com/asset/?id=180435792", weight = 1 }
- },
- walk = {
- { id = "http://www.roblox.com/asset/?id=180426354", weight = 10 }
- },
- run = {
- { id = "run.xml", weight = 10 }
- },
- jump = {
- { id = "http://www.roblox.com/asset/?id=125750702", weight = 10 }
- },
- fall = {
- { id = "http://www.roblox.com/asset/?id=180436148", weight = 10 }
- },
- climb = {
- { id = "http://www.roblox.com/asset/?id=180436334", weight = 10 }
- },
- sit = {
- { id = "http://www.roblox.com/asset/?id=178130996", weight = 10 }
- },
- toolnone = {
- { id = "http://www.roblox.com/asset/?id=182393478", weight = 10 }
- },
- toolslash = {
- { id = "http://www.roblox.com/asset/?id=129967390", weight = 10 }
- -- { id = "slash.xml", weight = 10 }
- },
- toollunge = {
- { id = "http://www.roblox.com/asset/?id=129967478", weight = 10 }
- },
- wave = {
- { id = "http://www.roblox.com/asset/?id=128777973", weight = 10 }
- },
- point = {
- { id = "http://www.roblox.com/asset/?id=128853357", weight = 10 }
- },
- dance1 = {
- { id = "http://www.roblox.com/asset/?id=182435998", weight = 10 },
- { id = "http://www.roblox.com/asset/?id=182491037", weight = 10 },
- { id = "http://www.roblox.com/asset/?id=182491065", weight = 10 }
- },
- dance2 = {
- { id = "http://www.roblox.com/asset/?id=182436842", weight = 10 },
- { id = "http://www.roblox.com/asset/?id=182491248", weight = 10 },
- { id = "http://www.roblox.com/asset/?id=182491277", weight = 10 }
- },
- dance3 = {
- { id = "http://www.roblox.com/asset/?id=182436935", weight = 10 },
- { id = "http://www.roblox.com/asset/?id=182491368", weight = 10 },
- { id = "http://www.roblox.com/asset/?id=182491423", weight = 10 }
- },
- laugh = {
- { id = "http://www.roblox.com/asset/?id=129423131", weight = 10 }
- },
- cheer = {
- { id = "http://www.roblox.com/asset/?id=129423030", weight = 10 }
- },
- }
- local dances = {"dance1", "dance2", "dance3"}
- -- Existance in this list signifies that it is an emote, the value indicates if it is a looping emote
- local emoteNames = { wave = false, point = false, dance1 = true, dance2 = true, dance3 = true, laugh = false, cheer = false}
- function configureAnimationSet(name, fileList)
- if (animTable[name] ~= nil) then
- for _, connection in pairs(animTable[name].connections) do
- connection:disconnect()
- end
- end
- animTable[name] = {}
- animTable[name].count = 0
- animTable[name].totalWeight = 0
- animTable[name].connections = {}
- -- check for config values
- local config = script:FindFirstChild(name)
- if (config ~= nil) then
- -- print("Loading anims " .. name)
- table.insert(animTable[name].connections, config.ChildAdded:connect(function(child) configureAnimationSet(name, fileList) end))
- table.insert(animTable[name].connections, config.ChildRemoved:connect(function(child) configureAnimationSet(name, fileList) end))
- local idx = 1
- for _, childPart in pairs(config:GetChildren()) do
- if (childPart:IsA("Animation")) then
- table.insert(animTable[name].connections, childPart.Changed:connect(function(property) configureAnimationSet(name, fileList) end))
- animTable[name][idx] = {}
- animTable[name][idx].anim = childPart
- local weightObject = childPart:FindFirstChild("Weight")
- if (weightObject == nil) then
- animTable[name][idx].weight = 1
- else
- animTable[name][idx].weight = weightObject.Value
- end
- animTable[name].count = animTable[name].count + 1
- animTable[name].totalWeight = animTable[name].totalWeight + animTable[name][idx].weight
- -- print(name .. " [" .. idx .. "] " .. animTable[name][idx].anim.AnimationId .. " (" .. animTable[name][idx].weight .. ")")
- idx = idx + 1
- end
- end
- end
- -- fallback to defaults
- if (animTable[name].count <= 0) then
- for idx, anim in pairs(fileList) do
- animTable[name][idx] = {}
- animTable[name][idx].anim = Instance.new("Animation")
- animTable[name][idx].anim.Name = name
- animTable[name][idx].anim.AnimationId = anim.id
- animTable[name][idx].weight = anim.weight
- animTable[name].count = animTable[name].count + 1
- animTable[name].totalWeight = animTable[name].totalWeight + anim.weight
- -- print(name .. " [" .. idx .. "] " .. anim.id .. " (" .. anim.weight .. ")")
- end
- end
- end
- -- Setup animation objects
- function scriptChildModified(child)
- local fileList = animNames[child.Name]
- if (fileList ~= nil) then
- configureAnimationSet(child.Name, fileList)
- end
- end
- script.ChildAdded:connect(scriptChildModified)
- script.ChildRemoved:connect(scriptChildModified)
- for name, fileList in pairs(animNames) do
- configureAnimationSet(name, fileList)
- end
- -- ANIMATION
- -- declarations
- local toolAnim = "None"
- local toolAnimTime = 0
- local jumpAnimTime = 0
- local jumpAnimDuration = 0.3
- local toolTransitionTime = 0.1
- local fallTransitionTime = 0.3
- local jumpMaxLimbVelocity = 0.75
- -- functions
- function stopAllAnimations()
- local oldAnim = currentAnim
- -- return to idle if finishing an emote
- if (emoteNames[oldAnim] ~= nil and emoteNames[oldAnim] == false) then
- oldAnim = "idle"
- end
- currentAnim = ""
- currentAnimInstance = nil
- if (currentAnimKeyframeHandler ~= nil) then
- currentAnimKeyframeHandler:disconnect()
- end
- if (currentAnimTrack ~= nil) then
- currentAnimTrack:Stop()
- currentAnimTrack:Destroy()
- currentAnimTrack = nil
- end
- return oldAnim
- end
- function setAnimationSpeed(speed)
- if speed ~= currentAnimSpeed then
- currentAnimSpeed = speed
- currentAnimTrack:AdjustSpeed(currentAnimSpeed)
- end
- end
- function keyFrameReachedFunc(frameName)
- if (frameName == "End") then
- local repeatAnim = currentAnim
- -- return to idle if finishing an emote
- if (emoteNames[repeatAnim] ~= nil and emoteNames[repeatAnim] == false) then
- repeatAnim = "idle"
- end
- local animSpeed = currentAnimSpeed
- playAnimation(repeatAnim, 0.0, Humanoid)
- setAnimationSpeed(animSpeed)
- end
- end
- -- Preload animations
- function playAnimation(animName, transitionTime, humanoid)
- local roll = math.random(1, animTable[animName].totalWeight)
- local origRoll = roll
- local idx = 1
- while (roll > animTable[animName][idx].weight) do
- roll = roll - animTable[animName][idx].weight
- idx = idx + 1
- end
- -- print(animName .. " " .. idx .. " [" .. origRoll .. "]")
- local anim = animTable[animName][idx].anim
- -- switch animation
- if (anim ~= currentAnimInstance) then
- if (currentAnimTrack ~= nil) then
- currentAnimTrack:Stop(transitionTime)
- currentAnimTrack:Destroy()
- end
- currentAnimSpeed = 1.0
- -- load it to the humanoid; get AnimationTrack
- currentAnimTrack = humanoid:LoadAnimation(anim)
- currentAnimTrack.Priority = Enum.AnimationPriority.Core
- -- play the animation
- currentAnimTrack:Play(transitionTime)
- currentAnim = animName
- currentAnimInstance = anim
- -- set up keyframe name triggers
- if (currentAnimKeyframeHandler ~= nil) then
- currentAnimKeyframeHandler:disconnect()
- end
- currentAnimKeyframeHandler = currentAnimTrack.KeyframeReached:connect(keyFrameReachedFunc)
- end
- end
- -------------------------------------------------------------------------------------------
- -------------------------------------------------------------------------------------------
- local toolAnimName = ""
- local toolAnimTrack = nil
- local toolAnimInstance = nil
- local currentToolAnimKeyframeHandler = nil
- function toolKeyFrameReachedFunc(frameName)
- if (frameName == "End") then
- -- print("Keyframe : ".. frameName)
- playToolAnimation(toolAnimName, 0.0, Humanoid)
- end
- end
- function playToolAnimation(animName, transitionTime, humanoid, priority)
- local roll = math.random(1, animTable[animName].totalWeight)
- local origRoll = roll
- local idx = 1
- while (roll > animTable[animName][idx].weight) do
- roll = roll - animTable[animName][idx].weight
- idx = idx + 1
- end
- -- print(animName .. " * " .. idx .. " [" .. origRoll .. "]")
- local anim = animTable[animName][idx].anim
- if (toolAnimInstance ~= anim) then
- if (toolAnimTrack ~= nil) then
- toolAnimTrack:Stop()
- toolAnimTrack:Destroy()
- transitionTime = 0
- end
- -- load it to the humanoid; get AnimationTrack
- toolAnimTrack = humanoid:LoadAnimation(anim)
- if priority then
- toolAnimTrack.Priority = priority
- end
- -- play the animation
- toolAnimTrack:Play(transitionTime)
- toolAnimName = animName
- toolAnimInstance = anim
- currentToolAnimKeyframeHandler = toolAnimTrack.KeyframeReached:connect(toolKeyFrameReachedFunc)
- end
- end
- function stopToolAnimations()
- local oldAnim = toolAnimName
- if (currentToolAnimKeyframeHandler ~= nil) then
- currentToolAnimKeyframeHandler:disconnect()
- end
- toolAnimName = ""
- toolAnimInstance = nil
- if (toolAnimTrack ~= nil) then
- toolAnimTrack:Stop()
- toolAnimTrack:Destroy()
- toolAnimTrack = nil
- end
- return oldAnim
- end
- -------------------------------------------------------------------------------------------
- -------------------------------------------------------------------------------------------
- function onRunning(speed)
- if speed > 0.01 then
- playAnimation("walk", 0.1, Humanoid)
- if currentAnimInstance and currentAnimInstance.AnimationId == "http://www.roblox.com/asset/?id=180426354" then
- setAnimationSpeed(speed / 14.5)
- end
- pose = "Running"
- else
- if emoteNames[currentAnim] == nil then
- playAnimation("idle", 0.1, Humanoid)
- pose = "Standing"
- end
- end
- end
- function onDied()
- pose = "Dead"
- end
- function onJumping()
- playAnimation("jump", 0.1, Humanoid)
- jumpAnimTime = jumpAnimDuration
- pose = "Jumping"
- end
- function onClimbing(speed)
- playAnimation("climb", 0.1, Humanoid)
- setAnimationSpeed(speed / 12.0)
- pose = "Climbing"
- end
- function onGettingUp()
- pose = "GettingUp"
- end
- function onFreeFall()
- if (jumpAnimTime <= 0) then
- playAnimation("fall", fallTransitionTime, Humanoid)
- end
- pose = "FreeFall"
- end
- function onFallingDown()
- pose = "FallingDown"
- end
- function onSeated()
- pose = "Seated"
- end
- function onPlatformStanding()
- pose = "PlatformStanding"
- end
- function onSwimming(speed)
- if speed > 0 then
- pose = "Running"
- else
- pose = "Standing"
- end
- end
- function getTool()
- for _, kid in ipairs(Figure:GetChildren()) do
- if kid.className == "Tool" then return kid end
- end
- return nil
- end
- function getToolAnim(tool)
- for _, c in ipairs(tool:GetChildren()) do
- if c.Name == "toolanim" and c.className == "StringValue" then
- return c
- end
- end
- return nil
- end
- function animateTool()
- if (toolAnim == "None") then
- playToolAnimation("toolnone", toolTransitionTime, Humanoid, Enum.AnimationPriority.Idle)
- return
- end
- if (toolAnim == "Slash") then
- playToolAnimation("toolslash", 0, Humanoid, Enum.AnimationPriority.Action)
- return
- end
- if (toolAnim == "Lunge") then
- playToolAnimation("toollunge", 0, Humanoid, Enum.AnimationPriority.Action)
- return
- end
- end
- function moveSit()
- RightShoulder.MaxVelocity = 0.15
- LeftShoulder.MaxVelocity = 0.15
- RightShoulder:SetDesiredAngle(3.14 /2)
- LeftShoulder:SetDesiredAngle(-3.14 /2)
- RightHip:SetDesiredAngle(3.14 /2)
- LeftHip:SetDesiredAngle(-3.14 /2)
- end
- local lastTick = 0
- function move(time)
- local amplitude = 1
- local frequency = 1
- local deltaTime = time - lastTick
- lastTick = time
- local climbFudge = 0
- local setAngles = false
- if (jumpAnimTime > 0) then
- jumpAnimTime = jumpAnimTime - deltaTime
- end
- if (pose == "FreeFall" and jumpAnimTime <= 0) then
- playAnimation("fall", fallTransitionTime, Humanoid)
- elseif (pose == "Seated") then
- playAnimation("sit", 0.5, Humanoid)
- return
- elseif (pose == "Running") then
- playAnimation("walk", 0.1, Humanoid)
- elseif (pose == "Dead" or pose == "GettingUp" or pose == "FallingDown" or pose == "Seated" or pose == "PlatformStanding") then
- -- print("Wha " .. pose)
- stopAllAnimations()
- amplitude = 0.1
- frequency = 1
- setAngles = true
- end
- if (setAngles) then
- local desiredAngle = amplitude * math.sin(time * frequency)
- RightShoulder:SetDesiredAngle(desiredAngle + climbFudge)
- LeftShoulder:SetDesiredAngle(desiredAngle - climbFudge)
- RightHip:SetDesiredAngle(-desiredAngle)
- LeftHip:SetDesiredAngle(-desiredAngle)
- end
- -- Tool Animation handling
- local tool = getTool()
- if tool and tool:FindFirstChild("Handle") then
- local animStringValueObject = getToolAnim(tool)
- if animStringValueObject then
- toolAnim = animStringValueObject.Value
- -- message recieved, delete StringValue
- animStringValueObject.Parent = nil
- toolAnimTime = time + .3
- end
- if time > toolAnimTime then
- toolAnimTime = 0
- toolAnim = "None"
- end
- animateTool()
- else
- stopToolAnimations()
- toolAnim = "None"
- toolAnimInstance = nil
- toolAnimTime = 0
- end
- end
- local events = {}
- local eventHum = Humanoid
- local function onUnhook()
- for i = 1, #events do
- events[i]:Disconnect()
- end
- events = {}
- end
- local function onHook()
- onUnhook()
- pose = eventHum.Sit and "Seated" or "Standing"
- events = {
- eventHum.Died:connect(onDied),
- eventHum.Running:connect(onRunning),
- eventHum.Jumping:connect(onJumping),
- eventHum.Climbing:connect(onClimbing),
- eventHum.GettingUp:connect(onGettingUp),
- eventHum.FreeFalling:connect(onFreeFall),
- eventHum.FallingDown:connect(onFallingDown),
- eventHum.Seated:connect(onSeated),
- eventHum.PlatformStanding:connect(onPlatformStanding),
- eventHum.Swimming:connect(onSwimming)
- }
- end
- onHook()
- -- setup emote chat hook
- game:GetService("Players").LocalPlayer.Chatted:connect(function(msg)
- local emote = ""
- if msg == "/e dance" then
- emote = dances[math.random(1, #dances)]
- elseif (string.sub(msg, 1, 3) == "/e ") then
- emote = string.sub(msg, 4)
- elseif (string.sub(msg, 1, 7) == "/emote ") then
- emote = string.sub(msg, 8)
- end
- if (pose == "Standing" and emoteNames[emote] ~= nil) then
- playAnimation(emote, 0.1, Humanoid)
- end
- end)
- -- main program
- -- initialize to idle
- playAnimation("idle", 0.1, Humanoid)
- pose = "Standing"
- spawn(function()
- while Figure.Parent ~= nil do
- local _, time = wait(0.1)
- move(time)
- end
- end)
- return {
- onRunning = onRunning,
- onDied = onDied,
- onJumping = onJumping,
- onClimbing = onClimbing,
- onGettingUp = onGettingUp,
- onFreeFall = onFreeFall,
- onFallingDown = onFallingDown,
- onSeated = onSeated,
- onPlatformStanding = onPlatformStanding,
- onHook = onHook,
- onUnhook = onUnhook
- }
- end
- return r6()
- end
- function _R15()
- local function r15()
- local Character = char
- local Humanoid = Character:WaitForChild("Humanoid")
- local pose = "Standing"
- local userNoUpdateOnLoopSuccess, userNoUpdateOnLoopValue = pcall(function() return UserSettings():IsUserFeatureEnabled("UserNoUpdateOnLoop") end)
- local userNoUpdateOnLoop = userNoUpdateOnLoopSuccess and userNoUpdateOnLoopValue
- local userAnimationSpeedDampeningSuccess, userAnimationSpeedDampeningValue = pcall(function() return UserSettings():IsUserFeatureEnabled("UserAnimationSpeedDampening") end)
- local userAnimationSpeedDampening = userAnimationSpeedDampeningSuccess and userAnimationSpeedDampeningValue
- local animateScriptEmoteHookFlagExists, animateScriptEmoteHookFlagEnabled = pcall(function()
- return UserSettings():IsUserFeatureEnabled("UserAnimateScriptEmoteHook")
- end)
- local FFlagAnimateScriptEmoteHook = animateScriptEmoteHookFlagExists and animateScriptEmoteHookFlagEnabled
- local AnimationSpeedDampeningObject = script:FindFirstChild("ScaleDampeningPercent")
- local HumanoidHipHeight = 2
- local EMOTE_TRANSITION_TIME = 0.1
- local currentAnim = ""
- local currentAnimInstance = nil
- local currentAnimTrack = nil
- local currentAnimKeyframeHandler = nil
- local currentAnimSpeed = 1.0
- local runAnimTrack = nil
- local runAnimKeyframeHandler = nil
- local animTable = {}
- local animNames = {
- idle = {
- { id = "http://www.roblox.com/asset/?id=507766666", weight = 1 },
- { id = "http://www.roblox.com/asset/?id=507766951", weight = 1 },
- { id = "http://www.roblox.com/asset/?id=507766388", weight = 9 }
- },
- walk = {
- { id = "http://www.roblox.com/asset/?id=507777826", weight = 10 }
- },
- run = {
- { id = "http://www.roblox.com/asset/?id=507767714", weight = 10 }
- },
- swim = {
- { id = "http://www.roblox.com/asset/?id=507784897", weight = 10 }
- },
- swimidle = {
- { id = "http://www.roblox.com/asset/?id=507785072", weight = 10 }
- },
- jump = {
- { id = "http://www.roblox.com/asset/?id=507765000", weight = 10 }
- },
- fall = {
- { id = "http://www.roblox.com/asset/?id=507767968", weight = 10 }
- },
- climb = {
- { id = "http://www.roblox.com/asset/?id=507765644", weight = 10 }
- },
- sit = {
- { id = "http://www.roblox.com/asset/?id=2506281703", weight = 10 }
- },
- toolnone = {
- { id = "http://www.roblox.com/asset/?id=507768375", weight = 10 }
- },
- toolslash = {
- { id = "http://www.roblox.com/asset/?id=522635514", weight = 10 }
- },
- toollunge = {
- { id = "http://www.roblox.com/asset/?id=522638767", weight = 10 }
- },
- wave = {
- { id = "http://www.roblox.com/asset/?id=507770239", weight = 10 }
- },
- point = {
- { id = "http://www.roblox.com/asset/?id=507770453", weight = 10 }
- },
- dance = {
- { id = "http://www.roblox.com/asset/?id=507771019", weight = 10 },
- { id = "http://www.roblox.com/asset/?id=507771955", weight = 10 },
- { id = "http://www.roblox.com/asset/?id=507772104", weight = 10 }
- },
- dance2 = {
- { id = "http://www.roblox.com/asset/?id=507776043", weight = 10 },
- { id = "http://www.roblox.com/asset/?id=507776720", weight = 10 },
- { id = "http://www.roblox.com/asset/?id=507776879", weight = 10 }
- },
- dance3 = {
- { id = "http://www.roblox.com/asset/?id=507777268", weight = 10 },
- { id = "http://www.roblox.com/asset/?id=507777451", weight = 10 },
- { id = "http://www.roblox.com/asset/?id=507777623", weight = 10 }
- },
- laugh = {
- { id = "http://www.roblox.com/asset/?id=507770818", weight = 10 }
- },
- cheer = {
- { id = "http://www.roblox.com/asset/?id=507770677", weight = 10 }
- },
- }
- -- Existance in this list signifies that it is an emote, the value indicates if it is a looping emote
- local emoteNames = { wave = false, point = false, dance = true, dance2 = true, dance3 = true, laugh = false, cheer = false}
- local PreloadAnimsUserFlag = false
- local PreloadedAnims = {}
- local successPreloadAnim, msgPreloadAnim = pcall(function()
- PreloadAnimsUserFlag = UserSettings():IsUserFeatureEnabled("UserPreloadAnimations")
- end)
- if not successPreloadAnim then
- PreloadAnimsUserFlag = false
- end
- math.randomseed(tick())
- function findExistingAnimationInSet(set, anim)
- if set == nil or anim == nil then
- return 0
- end
- for idx = 1, set.count, 1 do
- if set[idx].anim.AnimationId == anim.AnimationId then
- return idx
- end
- end
- return 0
- end
- function configureAnimationSet(name, fileList)
- if (animTable[name] ~= nil) then
- for _, connection in pairs(animTable[name].connections) do
- connection:disconnect()
- end
- end
- animTable[name] = {}
- animTable[name].count = 0
- animTable[name].totalWeight = 0
- animTable[name].connections = {}
- local allowCustomAnimations = true
- local success, msg = pcall(function() allowCustomAnimations = game:GetService("StarterPlayer").AllowCustomAnimations end)
- if not success then
- allowCustomAnimations = true
- end
- -- check for config values
- local config = script:FindFirstChild(name)
- if (allowCustomAnimations and config ~= nil) then
- table.insert(animTable[name].connections, config.ChildAdded:connect(function(child) configureAnimationSet(name, fileList) end))
- table.insert(animTable[name].connections, config.ChildRemoved:connect(function(child) configureAnimationSet(name, fileList) end))
- local idx = 0
- for _, childPart in pairs(config:GetChildren()) do
- if (childPart:IsA("Animation")) then
- local newWeight = 1
- local weightObject = childPart:FindFirstChild("Weight")
- if (weightObject ~= nil) then
- newWeight = weightObject.Value
- end
- animTable[name].count = animTable[name].count + 1
- idx = animTable[name].count
- animTable[name][idx] = {}
- animTable[name][idx].anim = childPart
- animTable[name][idx].weight = newWeight
- animTable[name].totalWeight = animTable[name].totalWeight + animTable[name][idx].weight
- table.insert(animTable[name].connections, childPart.Changed:connect(function(property) configureAnimationSet(name, fileList) end))
- table.insert(animTable[name].connections, childPart.ChildAdded:connect(function(property) configureAnimationSet(name, fileList) end))
- table.insert(animTable[name].connections, childPart.ChildRemoved:connect(function(property) configureAnimationSet(name, fileList) end))
- end
- end
- end
- -- fallback to defaults
- if (animTable[name].count <= 0) then
- for idx, anim in pairs(fileList) do
- animTable[name][idx] = {}
- animTable[name][idx].anim = Instance.new("Animation")
- animTable[name][idx].anim.Name = name
- animTable[name][idx].anim.AnimationId = anim.id
- animTable[name][idx].weight = anim.weight
- animTable[name].count = animTable[name].count + 1
- animTable[name].totalWeight = animTable[name].totalWeight + anim.weight
- end
- end
- -- preload anims
- if PreloadAnimsUserFlag then
- for i, animType in pairs(animTable) do
- for idx = 1, animType.count, 1 do
- if PreloadedAnims[animType[idx].anim.AnimationId] == nil then
- Humanoid:LoadAnimation(animType[idx].anim)
- PreloadedAnims[animType[idx].anim.AnimationId] = true
- end
- end
- end
- end
- end
- ------------------------------------------------------------------------------------------------------------
- function configureAnimationSetOld(name, fileList)
- if (animTable[name] ~= nil) then
- for _, connection in pairs(animTable[name].connections) do
- connection:disconnect()
- end
- end
- animTable[name] = {}
- animTable[name].count = 0
- animTable[name].totalWeight = 0
- animTable[name].connections = {}
- local allowCustomAnimations = true
- local success, msg = pcall(function() allowCustomAnimations = game:GetService("StarterPlayer").AllowCustomAnimations end)
- if not success then
- allowCustomAnimations = true
- end
- -- check for config values
- local config = script:FindFirstChild(name)
- if (allowCustomAnimations and config ~= nil) then
- table.insert(animTable[name].connections, config.ChildAdded:connect(function(child) configureAnimationSet(name, fileList) end))
- table.insert(animTable[name].connections, config.ChildRemoved:connect(function(child) configureAnimationSet(name, fileList) end))
- local idx = 1
- for _, childPart in pairs(config:GetChildren()) do
- if (childPart:IsA("Animation")) then
- table.insert(animTable[name].connections, childPart.Changed:connect(function(property) configureAnimationSet(name, fileList) end))
- animTable[name][idx] = {}
- animTable[name][idx].anim = childPart
- local weightObject = childPart:FindFirstChild("Weight")
- if (weightObject == nil) then
- animTable[name][idx].weight = 1
- else
- animTable[name][idx].weight = weightObject.Value
- end
- animTable[name].count = animTable[name].count + 1
- animTable[name].totalWeight = animTable[name].totalWeight + animTable[name][idx].weight
- idx = idx + 1
- end
- end
- end
- -- fallback to defaults
- if (animTable[name].count <= 0) then
- for idx, anim in pairs(fileList) do
- animTable[name][idx] = {}
- animTable[name][idx].anim = Instance.new("Animation")
- animTable[name][idx].anim.Name = name
- animTable[name][idx].anim.AnimationId = anim.id
- animTable[name][idx].weight = anim.weight
- animTable[name].count = animTable[name].count + 1
- animTable[name].totalWeight = animTable[name].totalWeight + anim.weight
- -- print(name .. " [" .. idx .. "] " .. anim.id .. " (" .. anim.weight .. ")")
- end
- end
- -- preload anims
- if PreloadAnimsUserFlag then
- for i, animType in pairs(animTable) do
- for idx = 1, animType.count, 1 do
- Humanoid:LoadAnimation(animType[idx].anim)
- end
- end
- end
- end
- -- Setup animation objects
- function scriptChildModified(child)
- local fileList = animNames[child.Name]
- if (fileList ~= nil) then
- configureAnimationSet(child.Name, fileList)
- end
- end
- script.ChildAdded:connect(scriptChildModified)
- script.ChildRemoved:connect(scriptChildModified)
- for name, fileList in pairs(animNames) do
- configureAnimationSet(name, fileList)
- end
- -- ANIMATION
- -- declarations
- local toolAnim = "None"
- local toolAnimTime = 0
- local jumpAnimTime = 0
- local jumpAnimDuration = 0.31
- local toolTransitionTime = 0.1
- local fallTransitionTime = 0.2
- local currentlyPlayingEmote = false
- -- functions
- function stopAllAnimations()
- local oldAnim = currentAnim
- -- return to idle if finishing an emote
- if (emoteNames[oldAnim] ~= nil and emoteNames[oldAnim] == false) then
- oldAnim = "idle"
- end
- if FFlagAnimateScriptEmoteHook and currentlyPlayingEmote then
- oldAnim = "idle"
- currentlyPlayingEmote = false
- end
- currentAnim = ""
- currentAnimInstance = nil
- if (currentAnimKeyframeHandler ~= nil) then
- currentAnimKeyframeHandler:disconnect()
- end
- if (currentAnimTrack ~= nil) then
- currentAnimTrack:Stop()
- currentAnimTrack:Destroy()
- currentAnimTrack = nil
- end
- -- clean up walk if there is one
- if (runAnimKeyframeHandler ~= nil) then
- runAnimKeyframeHandler:disconnect()
- end
- if (runAnimTrack ~= nil) then
- runAnimTrack:Stop()
- runAnimTrack:Destroy()
- runAnimTrack = nil
- end
- return oldAnim
- end
- function getHeightScale()
- if Humanoid then
- if not Humanoid.AutomaticScalingEnabled then
- return 1
- end
- local scale = Humanoid.HipHeight / HumanoidHipHeight
- if userAnimationSpeedDampening then
- if AnimationSpeedDampeningObject == nil then
- AnimationSpeedDampeningObject = script:FindFirstChild("ScaleDampeningPercent")
- end
- if AnimationSpeedDampeningObject ~= nil then
- scale = 1 + (Humanoid.HipHeight - HumanoidHipHeight) * AnimationSpeedDampeningObject.Value / HumanoidHipHeight
- end
- end
- return scale
- end
- return 1
- end
- local smallButNotZero = 0.0001
- function setRunSpeed(speed)
- local speedScaled = speed * 1.25
- local heightScale = getHeightScale()
- local runSpeed = speedScaled / heightScale
- if runSpeed ~= currentAnimSpeed then
- if runSpeed < 0.33 then
- currentAnimTrack:AdjustWeight(1.0)
- runAnimTrack:AdjustWeight(smallButNotZero)
- elseif runSpeed < 0.66 then
- local weight = ((runSpeed - 0.33) / 0.33)
- currentAnimTrack:AdjustWeight(1.0 - weight + smallButNotZero)
- runAnimTrack:AdjustWeight(weight + smallButNotZero)
- else
- currentAnimTrack:AdjustWeight(smallButNotZero)
- runAnimTrack:AdjustWeight(1.0)
- end
- currentAnimSpeed = runSpeed
- runAnimTrack:AdjustSpeed(runSpeed)
- currentAnimTrack:AdjustSpeed(runSpeed)
- end
- end
- function setAnimationSpeed(speed)
- if currentAnim == "walk" then
- setRunSpeed(speed)
- else
- if speed ~= currentAnimSpeed then
- currentAnimSpeed = speed
- currentAnimTrack:AdjustSpeed(currentAnimSpeed)
- end
- end
- end
- function keyFrameReachedFunc(frameName)
- if (frameName == "End") then
- if currentAnim == "walk" then
- if userNoUpdateOnLoop == true then
- if runAnimTrack.Looped ~= true then
- runAnimTrack.TimePosition = 0.0
- end
- if currentAnimTrack.Looped ~= true then
- currentAnimTrack.TimePosition = 0.0
- end
- else
- runAnimTrack.TimePosition = 0.0
- currentAnimTrack.TimePosition = 0.0
- end
- else
- local repeatAnim = currentAnim
- -- return to idle if finishing an emote
- if (emoteNames[repeatAnim] ~= nil and emoteNames[repeatAnim] == false) then
- repeatAnim = "idle"
- end
- if FFlagAnimateScriptEmoteHook and currentlyPlayingEmote then
- if currentAnimTrack.Looped then
- -- Allow the emote to loop
- return
- end
- repeatAnim = "idle"
- currentlyPlayingEmote = false
- end
- local animSpeed = currentAnimSpeed
- playAnimation(repeatAnim, 0.15, Humanoid)
- setAnimationSpeed(animSpeed)
- end
- end
- end
- function rollAnimation(animName)
- local roll = math.random(1, animTable[animName].totalWeight)
- local origRoll = roll
- local idx = 1
- while (roll > animTable[animName][idx].weight) do
- roll = roll - animTable[animName][idx].weight
- idx = idx + 1
- end
- return idx
- end
- local function switchToAnim(anim, animName, transitionTime, humanoid)
- -- switch animation
- if (anim ~= currentAnimInstance) then
- if (currentAnimTrack ~= nil) then
- currentAnimTrack:Stop(transitionTime)
- currentAnimTrack:Destroy()
- end
- if (runAnimTrack ~= nil) then
- runAnimTrack:Stop(transitionTime)
- runAnimTrack:Destroy()
- if userNoUpdateOnLoop == true then
- runAnimTrack = nil
- end
- end
- currentAnimSpeed = 1.0
- -- load it to the humanoid; get AnimationTrack
- currentAnimTrack = humanoid:LoadAnimation(anim)
- currentAnimTrack.Priority = Enum.AnimationPriority.Core
- -- play the animation
- currentAnimTrack:Play(transitionTime)
- currentAnim = animName
- currentAnimInstance = anim
- -- set up keyframe name triggers
- if (currentAnimKeyframeHandler ~= nil) then
- currentAnimKeyframeHandler:disconnect()
- end
- currentAnimKeyframeHandler = currentAnimTrack.KeyframeReached:connect(keyFrameReachedFunc)
- -- check to see if we need to blend a walk/run animation
- if animName == "walk" then
- local runAnimName = "run"
- local runIdx = rollAnimation(runAnimName)
- runAnimTrack = humanoid:LoadAnimation(animTable[runAnimName][runIdx].anim)
- runAnimTrack.Priority = Enum.AnimationPriority.Core
- runAnimTrack:Play(transitionTime)
- if (runAnimKeyframeHandler ~= nil) then
- runAnimKeyframeHandler:disconnect()
- end
- runAnimKeyframeHandler = runAnimTrack.KeyframeReached:connect(keyFrameReachedFunc)
- end
- end
- end
- function playAnimation(animName, transitionTime, humanoid)
- local idx = rollAnimation(animName)
- local anim = animTable[animName][idx].anim
- switchToAnim(anim, animName, transitionTime, humanoid)
- currentlyPlayingEmote = false
- end
- function playEmote(emoteAnim, transitionTime, humanoid)
- switchToAnim(emoteAnim, emoteAnim.Name, transitionTime, humanoid)
- currentlyPlayingEmote = true
- end
- -------------------------------------------------------------------------------------------
- -------------------------------------------------------------------------------------------
- local toolAnimName = ""
- local toolAnimTrack = nil
- local toolAnimInstance = nil
- local currentToolAnimKeyframeHandler = nil
- function toolKeyFrameReachedFunc(frameName)
- if (frameName == "End") then
- playToolAnimation(toolAnimName, 0.0, Humanoid)
- end
- end
- function playToolAnimation(animName, transitionTime, humanoid, priority)
- local idx = rollAnimation(animName)
- local anim = animTable[animName][idx].anim
- if (toolAnimInstance ~= anim) then
- if (toolAnimTrack ~= nil) then
- toolAnimTrack:Stop()
- toolAnimTrack:Destroy()
- transitionTime = 0
- end
- -- load it to the humanoid; get AnimationTrack
- toolAnimTrack = humanoid:LoadAnimation(anim)
- if priority then
- toolAnimTrack.Priority = priority
- end
- -- play the animation
- toolAnimTrack:Play(transitionTime)
- toolAnimName = animName
- toolAnimInstance = anim
- currentToolAnimKeyframeHandler = toolAnimTrack.KeyframeReached:connect(toolKeyFrameReachedFunc)
- end
- end
- function stopToolAnimations()
- local oldAnim = toolAnimName
- if (currentToolAnimKeyframeHandler ~= nil) then
- currentToolAnimKeyframeHandler:disconnect()
- end
- toolAnimName = ""
- toolAnimInstance = nil
- if (toolAnimTrack ~= nil) then
- toolAnimTrack:Stop()
- toolAnimTrack:Destroy()
- toolAnimTrack = nil
- end
- return oldAnim
- end
- -------------------------------------------------------------------------------------------
- -------------------------------------------------------------------------------------------
- -- STATE CHANGE HANDLERS
- function onRunning(speed)
- if speed > 0.75 then
- local scale = 16.0
- playAnimation("walk", 0.2, Humanoid)
- setAnimationSpeed(speed / scale)
- pose = "Running"
- else
- if emoteNames[currentAnim] == nil and not currentlyPlayingEmote then
- playAnimation("idle", 0.2, Humanoid)
- pose = "Standing"
- end
- end
- end
- function onDied()
- pose = "Dead"
- end
- function onJumping()
- playAnimation("jump", 0.1, Humanoid)
- jumpAnimTime = jumpAnimDuration
- pose = "Jumping"
- end
- function onClimbing(speed)
- local scale = 5.0
- playAnimation("climb", 0.1, Humanoid)
- setAnimationSpeed(speed / scale)
- pose = "Climbing"
- end
- function onGettingUp()
- pose = "GettingUp"
- end
- function onFreeFall()
- if (jumpAnimTime <= 0) then
- playAnimation("fall", fallTransitionTime, Humanoid)
- end
- pose = "FreeFall"
- end
- function onFallingDown()
- pose = "FallingDown"
- end
- function onSeated()
- pose = "Seated"
- end
- function onPlatformStanding()
- pose = "PlatformStanding"
- end
- -------------------------------------------------------------------------------------------
- -------------------------------------------------------------------------------------------
- function onSwimming(speed)
- if speed > 1.00 then
- local scale = 10.0
- playAnimation("swim", 0.4, Humanoid)
- setAnimationSpeed(speed / scale)
- pose = "Swimming"
- else
- playAnimation("swimidle", 0.4, Humanoid)
- pose = "Standing"
- end
- end
- function animateTool()
- if (toolAnim == "None") then
- playToolAnimation("toolnone", toolTransitionTime, Humanoid, Enum.AnimationPriority.Idle)
- return
- end
- if (toolAnim == "Slash") then
- playToolAnimation("toolslash", 0, Humanoid, Enum.AnimationPriority.Action)
- return
- end
- if (toolAnim == "Lunge") then
- playToolAnimation("toollunge", 0, Humanoid, Enum.AnimationPriority.Action)
- return
- end
- end
- function getToolAnim(tool)
- for _, c in ipairs(tool:GetChildren()) do
- if c.Name == "toolanim" and c.className == "StringValue" then
- return c
- end
- end
- return nil
- end
- local lastTick = 0
- function stepAnimate(currentTime)
- local amplitude = 1
- local frequency = 1
- local deltaTime = currentTime - lastTick
- lastTick = currentTime
- local climbFudge = 0
- local setAngles = false
- if (jumpAnimTime > 0) then
- jumpAnimTime = jumpAnimTime - deltaTime
- end
- if (pose == "FreeFall" and jumpAnimTime <= 0) then
- playAnimation("fall", fallTransitionTime, Humanoid)
- elseif (pose == "Seated") then
- playAnimation("sit", 0.5, Humanoid)
- return
- elseif (pose == "Running") then
- playAnimation("walk", 0.2, Humanoid)
- elseif (pose == "Dead" or pose == "GettingUp" or pose == "FallingDown" or pose == "Seated" or pose == "PlatformStanding") then
- stopAllAnimations()
- amplitude = 0.1
- frequency = 1
- setAngles = true
- end
- -- Tool Animation handling
- local tool = Character:FindFirstChildOfClass("Tool")
- if tool and tool:FindFirstChild("Handle") then
- local animStringValueObject = getToolAnim(tool)
- if animStringValueObject then
- toolAnim = animStringValueObject.Value
- -- message recieved, delete StringValue
- animStringValueObject.Parent = nil
- toolAnimTime = currentTime + .3
- end
- if currentTime > toolAnimTime then
- toolAnimTime = 0
- toolAnim = "None"
- end
- animateTool()
- else
- stopToolAnimations()
- toolAnim = "None"
- toolAnimInstance = nil
- toolAnimTime = 0
- end
- end
- -- connect events
- local events = {}
- local eventHum = Humanoid
- local function onUnhook()
- for i = 1, #events do
- events[i]:Disconnect()
- end
- events = {}
- end
- local function onHook()
- onUnhook()
- pose = eventHum.Sit and "Seated" or "Standing"
- events = {
- eventHum.Died:connect(onDied),
- eventHum.Running:connect(onRunning),
- eventHum.Jumping:connect(onJumping),
- eventHum.Climbing:connect(onClimbing),
- eventHum.GettingUp:connect(onGettingUp),
- eventHum.FreeFalling:connect(onFreeFall),
- eventHum.FallingDown:connect(onFallingDown),
- eventHum.Seated:connect(onSeated),
- eventHum.PlatformStanding:connect(onPlatformStanding),
- eventHum.Swimming:connect(onSwimming)
- }
- end
- onHook()
- -- setup emote chat hook
- game:GetService("Players").LocalPlayer.Chatted:connect(function(msg)
- local emote = ""
- if (string.sub(msg, 1, 3) == "/e ") then
- emote = string.sub(msg, 4)
- elseif (string.sub(msg, 1, 7) == "/emote ") then
- emote = string.sub(msg, 8)
- end
- if (pose == "Standing" and emoteNames[emote] ~= nil) then
- playAnimation(emote, EMOTE_TRANSITION_TIME, Humanoid)
- end
- end)
- --[[ emote bindable hook
- if FFlagAnimateScriptEmoteHook then
- script:WaitForChild("PlayEmote").OnInvoke = function(emote)
- -- Only play emotes when idling
- if pose ~= "Standing" then
- return
- end
- if emoteNames[emote] ~= nil then
- -- Default emotes
- playAnimation(emote, EMOTE_TRANSITION_TIME, Humanoid)
- return true
- elseif typeof(emote) == "Instance" and emote:IsA("Animation") then
- -- Non-default emotes
- playEmote(emote, EMOTE_TRANSITION_TIME, Humanoid)
- return true
- end
- -- Return false to indicate that the emote could not be played
- return false
- end
- end
- ]]
- -- initialize to idle
- playAnimation("idle", 0.1, Humanoid)
- pose = "Standing"
- -- loop to handle timed state transitions and tool animations
- spawn(function()
- while Character.Parent ~= nil do
- local _, currentGameTime = wait(0.1)
- stepAnimate(currentGameTime)
- end
- end)
- return {
- onRunning = onRunning,
- onDied = onDied,
- onJumping = onJumping,
- onClimbing = onClimbing,
- onGettingUp = onGettingUp,
- onFreeFall = onFreeFall,
- onFallingDown = onFallingDown,
- onSeated = onSeated,
- onPlatformStanding = onPlatformStanding,
- onHook = onHook,
- onUnhook = onUnhook
- }
- end
- return r15()
- end
- while true do
- wait(.1)
- if plr.Character ~= nil then
- char = plr.Character
- break
- end
- end
- function _Controller()
- local humanoid = char:WaitForChild("Humanoid")
- local animFuncs = {}
- if (humanoid.RigType == Enum.HumanoidRigType.R6) then
- animFuncs = _R6()
- else
- animFuncs = _R15()
- end
- print("Animation succes")
- return animFuncs
- end
- function _AnimationHandler()
- local AnimationHandler = {}
- AnimationHandler.__index = AnimationHandler
- function AnimationHandler.new(humanoid, animate)
- local self = setmetatable({}, AnimationHandler)
- self._AnimFuncs = _Controller()
- self.Humanoid = humanoid
- return self
- end
- function AnimationHandler:EnableDefault(bool)
- if (bool) then
- self._AnimFuncs.onHook()
- else
- self._AnimFuncs.onUnhook()
- end
- end
- function AnimationHandler:Run(name, ...)
- self._AnimFuncs[name](...)
- end
- return AnimationHandler
- end
- function _GravityController()
- local ZERO = Vector3.new(0, 0, 0)
- local UNIT_X = Vector3.new(1, 0, 0)
- local UNIT_Y = Vector3.new(0, 1, 0)
- local UNIT_Z = Vector3.new(0, 0, 1)
- local VEC_XY = Vector3.new(1, 0, 1)
- local IDENTITYCF = CFrame.new()
- local JUMPMODIFIER = 1.2
- local TRANSITION = 0.15
- local WALKF = 200 / 3
- local UIS = game:GetService("UserInputService")
- local RUNSERVICE = game:GetService("RunService")
- local InitObjects = _InitObjects()
- local AnimationHandler = _AnimationHandler()
- local StateTracker = _StateTracker()
- -- Class
- local GravityController = {}
- GravityController.__index = GravityController
- -- Private Functions
- local function getRotationBetween(u, v, axis)
- local dot, uxv = u:Dot(v), u:Cross(v)
- if (dot < -0.99999) then return CFrame.fromAxisAngle(axis, math.pi) end
- return CFrame.new(0, 0, 0, uxv.x, uxv.y, uxv.z, 1 + dot)
- end
- local function lookAt(pos, forward, up)
- local r = forward:Cross(up)
- local u = r:Cross(forward)
- return CFrame.fromMatrix(pos, r.Unit, u.Unit)
- end
- local function getMass(array)
- local mass = 0
- for _, part in next, array do
- if (part:IsA("BasePart")) then
- mass = mass + part:GetMass()
- end
- end
- return mass
- end
- -- Public Constructor
- local ExecutedPlayerModule = _PlayerModule()
- local ExecutedSounds = _sounds()
- function GravityController.new(player)
- local self = setmetatable({}, GravityController)
- --[[ Camera
- local loaded = player.PlayerScripts:WaitForChild("PlayerScriptsLoader"):WaitForChild("Loaded")
- if (not loaded.Value) then
- --loaded.Changed:Wait()
- end
- ]]
- local playerModule = ExecutedPlayerModule
- self.Controls = playerModule:GetControls()
- self.Camera = playerModule:GetCameras()
- -- Player and character
- self.Player = player
- self.Character = player.Character
- self.Humanoid = player.Character:WaitForChild("Humanoid")
- self.HRP = player.Character:WaitForChild("HumanoidRootPart")
- -- Animation
- self.AnimationHandler = AnimationHandler.new(self.Humanoid, self.Character:WaitForChild("Animate"))
- self.AnimationHandler:EnableDefault(false)
- local ssss = game:GetService("Players").LocalPlayer.PlayerScripts:FindFirstChild("SetState") or Instance.new("BindableEvent",game:GetService("Players").LocalPlayer.PlayerScripts)
- local soundState = ExecutedSounds
- ssss.Name = "SetState"
- self.StateTracker = StateTracker.new(self.Humanoid, soundState)
- self.StateTracker.Changed:Connect(function(name, speed)
- self.AnimationHandler:Run(name, speed)
- end)
- -- Collider and forces
- local collider, gyro, vForce, floor = InitObjects(self)
- floor.Touched:Connect(function() end)
- collider.Touched:Connect(function() end)
- self.Collider = collider
- self.VForce = vForce
- self.Gyro = gyro
- self.Floor = floor
- -- Attachment to parts
- self.LastPart = workspace.Terrain
- self.LastPartCFrame = IDENTITYCF
- -- Gravity properties
- self.GravityUp = UNIT_Y
- self.Ignores = {self.Character}
- function self.Camera.GetUpVector(this, oldUpVector)
- return self.GravityUp
- end
- -- Events etc
- self.Humanoid.PlatformStand = true
- self.CharacterMass = getMass(self.Character:GetDescendants())
- self.Character.AncestryChanged:Connect(function() self.CharacterMass = getMass(self.Character:GetDescendants()) end)
- self.JumpCon = RUNSERVICE.RenderStepped:Connect(function(dt)
- if (self.Controls:IsJumping()) then
- self:OnJumpRequest()
- end
- end)
- self.DeathCon = self.Humanoid.Died:Connect(function() self:Destroy() end)
- self.SeatCon = self.Humanoid.Seated:Connect(function(active) if (active) then self:Destroy() end end)
- self.HeartCon = RUNSERVICE.Heartbeat:Connect(function(dt) self:OnHeartbeatStep(dt) end)
- RUNSERVICE:BindToRenderStep("GravityStep", Enum.RenderPriority.Input.Value + 1, function(dt) self:OnGravityStep(dt) end)
- return self
- end
- -- Public Methods
- function GravityController:Destroy()
- self.JumpCon:Disconnect()
- self.DeathCon:Disconnect()
- self.SeatCon:Disconnect()
- self.HeartCon:Disconnect()
- RUNSERVICE:UnbindFromRenderStep("GravityStep")
- self.Collider:Destroy()
- self.VForce:Destroy()
- self.Gyro:Destroy()
- self.StateTracker:Destroy()
- self.Humanoid.PlatformStand = false
- self.AnimationHandler:EnableDefault(true)
- self.GravityUp = UNIT_Y
- end
- function GravityController:GetGravityUp(oldGravity)
- return oldGravity
- end
- function GravityController:IsGrounded(isJumpCheck)
- if (not isJumpCheck) then
- local parts = self.Floor:GetTouchingParts()
- for _, part in next, parts do
- if (not part:IsDescendantOf(self.Character)) then
- return true
- end
- end
- else
- if (self.StateTracker.Jumped) then
- return false
- end
- -- 1. check we are touching something with the collider
- local valid = {}
- local parts = self.Collider:GetTouchingParts()
- for _, part in next, parts do
- if (not part:IsDescendantOf(self.Character)) then
- table.insert(valid, part)
- end
- end
- if (#valid > 0) then
- -- 2. do a decently long downwards raycast
- local max = math.cos(self.Humanoid.MaxSlopeAngle)
- local ray = Ray.new(self.Collider.Position, -10 * self.GravityUp)
- local hit, pos, normal = workspace:FindPartOnRayWithWhitelist(ray, valid, true)
- -- 3. use slope to decide on jump
- if (hit and max <= self.GravityUp:Dot(normal)) then
- return true
- end
- end
- end
- return false
- end
- function GravityController:OnJumpRequest()
- if (not self.StateTracker.Jumped and self:IsGrounded(true)) then
- local hrpVel = self.HRP.Velocity
- self.HRP.Velocity = hrpVel + self.GravityUp*self.Humanoid.JumpPower*JUMPMODIFIER
- self.StateTracker:RequestedJump()
- end
- end
- function GravityController:GetMoveVector()
- return self.Controls:GetMoveVector()
- end
- function GravityController:OnHeartbeatStep(dt)
- local ray = Ray.new(self.Collider.Position, -1.1*self.GravityUp)
- local hit, pos, normal = workspace:FindPartOnRayWithIgnoreList(ray, self.Ignores)
- local lastPart = self.LastPart
- if (hit and lastPart and lastPart == hit) then
- local offset = self.LastPartCFrame:ToObjectSpace(self.HRP.CFrame)
- self.HRP.CFrame = hit.CFrame:ToWorldSpace(offset)
- end
- self.LastPart = hit
- self.LastPartCFrame = hit and hit.CFrame
- end
- function GravityController:OnGravityStep(dt)
- -- update gravity up vector
- local oldGravity = self.GravityUp
- local newGravity = self:GetGravityUp(oldGravity)
- local rotation = getRotationBetween(oldGravity, newGravity, workspace.CurrentCamera.CFrame.RightVector)
- rotation = IDENTITYCF:Lerp(rotation, TRANSITION)
- self.GravityUp = rotation * oldGravity
- -- get world move vector
- local camCF = workspace.CurrentCamera.CFrame
- local fDot = camCF.LookVector:Dot(newGravity)
- local cForward = math.abs(fDot) > 0.5 and -math.sign(fDot)*camCF.UpVector or camCF.LookVector
- local left = cForward:Cross(-newGravity).Unit
- local forward = -left:Cross(newGravity).Unit
- local move = self:GetMoveVector()
- local worldMove = forward*move.z - left*move.x
- worldMove = worldMove:Dot(worldMove) > 1 and worldMove.Unit or worldMove
- local isInputMoving = worldMove:Dot(worldMove) > 0
- -- get the desired character cframe
- local hrpCFLook = self.HRP.CFrame.LookVector
- local charF = hrpCFLook:Dot(forward)*forward + hrpCFLook:Dot(left)*left
- local charR = charF:Cross(newGravity).Unit
- local newCharCF = CFrame.fromMatrix(ZERO, charR, newGravity, -charF)
- local newCharRotation = IDENTITYCF
- if (isInputMoving) then
- newCharRotation = IDENTITYCF:Lerp(getRotationBetween(charF, worldMove, newGravity), 0.7)
- end
- -- calculate forces
- local g = workspace.Gravity
- local gForce = g * self.CharacterMass * (UNIT_Y - newGravity)
- local cVelocity = self.HRP.Velocity
- local tVelocity = self.Humanoid.WalkSpeed * worldMove
- local gVelocity = cVelocity:Dot(newGravity)*newGravity
- local hVelocity = cVelocity - gVelocity
- if (hVelocity:Dot(hVelocity) < 1) then
- hVelocity = ZERO
- end
- local dVelocity = tVelocity - hVelocity
- local walkForceM = math.min(10000, WALKF * self.CharacterMass * dVelocity.Magnitude / (dt*60))
- local walkForce = walkForceM > 0 and dVelocity.Unit*walkForceM or ZERO
- -- mouse lock
- local charRotation = newCharRotation * newCharCF
- if (self.Camera:IsCamRelative()) then
- local lv = workspace.CurrentCamera.CFrame.LookVector
- local hlv = lv - charRotation.UpVector:Dot(lv)*charRotation.UpVector
- charRotation = lookAt(ZERO, hlv, charRotation.UpVector)
- end
- -- get state
- self.StateTracker:OnStep(self.GravityUp, self:IsGrounded(), isInputMoving)
- -- update values
- self.VForce.Force = walkForce + gForce
- self.Gyro.CFrame = charRotation
- end
- return GravityController
- end
- function _Draw3D()
- local module = {}
- -- Style Guide
- module.StyleGuide = {
- Point = {
- Thickness = 0.5;
- Color = Color3.new(0, 1, 0);
- },
- Line = {
- Thickness = 0.1;
- Color = Color3.new(1, 1, 0);
- },
- Ray = {
- Thickness = 0.1;
- Color = Color3.new(1, 0, 1);
- },
- Triangle = {
- Thickness = 0.05;
- };
- CFrame = {
- Thickness = 0.1;
- RightColor3 = Color3.new(1, 0, 0);
- UpColor3 = Color3.new(0, 1, 0);
- BackColor3 = Color3.new(0, 0, 1);
- PartProperties = {
- Material = Enum.Material.SmoothPlastic;
- };
- }
- }
- -- CONSTANTS
- local WEDGE = Instance.new("WedgePart")
- WEDGE.Material = Enum.Material.SmoothPlastic
- WEDGE.Anchored = true
- WEDGE.CanCollide = false
- local PART = Instance.new("Part")
- PART.Size = Vector3.new(0.1, 0.1, 0.1)
- PART.Anchored = true
- PART.CanCollide = false
- PART.TopSurface = Enum.SurfaceType.Smooth
- PART.BottomSurface = Enum.SurfaceType.Smooth
- PART.Material = Enum.Material.SmoothPlastic
- -- Functions
- local function draw(properties, style)
- local part = PART:Clone()
- for k, v in next, properties do
- part[k] = v
- end
- if (style) then
- for k, v in next, style do
- if (k ~= "Thickness") then
- part[k] = v
- end
- end
- end
- return part
- end
- function module.Draw(parent, properties)
- properties.Parent = parent
- return draw(properties, nil)
- end
- function module.Point(parent, cf_v3)
- local thickness = module.StyleGuide.Point.Thickness
- return draw({
- Size = Vector3.new(thickness, thickness, thickness);
- CFrame = (typeof(cf_v3) == "CFrame" and cf_v3 or CFrame.new(cf_v3));
- Parent = parent;
- }, module.StyleGuide.Point)
- end
- function module.Line(parent, a, b)
- local thickness = module.StyleGuide.Line.Thickness
- return draw({
- CFrame = CFrame.new((a + b)/2, b);
- Size = Vector3.new(thickness, thickness, (b - a).Magnitude);
- Parent = parent;
- }, module.StyleGuide.Line)
- end
- function module.Ray(parent, origin, direction)
- local thickness = module.StyleGuide.Ray.Thickness
- return draw({
- CFrame = CFrame.new(origin + direction/2, origin + direction);
- Size = Vector3.new(thickness, thickness, direction.Magnitude);
- Parent = parent;
- }, module.StyleGuide.Ray)
- end
- function module.Triangle(parent, a, b, c)
- local ab, ac, bc = b - a, c - a, c - b
- local abd, acd, bcd = ab:Dot(ab), ac:Dot(ac), bc:Dot(bc)
- if (abd > acd and abd > bcd) then
- c, a = a, c
- elseif (acd > bcd and acd > abd) then
- a, b = b, a
- end
- ab, ac, bc = b - a, c - a, c - b
- local right = ac:Cross(ab).Unit
- local up = bc:Cross(right).Unit
- local back = bc.Unit
- local height = math.abs(ab:Dot(up))
- local width1 = math.abs(ab:Dot(back))
- local width2 = math.abs(ac:Dot(back))
- local thickness = module.StyleGuide.Triangle.Thickness
- local w1 = WEDGE:Clone()
- w1.Size = Vector3.new(thickness, height, width1)
- w1.CFrame = CFrame.fromMatrix((a + b)/2, right, up, back)
- w1.Parent = parent
- local w2 = WEDGE:Clone()
- w2.Size = Vector3.new(thickness, height, width2)
- w2.CFrame = CFrame.fromMatrix((a + c)/2, -right, up, -back)
- w2.Parent = parent
- for k, v in next, module.StyleGuide.Triangle do
- if (k ~= "Thickness") then
- w1[k] = v
- w2[k] = v
- end
- end
- return w1, w2
- end
- function module.CFrame(parent, cf)
- local origin = cf.Position
- local r = cf.RightVector
- local u = cf.UpVector
- local b = -cf.LookVector
- local thickness = module.StyleGuide.CFrame.Thickness
- local right = draw({
- CFrame = CFrame.new(origin + r/2, origin + r);
- Size = Vector3.new(thickness, thickness, r.Magnitude);
- Color = module.StyleGuide.CFrame.RightColor3;
- Parent = parent;
- }, module.StyleGuide.CFrame.PartProperties)
- local up = draw({
- CFrame = CFrame.new(origin + u/2, origin + u);
- Size = Vector3.new(thickness, thickness, r.Magnitude);
- Color = module.StyleGuide.CFrame.UpColor3;
- Parent = parent;
- }, module.StyleGuide.CFrame.PartProperties)
- local back = draw({
- CFrame = CFrame.new(origin + b/2, origin + b);
- Size = Vector3.new(thickness, thickness, u.Magnitude);
- Color = module.StyleGuide.CFrame.BackColor3;
- Parent = parent;
- }, module.StyleGuide.CFrame.PartProperties)
- return right, up, back
- end
- -- Return
- return module
- end
- function _Draw2D()
- local module = {}
- -- Style Guide
- module.StyleGuide = {
- Point = {
- BorderSizePixel = 0;
- Size = UDim2.new(0, 4, 0, 4);
- BorderColor3 = Color3.new(0, 0, 0);
- BackgroundColor3 = Color3.new(0, 1, 0);
- },
- Line = {
- Thickness = 1;
- BorderSizePixel = 0;
- BorderColor3 = Color3.new(0, 0, 0);
- BackgroundColor3 = Color3.new(0, 1, 0);
- },
- Ray = {
- Thickness = 1;
- BorderSizePixel = 0;
- BorderColor3 = Color3.new(0, 0, 0);
- BackgroundColor3 = Color3.new(0, 1, 0);
- },
- Triangle = {
- ImageTransparency = 0;
- ImageColor3 = Color3.new(0, 1, 0);
- }
- }
- -- CONSTANTS
- local HALF = Vector2.new(0.5, 0.5)
- local RIGHT = "rbxassetid://2798177521"
- local LEFT = "rbxassetid://2798177955"
- local IMG = Instance.new("ImageLabel")
- IMG.BackgroundTransparency = 1
- IMG.AnchorPoint = HALF
- IMG.BorderSizePixel = 0
- local FRAME = Instance.new("Frame")
- FRAME.BorderSizePixel = 0
- FRAME.Size = UDim2.new(0, 0, 0, 0)
- FRAME.BackgroundColor3 = Color3.new(1, 1, 1)
- -- Functions
- function draw(properties, style)
- local frame = FRAME:Clone()
- for k, v in next, properties do
- frame[k] = v
- end
- if (style) then
- for k, v in next, style do
- if (k ~= "Thickness") then
- frame[k] = v
- end
- end
- end
- return frame
- end
- function module.Draw(parent, properties)
- properties.Parent = parent
- return draw(properties, nil)
- end
- function module.Point(parent, v2)
- return draw({
- AnchorPoint = HALF;
- Position = UDim2.new(0, v2.x, 0, v2.y);
- Parent = parent;
- }, module.StyleGuide.Point)
- end
- function module.Line(parent, a, b)
- local v = (b - a)
- local m = (a + b)/2
- return draw({
- AnchorPoint = HALF;
- Position = UDim2.new(0, m.x, 0, m.y);
- Size = UDim2.new(0, module.StyleGuide.Line.Thickness, 0, v.magnitude);
- Rotation = math.deg(math.atan2(v.y, v.x)) - 90;
- BackgroundColor3 = Color3.new(1, 1, 0);
- Parent = parent;
- }, module.StyleGuide.Line)
- end
- function module.Ray(parent, origin, direction)
- local a, b = origin, origin + direction
- local v = (b - a)
- local m = (a + b)/2
- return draw({
- AnchorPoint = HALF;
- Position = UDim2.new(0, m.x, 0, m.y);
- Size = UDim2.new(0, module.StyleGuide.Ray.Thickness, 0, v.magnitude);
- Rotation = math.deg(math.atan2(v.y, v.x)) - 90;
- Parent = parent;
- }, module.StyleGuide.Ray)
- end
- function module.Triangle(parent, a, b, c)
- local ab, ac, bc = b - a, c - a, c - b
- local abd, acd, bcd = ab:Dot(ab), ac:Dot(ac), bc:Dot(bc)
- if (abd > acd and abd > bcd) then
- c, a = a, c
- elseif (acd > bcd and acd > abd) then
- a, b = b, a
- end
- ab, ac, bc = b - a, c - a, c - b
- local unit = bc.unit
- local height = unit:Cross(ab)
- local flip = (height >= 0)
- local theta = math.deg(math.atan2(unit.y, unit.x)) + (flip and 0 or 180)
- local m1 = (a + b)/2
- local m2 = (a + c)/2
- local w1 = IMG:Clone()
- w1.Image = flip and RIGHT or LEFT
- w1.AnchorPoint = HALF
- w1.Size = UDim2.new(0, math.abs(unit:Dot(ab)), 0, height)
- w1.Position = UDim2.new(0, m1.x, 0, m1.y)
- w1.Rotation = theta
- w1.Parent = parent
- local w2 = IMG:Clone()
- w2.Image = flip and LEFT or RIGHT
- w2.AnchorPoint = HALF
- w2.Size = UDim2.new(0, math.abs(unit:Dot(ac)), 0, height)
- w2.Position = UDim2.new(0, m2.x, 0, m2.y)
- w2.Rotation = theta
- w2.Parent = parent
- for k, v in next, module.StyleGuide.Triangle do
- w1[k] = v
- w2[k] = v
- end
- return w1, w2
- end
- -- Return
- return module
- end
- function _DrawClass()
- local Draw2DModule = _Draw2D()
- local Draw3DModule = _Draw3D()
- --
- local DrawClass = {}
- local DrawClassStorage = setmetatable({}, {__mode = "k"})
- DrawClass.__index = DrawClass
- function DrawClass.new(parent)
- local self = setmetatable({}, DrawClass)
- self.Parent = parent
- DrawClassStorage[self] = {}
- self.Draw3D = {}
- for key, func in next, Draw3DModule do
- self.Draw3D[key] = function(...)
- local returns = {func(self.Parent, ...)}
- for i = 1, #returns do
- table.insert(DrawClassStorage[self], returns[i])
- end
- return unpack(returns)
- end
- end
- self.Draw2D = {}
- for key, func in next, Draw2DModule do
- self.Draw2D[key] = function(...)
- local returns = {func(self.Parent, ...)}
- for i = 1, #returns do
- table.insert(DrawClassStorage[self], returns[i])
- end
- return unpack(returns)
- end
- end
- return self
- end
- --
- function DrawClass:Clear()
- local t = DrawClassStorage[self]
- while (#t > 0) do
- local part = table.remove(t)
- if (part) then
- part:Destroy()
- end
- end
- DrawClassStorage[self] = {}
- end
- --
- return DrawClass
- end
- --END TEST
- local PLAYERS = game:GetService("Players")
- local GravityController = _GravityController()
- local Controller = GravityController.new(PLAYERS.LocalPlayer)
- local DrawClass = _DrawClass()
- local PI2 = math.pi*2
- local ZERO = Vector3.new(0, 0, 0)
- local LOWER_RADIUS_OFFSET = 3
- local NUM_DOWN_RAYS = 24
- local ODD_DOWN_RAY_START_RADIUS = 3
- local EVEN_DOWN_RAY_START_RADIUS = 2
- local ODD_DOWN_RAY_END_RADIUS = 1.66666
- local EVEN_DOWN_RAY_END_RADIUS = 1
- local NUM_FEELER_RAYS = 9
- local FEELER_LENGTH = 2
- local FEELER_START_OFFSET = 2
- local FEELER_RADIUS = 3.5
- local FEELER_APEX_OFFSET = 1
- local FEELER_WEIGHTING = 8
- function GetGravityUp(self, oldGravityUp)
- local ignoreList = {}
- for i, player in next, PLAYERS:GetPlayers() do
- ignoreList[i] = player.Character
- end
- -- get the normal
- local hrpCF = self.HRP.CFrame
- local isR15 = (self.Humanoid.RigType == Enum.HumanoidRigType.R15)
- local origin = isR15 and hrpCF.p or hrpCF.p + 0.35*oldGravityUp
- local radialVector = math.abs(hrpCF.LookVector:Dot(oldGravityUp)) < 0.999 and hrpCF.LookVector:Cross(oldGravityUp) or hrpCF.RightVector:Cross(oldGravityUp)
- local centerRayLength = 25
- local centerRay = Ray.new(origin, -centerRayLength * oldGravityUp)
- local centerHit, centerHitPoint, centerHitNormal = workspace:FindPartOnRayWithIgnoreList(centerRay, ignoreList)
- --[[disable
- DrawClass:Clear()
- DrawClass.Draw3D.Ray(centerRay.Origin, centerRay.Direction)
- ]]
- local downHitCount = 0
- local totalHitCount = 0
- local centerRayHitCount = 0
- local evenRayHitCount = 0
- local oddRayHitCount = 0
- local mainDownNormal = ZERO
- if (centerHit) then
- mainDownNormal = centerHitNormal
- centerRayHitCount = 0
- end
- local downRaySum = ZERO
- for i = 1, NUM_DOWN_RAYS do
- local dtheta = PI2 * ((i-1)/NUM_DOWN_RAYS)
- local angleWeight = 0.25 + 0.75 * math.abs(math.cos(dtheta))
- local isEvenRay = (i%2 == 0)
- local startRadius = isEvenRay and EVEN_DOWN_RAY_START_RADIUS or ODD_DOWN_RAY_START_RADIUS
- local endRadius = isEvenRay and EVEN_DOWN_RAY_END_RADIUS or ODD_DOWN_RAY_END_RADIUS
- local downRayLength = centerRayLength
- local offset = CFrame.fromAxisAngle(oldGravityUp, dtheta) * radialVector
- local dir = (LOWER_RADIUS_OFFSET * -oldGravityUp + (endRadius - startRadius) * offset)
- local ray = Ray.new(origin + startRadius * offset, downRayLength * dir.unit)
- local hit, hitPoint, hitNormal = workspace:FindPartOnRayWithIgnoreList(ray, ignoreList)
- --[[disable
- DrawClass.Draw3D.Ray(ray.Origin, ray.Direction)
- ]]
- if (hit) then
- downRaySum = downRaySum + angleWeight * hitNormal
- downHitCount = downHitCount + 1
- if isEvenRay then
- evenRayHitCount = evenRayHitCount + 1
- else
- oddRayHitCount = oddRayHitCount + 1
- end
- end
- end
- local feelerHitCount = 0
- local feelerNormalSum = ZERO
- for i = 1, NUM_FEELER_RAYS do
- local dtheta = 2 * math.pi * ((i-1)/NUM_FEELER_RAYS)
- local angleWeight = 0.25 + 0.75 * math.abs(math.cos(dtheta))
- local offset = CFrame.fromAxisAngle(oldGravityUp, dtheta) * radialVector
- local dir = (FEELER_RADIUS * offset + LOWER_RADIUS_OFFSET * -oldGravityUp).unit
- local feelerOrigin = origin - FEELER_APEX_OFFSET * -oldGravityUp + FEELER_START_OFFSET * dir
- local ray = Ray.new(feelerOrigin, FEELER_LENGTH * dir)
- local hit, hitPoint, hitNormal = workspace:FindPartOnRayWithIgnoreList(ray, ignoreList)
- --[[disable
- DrawClass.Draw3D.Ray(ray.Origin, ray.Direction)
- ]]
- if (hit) then
- feelerNormalSum = feelerNormalSum + FEELER_WEIGHTING * angleWeight * hitNormal --* hitDistSqInv
- feelerHitCount = feelerHitCount + 1
- end
- end
- if (centerRayHitCount + downHitCount + feelerHitCount > 0) then
- local normalSum = mainDownNormal + downRaySum + feelerNormalSum
- if (normalSum ~= ZERO) then
- return normalSum.unit
- end
- end
- return oldGravityUp
- end
- Controller.GetGravityUp = GetGravityUp
- -- E is toggle
- game:GetService("ContextActionService"):BindAction("Toggle", function(action, state, input)
- if not (state == Enum.UserInputState.Begin) then
- return
- end
- if (Controller) then
- Controller:Destroy()
- Controller = nil
- else
- Controller = GravityController.new(PLAYERS.LocalPlayer)
- Controller.GetGravityUp = GetGravityUp
- end
- end, false, Enum.KeyCode.Z)
- print("end")
Add Comment
Please, Sign In to add comment