driver.cpp 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630
  1. /**
  2. \file vdigit/driver.cpp
  3. \brief wxvdigit - Display driver
  4. This driver is designed for wxGUI (vector digitizer) - to draw
  5. vector map layer to PseudoDC.
  6. This program is free software under the GNU General Public
  7. License (>=v2). Read the file COPYING that comes with GRASS
  8. for details.
  9. (C) 2008-2009 by Martin Landa, and the GRASS development team
  10. \author Martin Landa <landa.martin gmail.com>
  11. */
  12. #include <cmath>
  13. #include "driver.h"
  14. #define MSG 0
  15. #define WARN 1
  16. #define ERR 2
  17. static int print_error(const char *, const int);
  18. static int print_percent(int);
  19. static void print_sentence (PyObject*, const int, const char *);
  20. static PyObject *logStream;
  21. static int message_id = 1;
  22. /**
  23. \brief Initialize driver
  24. Allocate given structures.
  25. \param[in,out] PseudoDC device where to draw vector objects
  26. \param[in,out] PseudoDC device where to draw vector objects (tmp, selected)
  27. \return
  28. */
  29. DisplayDriver::DisplayDriver(gwxPseudoDC *device, gwxPseudoDC *deviceTmp,
  30. PyObject *log)
  31. {
  32. G_gisinit(""); /* GRASS functions */
  33. mapInfo = NULL;
  34. dc = device;
  35. dcTmp = deviceTmp;
  36. logStream = log;
  37. points = Vect_new_line_struct();
  38. pointsScreen = new wxList();
  39. cats = Vect_new_cats_struct();
  40. selected.field = -1;
  41. selected.cats = Vect_new_list();
  42. selected.ids = Vect_new_list();
  43. selected.idsDupl = Vect_new_list();
  44. drawSegments = false;
  45. G_set_error_routine(&print_error);
  46. G_set_percent_routine(&print_percent);
  47. // avoid GUI crash when G_fatal_error() is called (opening the vector map)
  48. // Vect_set_fatal_error(GV_FATAL_PRINT);
  49. }
  50. /**
  51. \brief Destroy driver
  52. Close the map, deallocate given structures.
  53. \param
  54. \return
  55. */
  56. DisplayDriver::~DisplayDriver()
  57. {
  58. if (mapInfo)
  59. CloseMap();
  60. G_unset_error_routine();
  61. G_unset_percent_routine();
  62. Vect_destroy_line_struct(points);
  63. delete pointsScreen;
  64. Vect_destroy_cats_struct(cats);
  65. Vect_destroy_list(selected.cats);
  66. Vect_destroy_list(selected.ids);
  67. Vect_destroy_list(selected.idsDupl);
  68. }
  69. /**
  70. \brief Set device for drawing
  71. \param[in,out] PseudoDC device where to draw vector objects
  72. \return
  73. */
  74. void DisplayDriver::SetDevice(void *device)
  75. {
  76. dc = (gwxPseudoDC *) device;
  77. return;
  78. }
  79. /*
  80. \brief Close vector map layer
  81. \param void
  82. \return 0 on success
  83. \return non-zero on error
  84. */
  85. int DisplayDriver::CloseMap()
  86. {
  87. int ret;
  88. ret = -1;
  89. if (mapInfo) {
  90. if (mapInfo->mode == GV_MODE_RW) {
  91. /* rebuild topology */
  92. Vect_build_partial(mapInfo, GV_BUILD_NONE);
  93. Vect_build(mapInfo);
  94. }
  95. /* close map and store topo/cidx */
  96. ret = Vect_close(mapInfo);
  97. G_free ((void *) mapInfo);
  98. mapInfo = NULL;
  99. }
  100. return ret;
  101. }
  102. /**
  103. \brief Open vector map layer
  104. \param[in] mapname name of vector map
  105. \param[in] mapset name of mapset where the vector map layer is stored
  106. \return topo level
  107. \return -1 on error
  108. */
  109. int DisplayDriver::OpenMap(const char* mapname, const char *mapset, bool update)
  110. {
  111. int ret;
  112. if (!mapInfo)
  113. mapInfo = (struct Map_info *) G_malloc (sizeof (struct Map_info));
  114. // define open level (level 2: topology)
  115. Vect_set_open_level(2);
  116. // avoid GUI crash when G_fatal_error() is called (opening the vector map)
  117. Vect_set_fatal_error(GV_FATAL_PRINT);
  118. // open existing map
  119. if (!update) {
  120. ret = Vect_open_old(mapInfo, (char*) mapname, (char *) mapset);
  121. }
  122. else {
  123. ret = Vect_open_update(mapInfo, (char*) mapname, (char *) mapset);
  124. }
  125. if (ret == -1) { // error
  126. G_free((void *) mapInfo);
  127. mapInfo = NULL;
  128. }
  129. return ret;
  130. }
  131. /**
  132. \brief Reload vector map layer
  133. Close and open again. Needed for modification using v.edit.
  134. TODO: Get rid of that...
  135. \param
  136. \return
  137. */
  138. void DisplayDriver::ReloadMap()
  139. {
  140. // char* name = G_store(Vect_get_map_name(mapInfo)); ???
  141. char* name = G_store(mapInfo->name);
  142. char* mapset = G_store(Vect_get_mapset(mapInfo));
  143. Vect_close(mapInfo);
  144. mapInfo = NULL;
  145. OpenMap(name, mapset, false); // used only for v.edit
  146. //Vect_build_partial(mapInfo, GV_BUILD_NONE, stderr);
  147. //Vect_build(mapInfo, stderr);
  148. return;
  149. }
  150. /*
  151. \brief Conversion from geographic coordinates (east, north)
  152. to screen (x, y)
  153. TODO: 3D stuff...
  154. \param[in] east,north,depth geographical coordinates
  155. \param[out] x, y, z screen coordinates
  156. \return
  157. */
  158. void DisplayDriver::Cell2Pixel(double east, double north, double depth,
  159. double *x, double *y, double *z)
  160. {
  161. double n, w;
  162. /*
  163. *x = int((east - region.map_west) / region.map_res);
  164. *y = int((region.map_north - north) / region.map_res);
  165. */
  166. w = region.center_easting - (region.map_width / 2) * region.map_res;
  167. n = region.center_northing + (region.map_height / 2) * region.map_res;
  168. /*
  169. *x = int((east - w) / region.map_res);
  170. *y = int((n - north) / region.map_res);
  171. */
  172. if (x)
  173. *x = (east - w) / region.map_res;
  174. if (y)
  175. *y = (n - north) / region.map_res;
  176. if (z)
  177. *z = 0.;
  178. return;
  179. }
  180. /**
  181. \brief Calculate distance in pixels
  182. \todo LL projection
  183. \param dist real distance
  184. */
  185. double DisplayDriver::DistanceInPixels(double dist)
  186. {
  187. double x;
  188. Cell2Pixel(region.map_west + dist, region.map_north, 0.0, &x, NULL, NULL);
  189. return std::sqrt(x * x);
  190. }
  191. /**
  192. \brief Set geographical region
  193. Region must be upgraded because of Cell2Pixel().
  194. \param[in] north,south,east,west,ns_res,ew_res region settings
  195. \return
  196. */
  197. void DisplayDriver::SetRegion(double north, double south, double east, double west,
  198. double ns_res, double ew_res,
  199. double center_easting, double center_northing,
  200. double map_width, double map_height)
  201. {
  202. region.box.N = north;
  203. region.box.S = south;
  204. region.box.E = east;
  205. region.box.W = west;
  206. region.box.T = PORT_DOUBLE_MAX;
  207. region.box.B = -PORT_DOUBLE_MAX;
  208. region.ns_res = ns_res;
  209. region.ew_res = ew_res;
  210. region.center_easting = center_easting;
  211. region.center_northing = center_northing;
  212. region.map_width = map_width;
  213. region.map_height = map_height;
  214. // calculate real region
  215. region.map_res = (region.ew_res > region.ns_res) ? region.ew_res : region.ns_res;
  216. region.map_west = region.center_easting - (region.map_width / 2.) * region.map_res;
  217. region.map_north = region.center_northing + (region.map_height / 2.) * region.map_res;
  218. return;
  219. }
  220. /*
  221. \brief Set settings for displaying vector feature
  222. E.g. line width, color, ...
  223. \param[in] lineWidth,... settgings
  224. \return
  225. */
  226. void DisplayDriver::UpdateSettings(unsigned long highlight,
  227. bool ehighlightDupl, unsigned long chighlightDupl,
  228. bool ePoint, unsigned long cPoint, /* enabled, color */
  229. bool eLine, unsigned long cLine,
  230. bool eBoundaryNo, unsigned long cBoundaryNo,
  231. bool eBoundaryOne, unsigned long cBoundaryOne,
  232. bool eBoundaryTwo, unsigned long cBoundaryTwo,
  233. bool eCentroidIn, unsigned long cCentroidIn,
  234. bool eCentroidOut, unsigned long cCentroidOut,
  235. bool eCentroidDup, unsigned long cCentroidDup,
  236. bool eNodeOne, unsigned long cNodeOne,
  237. bool eNodeTwo, unsigned long cNodeTwo,
  238. bool eVertex, unsigned long cVertex,
  239. bool eArea, unsigned long cArea,
  240. bool eDirection, unsigned long cDirection,
  241. int lineWidth, int alpha)
  242. {
  243. settings.highlight.Set(highlight);
  244. settings.highlightDupl.enabled = ehighlightDupl;
  245. settings.highlightDupl.color.Set(chighlightDupl);
  246. settings.point.enabled = ePoint;
  247. settings.point.color.Set(cPoint);
  248. settings.line.enabled = eLine;
  249. settings.line.color.Set(cLine);
  250. settings.boundaryNo.enabled = eBoundaryNo;
  251. settings.boundaryNo.color.Set(cBoundaryNo);
  252. settings.boundaryOne.enabled = eBoundaryOne;
  253. settings.boundaryOne.color.Set(cBoundaryOne);
  254. settings.boundaryTwo.enabled = eBoundaryTwo;
  255. settings.boundaryTwo.color.Set(cBoundaryTwo);
  256. settings.centroidIn.enabled = eCentroidIn;
  257. settings.centroidIn.color.Set(cCentroidIn);
  258. settings.centroidOut.enabled = eCentroidOut;
  259. settings.centroidOut.color.Set(cCentroidOut);
  260. settings.centroidDup.enabled = eCentroidDup;
  261. settings.centroidDup.color.Set(cCentroidDup);
  262. settings.nodeOne.enabled = eNodeOne;
  263. settings.nodeOne.color.Set(cNodeOne);
  264. settings.nodeTwo.enabled = eNodeTwo;
  265. settings.nodeTwo.color.Set(cNodeTwo);
  266. settings.vertex.enabled = eVertex;
  267. settings.vertex.color.Set(cVertex);
  268. settings.area.enabled = eArea;
  269. settings.area.color.Set(cArea);
  270. settings.area.color.Set(settings.area.color.Red(),
  271. settings.area.color.Green(),
  272. settings.area.color.Blue(),
  273. alpha);
  274. settings.direction.enabled = eDirection;
  275. settings.direction.color.Set(cDirection);
  276. settings.lineWidth = lineWidth;
  277. return;
  278. }
  279. /**
  280. \brief Prints gId: dcIds
  281. Useful for debugging purposes.
  282. \param
  283. \return
  284. */
  285. void DisplayDriver::PrintIds()
  286. {
  287. std::cerr << "topology.highlight: " << topology.highlight << std::endl;
  288. std::cerr << "topology.point: " << topology.point << std::endl;
  289. std::cerr << "topology.line: " << topology.line << std::endl;
  290. std::cerr << "topology.boundaryNo: " << topology.boundaryNo << std::endl;
  291. std::cerr << "topology.boundaryOne: " << topology.boundaryOne << std::endl;
  292. std::cerr << "topology.boundaryTwo: " << topology.boundaryTwo << std::endl;
  293. std::cerr << "topology.centroidIn: " << topology.centroidIn << std::endl;
  294. std::cerr << "topology.centroidOut: " << topology.centroidOut << std::endl;
  295. std::cerr << "topology.centroidDup: " << topology.centroidDup << std::endl;
  296. std::cerr << "topology.nodeOne: " << topology.nodeOne << std::endl;
  297. std::cerr << "topology.nodeTwo: " << topology.nodeTwo << std::endl;
  298. std::cerr << "topology.vertex: " << topology.vertex << std::endl;
  299. std::cerr << std::endl << "nobjects: "
  300. << topology.point * 2 + // cross
  301. topology.line +
  302. topology.boundaryNo +
  303. topology.boundaryOne +
  304. topology.boundaryTwo +
  305. topology.centroidIn * 2 +
  306. topology.centroidOut * 2 +
  307. topology.centroidDup * 2 +
  308. topology.nodeOne * 2 +
  309. topology.nodeTwo * 2 +
  310. topology.vertex * 2 << std::endl;
  311. std::cerr << "selected: ";
  312. for (int i = 0; i < selected.ids->n_values; i++) {
  313. std::cerr << selected.ids->value[i] << " ";
  314. }
  315. std::cerr << std::endl;
  316. return;
  317. }
  318. /**
  319. \brief Reset topology structure.
  320. \return
  321. */
  322. void DisplayDriver::ResetTopology()
  323. {
  324. topology.highlight = 0;
  325. topology.point = 0;
  326. topology.line = 0;
  327. topology.boundaryNo = 0;
  328. topology.boundaryOne = 0;
  329. topology.boundaryTwo = 0;
  330. topology.centroidIn = 0;
  331. topology.centroidOut = 0;
  332. topology.centroidDup = 0;
  333. topology.nodeOne = 0;
  334. topology.nodeTwo = 0;
  335. topology.vertex = 0;
  336. return;
  337. }
  338. /**
  339. \brief Convert vect list to std::vector
  340. \param list vect list
  341. \return std::vector
  342. */
  343. std::vector<int> DisplayDriver::ListToVector(struct ilist *list)
  344. {
  345. std::vector<int> vect;
  346. if (!list)
  347. return vect;
  348. for (int i = 0; i < list->n_values; i++) {
  349. vect.push_back(list->value[i]);
  350. }
  351. return vect;
  352. }
  353. /**
  354. \brief Convert std::vector to vect list
  355. \param list vect list
  356. \param vec std::vector instance
  357. \return number of items
  358. \return -1 on error
  359. */
  360. int DisplayDriver::VectorToList(struct ilist *list, const std::vector<int>& vec)
  361. {
  362. if (!list)
  363. return -1;
  364. Vect_reset_list(list);
  365. for (std::vector<int>::const_iterator i = vec.begin(), e = vec.end();
  366. i != e; ++i) {
  367. Vect_list_append(list, *i);
  368. }
  369. return list->n_values;
  370. }
  371. /**
  372. \brief Get bounding box of (opened) vector map layer
  373. \return (w,s,b,e,n,t)
  374. */
  375. std::vector<double> DisplayDriver::GetMapBoundingBox()
  376. {
  377. std::vector<double> region;
  378. struct bound_box bbox;
  379. if (!mapInfo) {
  380. return region;
  381. }
  382. Vect_get_map_box(mapInfo, &bbox);
  383. region.push_back(bbox.W);
  384. region.push_back(bbox.S);
  385. region.push_back(bbox.B);
  386. region.push_back(bbox.E);
  387. region.push_back(bbox.N);
  388. region.push_back(bbox.T);
  389. return region;
  390. }
  391. /*
  392. \brief Print one message, prefix inserted before each new line
  393. From lib/gis/error.c
  394. */
  395. void print_sentence (PyObject *pyFd, const int type, const char *msg)
  396. {
  397. char prefix[256];
  398. const char *start;
  399. char* sentence;
  400. switch (type) {
  401. case MSG:
  402. sprintf (prefix, "GRASS_INFO_MESSAGE(%d,%d): Vdigit: ", getpid(), message_id);
  403. break;
  404. case WARN:
  405. sprintf (prefix, "GRASS_INFO_WARNING(%d,%d): Vdigit: ", getpid(), message_id);
  406. break;
  407. case ERR:
  408. sprintf (prefix, "GRASS_INFO_ERROR(%d,%d): Vdigit: ", getpid(), message_id);
  409. break;
  410. }
  411. start = msg;
  412. PyFile_WriteString("\n", pyFd);
  413. while (*start != '\0') {
  414. const char *next = start;
  415. PyFile_WriteString(prefix, pyFd);
  416. while ( *next != '\0' ) {
  417. next++;
  418. if ( *next == '\n' ) {
  419. next++;
  420. break;
  421. }
  422. }
  423. sentence = (char *) G_malloc((next - start + 1) * sizeof (char));
  424. strncpy(sentence, start, next - start + 1);
  425. sentence[next-start] = '\0';
  426. PyFile_WriteString(sentence, pyFd);
  427. G_free((void *)sentence);
  428. PyFile_WriteString("\n", pyFd);
  429. start = next;
  430. }
  431. PyFile_WriteString("\n", pyFd);
  432. sprintf(prefix, "GRASS_INFO_END(%d,%d)\n", getpid(), message_id);
  433. PyFile_WriteString(prefix, pyFd);
  434. message_id++;
  435. }
  436. /*!
  437. \brief Print error/warning/message
  438. \param msg message buffer
  439. \param type message type
  440. \return 0
  441. */
  442. int print_error(const char *msg, const int type)
  443. {
  444. if (logStream) {
  445. print_sentence(logStream, type, msg);
  446. }
  447. else {
  448. fprintf(stderr, "Vdigit: %s\n", msg);
  449. }
  450. return 0;
  451. }
  452. /*!
  453. \brief Print percentage information
  454. \param x value
  455. \return 0
  456. */
  457. int print_percent(int x)
  458. {
  459. char msg[256];
  460. if (logStream) {
  461. sprintf(msg, "GRASS_INFO_PERCENT: %d\n", x);
  462. PyFile_WriteString(msg, logStream);
  463. }
  464. else {
  465. fprintf(stderr, "GRASS_INFO_PERCENT: %d\n", x);
  466. }
  467. return 0;
  468. }
  469. /**
  470. \brief Check if vector map is 3D
  471. \return True for 3D otherwise False
  472. */
  473. bool DisplayDriver::Is3D()
  474. {
  475. return (bool) Vect_is_3d(mapInfo);
  476. }