mobileTextEditor.js 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307
  1. 'use strict';
  2. exports.__esModule = true;
  3. var _unicode = require('./../helpers/unicode');
  4. var _event = require('./../helpers/dom/event');
  5. var _element = require('./../helpers/dom/element');
  6. var _baseEditor = require('./_baseEditor');
  7. var _baseEditor2 = _interopRequireDefault(_baseEditor);
  8. var _eventManager = require('./../eventManager');
  9. var _eventManager2 = _interopRequireDefault(_eventManager);
  10. function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
  11. var MobileTextEditor = _baseEditor2.default.prototype.extend();
  12. var domDimensionsCache = {};
  13. /**
  14. * @private
  15. * @editor MobileTextEditor
  16. * @class MobileTextEditor
  17. */
  18. var createControls = function createControls() {
  19. this.controls = {};
  20. this.controls.leftButton = document.createElement('DIV');
  21. this.controls.leftButton.className = 'leftButton';
  22. this.controls.rightButton = document.createElement('DIV');
  23. this.controls.rightButton.className = 'rightButton';
  24. this.controls.upButton = document.createElement('DIV');
  25. this.controls.upButton.className = 'upButton';
  26. this.controls.downButton = document.createElement('DIV');
  27. this.controls.downButton.className = 'downButton';
  28. for (var button in this.controls) {
  29. if (Object.prototype.hasOwnProperty.call(this.controls, button)) {
  30. this.positionControls.appendChild(this.controls[button]);
  31. }
  32. }
  33. };
  34. MobileTextEditor.prototype.valueChanged = function () {
  35. return this.initValue != this.getValue();
  36. };
  37. MobileTextEditor.prototype.init = function () {
  38. var that = this;
  39. this.eventManager = new _eventManager2.default(this.instance);
  40. this.createElements();
  41. this.bindEvents();
  42. this.instance.addHook('afterDestroy', function () {
  43. that.destroy();
  44. });
  45. };
  46. MobileTextEditor.prototype.getValue = function () {
  47. return this.TEXTAREA.value;
  48. };
  49. MobileTextEditor.prototype.setValue = function (newValue) {
  50. this.initValue = newValue;
  51. this.TEXTAREA.value = newValue;
  52. };
  53. MobileTextEditor.prototype.createElements = function () {
  54. this.editorContainer = document.createElement('DIV');
  55. this.editorContainer.className = 'htMobileEditorContainer';
  56. this.cellPointer = document.createElement('DIV');
  57. this.cellPointer.className = 'cellPointer';
  58. this.moveHandle = document.createElement('DIV');
  59. this.moveHandle.className = 'moveHandle';
  60. this.inputPane = document.createElement('DIV');
  61. this.inputPane.className = 'inputs';
  62. this.positionControls = document.createElement('DIV');
  63. this.positionControls.className = 'positionControls';
  64. this.TEXTAREA = document.createElement('TEXTAREA');
  65. (0, _element.addClass)(this.TEXTAREA, 'handsontableInput');
  66. this.inputPane.appendChild(this.TEXTAREA);
  67. this.editorContainer.appendChild(this.cellPointer);
  68. this.editorContainer.appendChild(this.moveHandle);
  69. this.editorContainer.appendChild(this.inputPane);
  70. this.editorContainer.appendChild(this.positionControls);
  71. createControls.call(this);
  72. document.body.appendChild(this.editorContainer);
  73. };
  74. MobileTextEditor.prototype.onBeforeKeyDown = function (event) {
  75. var instance = this;
  76. var that = instance.getActiveEditor();
  77. if (event.target !== that.TEXTAREA || (0, _event.isImmediatePropagationStopped)(event)) {
  78. return;
  79. }
  80. switch (event.keyCode) {
  81. case _unicode.KEY_CODES.ENTER:
  82. that.close();
  83. event.preventDefault(); // don't add newline to field
  84. break;
  85. case _unicode.KEY_CODES.BACKSPACE:
  86. (0, _event.stopImmediatePropagation)(event); // backspace, delete, home, end should only work locally when cell is edited (not in table context)
  87. break;
  88. default:
  89. break;
  90. }
  91. };
  92. MobileTextEditor.prototype.open = function () {
  93. this.instance.addHook('beforeKeyDown', this.onBeforeKeyDown);
  94. (0, _element.addClass)(this.editorContainer, 'active');
  95. (0, _element.removeClass)(this.cellPointer, 'hidden');
  96. this.updateEditorPosition();
  97. };
  98. MobileTextEditor.prototype.focus = function () {
  99. this.TEXTAREA.focus();
  100. (0, _element.setCaretPosition)(this.TEXTAREA, this.TEXTAREA.value.length);
  101. };
  102. MobileTextEditor.prototype.close = function () {
  103. this.TEXTAREA.blur();
  104. this.instance.removeHook('beforeKeyDown', this.onBeforeKeyDown);
  105. (0, _element.removeClass)(this.editorContainer, 'active');
  106. };
  107. MobileTextEditor.prototype.scrollToView = function () {
  108. var coords = this.instance.getSelectedRange().highlight;
  109. this.instance.view.scrollViewport(coords);
  110. };
  111. MobileTextEditor.prototype.hideCellPointer = function () {
  112. if (!(0, _element.hasClass)(this.cellPointer, 'hidden')) {
  113. (0, _element.addClass)(this.cellPointer, 'hidden');
  114. }
  115. };
  116. MobileTextEditor.prototype.updateEditorPosition = function (x, y) {
  117. if (x && y) {
  118. x = parseInt(x, 10);
  119. y = parseInt(y, 10);
  120. this.editorContainer.style.top = y + 'px';
  121. this.editorContainer.style.left = x + 'px';
  122. } else {
  123. var selection = this.instance.getSelected(),
  124. selectedCell = this.instance.getCell(selection[0], selection[1]);
  125. // cache sizes
  126. if (!domDimensionsCache.cellPointer) {
  127. domDimensionsCache.cellPointer = {
  128. height: (0, _element.outerHeight)(this.cellPointer),
  129. width: (0, _element.outerWidth)(this.cellPointer)
  130. };
  131. }
  132. if (!domDimensionsCache.editorContainer) {
  133. domDimensionsCache.editorContainer = {
  134. width: (0, _element.outerWidth)(this.editorContainer)
  135. };
  136. }
  137. if (selectedCell !== undefined) {
  138. var scrollLeft = this.instance.view.wt.wtOverlays.leftOverlay.trimmingContainer == window ? 0 : (0, _element.getScrollLeft)(this.instance.view.wt.wtOverlays.leftOverlay.holder);
  139. var scrollTop = this.instance.view.wt.wtOverlays.topOverlay.trimmingContainer == window ? 0 : (0, _element.getScrollTop)(this.instance.view.wt.wtOverlays.topOverlay.holder);
  140. var selectedCellOffset = (0, _element.offset)(selectedCell),
  141. selectedCellWidth = (0, _element.outerWidth)(selectedCell),
  142. currentScrollPosition = {
  143. x: scrollLeft,
  144. y: scrollTop
  145. };
  146. this.editorContainer.style.top = parseInt(selectedCellOffset.top + (0, _element.outerHeight)(selectedCell) - currentScrollPosition.y + domDimensionsCache.cellPointer.height, 10) + 'px';
  147. this.editorContainer.style.left = parseInt(window.innerWidth / 2 - domDimensionsCache.editorContainer.width / 2, 10) + 'px';
  148. if (selectedCellOffset.left + selectedCellWidth / 2 > parseInt(this.editorContainer.style.left, 10) + domDimensionsCache.editorContainer.width) {
  149. this.editorContainer.style.left = window.innerWidth - domDimensionsCache.editorContainer.width + 'px';
  150. } else if (selectedCellOffset.left + selectedCellWidth / 2 < parseInt(this.editorContainer.style.left, 10) + 20) {
  151. this.editorContainer.style.left = 0 + 'px';
  152. }
  153. this.cellPointer.style.left = parseInt(selectedCellOffset.left - domDimensionsCache.cellPointer.width / 2 - (0, _element.offset)(this.editorContainer).left + selectedCellWidth / 2 - currentScrollPosition.x, 10) + 'px';
  154. }
  155. }
  156. };
  157. MobileTextEditor.prototype.updateEditorData = function () {
  158. var selected = this.instance.getSelected(),
  159. selectedValue = this.instance.getDataAtCell(selected[0], selected[1]);
  160. this.row = selected[0];
  161. this.col = selected[1];
  162. this.setValue(selectedValue);
  163. this.updateEditorPosition();
  164. };
  165. MobileTextEditor.prototype.prepareAndSave = function () {
  166. var val;
  167. if (!this.valueChanged()) {
  168. return;
  169. }
  170. if (this.instance.getSettings().trimWhitespace) {
  171. val = [[String.prototype.trim.call(this.getValue())]];
  172. } else {
  173. val = [[this.getValue()]];
  174. }
  175. this.saveValue(val);
  176. };
  177. MobileTextEditor.prototype.bindEvents = function () {
  178. var that = this;
  179. this.eventManager.addEventListener(this.controls.leftButton, 'touchend', function (event) {
  180. that.prepareAndSave();
  181. that.instance.selection.transformStart(0, -1, null, true);
  182. that.updateEditorData();
  183. event.preventDefault();
  184. });
  185. this.eventManager.addEventListener(this.controls.rightButton, 'touchend', function (event) {
  186. that.prepareAndSave();
  187. that.instance.selection.transformStart(0, 1, null, true);
  188. that.updateEditorData();
  189. event.preventDefault();
  190. });
  191. this.eventManager.addEventListener(this.controls.upButton, 'touchend', function (event) {
  192. that.prepareAndSave();
  193. that.instance.selection.transformStart(-1, 0, null, true);
  194. that.updateEditorData();
  195. event.preventDefault();
  196. });
  197. this.eventManager.addEventListener(this.controls.downButton, 'touchend', function (event) {
  198. that.prepareAndSave();
  199. that.instance.selection.transformStart(1, 0, null, true);
  200. that.updateEditorData();
  201. event.preventDefault();
  202. });
  203. this.eventManager.addEventListener(this.moveHandle, 'touchstart', function (event) {
  204. if (event.touches.length == 1) {
  205. var touch = event.touches[0];
  206. var onTouchPosition = {
  207. x: that.editorContainer.offsetLeft,
  208. y: that.editorContainer.offsetTop
  209. };
  210. var onTouchOffset = {
  211. x: touch.pageX - onTouchPosition.x,
  212. y: touch.pageY - onTouchPosition.y
  213. };
  214. that.eventManager.addEventListener(this, 'touchmove', function (event) {
  215. var touch = event.touches[0];
  216. that.updateEditorPosition(touch.pageX - onTouchOffset.x, touch.pageY - onTouchOffset.y);
  217. that.hideCellPointer();
  218. event.preventDefault();
  219. });
  220. }
  221. });
  222. this.eventManager.addEventListener(document.body, 'touchend', function (event) {
  223. if (!(0, _element.isChildOf)(event.target, that.editorContainer) && !(0, _element.isChildOf)(event.target, that.instance.rootElement)) {
  224. that.close();
  225. }
  226. });
  227. this.eventManager.addEventListener(this.instance.view.wt.wtOverlays.leftOverlay.holder, 'scroll', function (event) {
  228. if (that.instance.view.wt.wtOverlays.leftOverlay.trimmingContainer != window) {
  229. that.hideCellPointer();
  230. }
  231. });
  232. this.eventManager.addEventListener(this.instance.view.wt.wtOverlays.topOverlay.holder, 'scroll', function (event) {
  233. if (that.instance.view.wt.wtOverlays.topOverlay.trimmingContainer != window) {
  234. that.hideCellPointer();
  235. }
  236. });
  237. };
  238. MobileTextEditor.prototype.destroy = function () {
  239. this.eventManager.clear();
  240. this.editorContainer.parentNode.removeChild(this.editorContainer);
  241. };
  242. exports.default = MobileTextEditor;