cats.cpp 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475
  1. /**
  2. \file cats.cpp
  3. \brief Category management
  4. This program is free software under the GNU General Public
  5. License (>=v2). Read the file COPYING that comes with GRASS
  6. for details.
  7. (C) 2008 by The GRASS development team
  8. \author Martin Landa <landa.martin gmail.com>
  9. \date 2008
  10. */
  11. extern "C" {
  12. #include <grass/dbmi.h>
  13. }
  14. #include "driver.h"
  15. #include "digit.h"
  16. /**
  17. \brief Initialize cats structure.
  18. \return 0 on success
  19. \return -1 on error
  20. */
  21. int Digit::InitCats()
  22. {
  23. int ndblinks, nfields, field, ncats;
  24. int cat, type, id;
  25. struct field_info *fi;
  26. if (!cats.empty()) {
  27. cats.clear();
  28. }
  29. if (!display->mapInfo) {
  30. /* DisplayMsg(); */
  31. return -1;
  32. }
  33. /* initialization */
  34. ndblinks = Vect_get_num_dblinks(display->mapInfo);
  35. for (int i = 0; i < ndblinks; i++) {
  36. fi = Vect_get_dblink(display->mapInfo, i);
  37. if (fi) {
  38. cats[fi->number] = PORT_INT_MIN;
  39. }
  40. }
  41. /* find max category */
  42. nfields = Vect_cidx_get_num_fields (display->mapInfo);
  43. G_debug(2, "wxDigit.InitCats(): nfields=%d", nfields);
  44. for (int i = 0; i < nfields; i++ ) {
  45. field = Vect_cidx_get_field_number(display->mapInfo, i);
  46. ncats = Vect_cidx_get_num_cats_by_index(display->mapInfo, i);
  47. if (field <= 0) {
  48. continue;
  49. }
  50. for (int j = 0; j < ncats; j++) {
  51. Vect_cidx_get_cat_by_index (display->mapInfo, i, j, &cat, &type, &id);
  52. if (cat > cats[field])
  53. cats[field] = cat;
  54. }
  55. G_debug(3, "wxDigit.InitCats(): layer=%d, cat=%d", field, cats[field]);
  56. }
  57. /* set default values */
  58. for(std::map<int, int>::const_iterator b = cats.begin(), e = cats.end();
  59. b != e; ++b ) {
  60. if (b->second == PORT_INT_MIN) {
  61. cats[b->first] = 0; /* first category 1 */
  62. G_debug(3, "wxDigit.InitCats(): layer=%d, cat=%d", b->first, cats[b->first]);
  63. }
  64. }
  65. return 0;
  66. }
  67. /**
  68. \brief Get max category number for layer
  69. \param layer layer number
  70. \return category number (0 if no category found)
  71. \return -1 on error
  72. */
  73. int Digit::GetCategory(int layer)
  74. {
  75. if (cats.find(layer) != cats.end()) {
  76. G_debug(3, "v.digit.GetCategory(): layer=%d, cat=%d", layer, cats[layer]);
  77. return cats[layer];
  78. }
  79. return 0;
  80. }
  81. /**
  82. \brief Set max category number for layer
  83. \param layer layer number
  84. \param cats category number to be set
  85. \return previously set category
  86. \return -1 if layer not available
  87. */
  88. int Digit::SetCategory(int layer, int cat)
  89. {
  90. int old_cat;
  91. if (cats.find(layer) != cats.end()) {
  92. old_cat = cats[layer];
  93. }
  94. else {
  95. old_cat = -1;
  96. }
  97. cats[layer] = cat;
  98. G_debug(3, "wxDigit.SetCategory(): layer=%d, cat=%d old_cat=%d",
  99. layer, cat, old_cat);
  100. return old_cat;
  101. }
  102. /**
  103. Get list of layers
  104. Requires InitCats() to be called.
  105. \return list of layers
  106. */
  107. std::vector<int> Digit::GetLayers()
  108. {
  109. std::vector<int> layers;
  110. for(std::map<int, int>::const_iterator b = cats.begin(), e = cats.end();
  111. b != e; ++b ) {
  112. layers.push_back(b->first);
  113. }
  114. return layers;
  115. }
  116. /**
  117. \brief Get list of layer/category(ies) for selected feature.
  118. \param line feature id (-1 for first selected feature)
  119. \return list of layer/cats
  120. */
  121. std::map<int, std::vector<int> > Digit::GetLineCats(int line_id)
  122. {
  123. std::map<int, std::vector<int> > lc;
  124. int line;
  125. struct line_cats *Cats;
  126. if (!display->mapInfo) {
  127. display->DisplayMsg();
  128. return lc;
  129. }
  130. if (line_id == -1 && display->selected.ids->n_values < 1) {
  131. /* GetLineCatsMsg(line_id); */
  132. return lc;
  133. }
  134. line = line_id;
  135. if (line_id == -1) {
  136. line = display->selected.ids->value[0];
  137. }
  138. if (!Vect_line_alive(display->mapInfo, line)) {
  139. display->DeadLineMsg(line);
  140. return lc;
  141. }
  142. Cats = Vect_new_cats_struct();
  143. if (Vect_read_line(display->mapInfo, NULL, Cats, line) < 0) {
  144. Vect_destroy_cats_struct(Cats);
  145. display->ReadLineMsg(line);
  146. return lc;
  147. }
  148. for (int i = 0; i < Cats->n_cats; i++) {
  149. if (lc.find(Cats->field[i]) == lc.end()) {
  150. std::vector<int> cats;
  151. lc[Cats->field[i]] = cats;
  152. }
  153. lc[Cats->field[i]].push_back(Cats->cat[i]);
  154. }
  155. Vect_destroy_cats_struct(Cats);
  156. return lc;
  157. }
  158. /**
  159. \brief Set categories for given feature/layer
  160. \param line feature id (-1 for first selected feature)
  161. \param layer layer number
  162. \param cats list of cats
  163. \param add True for add, False for delete
  164. \return new feature id (feature need to be rewritten)
  165. \return -1 on error
  166. */
  167. int Digit::SetLineCats(int line_id, int layer, std::vector<int> cats, bool add)
  168. {
  169. int line, ret, type;
  170. struct line_pnts *Points;
  171. struct line_cats *Cats;
  172. if (!display->mapInfo) {
  173. display->DisplayMsg();
  174. return -1;
  175. }
  176. if (line_id == -1 && display->selected.ids->n_values < 1) {
  177. display->GetLineCatsMsg(line_id);
  178. return -1;
  179. }
  180. if (line_id == -1) {
  181. line = display->selected.ids->value[0];
  182. }
  183. else {
  184. line = line_id;
  185. }
  186. if (!Vect_line_alive(display->mapInfo, line)) {
  187. display->DeadLineMsg(line);
  188. return -1;
  189. }
  190. Points = Vect_new_line_struct();
  191. Cats = Vect_new_cats_struct();
  192. type = Vect_read_line(display->mapInfo, Points, Cats, line);
  193. if (type < 0) {
  194. Vect_destroy_line_struct(Points);
  195. Vect_destroy_cats_struct(Cats);
  196. display->ReadLineMsg(line);
  197. return -1;
  198. }
  199. for (std::vector<int>::const_iterator c = cats.begin(), e = cats.end();
  200. c != e; ++c) {
  201. if (add) {
  202. Vect_cat_set(Cats, layer, *c);
  203. }
  204. else {
  205. Vect_field_cat_del(Cats, layer, *c);
  206. }
  207. G_debug(3, "Digit.SetLineCats(): layer=%d, cat=%d, add=%d",
  208. layer, *c, add);
  209. }
  210. /* register changeset */
  211. // AddActionToChangeset(changesets.size(), REWRITE, display->selected.ids->value[0]);
  212. ret = Vect_rewrite_line(display->mapInfo, line, type,
  213. Points, Cats);
  214. /* TODO
  215. updates feature id (id is changed since line has been rewriten)
  216. if (ret > 0) {
  217. changesets[changesets.size()-1][0].line = ret;
  218. }
  219. else {
  220. changesets.erase(changesets.size()-1);
  221. }
  222. */
  223. if (line_id == -1) {
  224. /* update line id since the line was rewritten */
  225. display->selected.ids->value[0] = ret;
  226. }
  227. Vect_destroy_line_struct(Points);
  228. Vect_destroy_cats_struct(Cats);
  229. return ret;
  230. }
  231. /**
  232. \brief Copy categories from one vector feature to other
  233. \param fromId list of 'from' feature ids
  234. \param toId list of 'to' feature ids
  235. \param copyAttrb duplicate attribures instead of copying categories
  236. \return number of modified features
  237. \return -1 on error
  238. */
  239. int Digit::CopyCats(std::vector<int> fromId, std::vector<int> toId, bool copyAttrb)
  240. {
  241. int fline, tline, nlines, type;
  242. int cat;
  243. struct line_pnts *Points;
  244. struct line_cats *Cats_from, *Cats_to;
  245. Points = Vect_new_line_struct();
  246. Cats_from = Vect_new_cats_struct();
  247. Cats_to = Vect_new_cats_struct();
  248. nlines = 0;
  249. for (std::vector<int>::const_iterator fi = fromId.begin(), fe = fromId.end();
  250. fi != fe; ++fi) {
  251. fline = *fi;
  252. if (!Vect_line_alive(display->mapInfo, fline))
  253. continue;
  254. type = Vect_read_line(display->mapInfo, NULL, Cats_from, fline);
  255. if (type < 0) {
  256. display->ReadLineMsg(fline);
  257. return -1;
  258. }
  259. for(std::vector<int>::const_iterator ti = toId.begin(), te = toId.end();
  260. ti != te; ++ti) {
  261. tline = *ti;
  262. if (!Vect_line_alive(display->mapInfo, tline))
  263. continue;
  264. type = Vect_read_line(display->mapInfo, Points, Cats_to, tline);
  265. if (type < 0) {
  266. display->ReadLineMsg(tline);
  267. return -1;
  268. }
  269. for (int i = 0; i < Cats_from->n_cats; i++) {
  270. if (!copyAttrb) {
  271. cat = Cats_from->cat[i]; /* duplicate category */
  272. }
  273. else {
  274. /* duplicate attributes */
  275. struct field_info *fi;
  276. char buf[GSQL_MAX];
  277. dbDriver *driver;
  278. dbHandle handle;
  279. dbCursor cursor;
  280. dbTable *table;
  281. dbColumn *column;
  282. dbValue *value;
  283. dbString stmt, value_string;
  284. int col, ncols;
  285. int more, ctype;
  286. cat = ++cats[Cats_from->field[i]];
  287. fi = Vect_get_field(display->mapInfo, Cats_from->field[i]);
  288. if (fi == NULL) {
  289. display->DblinkMsg(Cats_from->field[i]);
  290. return -1;
  291. }
  292. driver = db_start_driver(fi->driver);
  293. if (driver == NULL) {
  294. display->DbDriverMsg(fi->driver);
  295. return -1;
  296. }
  297. db_init_handle (&handle);
  298. db_set_handle (&handle, fi->database, NULL);
  299. if (db_open_database(driver, &handle) != DB_OK) {
  300. db_shutdown_driver(driver);
  301. display->DbDatabaseMsg(fi->driver, fi->database);
  302. return -1;
  303. }
  304. db_init_string (&stmt);
  305. sprintf (buf, "SELECT * FROM %s WHERE %s=%d",
  306. fi->table, fi->key, Cats_from->cat[i]);
  307. db_set_string(&stmt, buf);
  308. if (db_open_select_cursor(driver, &stmt, &cursor, DB_SEQUENTIAL) != DB_OK) {
  309. db_close_database(driver);
  310. db_shutdown_driver(driver);
  311. display->DbSelectCursorMsg(db_get_string(&stmt));
  312. return -1;
  313. }
  314. table = db_get_cursor_table(&cursor);
  315. ncols = db_get_table_number_of_columns(table);
  316. sprintf(buf, "INSERT INTO %s VALUES (", fi->table);
  317. db_set_string(&stmt, buf);
  318. /* fetch the data */
  319. while (1) {
  320. if (db_fetch(&cursor, DB_NEXT, &more) != DB_OK) {
  321. db_close_database(driver);
  322. db_shutdown_driver(driver);
  323. return -1;
  324. }
  325. if (!more)
  326. break;
  327. for (col = 0; col < ncols; col++) {
  328. if (col > 0)
  329. db_append_string(&stmt, ",");
  330. column = db_get_table_column(table, col);
  331. if (strcmp(db_get_column_name(column), fi->key) == 0) {
  332. sprintf(buf, "%d", cat);
  333. db_append_string(&stmt, buf);
  334. continue;
  335. }
  336. value = db_get_column_value(column);
  337. db_convert_column_value_to_string(column, &value_string);
  338. if (db_test_value_isnull(value))
  339. db_append_string(&stmt, "NULL");
  340. else {
  341. ctype = db_sqltype_to_Ctype(db_get_column_sqltype(column));
  342. if (ctype != DB_C_TYPE_STRING)
  343. db_append_string(&stmt, db_get_string(&value_string));
  344. else {
  345. sprintf(buf, "'%s'", db_get_string(&value_string));
  346. db_append_string(&stmt, buf);
  347. }
  348. }
  349. }
  350. }
  351. db_append_string(&stmt, ")");
  352. if (db_execute_immediate (driver, &stmt) != DB_OK ) {
  353. db_close_database(driver);
  354. db_shutdown_driver(driver);
  355. display->DbExecuteMsg(db_get_string(&stmt));
  356. return -1;
  357. }
  358. db_close_database(driver);
  359. db_shutdown_driver(driver);
  360. }
  361. if (Vect_cat_set(Cats_to, Cats_from->field[i], cat) < 1) {
  362. continue;
  363. }
  364. }
  365. if (Vect_rewrite_line(display->mapInfo, tline, type, Points, Cats_to) < 0) {
  366. display->WriteLineMsg();
  367. return -1;
  368. }
  369. G_debug(1, "Digit::CopyCats(): fline=%d, tline=%d", fline, tline);
  370. nlines++;
  371. }
  372. }
  373. Vect_destroy_line_struct(Points);
  374. Vect_destroy_cats_struct(Cats_from);
  375. Vect_destroy_cats_struct(Cats_to);
  376. return nlines;
  377. }