From b55ad50b9ff7db3db481194c8233a7209dd19d12 Mon Sep 17 00:00:00 2001 From: riperiperi Date: Mon, 24 Jun 2019 23:24:01 +0100 Subject: [PATCH] [UI, World] Improvements to placement, category limits. - Multifloor targeting: Interactions, Go Here and Object Movement can now be done across all visible floors, rather than just the top. (build tools are unchanged) - Lot Category Improvements: Objects with specific category limitations now show them in the catalog. Attempting to buy them shows a warning (that can be dismissed) - The heatwave has intensified, "blended" snow lots are now properly grass. - Fix to 2 tile casino bar. --- .../Servers/Lot/Domain/LotContainer.cs | 2 + .../tso.client/UI/Panels/UILotControl.cs | 22 +++- .../tso.client/UI/Panels/UIObjectHolder.cs | 113 ++++++++++++------ .../tso.client/UI/Panels/UIQueryPanel.cs | 31 +++-- .../Content/Objects/Casino_2-Tile_Bar_CC.iff | Bin 895536 -> 895510 bytes .../english.dir/_f115_neighpagestrings.cst | 4 +- TSOClient/tso.world/RC/World2DRC.cs | 4 +- TSOClient/tso.world/World.cs | 40 +++++-- TSOClient/tso.world/World2D.cs | 3 +- .../tso.world/components/TerrainComponent.cs | 5 +- TSOClient/tso.world/model/Blueprint.cs | 5 + 11 files changed, 165 insertions(+), 64 deletions(-) diff --git a/TSOClient/FSO.Server/Servers/Lot/Domain/LotContainer.cs b/TSOClient/FSO.Server/Servers/Lot/Domain/LotContainer.cs index 07d5587f8..160ff4374 100644 --- a/TSOClient/FSO.Server/Servers/Lot/Domain/LotContainer.cs +++ b/TSOClient/FSO.Server/Servers/Lot/Domain/LotContainer.cs @@ -108,6 +108,8 @@ public class LotContainer 0x50907E06, //flies - controller 0x3161BB5B, //job controller + 0x475CC813, //water balloon controller + 0x5157DDF2, //cat carrier 0x3278BD34, //dog carrier }; diff --git a/TSOClient/tso.client/UI/Panels/UILotControl.cs b/TSOClient/tso.client/UI/Panels/UILotControl.cs index e6b19880e..7d01cf940 100644 --- a/TSOClient/tso.client/UI/Panels/UILotControl.cs +++ b/TSOClient/tso.client/UI/Panels/UILotControl.cs @@ -422,6 +422,18 @@ private void OnMouse(UIMouseEventType type, UpdateState state) } } + private short GetFloorBlockableHover(Point pt) + { + var tilePos = World.EstTileAtPosWithScroll3D(new Vector2(pt.X, pt.Y) / FSOEnvironment.DPIScaleFactor); + var newHover = World.GetObjectIDAtScreenPos(pt.X, + pt.Y, + GameFacade.GraphicsDevice); + + var hobj = vm.GetObjectById(newHover); + if (hobj == null || hobj.Position.Level < tilePos.Z) newHover = 0; + return newHover; + } + public void Click(Point pt, UpdateState state) { if (!LiveMode) @@ -434,9 +446,9 @@ public void Click(Point pt, UpdateState state) { VMEntity obj; //get new pie menu, make new pie menu panel for it - var tilePos = World.EstTileAtPosWithScroll(new Vector2(pt.X, pt.Y) / FSOEnvironment.DPIScaleFactor); + var tilePos = World.EstTileAtPosWithScroll3D(new Vector2(pt.X, pt.Y) / FSOEnvironment.DPIScaleFactor); - LotTilePos targetPos = LotTilePos.FromBigTile((short)tilePos.X, (short)tilePos.Y, World.State.Level); + LotTilePos targetPos = LotTilePos.FromBigTile((short)tilePos.X, (short)tilePos.Y, (sbyte)tilePos.Z); if (vm.Context.SolidToAvatars(targetPos).Solid) targetPos = LotTilePos.OUT_OF_WORLD; GotoObject.SetPosition(targetPos, Direction.NORTH, vm.Context); @@ -445,6 +457,8 @@ public void Click(Point pt, UpdateState state) pt.Y, GameFacade.GraphicsDevice); + var hobj = vm.GetObjectById(newHover); + if (hobj == null || hobj.Position.Level < tilePos.Z) newHover = 0; ObjectHover = newHover; bool objSelected = ObjectHover > 0; @@ -574,9 +588,7 @@ public void LiveModeUpdate(UpdateState state, bool scrolled) OldMX = state.MouseState.X; OldMY = state.MouseState.Y; var scaled = GetScaledPoint(state.MouseState.Position); - var newHover = World.GetObjectIDAtScreenPos(scaled.X, - scaled.Y, - GameFacade.GraphicsDevice); + var newHover = GetFloorBlockableHover(scaled); if (ObjectHover != newHover) { diff --git a/TSOClient/tso.client/UI/Panels/UIObjectHolder.cs b/TSOClient/tso.client/UI/Panels/UIObjectHolder.cs index 5859ec49c..9c382d2c9 100644 --- a/TSOClient/tso.client/UI/Panels/UIObjectHolder.cs +++ b/TSOClient/tso.client/UI/Panels/UIObjectHolder.cs @@ -49,6 +49,7 @@ public class UIObjectHolder //controls the object holder interface public bool Roommate; public bool DonateMode; + private bool Locked; public event HolderEventHandler OnPickup; public event HolderEventHandler OnDelete; @@ -131,7 +132,7 @@ public void MoveSelected(Vector2 pos, sbyte level) { for (int i = 0; i < 4; i++) { - status = Holding.Group.ChangePosition(LotTilePos.FromBigTile((short)pos.X, (short)pos.Y, World.State.Level), dir, vm.Context, VMPlaceRequestFlags.UserPlacement).Status; + status = Holding.Group.ChangePosition(LotTilePos.FromBigTile((short)pos.X, (short)pos.Y, level), dir, vm.Context, VMPlaceRequestFlags.UserPlacement).Status; if (status != VMPlacementError.MustBeAgainstWall) break; dir = (Direction)((((int)dir << 6) & 255) | ((int)dir >> 2)); } @@ -143,7 +144,7 @@ public void MoveSelected(Vector2 pos, sbyte level) Holding.Group.ChangePosition(LotTilePos.OUT_OF_WORLD, Holding.Dir, vm.Context, VMPlaceRequestFlags.UserPlacement); Holding.Group.SetVisualPosition(new Vector3(pos, - (((Holding.Group.Objects[0].GetValue(VMStackObjectVariable.AllowedHeightFlags) & 1) == 1) ? 0 : 4f / 5f) + (World.State.Level-1)*2.95f), + (((Holding.Group.Objects[0].GetValue(VMStackObjectVariable.AllowedHeightFlags) & 1) == 1) ? 0 : 4f / 5f) + (level-1)*2.95f), //^ if we can't be placed on the floor, default to table height. Holding.Dir, vm.Context); } @@ -152,7 +153,7 @@ public void MoveSelected(Vector2 pos, sbyte level) { var target = Holding.Group.Objects[i]; var tpos = target.VisualPosition; - tpos.Z = (World.State.Level - 1)*2.95f; + tpos.Z = (level - 1)*2.95f; Holding.CursorTiles[i].MultitileGroup.SetVisualPosition(tpos, Holding.Dir, vm.Context); } Holding.CanPlace = status; @@ -209,6 +210,37 @@ public void MouseDown(UpdateState state) } } + private void InventoryPlaceHolding() + { + var pos = Holding.Group.BaseObject.Position; + vm.SendCommand(new VMNetPlaceInventoryCmd + { + ObjectPID = Holding.InventoryPID, + dir = Holding.Dir, + level = pos.Level, + x = pos.x, + y = pos.y, + + Mode = (DonateMode) ? PurchaseMode.Donate : PurchaseMode.Normal + }); + } + + private void BuyHolding() + { + var pos = Holding.Group.BaseObject.Position; + var GUID = (Holding.Group.MultiTile) ? Holding.Group.BaseObject.MasterDefinition.GUID : Holding.Group.BaseObject.Object.OBJ.GUID; + vm.SendCommand(new VMNetBuyObjectCmd + { + GUID = GUID, + dir = Holding.Dir, + level = pos.Level, + x = pos.x, + y = pos.y, + + Mode = (DonateMode) ? PurchaseMode.Donate : PurchaseMode.Normal + }); + } + public void MouseUp(UpdateState state) { MouseIsDown = false; @@ -216,12 +248,13 @@ public void MouseUp(UpdateState state) { if (Holding.CanPlace == VMPlacementError.Success) { - HITVM.Get().PlaySoundEvent((Holding.IsBought) ? UISounds.ObjectMovePlace : UISounds.ObjectPlace); //ExecuteEntryPoint(11); //User Placement var putDown = Holding; var pos = Holding.Group.BaseObject.Position; + var badCategory = ((Holding.Group.BaseObject as VMGameObject)?.Disabled ?? 0).HasFlag(VMGameObjectDisableFlags.LotCategoryWrong); if (Holding.IsBought) { + HITVM.Get().PlaySoundEvent(UISounds.ObjectMovePlace); vm.SendCommand(new VMNetMoveObjectCmd { ObjectID = Holding.MoveTarget, @@ -231,32 +264,29 @@ public void MouseUp(UpdateState state) y = pos.y }); } - else if (Holding.InventoryPID > 0) - { - vm.SendCommand(new VMNetPlaceInventoryCmd + else { + if (badCategory) { - ObjectPID = Holding.InventoryPID, - dir = Holding.Dir, - level = pos.Level, - x = pos.x, - y = pos.y, - - Mode = (DonateMode) ? PurchaseMode.Donate : PurchaseMode.Normal - }); - } - else - { - var GUID = (Holding.Group.MultiTile)? Holding.Group.BaseObject.MasterDefinition.GUID : Holding.Group.BaseObject.Object.OBJ.GUID; - vm.SendCommand(new VMNetBuyObjectCmd + Locked = true; + UIAlert.YesNo(GameFacade.Strings.GetString("245", "5"), GameFacade.Strings.GetString("245", (Holding.InventoryPID > 0)?"7":"6"), true, + (confirm) => + { + Locked = false; + if (!confirm) return; + HITVM.Get().PlaySoundEvent(UISounds.ObjectPlace); + if (Holding.InventoryPID > 0) InventoryPlaceHolding(); + else BuyHolding(); + ClearSelected(); + if (OnPutDown != null) OnPutDown(putDown, state); //call this after so that buy mode etc can produce more. + }); + return; + } else { - GUID = GUID, - dir = Holding.Dir, - level = pos.Level, - x = pos.x, - y = pos.y, - - Mode = (DonateMode) ? PurchaseMode.Donate : PurchaseMode.Normal - }); + HITVM.Get().PlaySoundEvent(UISounds.ObjectPlace); + if (Holding.InventoryPID > 0) InventoryPlaceHolding(); + else BuyHolding(); + } + } ClearSelected(); if (OnPutDown != null) OnPutDown(putDown, state); //call this after so that buy mode etc can produce more. @@ -416,11 +446,24 @@ private Point GetScaledPoint(Point TapPoint) return ((TapPoint - screenMiddle).ToVector2() / World.BackbufferScale).ToPoint() + screenMiddle; } + private short GetFloorBlockableHover(Point pt) + { + var tilePos = World.EstTileAtPosWithScroll3D(new Vector2(pt.X, pt.Y) / FSOEnvironment.DPIScaleFactor); + var newHover = World.GetObjectIDAtScreenPos(pt.X, + pt.Y, + GameFacade.GraphicsDevice); + + var hobj = vm.GetObjectById(newHover); + if (hobj == null || hobj.Position.Level < tilePos.Z) newHover = 0; + return newHover; + } + public void Update(UpdateState state, bool scrolled) { LastState = state; if (ShowTooltip) state.UIState.TooltipProperties.UpdateDead = false; MouseClicked = (MouseIsDown && (!MouseWasDown)); + if (Locked) return; CursorType cur = CursorType.SimsMove; if (Holding != null) @@ -455,8 +498,8 @@ public void Update(UpdateState state, bool scrolled) cur = CursorType.SimsRotate; if (Math.Sqrt(xDiff * xDiff + yDiff * yDiff) > 64) { - var from = World.EstTileAtPosWithScroll(new Vector2(MouseDownX, MouseDownY)); - var target = World.EstTileAtPosWithScroll(state.MouseState.Position.ToVector2()); + var from = World.EstTileAtPosWithScroll(new Vector2(MouseDownX, MouseDownY), Holding.Level); + var target = World.EstTileAtPosWithScroll(state.MouseState.Position.ToVector2(), Holding.Level); var vec = target - from; var dir = Math.Atan2(vec.Y, vec.X); @@ -503,15 +546,15 @@ public void Update(UpdateState state, bool scrolled) else { var scaled = GetScaledPoint(state.MouseState.Position); - var tilePos = World.EstTileAtPosWithScroll(new Vector2(scaled.X, scaled.Y) / FSOEnvironment.DPIScaleFactor) + Holding.TilePosOffset; - MoveSelected(tilePos, 1); + var tilePos = World.EstTileAtPosWithScroll3D(new Vector2(scaled.X, scaled.Y) / FSOEnvironment.DPIScaleFactor + Holding.MousePosOffset); + MoveSelected(new Vector2(tilePos.X, tilePos.Y), (sbyte)tilePos.Z); // + Holding.TilePosOffset } } else if (MouseClicked) { //not holding an object, but one can be selected var scaled = GetScaledPoint(state.MouseState.Position); - var newHover = World.GetObjectIDAtScreenPos(scaled.X, scaled.Y, GameFacade.GraphicsDevice); + var newHover = GetFloorBlockableHover(scaled); //World.GetObjectIDAtScreenPos(scaled.X, scaled.Y, GameFacade.GraphicsDevice); if (MouseClicked && (newHover != 0) && (vm.GetObjectById(newHover) is VMGameObject)) { var objGroup = vm.GetObjectById(newHover).MultitileGroup; @@ -519,7 +562,7 @@ public void Update(UpdateState state, bool scrolled) var allowMove = vm.PlatformState.Validator.CanMoveObject((VMAvatar)ParentControl.ActiveEntity, objGroup.BaseObject); var success = (Roommate || objGroup.SalePrice > -1)?objGroup.BaseObject.IsUserMovable(vm.Context, false): VMPlacementError.ObjectNotOwnedByYou; if (GameFacade.EnableMod) success = VMPlacementError.Success; - if (objBasePos.Level != World.State.Level) success = VMPlacementError.CantEffectFirstLevelFromSecondLevel; + //if (objBasePos.Level != World.State.Level) success = VMPlacementError.CantEffectFirstLevelFromSecondLevel; if (success == VMPlacementError.Success) { var ghostGroup = vm.Context.GhostCopyGroup(objGroup); @@ -533,6 +576,7 @@ public void Update(UpdateState state, bool scrolled) Holding.CanDelete = canDelete; Holding.DeleteError = canDelete ? VMPlacementError.CannotDeleteObject : VMPlacementError.ObjectNotOwnedByYou; Holding.MoveTarget = newHover; + Holding.MousePosOffset = (objGroup.BaseObject.WorldUI.GetScreenPos(World.State) - GetScaledPoint(state.MouseState.Position).ToVector2()) / FSOEnvironment.DPIScaleFactor; Holding.TilePosOffset = new Vector2(objBasePos.x / 16f, objBasePos.y / 16f) - World.EstTileAtPosWithScroll(GetScaledPoint(state.MouseState.Position).ToVector2() / FSOEnvironment.DPIScaleFactor); if (OnPickup != null) OnPickup(Holding, state); //ExecuteEntryPoint(12); //User Pickup @@ -578,6 +622,7 @@ public class UIObjectSelection public Direction Dir = Direction.NORTH; public Vector2 TilePos; public Vector2 TilePosOffset; + public Vector2 MousePosOffset; public bool Clicked; public VMPlacementError CanPlace; public sbyte Level; diff --git a/TSOClient/tso.client/UI/Panels/UIQueryPanel.cs b/TSOClient/tso.client/UI/Panels/UIQueryPanel.cs index cf00e5efa..4a34ec09b 100644 --- a/TSOClient/tso.client/UI/Panels/UIQueryPanel.cs +++ b/TSOClient/tso.client/UI/Panels/UIQueryPanel.cs @@ -220,6 +220,7 @@ public int Tab } private string[] AdStrings; + private string[] CategoryStrings; private int _Mode; public int Mode @@ -263,6 +264,13 @@ public UIQueryPanel(UILotControl parent, LotView.World world) { AdStrings[i] = ((i<7)?str.Substring(0,str.Length-2)+"{0}":str) + "\r\n"; } + CategoryStrings = new string[11]; + for (int i = 0; i < 10; i++) + { + CategoryStrings[i] = GameFacade.Strings.GetString("f115", (i + 73).ToString()); + } + CategoryStrings[10] = GameFacade.Strings.GetString("f115", "98"); + var useSmall = (GlobalSettings.Default.GraphicsWidth < 1024) || FSOEnvironment.UIZoomFactor > 1f; var script = this.RenderScript("querypanel"+(useSmall?"":"1024")+".uis"); @@ -504,18 +512,25 @@ public void SetInfo(VM vm, VMEntity entity, bool bought) { motivesString.AppendFormat(GameFacade.Strings.GetString("206", "19") + "${0}\r\n", price); } - if (def.RatingHunger != 0) { motivesString.AppendFormat(AdStrings[0], def.RatingHunger); } - if (def.RatingComfort != 0) { motivesString.AppendFormat(AdStrings[1], def.RatingComfort); } - if (def.RatingHygiene != 0) { motivesString.AppendFormat(AdStrings[2], def.RatingHygiene); } - if (def.RatingBladder != 0) { motivesString.AppendFormat(AdStrings[3], def.RatingBladder); } - if (def.RatingEnergy != 0) { motivesString.AppendFormat(AdStrings[4], def.RatingEnergy); } - if (def.RatingFun != 0) { motivesString.AppendFormat(AdStrings[5], def.RatingFun); } - if (def.RatingRoom != 0) { motivesString.AppendFormat(AdStrings[6], def.RatingRoom); } + + var catFlags = def.LotCategories; + for (int i = 1; i < 12; i++) + { + if ((catFlags & (1 << i)) > 0) motivesString.AppendLine(CategoryStrings[i - 1]); + } + + if (def.RatingHunger != 0) { motivesString.AppendFormat(AdStrings[0], (short)def.RatingHunger); } + if (def.RatingComfort != 0) { motivesString.AppendFormat(AdStrings[1], (short)def.RatingComfort); } + if (def.RatingHygiene != 0) { motivesString.AppendFormat(AdStrings[2], (short)def.RatingHygiene); } + if (def.RatingBladder != 0) { motivesString.AppendFormat(AdStrings[3], (short)def.RatingBladder); } + if (def.RatingEnergy != 0) { motivesString.AppendFormat(AdStrings[4], (short)def.RatingEnergy); } + if (def.RatingFun != 0) { motivesString.AppendFormat(AdStrings[5], (short)def.RatingFun); } + if (def.RatingRoom != 0) { motivesString.AppendFormat(AdStrings[6], (short)def.RatingRoom); } var sFlags = def.RatingSkillFlags; for (int i = 0; i < 7; i++) { - if ((sFlags & (1 << i)) > 0) motivesString.Append(AdStrings[i+7]); + if ((sFlags & (1 << i)) > 0) motivesString.Append(AdStrings[i + 7]); } MotivesText.CurrentText = motivesString.ToString(); diff --git a/TSOClient/tso.content/Content/Objects/Casino_2-Tile_Bar_CC.iff b/TSOClient/tso.content/Content/Objects/Casino_2-Tile_Bar_CC.iff index 6c5b521f7050d7bc3b95a47e47fd24f660093f15..1c49e540318c562102f8b82e90916a0464cc18db 100644 GIT binary patch delta 3457 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 p_y65m;S state.Level || !obj.Visible || obj.CutawayHidden) continue; var objR = (ObjectComponentRC)obj; var intr = objR.IntersectsBounds(ray); if (obj.Container != null && intr != null) intr = intr.Value - 1.5f; @@ -255,7 +255,7 @@ public override short GetObjectIDAtScreenPos(int x, int y, GraphicsDevice gd, Wo foreach (var sim in Blueprint.Avatars) { - if (!sim.Visible) continue; + if (!sim.Visible || sim.Level > state.Level) continue; var pos = sim.GetPelvisPosition()*3; pos = new Vector3(pos.X, pos.Z, pos.Y) + new Vector3(1.5f, 0, 1.5f); var box = new BoundingBox(pos - new Vector3(0.5f, 2, 0.5f), pos + new Vector3(0.5f, 2, 0.5f)); diff --git a/TSOClient/tso.world/World.cs b/TSOClient/tso.world/World.cs index cd9b9e3d9..92e9b6d9c 100644 --- a/TSOClient/tso.world/World.cs +++ b/TSOClient/tso.world/World.cs @@ -653,11 +653,12 @@ public void Force2DPredraw(GraphicsDevice device) return tMin; } - public Vector2 EstTileAtPosWithScroll(Vector2 pos) + public Vector2 EstTileAtPosWithScroll(Vector2 pos, sbyte level = -1) { + if (level == -1) level = State.Level; pos *= new Vector2(FSOEnvironment.DPIScaleFactor); var sPos = new Vector3(pos, 0); - + var p1 = State.Device.Viewport.Unproject(sPos, State.Camera.Projection, State.Camera.View, Matrix.Identity); sPos.Z = 1; var p2 = State.Device.Viewport.Unproject(sPos, State.Camera.Projection, State.Camera.View, Matrix.Identity); @@ -665,9 +666,9 @@ public Vector2 EstTileAtPosWithScroll(Vector2 pos) dir.Normalize(); var ray = new Ray(p1, p2 - p1); ray.Direction.Normalize(); - ray.Position -= new Vector3(0, (State.Level-1) * 2.95f * 3, 0); - - var baseBox = new BoundingBox(new Vector3(0, -5000, 0), new Vector3(Blueprint.Width*3, 5000, Blueprint.Height*3)); + ray.Position -= new Vector3(0, (level - 1) * 2.95f * 3, 0); + + var baseBox = new BoundingBox(new Vector3(0, -5000, 0), new Vector3(Blueprint.Width * 3, 5000, Blueprint.Height * 3)); if (baseBox.Contains(ray.Position) != ContainmentType.Contains) { //move ray start inside box @@ -685,15 +686,15 @@ public Vector2 EstTileAtPosWithScroll(Vector2 pos) var py = (ray.Direction.Z > 0); int iteration = 0; - while (mx >= 0 && mx < Blueprint.Width && my >= 0 && my= 0 && mx < Blueprint.Width && my >= 0 && my < Blueprint.Width) { //test triangle 1. (centre of tile down xz, we lean towards positive x) var plane = new Plane( - new Vector3(mx * 3, Blueprint.GetAltPoint(mx, my) * Blueprint.TerrainFactor*3, my * 3), - new Vector3(mx * 3+3, Blueprint.GetAltPoint(mx+1, my) * Blueprint.TerrainFactor*3, my * 3), - new Vector3(mx * 3+3, Blueprint.GetAltPoint(mx+1, my+1) * Blueprint.TerrainFactor*3, my * 3+3) + new Vector3(mx * 3, Blueprint.GetAltPoint(mx, my) * Blueprint.TerrainFactor * 3, my * 3), + new Vector3(mx * 3 + 3, Blueprint.GetAltPoint(mx + 1, my) * Blueprint.TerrainFactor * 3, my * 3), + new Vector3(mx * 3 + 3, Blueprint.GetAltPoint(mx + 1, my + 1) * Blueprint.TerrainFactor * 3, my * 3 + 3) ); - var tBounds = new BoundingBox(new Vector3(mx*3, -5000, my*3), new Vector3(mx*3+3, 5000, my*3+3)); + var tBounds = new BoundingBox(new Vector3(mx * 3, -5000, my * 3), new Vector3(mx * 3 + 3, 5000, my * 3 + 3)); var t1 = ray.Intersects(plane); var t2 = BoxRC2(ray, 3); @@ -705,11 +706,12 @@ public Vector2 EstTileAtPosWithScroll(Vector2 pos) var tentative = ray.Position + ray.Direction * (t1.Value + 0.00001f); //did it hit the correct side of the triangle? - var mySide = ((tentative.X / 3)%1) - ((tentative.Z / 3)%1); + var mySide = ((tentative.X / 3) % 1) - ((tentative.Z / 3) % 1); if (mySide >= 0) { return new Vector2(tentative.X / 3, tentative.Z / 3); - } else + } + else { //test the other side (positive z) plane = new Plane( @@ -749,6 +751,20 @@ public Vector2 EstTileAtPosWithScroll(Vector2 pos) return new Vector2(0, 0); } + public Vector3 EstTileAtPosWithScroll3D(Vector2 pos, sbyte startFloor = -1) + { + if (startFloor == -1) startFloor = State.Level; + for (sbyte floor = startFloor; floor > 0; floor--) + { + var result = EstTileAtPosWithScroll(pos, floor); + if (floor == 1 || (Blueprint.TileInbounds(result) && Blueprint.GetFloor((short)result.X, (short)result.Y, floor).Pattern != 0)) + { + return new Vector3(result, floor); + } + } + return new Vector3(EstTileAtPosWithScroll(pos), State.Level); + } + /// /// Gets the ID of the object at a given position. /// diff --git a/TSOClient/tso.world/World2D.cs b/TSOClient/tso.world/World2D.cs index 4af1738b5..0301877da 100644 --- a/TSOClient/tso.world/World2D.cs +++ b/TSOClient/tso.world/World2D.cs @@ -125,7 +125,7 @@ public virtual short GetObjectIDAtScreenPos(int x, int y, GraphicsDevice gd, Wor var tilePosition = obj.Position; - if (obj.Level != state.Level) continue; + if (obj.Level > state.Level) continue; var oPx = state.WorldSpace.GetScreenFromTile(tilePosition); obj.ValidateSprite(state); @@ -143,6 +143,7 @@ public virtual short GetObjectIDAtScreenPos(int x, int y, GraphicsDevice gd, Wor state._3D.Begin(gd); foreach (var avatar in Blueprint.Avatars) { + if (avatar.Level > state.Level) continue; _2d.OffsetPixel(state.WorldSpace.GetScreenFromTile(avatar.Position)); _2d.OffsetTile(avatar.Position); avatar.Draw(gd, state); diff --git a/TSOClient/tso.world/components/TerrainComponent.cs b/TSOClient/tso.world/components/TerrainComponent.cs index 3e3d98e4c..9679bc6ef 100644 --- a/TSOClient/tso.world/components/TerrainComponent.cs +++ b/TSOClient/tso.world/components/TerrainComponent.cs @@ -96,7 +96,10 @@ public void ForceSnow(bool toGrass) if (LightType == TerrainType.SNOW) { LightType = TerrainType.GRASS; - if (DarkType == TerrainType.SNOW) DarkType = TerrainType.GRASS; + } + if (DarkType == TerrainType.SNOW) + { + DarkType = TerrainType.GRASS; } } else diff --git a/TSOClient/tso.world/model/Blueprint.cs b/TSOClient/tso.world/model/Blueprint.cs index 7f996e5e8..ed576c715 100644 --- a/TSOClient/tso.world/model/Blueprint.cs +++ b/TSOClient/tso.world/model/Blueprint.cs @@ -299,6 +299,11 @@ public FloorTile GetFloor(short tileX, short tileY, sbyte level) return Floors[level-1][offset]; } + public bool TileInbounds(Vector2 tile) + { + return (tile.X >= 0 && tile.Y >= 0 && tile.X < Width && tile.Y < Height); + } + public void ChangeObjectLocation(ObjectComponent component, LotTilePos pos) { short tileX = (pos.x < 0) ? (short)0 : pos.TileX;