Enum.cpp 7.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189
  1. /******************************************************************************/
  2. #include "stdafx.h"
  3. /******************************************************************************/
  4. /******************************************************************************/
  5. EditEnum& EditEnum::setName(C Str &name ) {T.name =name ; name_time.getUTC(); return T;}
  6. EditEnum& EditEnum::setRemoved(bool removed) {T.removed=removed; removed_time.getUTC(); return T;}
  7. bool EditEnum::newer(C EditEnum &src)C
  8. {
  9. return removed_time>src.removed_time || name_time>src.name_time || order_time>src.order_time;
  10. }
  11. bool EditEnum::equal(C EditEnum &src)C
  12. {
  13. return removed_time==src.removed_time && name_time==src.name_time && order_time==src.order_time;
  14. }
  15. bool EditEnum::sync(C EditEnum &src)
  16. {
  17. bool changed=false;
  18. // 'order_time' is synced in 'EditEnums.sync'
  19. changed|=Sync(removed_time, src.removed_time, removed, src.removed);
  20. changed|=Sync( name_time, src. name_time, name, src. name);
  21. return changed;
  22. }
  23. bool EditEnum::undo(C EditEnum &src)
  24. {
  25. bool changed=false;
  26. changed|=Undo(removed_time, src.removed_time, removed, src.removed);
  27. changed|=Undo( name_time, src. name_time, name, src. name);
  28. changed|=Undo( order_time, src. order_time);
  29. return changed;
  30. }
  31. bool EditEnum::save(File &f)C
  32. {
  33. f.cmpUIntV(2);
  34. f<<id<<name<<removed<<removed_time<<name_time<<order_time;
  35. return f.ok();
  36. }
  37. bool EditEnum::load(File &f)
  38. {
  39. switch(f.decUIntV())
  40. {
  41. case 2:
  42. {
  43. f>>id>>name>>removed>>removed_time>>name_time>>order_time;
  44. if(f.ok())return true;
  45. }break;
  46. case 1:
  47. {
  48. f>>id; GetStr2(f, name); f>>removed>>removed_time>>name_time>>order_time;
  49. if(f.ok())return true;
  50. }break;
  51. case 0:
  52. {
  53. f>>id>>removed>>removed_time>>name_time>>order_time;
  54. GetStr(f, name);
  55. if(f.ok())return true;
  56. }break;
  57. }
  58. return false;
  59. }
  60. int EditEnums::FindI(C Memc<EditEnum> &enums, C UID &enum_id) {REPA(enums)if(enums[i].id==enum_id)return i; return -1;}
  61. EditEnum* EditEnums::Find( Memc<EditEnum> &enums, C UID &enum_id) {int i=FindI(enums, enum_id); return (i>=0) ? &enums[i] : null;}
  62. C EditEnum* EditEnums::Find(C Memc<EditEnum> &enums, C UID &enum_id) {return Find(ConstCast(enums), enum_id);}
  63. void EditEnums::del() {clear(); type=DEFAULT; type_time.zero();}
  64. int EditEnums::findI(C UID &enum_id)C {return FindI(T, enum_id);}
  65. EditEnum* EditEnums::find(C UID &enum_id) {return Find (T, enum_id);}
  66. C EditEnum* EditEnums::find(C UID &enum_id)C {return Find (T, enum_id);}
  67. bool EditEnums::newer(C EditEnums &src)C
  68. {
  69. REPA(T){C EditEnum &e=T[i], *s=src.find(e.id); if(!s || e.newer(*s))return true;}
  70. return type_time>src.type_time;
  71. }
  72. bool EditEnums::equal(C EditEnums &src)C
  73. {
  74. if(elms()!=src.elms())return false;
  75. REPA(T){C EditEnum &e=T[i], *s=src.find(e.id); if(!s || !e.equal(*s))return false;}
  76. return type_time==src.type_time;
  77. }
  78. bool EditEnums::sync(C EditEnums &src)
  79. {
  80. bool changed=false;
  81. Memc<EditEnum> this_enums, src_enums=src;
  82. Swap(SCAST(Memc<EditEnum>, T), this_enums);
  83. for(; this_enums.elms() || src_enums.elms(); )
  84. {
  85. EditEnum *t=this_enums.addr(0), // this elm
  86. *s= src_enums.addr(0), // src elm
  87. *t_in_s=(t ? Find( src_enums, t->id) : null), // this in src
  88. *s_in_t=(s ? Find(this_enums, s->id) : null); // src in this
  89. if(t && t_in_s && t_in_s->order_time>t->order_time) // this elm is present in src and there it has newer order
  90. {
  91. changed=true; t_in_s->sync(*t); this_enums.removeData(t, true);
  92. }else
  93. if(s && s_in_t && s_in_t->order_time>s->order_time) // src elm is present in this and there it has newer order
  94. {
  95. changed=true; s_in_t->sync(*s); src_enums.removeData(s, true);
  96. }else
  97. if(t && (!s || t->order_time>=s->order_time)) // there is this elm which is same or newer than src elm
  98. {
  99. if(t_in_s){changed|=t->sync(*t_in_s); src_enums.removeData(t_in_s, true);}
  100. Swap(*t, New()); this_enums.removeData(t, true);
  101. }else
  102. if(s)
  103. {
  104. changed=true; // always changed because we're taking src element instead of this elm
  105. if(s_in_t){changed|=s->sync(*s_in_t); this_enums.removeData(s_in_t, true);}
  106. Swap(*s, New()); src_enums.removeData(s, true);
  107. }else Exit("EditEnums::sync");
  108. }
  109. changed|=Sync(type_time, src.type_time, type, src.type);
  110. return changed;
  111. }
  112. void EditEnums::undo(C EditEnums &src)
  113. {
  114. Memc<EditEnum> this_enums, src_enums=src;
  115. Swap(SCAST(Memc<EditEnum>, T), this_enums);
  116. for(; this_enums.elms() || src_enums.elms(); )
  117. {
  118. EditEnum *t=this_enums.addr(0), // this elm
  119. *s= src_enums.addr(0), // src elm
  120. *t_in_s=(t ? Find( src_enums, t->id) : null), // this in src
  121. *s_in_t=(s ? Find(this_enums, s->id) : null); // src in this
  122. if(t && !t_in_s) // this elm is present and it is not found in src (which means that it was created later, so add it first as removed)
  123. {
  124. if(!t->removed){t->removed=true; t->removed_time++;} // mark as removed if necessary
  125. Swap(*t, New()); this_enums.removeData(t, true);
  126. }else
  127. if(s)
  128. {
  129. if(s_in_t) // 's_in_t' is newer, so undo and insert this one
  130. {
  131. s_in_t->undo(*s); Swap(*s_in_t, New());
  132. }else
  133. {
  134. Swap(*s, New());
  135. }
  136. this_enums.removeData(s_in_t, true);
  137. src_enums.removeData(s , true);
  138. }else Exit("EditEnums::undo");
  139. }
  140. Undo(type_time, src.type_time, type, src.type);
  141. }
  142. int EditEnums::move(C UID &enum_id, int index)
  143. {
  144. int elm=FindI(T, enum_id);
  145. if(InRange(elm, T))
  146. {
  147. T[elm].order_time.getUTC();
  148. moveElm(elm, index);
  149. if(elm>=index)index++;
  150. }
  151. return index;
  152. }
  153. void EditEnums::copyTo(Enum &e, C Str &name)C
  154. {
  155. Memt<Enum::Elm> elms; FREPA(T)if(!T[i].removed)elms.New().set(T[i].name, T[i].id);
  156. e.create(NameToEnum(name), elms);
  157. }
  158. void EditEnums::create(C Enum &src)
  159. {
  160. del();
  161. FREPA(src)New().setName(src[i].name).id=src[i].id;
  162. }
  163. bool EditEnums::save(File &f)C
  164. {
  165. f.cmpUIntV(1);
  166. f<<type<<type_time;
  167. return ::EE::Memc< ::EditEnum>::save(f);
  168. }
  169. bool EditEnums::load(File &f)
  170. {
  171. switch(f.decUIntV())
  172. {
  173. case 1: f>>type>>type_time; return ::EE::Memc< ::EditEnum>::load(f);
  174. case 0: type=DEFAULT; type_time.zero(); return ::EE::Memc< ::EditEnum>::load(f);
  175. }
  176. del(); return false;
  177. }
  178. bool EditEnums::load(C Str &name)
  179. {
  180. File f; if(f.readTry(name))return load(f); del(); return false;
  181. }
  182. EditEnum::EditEnum() : removed(false), id(UID().randomizeValid()) {}
  183. EditEnums::EditEnums() : type(DEFAULT) {}
  184. /******************************************************************************/