From b17761559114b53c7e967b98f940f6f69d501f77 Mon Sep 17 00:00:00 2001 From: MaxED Date: Sat, 10 Sep 2016 20:45:41 +0000 Subject: [PATCH] Fixed: in some cases lines with only back side were created after dragging map elements when using "Merge Dragger Vertices Only" drag mode. Fixed: in some cases invalid sectors were created after dragging map elements when using "Merge Dragged Geometry" and "Replace with Dragged Geometry" drag modes, when at least one of dragged linedef was facing into an enclosed void area. Fixed a crash after drawing lines over a sector with unclosed/invalid boundaries. Fixed(?): probably fixed a crash when closing Script Editor with an unsaved file during program termination. --- Source/Core/Geometry/Tools.cs | 25 ++++++++------ Source/Core/Map/MapSet.cs | 44 ++++++++++++++----------- Source/Core/VisualModes/VisualMode.cs | 12 +++---- Source/Core/Windows/ScriptEditorForm.cs | 2 +- 4 files changed, 46 insertions(+), 37 deletions(-) diff --git a/Source/Core/Geometry/Tools.cs b/Source/Core/Geometry/Tools.cs index d8545c8d..96ad70c3 100644 --- a/Source/Core/Geometry/Tools.cs +++ b/Source/Core/Geometry/Tools.cs @@ -2256,11 +2256,14 @@ namespace CodeImp.DoomBuilder.Geometry private static bool SectorWasInvalid(Sector s) { - if(s.Sidedefs.Count < 3 || s.FlatVertices.Length < 3) + if(s.Sidedefs == null || s.Sidedefs.Count < 3 || s.FlatVertices.Length < 3) { // Collect changed lines HashSet changedlines = new HashSet(); - foreach(Sidedef side in s.Sidedefs) changedlines.Add(side.Line); + if(s.Sidedefs != null) + { + foreach(Sidedef side in s.Sidedefs) changedlines.Add(side.Line); + } // Delete sector s.Dispose(); @@ -2387,24 +2390,26 @@ namespace CodeImp.DoomBuilder.Geometry } //mxd - public static Sector FindPotentialSector(Linedef line, bool front) + public static Sector FindSectorContaining(Linedef line, bool front) { List sectorsides = FindPotentialSectorAt(line, front); if(sectorsides == null) return null; Sector result = null; + bool foundstartline = false; - // Proceed only if all sectorsides reference the same sector + // Proceed only if all sectorsides reference the same sector and the start line is among them foreach(LinedefSide sectorside in sectorsides) { Sidedef target = (sectorside.Front ? sectorside.Line.Front : sectorside.Line.Back); - if(target != null) - { - if(result == null) result = target.Sector; - else if(result != target.Sector) return null; // Fial... - } + if(target == null) return null; // Fial... + + if(result == null) result = target.Sector; + else if(result != target.Sector) return null; // Fial... + + if(sectorside.Line == line) foundstartline = true; } - return result; + return (foundstartline ? result : null); } #endregion diff --git a/Source/Core/Map/MapSet.cs b/Source/Core/Map/MapSet.cs index 12368fbc..0640ad05 100644 --- a/Source/Core/Map/MapSet.cs +++ b/Source/Core/Map/MapSet.cs @@ -2133,29 +2133,29 @@ namespace CodeImp.DoomBuilder.Map EndAddRemove(); + // Collect changed lines... We need those in by-vertex-index order + // (otherwise SectorBuilder logic in some cases will incorrectly assign sector propertes) + List markedverts = GetMarkedVertices(true); + List changedlines = new List(markedverts.Count / 2); + HashSet changedlineshash = new HashSet(); + foreach(Vertex v in markedverts) + { + foreach(Linedef l in v.Linedefs) + { + if(!changedlineshash.Contains(l)) + { + changedlines.Add(l); + changedlineshash.Add(l); + } + } + } + //mxd. Correct sector references if(mergemode != MergeGeometryMode.CLASSIC) { // Linedefs cache needs to be up to date... Update(true, false); - - // Collect changed lines... We need those in by-vertex-index order - // (otherwise SectorBuilder logic in some cases will incorrectly assign sector propertes) - List markedverts = GetMarkedVertices(true); - List changedlines = new List(markedverts.Count / 2); - HashSet changedlineshash = new HashSet(); - foreach(Vertex v in markedverts) - { - foreach(Linedef l in v.Linedefs) - { - if(!changedlineshash.Contains(l)) - { - changedlines.Add(l); - changedlineshash.Add(l); - } - } - } - + // Fix stuff... CorrectSectorReferences(changedlines, true); CorrectOuterSides(new HashSet(changedlines)); @@ -2165,6 +2165,10 @@ namespace CodeImp.DoomBuilder.Map HashSet changedsectors = GetSectorsFromLinedefs(changedlines); foreach(Sector s in changedsectors) s.Marked = true; } + else + { + FlipBackwardLinedefs(changedlines); + } return true; } @@ -2436,7 +2440,7 @@ namespace CodeImp.DoomBuilder.Map foreach(Linedef line in linesmissingfront) { // Line is now inside a sector? (check from the missing side!) - Sector nearest = Tools.FindPotentialSector(line, true); + Sector nearest = Tools.FindSectorContaining(line, true); // We can reattach our line! if(nearest != null) linefrontsectorref[line] = nearest; @@ -2446,7 +2450,7 @@ namespace CodeImp.DoomBuilder.Map foreach(Linedef line in linesmissingback) { // Line is now inside a sector? (check from the missing side!) - Sector nearest = Tools.FindPotentialSector(line, false); + Sector nearest = Tools.FindSectorContaining(line, false); // We can reattach our line! if(nearest != null) linebacksectorref[line] = nearest; diff --git a/Source/Core/VisualModes/VisualMode.cs b/Source/Core/VisualModes/VisualMode.cs index d4423e07..0defeeed 100644 --- a/Source/Core/VisualModes/VisualMode.cs +++ b/Source/Core/VisualModes/VisualMode.cs @@ -508,7 +508,7 @@ namespace CodeImp.DoomBuilder.VisualModes // This preforms visibility culling protected void DoCulling() { - Dictionary visiblelines = new Dictionary(200); + HashSet visiblelines = new HashSet(); Vector2D campos2d = General.Map.VisualCamera.Position; // Make collections @@ -528,10 +528,10 @@ namespace CodeImp.DoomBuilder.VisualModes foreach(Linedef ld in block.Lines) { // Line not already processed? - if(!visiblelines.ContainsKey(ld)) + if(!visiblelines.Contains(ld)) { // Add line if not added yet - visiblelines.Add(ld, ld); + visiblelines.Add(ld); // Which side of the line is the camera on? if(ld.SideOfLine(campos2d) < 0) @@ -565,12 +565,12 @@ namespace CodeImp.DoomBuilder.VisualModes { // Create new visual thing vt = CreateVisualThing(t); - allthings.Add(t, vt); + allthings[t] = vt; } if(vt != null && !visiblethings.ContainsKey(vt.Thing)) { - visiblethings.Add(vt.Thing, vt); + visiblethings[vt.Thing] = vt; } } } @@ -579,7 +579,7 @@ namespace CodeImp.DoomBuilder.VisualModes if(processgeometry) { // Find camera sector - Linedef nld = MapSet.NearestLinedef(visiblelines.Values, campos2d); + Linedef nld = MapSet.NearestLinedef(visiblelines, campos2d); if(nld != null) { General.Map.VisualCamera.Sector = GetCameraSectorFromLinedef(nld); diff --git a/Source/Core/Windows/ScriptEditorForm.cs b/Source/Core/Windows/ScriptEditorForm.cs index 64d1b7eb..274479c2 100644 --- a/Source/Core/Windows/ScriptEditorForm.cs +++ b/Source/Core/Windows/ScriptEditorForm.cs @@ -96,7 +96,7 @@ namespace CodeImp.DoomBuilder.Windows editor.SaveSettings(); // Only when closed by the user - if(!appclose) + if(!appclose && (e.CloseReason == CloseReason.UserClosing || e.CloseReason == CloseReason.FormOwnerClosing)) { // Remember if scipts are changed General.Map.ApplyScriptChanged();