File indexing completed on 2024-09-08 11:04:03
0001 /*************************************************************************** 0002 * Copyright (C) 2003-2004 by David Saxton * 0003 * david@bluehaze.org * 0004 * * 0005 * This program is free software; you can redistribute it and/or modify * 0006 * it under the terms of the GNU General Public License as published by * 0007 * the Free Software Foundation; either version 2 of the License, or * 0008 * (at your option) any later version. * 0009 ***************************************************************************/ 0010 0011 #include "conrouter.h" 0012 #include "icndocument.h" 0013 #include "utils.h" 0014 0015 #include <cassert> 0016 #include <cmath> 0017 #include <cstdlib> 0018 0019 #include <ktechlab_debug.h> 0020 0021 ConRouter::ConRouter(ICNDocument *cv) 0022 { 0023 p_icnDocument = cv; 0024 m_lcx = m_lcy = 0; 0025 } 0026 0027 ConRouter::~ConRouter() 0028 { 0029 } 0030 0031 QPointList ConRouter::pointList(bool reverse) const 0032 { 0033 QPointList pointList; 0034 0035 if (reverse) { 0036 bool notDone = m_cellPointList.size() > 0; 0037 for (QPointList::const_iterator it = (--m_cellPointList.constEnd()); notDone; --it) { 0038 pointList.append(toCanvas(&*it)); 0039 if (it == m_cellPointList.begin()) 0040 notDone = false; 0041 } 0042 } else { 0043 const QPointList::const_iterator end = m_cellPointList.end(); 0044 for (QPointList::const_iterator it = m_cellPointList.begin(); it != end; ++it) { 0045 pointList.append(toCanvas(&*it)); 0046 } 0047 } 0048 0049 return pointList; 0050 } 0051 0052 QPointListList ConRouter::splitPoints(const QPoint &pos) const 0053 { 0054 const QPoint split = fromCanvas(&pos); 0055 0056 QList<QPointList> list; 0057 0058 // Check that the point is in the connector points, and not at the start or end 0059 bool found = false; 0060 QPointList::const_iterator end = m_cellPointList.end(); 0061 0062 double dl[] = {0.0, 1.1, 1.5}; // sqrt(2) < 1.5 < sqrt(5) 0063 for (unsigned i = 0; (i < 3) && !found; ++i) { 0064 for (QPointList::const_iterator it = m_cellPointList.begin(); it != end && !found; ++it) { 0065 QPointList::const_iterator fromLast = --m_cellPointList.constEnd(); 0066 if (qpoint_distance(*it, split) <= dl[i] && it != m_cellPointList.begin() && it != fromLast) // m_cellPointList.fromLast() ) 0067 found = true; 0068 } 0069 } 0070 0071 QPointList first; 0072 QPointList second; 0073 0074 if (!found) { 0075 qCWarning(KTL_LOG) << "ConRouter::splitConnectorPoints: Could not find point (" << pos.x() << ", " << pos.y() << ") in connector points"; 0076 qCWarning(KTL_LOG) << "ConRouter::splitConnectorPoints: Returning generic list"; 0077 0078 first.append(toCanvas(m_cellPointList.first())); 0079 first.append(pos); 0080 second.append(pos); 0081 second.append(toCanvas(m_cellPointList.last())); 0082 0083 list.append(first); 0084 list.append(second); 0085 0086 return list; 0087 } 0088 0089 // Now add the points to the two lists 0090 bool gotToSplit = false; 0091 for (QPointList::const_iterator it = m_cellPointList.begin(); it != end; ++it) { 0092 QPoint canvasPoint = toCanvas(&*it); 0093 if (*it == split) { 0094 gotToSplit = true; 0095 first.append(canvasPoint); 0096 second.prepend(canvasPoint); 0097 } else if (!gotToSplit) { 0098 first.append(canvasPoint); 0099 } else /*if (gotToSplit)*/ 0100 { 0101 second.append(canvasPoint); 0102 } 0103 } 0104 0105 list.append(first); 0106 list.append(second); 0107 0108 return list; 0109 } 0110 0111 QPointListList ConRouter::dividePoints(uint n) const 0112 { 0113 // Divide the points up into n pieces... 0114 0115 QPointList points = m_cellPointList; 0116 assert(n != 0); 0117 if (points.size() == 0) { 0118 points += QPoint(toCanvas(m_lcx), toCanvas(m_lcy)); 0119 } 0120 0121 const float avgLength = float(points.size() - 1) / float(n); 0122 0123 QPointListList pll; 0124 for (uint i = 0; i < n; ++i) { 0125 QPointList pl; 0126 // Get the points between (pos) and (pos+avgLength) 0127 const int endPos = roundDouble(avgLength * (i + 1)); 0128 const int startPos = roundDouble(avgLength * i); 0129 // const QPointList::iterator end = ++points.at(endPos); 0130 // for ( QPointList::iterator it = points.at(startPos); it != end; ++it ) 0131 for (int pos = startPos; pos < endPos; ++pos) { 0132 // pl += toCanvas(*it); 0133 pl += toCanvas(points.at(pos)); 0134 } 0135 pll += pl; 0136 } 0137 return pll; 0138 } 0139 0140 void ConRouter::checkACell(int x, int y, Cell *prev, int prevX, int prevY, int nextScore) 0141 { 0142 // if ( !p_icnDocument->isValidCellReference(x,y) ) return; 0143 if (!cellsPtr->haveCell(x, y)) 0144 return; 0145 0146 Cell *c = &cellsPtr->cell(x, y); 0147 if (c->permanent) 0148 return; 0149 0150 int newScore = nextScore + c->CIpenalty + c->Cpenalty; 0151 0152 // Check for changing direction 0153 if (x != prevX && prev->prevX == prevX) 0154 newScore += 5; 0155 else if (y != prevY && prev->prevY == prevY) 0156 newScore += 5; 0157 0158 if (c->bestScore < newScore) 0159 return; 0160 0161 // We only want to change the previous cell if the score is different, 0162 // or the score is the same but this cell allows the connector 0163 // to travel in the same direction 0164 0165 if (c->bestScore == newScore && x != prevX && y != prevY) 0166 return; 0167 0168 c->bestScore = newScore; 0169 c->prevX = prevX; 0170 c->prevY = prevY; 0171 0172 if (!c->addedToLabels) { 0173 c->addedToLabels = true; 0174 Point point; 0175 point.x = x; 0176 point.y = y; 0177 point.prevX = prevX; 0178 point.prevY = prevY; 0179 TempLabelMap::iterator it = tempLabels.insert(std::make_pair(newScore, point)); 0180 c->point = &it->second; 0181 } else { 0182 c->point->prevX = prevX; 0183 c->point->prevY = prevY; 0184 } 0185 } 0186 0187 void ConRouter::checkCell(int x, int y) 0188 { 0189 Cell *c = &cellsPtr->cell(x, y); 0190 0191 c->permanent = true; 0192 int nextScore = c->bestScore + 1; 0193 0194 // Check the surrounding cells (up, left, right, down) 0195 checkACell(x, y - 1, c, x, y, nextScore); 0196 checkACell(x - 1, y, c, x, y, nextScore); 0197 checkACell(x + 1, y, c, x, y, nextScore); 0198 checkACell(x, y + 1, c, x, y, nextScore); 0199 } 0200 0201 bool ConRouter::needsRouting(int sx, int sy, int ex, int ey) const 0202 { 0203 if (m_cellPointList.size() < 2) { 0204 // Better be on the safe side... 0205 return true; 0206 } 0207 0208 const int scx = fromCanvas(sx); 0209 const int scy = fromCanvas(sy); 0210 const int ecx = fromCanvas(ex); 0211 const int ecy = fromCanvas(ey); 0212 0213 const int psx = m_cellPointList.first().x(); 0214 const int psy = m_cellPointList.first().y(); 0215 const int pex = m_cellPointList.last().x(); 0216 const int pey = m_cellPointList.last().y(); 0217 0218 return (psx != scx || psy != scy || pex != ecx || pey != ecy) && (pex != scx || pey != scy || psx != ecx || psy != ecy); 0219 } 0220 0221 void ConRouter::setRoutePoints(const QPointList &pointList) 0222 { 0223 m_cellPointList = pointList; 0224 removeDuplicatePoints(); 0225 } 0226 0227 void ConRouter::setPoints(const QPointList &pointList, bool reverse) 0228 { 0229 if (pointList.size() == 0) 0230 return; 0231 0232 QPointList cellPointList; 0233 0234 QPoint prevCellPoint = fromCanvas(*pointList.begin()); 0235 cellPointList.append(prevCellPoint); 0236 const QPointList::const_iterator end = pointList.end(); 0237 for (QPointList::const_iterator it = pointList.begin(); it != end; ++it) { 0238 QPoint cellPoint = fromCanvas(*it); 0239 0240 while (prevCellPoint != cellPoint) { 0241 cellPointList.append(prevCellPoint); 0242 0243 if (prevCellPoint.x() < cellPoint.x()) 0244 prevCellPoint.setX(prevCellPoint.x() + 1); 0245 else if (prevCellPoint.x() > cellPoint.x()) 0246 prevCellPoint.setX(prevCellPoint.x() - 1); 0247 if (prevCellPoint.y() < cellPoint.y()) 0248 prevCellPoint.setY(prevCellPoint.y() + 1); 0249 else if (prevCellPoint.y() > cellPoint.y()) 0250 prevCellPoint.setY(prevCellPoint.y() - 1); 0251 }; 0252 0253 prevCellPoint = cellPoint; 0254 } 0255 cellPointList.append(prevCellPoint); 0256 0257 if (reverse) { 0258 m_cellPointList.clear(); 0259 const QPointList::iterator begin = cellPointList.begin(); 0260 for (QPointList::iterator it = --cellPointList.end(); it != begin; --it) { 0261 m_cellPointList += *it; 0262 } 0263 m_cellPointList += *begin; 0264 } else { 0265 m_cellPointList = cellPointList; 0266 } 0267 0268 removeDuplicatePoints(); 0269 } 0270 0271 void ConRouter::translateRoute(int dx, int dy) 0272 { 0273 if (dx == 0 && dy == 0) 0274 return; 0275 0276 m_lcx += dx; 0277 m_lcy += dy; 0278 0279 // const QPoint ds = QPoint( fromCanvas(dx), fromCanvas(dy) ); 0280 const QPoint ds = QPoint(dx / 8, dy / 8); 0281 0282 QPointList::iterator end = m_cellPointList.end(); 0283 for (QPointList::iterator it = m_cellPointList.begin(); it != end; ++it) { 0284 (*it) += ds; 0285 } 0286 0287 removeDuplicatePoints(); 0288 } 0289 0290 void ConRouter::mapRoute(int sx, int sy, int ex, int ey) 0291 { 0292 const int scx = fromCanvas(sx); 0293 const int scy = fromCanvas(sy); 0294 const int ecx = fromCanvas(ex); 0295 const int ecy = fromCanvas(ey); 0296 0297 cellsPtr = p_icnDocument->cells(); 0298 0299 if (!cellsPtr->haveCell(scx, scy) || !cellsPtr->haveCell(ecx, ecy)) { 0300 qCDebug(KTL_LOG) << "cellPtr doesn't have cells, giving up"; 0301 return; 0302 } 0303 0304 m_cellPointList.clear(); 0305 m_lcx = ecx; 0306 m_lcy = ecy; 0307 0308 // First, lets try some common connector routes (which will not necesssarily 0309 // be shortest, but they will be neat, and cut down on overall CPU usage) 0310 // If that fails, we will resort to a shortest-route algorithm to find an 0311 // appropriate route. 0312 0313 // Connector configuration: Line 0314 { 0315 bool ok = checkLineRoute(scx, scy, ecx, ecy, 4 * ICNDocument::hs_connector, 0); 0316 if (ok) { 0317 return; 0318 } else { 0319 m_cellPointList.clear(); 0320 } 0321 } 0322 0323 // Corner 1 0324 { 0325 bool ok = checkLineRoute(scx, scy, ecx, ecy, 2 * ICNDocument::hs_connector, 0); 0326 if (!ok) { 0327 m_cellPointList.clear(); 0328 } else { 0329 ok = checkLineRoute(scx, scy, ecx, ecy, ICNDocument::hs_connector - 1, 0); 0330 if (ok) { 0331 return; 0332 } else { 0333 m_cellPointList.clear(); 0334 } 0335 } 0336 } 0337 0338 // Corner 2 0339 { 0340 bool ok = checkLineRoute(scx, scy, ecx, ecy, 2 * ICNDocument::hs_connector, 0); 0341 if (!ok) { 0342 m_cellPointList.clear(); 0343 } else { 0344 ok = checkLineRoute(scx, scy, ecx, ecy, ICNDocument::hs_connector - 1, 0); 0345 if (ok) { 0346 return; 0347 } else { 0348 m_cellPointList.clear(); 0349 } 0350 } 0351 } 0352 0353 // It seems we must resort to brute-force route-checking 0354 { 0355 cellsPtr->reset(); 0356 0357 xcells = p_icnDocument->canvas()->width() / 8; 0358 ycells = p_icnDocument->canvas()->height() / 8; 0359 0360 // Now to map out the shortest routes to the cells 0361 Cell *const startCell = &cellsPtr->cell(ecx, ecy); 0362 startCell->permanent = true; 0363 startCell->bestScore = 0; 0364 startCell->prevX = startCellPos; 0365 startCell->prevY = startCellPos; 0366 0367 tempLabels.clear(); 0368 checkCell(ecx, ecy); 0369 0370 // Daniel: I changed it from a do while to a while otherwise 0371 // in rare cases the iterator can end up as end(). 0372 while (tempLabels.size() > 0 && !cellsPtr->cell(scx, scy).permanent) { 0373 TempLabelMap::iterator it = tempLabels.begin(); 0374 checkCell(it->second.x, it->second.y); 0375 tempLabels.erase(it); 0376 } 0377 0378 // Now, retrace the shortest route from the endcell to get out points :) 0379 int x = scx, y = scy; 0380 bool ok = true; 0381 0382 do { 0383 m_cellPointList.append(QPoint(x, y)); 0384 int newx = cellsPtr->cell(x, y).prevX; 0385 int newy = cellsPtr->cell(x, y).prevY; 0386 if (newx == x && newy == y) { 0387 ok = false; 0388 } 0389 x = newx; 0390 y = newy; 0391 } while (cellsPtr->haveCell(x, y) && (x != startCellPos) && (y != startCellPos) && ok); 0392 0393 // And append the last point... 0394 m_cellPointList.append(QPoint(ecx, ecy)); 0395 } 0396 0397 removeDuplicatePoints(); 0398 } 0399 0400 bool ConRouter::checkLineRoute(int scx, int scy, int ecx, int ecy, int maxConScore, int maxCIScore) 0401 { 0402 if ((scx != ecx) && (scy != ecy)) 0403 return false; 0404 0405 const bool isHorizontal = scy == ecy; 0406 0407 int start = 0, end = 0, x = 0, y = 0, dd = 0; 0408 if (isHorizontal) { 0409 dd = (scx < ecx) ? 1 : -1; 0410 start = scx; 0411 end = ecx + dd; 0412 y = scy; 0413 } else { 0414 dd = (scy < ecy) ? 1 : -1; 0415 start = scy; 0416 end = ecy + dd; 0417 x = scx; 0418 } 0419 0420 Cells *cells = p_icnDocument->cells(); 0421 0422 if (isHorizontal) { 0423 for (int x = start; x != end; x += dd) { 0424 if (std::abs(x - start) > 1 && std::abs(x - end) > 1 && (cells->cell(x, y).CIpenalty > maxCIScore || cells->cell(x, y).Cpenalty > maxConScore)) { 0425 return false; 0426 } else 0427 m_cellPointList.append(QPoint(x, y)); 0428 } 0429 } else { 0430 for (int y = start; y != end; y += dd) { 0431 if (std::abs(y - start) > 1 && std::abs(y - end) > 1 && (cells->cell(x, y).CIpenalty > maxCIScore || cells->cell(x, y).Cpenalty > maxConScore)) { 0432 return false; 0433 } else { 0434 m_cellPointList.append(QPoint(x, y)); 0435 } 0436 } 0437 } 0438 0439 m_cellPointList.prepend(QPoint(scx, scy)); 0440 m_cellPointList.append(QPoint(ecx, ecy)); 0441 removeDuplicatePoints(); 0442 return true; 0443 } 0444 0445 void ConRouter::removeDuplicatePoints() 0446 { 0447 QPoint invalid(-(1 << 30), -(1 << 30)); 0448 QPoint prev = invalid; 0449 0450 const QPointList::iterator end = m_cellPointList.end(); 0451 for (QPointList::iterator it = m_cellPointList.begin(); it != end; ++it) { 0452 if (*it == prev) { 0453 *it = invalid; 0454 } else { 0455 prev = *it; 0456 } 0457 } 0458 m_cellPointList.removeAll(invalid); 0459 }