|
@@ -1008,127 +1008,104 @@ void CZonePlacer::assignZones(vstd::RNG * rand)
|
|
void CZonePlacer::dropRandomRoads(vstd::RNG * rand)
|
|
void CZonePlacer::dropRandomRoads(vstd::RNG * rand)
|
|
{
|
|
{
|
|
auto zones = map.getZones();
|
|
auto zones = map.getZones();
|
|
|
|
+ bool anyDropped;
|
|
|
|
|
|
- //First, build a graph of road connections
|
|
|
|
- std::map<TRmgTemplateZoneId, std::set<TRmgTemplateZoneId>> roadGraph;
|
|
|
|
- std::vector<rmg::ZoneConnection> randomConnections;
|
|
|
|
- std::vector<rmg::ZoneConnection> fixedConnections;
|
|
|
|
-
|
|
|
|
- //Collect all road connections and build initial graph
|
|
|
|
- for(const auto & zone : zones)
|
|
|
|
- {
|
|
|
|
- for(const auto & connection : zone.second->getConnections())
|
|
|
|
- {
|
|
|
|
- if(connection.getRoadOption() == rmg::ERoadOption::ROAD_TRUE)
|
|
|
|
- {
|
|
|
|
- roadGraph[connection.getZoneA()].insert(connection.getZoneB());
|
|
|
|
- roadGraph[connection.getZoneB()].insert(connection.getZoneA());
|
|
|
|
- fixedConnections.push_back(connection);
|
|
|
|
- }
|
|
|
|
- else if(connection.getRoadOption() == rmg::ERoadOption::ROAD_RANDOM)
|
|
|
|
- {
|
|
|
|
- roadGraph[connection.getZoneA()].insert(connection.getZoneB());
|
|
|
|
- roadGraph[connection.getZoneB()].insert(connection.getZoneA());
|
|
|
|
- randomConnections.push_back(connection);
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
-
|
|
|
|
- //Find all connected components in the initial graph
|
|
|
|
- std::map<TRmgTemplateZoneId, int> zoneToComponent;
|
|
|
|
- int numComponents = 0;
|
|
|
|
-
|
|
|
|
- auto dfsComponent = [&](TRmgTemplateZoneId start, int component)
|
|
|
|
|
|
+ do
|
|
{
|
|
{
|
|
- std::stack<TRmgTemplateZoneId> stack;
|
|
|
|
- stack.push(start);
|
|
|
|
|
|
+ anyDropped = false;
|
|
|
|
+ std::map<TRmgTemplateZoneId, std::set<TRmgTemplateZoneId>> roadGraph;
|
|
|
|
+ std::set<rmg::ZoneConnection> randomConnections;
|
|
|
|
|
|
- while(!stack.empty())
|
|
|
|
|
|
+ //Build graph and collect random connections
|
|
|
|
+ for(const auto & zone : zones)
|
|
{
|
|
{
|
|
- auto current = stack.top();
|
|
|
|
- stack.pop();
|
|
|
|
-
|
|
|
|
- if(zoneToComponent.find(current) != zoneToComponent.end())
|
|
|
|
- continue;
|
|
|
|
-
|
|
|
|
- zoneToComponent[current] = component;
|
|
|
|
-
|
|
|
|
- for(auto neighbor : roadGraph[current])
|
|
|
|
|
|
+ for(const auto & connection : zone.second->getConnections())
|
|
{
|
|
{
|
|
- if(zoneToComponent.find(neighbor) == zoneToComponent.end())
|
|
|
|
|
|
+ if(connection.getRoadOption() == rmg::ERoadOption::ROAD_TRUE)
|
|
{
|
|
{
|
|
- stack.push(neighbor);
|
|
|
|
|
|
+ roadGraph[connection.getZoneA()].insert(connection.getZoneB());
|
|
|
|
+ roadGraph[connection.getZoneB()].insert(connection.getZoneA());
|
|
}
|
|
}
|
|
|
|
+ else if(connection.getRoadOption() == rmg::ERoadOption::ROAD_RANDOM)
|
|
|
|
+ {
|
|
|
|
+ roadGraph[connection.getZoneA()].insert(connection.getZoneB());
|
|
|
|
+ roadGraph[connection.getZoneB()].insert(connection.getZoneA());
|
|
|
|
+ randomConnections.insert(connection);
|
|
|
|
+ }
|
|
|
|
+ // ROAD_FALSE connections are ignored
|
|
}
|
|
}
|
|
}
|
|
}
|
|
- };
|
|
|
|
|
|
+ logGlobal->info("Remaining random connections: %d", randomConnections.size());
|
|
|
|
|
|
- //Find all components
|
|
|
|
- for(const auto & zone : zones)
|
|
|
|
- {
|
|
|
|
- if(zoneToComponent.find(zone.first) == zoneToComponent.end() && !roadGraph[zone.first].empty())
|
|
|
|
- {
|
|
|
|
- dfsComponent(zone.first, numComponents++);
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
|
|
+ if(randomConnections.empty())
|
|
|
|
+ break;
|
|
|
|
|
|
- //Process each component separately
|
|
|
|
- for(int component = 0; component < numComponents; component++)
|
|
|
|
- {
|
|
|
|
- //Get random connections for this component
|
|
|
|
- std::vector<rmg::ZoneConnection> componentRandomConnections;
|
|
|
|
- for(const auto & conn : randomConnections)
|
|
|
|
|
|
+ //Convert to vector for shuffling
|
|
|
|
+ std::vector<rmg::ZoneConnection> shuffledConnections(randomConnections.begin(), randomConnections.end());
|
|
|
|
+ RandomGeneratorUtil::randomShuffle(shuffledConnections, *rand);
|
|
|
|
+
|
|
|
|
+ //Try each random connection in shuffled order
|
|
|
|
+ for(const auto & conn : shuffledConnections)
|
|
{
|
|
{
|
|
- if(zoneToComponent[conn.getZoneA()] == component)
|
|
|
|
|
|
+ auto zoneA = conn.getZoneA();
|
|
|
|
+ auto zoneB = conn.getZoneB();
|
|
|
|
+
|
|
|
|
+ //Check if either zone would become isolated by removing this connection
|
|
|
|
+ if(roadGraph[zoneA].size() <= 1 || roadGraph[zoneB].size() <= 1)
|
|
{
|
|
{
|
|
- componentRandomConnections.push_back(conn);
|
|
|
|
|
|
+ //Can't remove this connection as it would isolate a zone
|
|
|
|
+ continue;
|
|
}
|
|
}
|
|
- }
|
|
|
|
-
|
|
|
|
- //Shuffle random connections
|
|
|
|
- RandomGeneratorUtil::randomShuffle(componentRandomConnections, *rand);
|
|
|
|
|
|
|
|
- //Try removing each random connection
|
|
|
|
- for(const auto & conn : componentRandomConnections)
|
|
|
|
- {
|
|
|
|
//Temporarily remove this connection
|
|
//Temporarily remove this connection
|
|
- roadGraph[conn.getZoneA()].erase(conn.getZoneB());
|
|
|
|
- roadGraph[conn.getZoneB()].erase(conn.getZoneA());
|
|
|
|
|
|
+ roadGraph[zoneA].erase(zoneB);
|
|
|
|
+ roadGraph[zoneB].erase(zoneA);
|
|
|
|
|
|
- //Check if graph remains connected
|
|
|
|
|
|
+ //Check if graph remains connected as a whole
|
|
bool canRemove = true;
|
|
bool canRemove = true;
|
|
std::set<TRmgTemplateZoneId> visited;
|
|
std::set<TRmgTemplateZoneId> visited;
|
|
|
|
|
|
- //Start DFS from any zone in this component
|
|
|
|
- auto startZone = conn.getZoneA();
|
|
|
|
- std::stack<TRmgTemplateZoneId> stack;
|
|
|
|
- stack.push(startZone);
|
|
|
|
|
|
+ // Get all zones that have road connections
|
|
|
|
+ std::set<TRmgTemplateZoneId> zonesWithRoads;
|
|
|
|
+ for(const auto & entry : roadGraph)
|
|
|
|
+ {
|
|
|
|
+ if(!entry.second.empty())
|
|
|
|
+ {
|
|
|
|
+ zonesWithRoads.insert(entry.first);
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
|
|
- while(!stack.empty())
|
|
|
|
|
|
+ if(!zonesWithRoads.empty())
|
|
{
|
|
{
|
|
- auto current = stack.top();
|
|
|
|
- stack.pop();
|
|
|
|
|
|
+ //Start DFS from any zone that has roads
|
|
|
|
+ TRmgTemplateZoneId startZone = *zonesWithRoads.begin();
|
|
|
|
+
|
|
|
|
+ std::stack<TRmgTemplateZoneId> stack;
|
|
|
|
+ stack.push(startZone);
|
|
|
|
|
|
- if(visited.find(current) != visited.end())
|
|
|
|
- continue;
|
|
|
|
|
|
+ while(!stack.empty())
|
|
|
|
+ {
|
|
|
|
+ auto current = stack.top();
|
|
|
|
+ stack.pop();
|
|
|
|
|
|
- visited.insert(current);
|
|
|
|
|
|
+ if(visited.find(current) != visited.end())
|
|
|
|
+ continue;
|
|
|
|
|
|
- for(auto neighbor : roadGraph[current])
|
|
|
|
- {
|
|
|
|
- if(visited.find(neighbor) == visited.end())
|
|
|
|
|
|
+ visited.insert(current);
|
|
|
|
+
|
|
|
|
+ for(auto neighbor : roadGraph[current])
|
|
{
|
|
{
|
|
- stack.push(neighbor);
|
|
|
|
|
|
+ if(visited.find(neighbor) == visited.end())
|
|
|
|
+ {
|
|
|
|
+ stack.push(neighbor);
|
|
|
|
+ }
|
|
}
|
|
}
|
|
}
|
|
}
|
|
- }
|
|
|
|
|
|
|
|
- //Check if all zones in this component are still reachable
|
|
|
|
- for(const auto & zone : zones)
|
|
|
|
- {
|
|
|
|
- if(zoneToComponent[zone.first] == component && !roadGraph[zone.first].empty())
|
|
|
|
|
|
+ //Check if all zones with roads are still reachable
|
|
|
|
+ for(auto zoneId : zonesWithRoads)
|
|
{
|
|
{
|
|
- if(visited.find(zone.first) == visited.end())
|
|
|
|
|
|
+ if(visited.find(zoneId) == visited.end())
|
|
{
|
|
{
|
|
canRemove = false;
|
|
canRemove = false;
|
|
break;
|
|
break;
|
|
@@ -1138,24 +1115,44 @@ void CZonePlacer::dropRandomRoads(vstd::RNG * rand)
|
|
|
|
|
|
if(!canRemove)
|
|
if(!canRemove)
|
|
{
|
|
{
|
|
- //Restore connection if removing it would break connectivity
|
|
|
|
- roadGraph[conn.getZoneA()].insert(conn.getZoneB());
|
|
|
|
- roadGraph[conn.getZoneB()].insert(conn.getZoneA());
|
|
|
|
|
|
+ //Restore connection and try next one
|
|
|
|
+ roadGraph[zoneA].insert(zoneB);
|
|
|
|
+ roadGraph[zoneB].insert(zoneA);
|
|
|
|
+ continue;
|
|
}
|
|
}
|
|
- else
|
|
|
|
|
|
+
|
|
|
|
+ //Found a connection we can remove - update in both zones that contain it
|
|
|
|
+ auto & zonePtr = zones[zoneA];
|
|
|
|
+ zonePtr->setRoadOption(conn.getId(), rmg::ERoadOption::ROAD_FALSE);
|
|
|
|
+
|
|
|
|
+ auto & otherZonePtr = zones[zoneB];
|
|
|
|
+ otherZonePtr->setRoadOption(conn.getId(), rmg::ERoadOption::ROAD_FALSE);
|
|
|
|
+
|
|
|
|
+ logGlobal->info("Dropped random road between %d and %d", zoneA, zoneB);
|
|
|
|
+ anyDropped = true;
|
|
|
|
+ break; //Exit loop and rebuild graph
|
|
|
|
+ }
|
|
|
|
+ } while(anyDropped);
|
|
|
|
+
|
|
|
|
+ // Use a set to track processed connection IDs to avoid duplicates
|
|
|
|
+ std::set<int> processedConnectionIds;
|
|
|
|
+
|
|
|
|
+ // Process each zone's connections
|
|
|
|
+ for(auto & zonePtr : zones)
|
|
|
|
+ {
|
|
|
|
+ for(auto & connection : zonePtr.second->getConnections())
|
|
|
|
+ {
|
|
|
|
+ if(connection.getRoadOption() == rmg::ERoadOption::ROAD_RANDOM)
|
|
{
|
|
{
|
|
- //Mark this connection as having no road
|
|
|
|
- for(auto & zonePtr : zones)
|
|
|
|
- {
|
|
|
|
- for(auto & connection : zonePtr.second->getConnections())
|
|
|
|
- {
|
|
|
|
- if(connection.getId() == conn.getId())
|
|
|
|
- {
|
|
|
|
- const_cast<rmg::ZoneConnection&>(connection) = conn; //FIXME: avoid const_cast
|
|
|
|
- break;
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
|
|
+ auto id = connection.getId();
|
|
|
|
+ // Only process each connection once
|
|
|
|
+ if(vstd::contains(processedConnectionIds, id))
|
|
|
|
+ continue;
|
|
|
|
+
|
|
|
|
+ processedConnectionIds.insert(id);
|
|
|
|
+
|
|
|
|
+ // Use the new setRoadOption method
|
|
|
|
+ zonePtr.second->setRoadOption(id, rmg::ERoadOption::ROAD_TRUE);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|