merges.js 1.6 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556
  1. const _ = require('../../../utils/under-dash');
  2. const Range = require('../../../doc/range');
  3. const colCache = require('../../../utils/col-cache');
  4. const Enums = require('../../../doc/enums');
  5. class Merges {
  6. constructor() {
  7. // optional mergeCells is array of ranges (like the xml)
  8. this.merges = {};
  9. }
  10. add(merge) {
  11. // merge is {address, master}
  12. if (this.merges[merge.master]) {
  13. this.merges[merge.master].expandToAddress(merge.address);
  14. } else {
  15. const range = `${merge.master}:${merge.address}`;
  16. this.merges[merge.master] = new Range(range);
  17. }
  18. }
  19. get mergeCells() {
  20. return _.map(this.merges, merge => merge.range);
  21. }
  22. reconcile(mergeCells, rows) {
  23. // reconcile merge list with merge cells
  24. _.each(mergeCells, merge => {
  25. const dimensions = colCache.decode(merge);
  26. for (let i = dimensions.top; i <= dimensions.bottom; i++) {
  27. const row = rows[i - 1];
  28. for (let j = dimensions.left; j <= dimensions.right; j++) {
  29. const cell = row.cells[j - 1];
  30. if (!cell) {
  31. // nulls are not included in document - so if master cell has no value - add a null one here
  32. row.cells[j] = {
  33. type: Enums.ValueType.Null,
  34. address: colCache.encodeAddress(i, j),
  35. };
  36. } else if (cell.type === Enums.ValueType.Merge) {
  37. cell.master = dimensions.tl;
  38. }
  39. }
  40. }
  41. });
  42. }
  43. getMasterAddress(address) {
  44. // if address has been merged, return its master's address. Assumes reconcile has been called
  45. const range = this.hash[address];
  46. return range && range.tl;
  47. }
  48. }
  49. module.exports = Merges;