ItemSelector.js 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382
  1. /*
  2. * Note that this control will most likely remain as an example, and not as a core Ext form
  3. * control. However, the API will be changing in a future release and so should not yet be
  4. * treated as a final, stable API at this time.
  5. */
  6. /**
  7. * A control that allows selection of between two Ext.ux.form.MultiSelect controls.
  8. */
  9. Ext.define('Ext.ux.form.ItemSelector', {
  10. extend: 'Ext.ux.form.MultiSelect',
  11. alias: ['widget.itemselectorfield', 'widget.itemselector'],
  12. alternateClassName: ['Ext.ux.ItemSelector'],
  13. requires: [
  14. 'Ext.button.Button',
  15. 'Ext.ux.form.MultiSelect'
  16. ],
  17. /**
  18. * @cfg {Boolean} [hideNavIcons=false] True to hide the navigation icons
  19. */
  20. hideNavIcons:false,
  21. /**
  22. * @cfg {Array} buttons Defines the set of buttons that should be displayed in between the ItemSelector
  23. * fields. Defaults to <tt>['top', 'up', 'add', 'remove', 'down', 'bottom']</tt>. These names are used
  24. * to build the button CSS class names, and to look up the button text labels in {@link #buttonsText}.
  25. * This can be overridden with a custom Array to change which buttons are displayed or their order.
  26. */
  27. buttons: ['top', 'up', 'add', 'remove', 'down', 'bottom'],
  28. /**
  29. * @cfg {Object} buttonsText The tooltips for the {@link #buttons}.
  30. * Labels for buttons.
  31. */
  32. buttonsText: {
  33. top: "Move to Top",
  34. up: "Move Up",
  35. add: "Add to Selected",
  36. remove: "Remove from Selected",
  37. down: "Move Down",
  38. bottom: "Move to Bottom"
  39. },
  40. initComponent: function() {
  41. var me = this;
  42. me.ddGroup = me.id + '-dd';
  43. me.callParent();
  44. // bindStore must be called after the fromField has been created because
  45. // it copies records from our configured Store into the fromField's Store
  46. me.bindStore(me.store);
  47. },
  48. createList: function(title){
  49. var me = this;
  50. return Ext.create('Ext.ux.form.MultiSelect', {
  51. submitValue: false,
  52. flex: 1,
  53. dragGroup: me.ddGroup,
  54. dropGroup: me.ddGroup,
  55. title: title,
  56. store: {
  57. model: me.store.model,
  58. data: []
  59. },
  60. displayField: me.displayField,
  61. disabled: me.disabled,
  62. listeners: {
  63. boundList: {
  64. scope: me,
  65. itemdblclick: me.onItemDblClick,
  66. drop: me.syncValue
  67. }
  68. }
  69. });
  70. },
  71. setupItems: function() {
  72. var me = this;
  73. me.fromField = me.createList(me.fromTitle);
  74. me.toField = me.createList(me.toTitle);
  75. return {
  76. border: false,
  77. layout: {
  78. type: 'hbox',
  79. align: 'stretch'
  80. },
  81. items: [
  82. me.fromField,
  83. {
  84. xtype: 'container',
  85. margins: '0 4',
  86. width: 22,
  87. layout: {
  88. type: 'vbox',
  89. pack: 'center'
  90. },
  91. items: me.createButtons()
  92. },
  93. me.toField
  94. ]
  95. };
  96. },
  97. createButtons: function(){
  98. var me = this,
  99. buttons = [];
  100. if (!me.hideNavIcons) {
  101. Ext.Array.forEach(me.buttons, function(name) {
  102. buttons.push({
  103. xtype: 'button',
  104. tooltip: me.buttonsText[name],
  105. handler: me['on' + Ext.String.capitalize(name) + 'BtnClick'],
  106. cls: Ext.baseCSSPrefix + 'form-itemselector-btn',
  107. iconCls: Ext.baseCSSPrefix + 'form-itemselector-' + name,
  108. navBtn: true,
  109. scope: me,
  110. margin: '4 0 0 0'
  111. });
  112. });
  113. }
  114. return buttons;
  115. },
  116. /**
  117. * Get the selected records from the specified list.
  118. *
  119. * Records will be returned *in store order*, not in order of selection.
  120. * @param {Ext.view.BoundList} list The list to read selections from.
  121. * @return {Ext.data.Model[]} The selected records in store order.
  122. *
  123. */
  124. getSelections: function(list) {
  125. var store = list.getStore();
  126. return Ext.Array.sort(list.getSelectionModel().getSelection(), function(a, b) {
  127. a = store.indexOf(a);
  128. b = store.indexOf(b);
  129. if (a < b) {
  130. return -1;
  131. } else if (a > b) {
  132. return 1;
  133. }
  134. return 0;
  135. });
  136. },
  137. onTopBtnClick : function() {
  138. var list = this.toField.boundList,
  139. store = list.getStore(),
  140. selected = this.getSelections(list);
  141. store.suspendEvents();
  142. store.remove(selected, true);
  143. store.insert(0, selected);
  144. store.resumeEvents();
  145. list.refresh();
  146. this.syncValue();
  147. list.getSelectionModel().select(selected);
  148. },
  149. onBottomBtnClick : function() {
  150. var list = this.toField.boundList,
  151. store = list.getStore(),
  152. selected = this.getSelections(list);
  153. store.suspendEvents();
  154. store.remove(selected, true);
  155. store.add(selected);
  156. store.resumeEvents();
  157. list.refresh();
  158. this.syncValue();
  159. list.getSelectionModel().select(selected);
  160. },
  161. onUpBtnClick : function() {
  162. var list = this.toField.boundList,
  163. store = list.getStore(),
  164. selected = this.getSelections(list),
  165. rec,
  166. i = 0,
  167. len = selected.length,
  168. index = 0;
  169. // Move each selection up by one place if possible
  170. store.suspendEvents();
  171. for (; i < len; ++i, index++) {
  172. rec = selected[i];
  173. index = Math.max(index, store.indexOf(rec) - 1);
  174. store.remove(rec, true);
  175. store.insert(index, rec);
  176. }
  177. store.resumeEvents();
  178. list.refresh();
  179. this.syncValue();
  180. list.getSelectionModel().select(selected);
  181. },
  182. onDownBtnClick : function() {
  183. var list = this.toField.boundList,
  184. store = list.getStore(),
  185. selected = this.getSelections(list),
  186. rec,
  187. i = selected.length - 1,
  188. index = store.getCount() - 1;
  189. // Move each selection down by one place if possible
  190. store.suspendEvents();
  191. for (; i > -1; --i, index--) {
  192. rec = selected[i];
  193. index = Math.min(index, store.indexOf(rec) + 1);
  194. store.remove(rec, true);
  195. store.insert(index, rec);
  196. }
  197. store.resumeEvents();
  198. list.refresh();
  199. this.syncValue();
  200. list.getSelectionModel().select(selected);
  201. },
  202. onAddBtnClick : function() {
  203. var me = this,
  204. selected = me.getSelections(me.fromField.boundList);
  205. me.moveRec(true, selected);
  206. me.toField.boundList.getSelectionModel().select(selected);
  207. },
  208. onRemoveBtnClick : function() {
  209. var me = this,
  210. selected = me.getSelections(me.toField.boundList);
  211. me.moveRec(false, selected);
  212. me.fromField.boundList.getSelectionModel().select(selected);
  213. },
  214. moveRec: function(add, recs) {
  215. var me = this,
  216. fromField = me.fromField,
  217. toField = me.toField,
  218. fromStore = add ? fromField.store : toField.store,
  219. toStore = add ? toField.store : fromField.store;
  220. fromStore.suspendEvents();
  221. toStore.suspendEvents();
  222. fromStore.remove(recs);
  223. toStore.add(recs);
  224. fromStore.resumeEvents();
  225. toStore.resumeEvents();
  226. fromField.boundList.refresh();
  227. toField.boundList.refresh();
  228. me.syncValue();
  229. },
  230. // Synchronizes the submit value with the current state of the toStore
  231. syncValue: function() {
  232. var me = this;
  233. me.mixins.field.setValue.call(me, me.setupValue(me.toField.store.getRange()));
  234. },
  235. onItemDblClick: function(view, rec) {
  236. this.moveRec(view === this.fromField.boundList, rec);
  237. },
  238. setValue: function(value) {
  239. var me = this,
  240. fromField = me.fromField,
  241. toField = me.toField,
  242. fromStore = fromField.store,
  243. toStore = toField.store,
  244. selected;
  245. // Wait for from store to be loaded
  246. if (!me.fromStorePopulated) {
  247. me.fromField.store.on({
  248. load: Ext.Function.bind(me.setValue, me, [value]),
  249. single: true
  250. });
  251. return;
  252. }
  253. value = me.setupValue(value);
  254. me.mixins.field.setValue.call(me, value);
  255. selected = me.getRecordsForValue(value);
  256. // Clear both left and right Stores.
  257. // Both stores must not fire events during this process.
  258. fromStore.suspendEvents();
  259. toStore.suspendEvents();
  260. fromStore.removeAll();
  261. toStore.removeAll();
  262. // Reset fromStore
  263. me.populateFromStore(me.store);
  264. // Copy selection across to toStore
  265. Ext.Array.forEach(selected, function(rec){
  266. // In the from store, move it over
  267. if (fromStore.indexOf(rec) > -1) {
  268. fromStore.remove(rec);
  269. }
  270. toStore.add(rec);
  271. });
  272. // Stores may now fire events
  273. fromStore.resumeEvents();
  274. toStore.resumeEvents();
  275. // Refresh both sides and then update the app layout
  276. Ext.suspendLayouts();
  277. fromField.boundList.refresh();
  278. toField.boundList.refresh();
  279. Ext.resumeLayouts(true);
  280. },
  281. onBindStore: function(store, initial) {
  282. var me = this;
  283. if (me.fromField) {
  284. me.fromField.store.removeAll()
  285. me.toField.store.removeAll();
  286. // Add everything to the from field as soon as the Store is loaded
  287. if (store.getCount()) {
  288. me.populateFromStore(store);
  289. } else {
  290. me.store.on('load', me.populateFromStore, me);
  291. }
  292. }
  293. },
  294. populateFromStore: function(store) {
  295. var fromStore = this.fromField.store;
  296. // Flag set when the fromStore has been loaded
  297. this.fromStorePopulated = true;
  298. fromStore.add(store.getRange());
  299. // setValue waits for the from Store to be loaded
  300. fromStore.fireEvent('load', fromStore);
  301. },
  302. onEnable: function(){
  303. var me = this;
  304. me.callParent();
  305. me.fromField.enable();
  306. me.toField.enable();
  307. Ext.Array.forEach(me.query('[navBtn]'), function(btn){
  308. btn.enable();
  309. });
  310. },
  311. onDisable: function(){
  312. var me = this;
  313. me.callParent();
  314. me.fromField.disable();
  315. me.toField.disable();
  316. Ext.Array.forEach(me.query('[navBtn]'), function(btn){
  317. btn.disable();
  318. });
  319. },
  320. onDestroy: function(){
  321. this.bindStore(null);
  322. this.callParent();
  323. }
  324. });