react-planner-viewer
Version:
react-planner-viewer is a React Component for view plans builded with react-planner in 2D mode
217 lines (171 loc) • 24.4 kB
JavaScript
import { List, Map } from 'immutable';
import { SELECT_TOOL_DRAWING_ITEM, UPDATE_DRAWING_ITEM, END_DRAWING_ITEM, BEGIN_DRAGGING_ITEM, UPDATE_DRAGGING_ITEM, END_DRAGGING_ITEM, BEGIN_ROTATING_ITEM, UPDATE_ROTATING_ITEM, END_ROTATING_ITEM, SELECT_ITEM, MODE_IDLE, MODE_DRAWING_ITEM, MODE_DRAGGING_ITEM, MODE_ROTATING_ITEM } from '../constants';
import { addItem, removeItem, unselect, select, unselectAll } from '../utils/layer-operations';
import * as Geometry from '../utils/geometry';
export default function (state, action) {
switch (action.type) {
case SELECT_ITEM:
return selectItem(state, action.layerID, action.itemID);
case SELECT_TOOL_DRAWING_ITEM:
return selectToolDrawingItem(state, action.sceneComponentType);
case UPDATE_DRAWING_ITEM:
return updateDrawingItem(state, action.layerID, action.x, action.y);
case END_DRAWING_ITEM:
return endDrawingItem(state, action.layerID, action.x, action.y);
case BEGIN_DRAGGING_ITEM:
return beginDraggingItem(state, action.layerID, action.itemID, action.x, action.y);
case UPDATE_DRAGGING_ITEM:
return updateDraggingItem(state, action.x, action.y);
case END_DRAGGING_ITEM:
return endDraggingItem(state, action.x, action.y);
case BEGIN_ROTATING_ITEM:
return beginRotatingItem(state, action.layerID, action.itemID, action.x, action.y);
case UPDATE_ROTATING_ITEM:
return updateRotatingItem(state, action.x, action.y);
case END_ROTATING_ITEM:
return endRotatingItem(state, action.x, action.y);
default:
return state;
}
}
function selectToolDrawingItem(state, sceneComponentType) {
return state.merge({
mode: MODE_DRAWING_ITEM,
drawingSupport: Map({
type: sceneComponentType
})
});
}
/** holes operations **/
function updateDrawingItem(state, layerID, x, y) {
var drawingSupport = state.drawingSupport,
catalog = state.catalog;
var scene = state.scene.updateIn(['layers', layerID], function (layer) {
return layer.withMutations(function (layer) {
if (drawingSupport.has('currentID')) {
layer.updateIn(['items', drawingSupport.get('currentID')], function (item) {
return item.merge({ x: x, y: y });
});
} else {
var _addItem = addItem(layer, drawingSupport.get('type'), x, y, 200, 100, 0, catalog),
item = _addItem.item;
select(layer, 'items', item.id);
drawingSupport = drawingSupport.set('currentID', item.id);
}
});
});
return state.merge({
scene: scene,
drawingSupport: drawingSupport
});
}
function endDrawingItem(state, layerID, x, y) {
var catalog = state.catalog;
state = updateDrawingItem(state, layerID, x, y, catalog);
var scene = state.scene.updateIn(['layers', layerID], function (layer) {
return unselectAll(layer);
});
return state.merge({
scene: scene,
sceneHistory: state.sceneHistory.push(scene),
drawingSupport: Map({
type: state.drawingSupport.get('type')
})
});
}
function beginDraggingItem(state, layerID, itemID, x, y) {
var item = state.getIn(['scene', 'layers', layerID, 'items', itemID]);
return state.merge({
mode: MODE_DRAGGING_ITEM,
draggingSupport: Map({
layerID: layerID,
itemID: itemID,
startPointX: x,
startPointY: y,
originalX: item.x,
originalY: item.y
})
});
}
function updateDraggingItem(state, x, y) {
var draggingSupport = state.draggingSupport,
scene = state.scene;
var layerID = draggingSupport.get('layerID');
var itemID = draggingSupport.get('itemID');
var startPointX = draggingSupport.get('startPointX');
var startPointY = draggingSupport.get('startPointY');
var originalX = draggingSupport.get('originalX');
var originalY = draggingSupport.get('originalY');
var diffX = startPointX - x;
var diffY = startPointY - y;
var item = scene.getIn(['layers', layerID, 'items', itemID]);
item = item.merge({
x: originalX - diffX,
y: originalY - diffY
});
return state.merge({
scene: scene.mergeIn(['layers', layerID, 'items', itemID], item)
});
}
function endDraggingItem(state, x, y) {
state = updateDraggingItem(state, x, y);
return state.merge({
mode: MODE_IDLE,
sceneHistory: state.sceneHistory.push(state.scene)
});
}
function beginRotatingItem(state, layerID, itemID, x, y) {
var item = state.getIn(['scene', 'layers', layerID, 'items', itemID]);
return state.merge({
mode: MODE_ROTATING_ITEM,
rotatingSupport: Map({
layerID: layerID,
itemID: itemID
})
});
}
function updateRotatingItem(state, x, y) {
var rotatingSupport = state.rotatingSupport,
scene = state.scene;
var layerID = rotatingSupport.get('layerID');
var itemID = rotatingSupport.get('itemID');
var item = state.getIn(['scene', 'layers', layerID, 'items', itemID]);
var deltaX = x - item.x;
var deltaY = y - item.y;
var rotation = Math.atan2(deltaY, deltaX) * 180 / Math.PI - 90;
if (-5 < rotation && rotation < 5) rotation = 0;
if (-95 < rotation && rotation < -85) rotation = -90;
if (-185 < rotation && rotation < -175) rotation = -180;
if (85 < rotation && rotation < 90) rotation = 90;
if (-270 < rotation && rotation < -265) rotation = 90;
item = item.merge({
rotation: rotation
});
return state.merge({
scene: scene.mergeIn(['layers', layerID, 'items', itemID], item)
});
}
function endRotatingItem(state, x, y) {
state = updateRotatingItem(state, x, y);
return state.merge({
mode: MODE_IDLE,
sceneHistory: state.sceneHistory.push(state.scene)
});
}
function selectItem(state, layerID, itemID) {
var scene = state.scene;
scene = scene.merge({
layers: scene.layers.map(unselectAll),
selectedLayer: layerID
});
scene = scene.updateIn(['layers', layerID], function (layer) {
return layer.withMutations(function (layer) {
select(layer, 'items', itemID);
});
});
return state.merge({
scene: scene,
sceneHistory: state.sceneHistory.push(scene)
});
}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../src/reducers/items-reducer.js"],"names":["List","Map","SELECT_TOOL_DRAWING_ITEM","UPDATE_DRAWING_ITEM","END_DRAWING_ITEM","BEGIN_DRAGGING_ITEM","UPDATE_DRAGGING_ITEM","END_DRAGGING_ITEM","BEGIN_ROTATING_ITEM","UPDATE_ROTATING_ITEM","END_ROTATING_ITEM","SELECT_ITEM","MODE_IDLE","MODE_DRAWING_ITEM","MODE_DRAGGING_ITEM","MODE_ROTATING_ITEM","addItem","removeItem","unselect","select","unselectAll","Geometry","state","action","type","selectItem","layerID","itemID","selectToolDrawingItem","sceneComponentType","updateDrawingItem","x","y","endDrawingItem","beginDraggingItem","updateDraggingItem","endDraggingItem","beginRotatingItem","updateRotatingItem","endRotatingItem","merge","mode","drawingSupport","catalog","scene","updateIn","layer","withMutations","has","get","item","id","set","sceneHistory","push","getIn","draggingSupport","startPointX","startPointY","originalX","originalY","diffX","diffY","mergeIn","rotatingSupport","deltaX","deltaY","rotation","Math","atan2","PI","layers","map","selectedLayer"],"mappings":"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","file":"items-reducer.js","sourcesContent":["import {List, Map} from 'immutable';\n\nimport {\n  SELECT_TOOL_DRAWING_ITEM,\n  UPDATE_DRAWING_ITEM,\n  END_DRAWING_ITEM,\n  BEGIN_DRAGGING_ITEM,\n  UPDATE_DRAGGING_ITEM,\n  END_DRAGGING_ITEM,\n  BEGIN_ROTATING_ITEM,\n  UPDATE_ROTATING_ITEM,\n  END_ROTATING_ITEM,\n  SELECT_ITEM,\n\n  MODE_IDLE,\n  MODE_DRAWING_ITEM,\n  MODE_DRAGGING_ITEM,\n  MODE_ROTATING_ITEM\n} from '../constants';\n\nimport {addItem, removeItem, unselect, select, unselectAll} from '../utils/layer-operations';\nimport * as Geometry from '../utils/geometry';\n\nexport default function (state, action) {\n  switch (action.type) {\n    case SELECT_ITEM:\n      return selectItem(state, action.layerID, action.itemID);\n\n    case SELECT_TOOL_DRAWING_ITEM:\n      return selectToolDrawingItem(state, action.sceneComponentType);\n\n    case UPDATE_DRAWING_ITEM:\n      return updateDrawingItem(state, action.layerID, action.x, action.y);\n\n    case END_DRAWING_ITEM:\n      return endDrawingItem(state, action.layerID, action.x, action.y);\n\n    case BEGIN_DRAGGING_ITEM:\n      return beginDraggingItem(state, action.layerID, action.itemID, action.x, action.y);\n\n    case UPDATE_DRAGGING_ITEM:\n      return updateDraggingItem(state, action.x, action.y);\n\n    case END_DRAGGING_ITEM:\n      return endDraggingItem(state, action.x, action.y);\n\n    case BEGIN_ROTATING_ITEM:\n      return beginRotatingItem(state, action.layerID, action.itemID, action.x, action.y);\n\n    case UPDATE_ROTATING_ITEM:\n      return updateRotatingItem(state, action.x, action.y);\n\n    case END_ROTATING_ITEM:\n      return endRotatingItem(state, action.x, action.y);\n\n    default:\n      return state;\n  }\n}\n\nfunction selectToolDrawingItem(state, sceneComponentType) {\n  return state.merge({\n    mode: MODE_DRAWING_ITEM,\n    drawingSupport: Map({\n      type: sceneComponentType\n    })\n  });\n}\n\n/** holes operations **/\nfunction updateDrawingItem(state, layerID, x, y) {\n  let {drawingSupport, catalog} = state;\n\n  let scene = state.scene.updateIn(['layers', layerID], layer => layer.withMutations(layer => {\n\n    if (drawingSupport.has('currentID')) {\n      layer.updateIn(['items', drawingSupport.get('currentID')], item => item.merge({x, y}));\n    } else {\n      let {item} = addItem(layer, drawingSupport.get('type'), x, y, 200, 100, 0, catalog);\n      select(layer, 'items', item.id);\n      drawingSupport = drawingSupport.set('currentID', item.id);\n    }\n  }));\n\n  return state.merge({\n    scene,\n    drawingSupport\n  });\n}\n\nfunction endDrawingItem(state, layerID, x, y) {\n  let catalog = state.catalog;\n  state = updateDrawingItem(state, layerID, x, y, catalog);\n  let scene = state.scene.updateIn(['layers', layerID], layer => unselectAll(layer));\n  return state.merge({\n    scene,\n    sceneHistory: state.sceneHistory.push(scene),\n    drawingSupport: Map({\n      type: state.drawingSupport.get('type')\n    })\n  });\n\n}\n\nfunction beginDraggingItem(state, layerID, itemID, x, y) {\n\n  let item = state.getIn(['scene', 'layers', layerID, 'items', itemID]);\n\n  return state.merge({\n    mode: MODE_DRAGGING_ITEM,\n    draggingSupport: Map({\n      layerID,\n      itemID,\n      startPointX: x,\n      startPointY: y,\n      originalX: item.x,\n      originalY: item.y\n    })\n  });\n}\n\nfunction updateDraggingItem(state, x, y) {\n  let {draggingSupport, scene} = state;\n\n  let layerID = draggingSupport.get('layerID');\n  let itemID = draggingSupport.get('itemID');\n  let startPointX = draggingSupport.get('startPointX');\n  let startPointY = draggingSupport.get('startPointY');\n  let originalX = draggingSupport.get('originalX');\n  let originalY = draggingSupport.get('originalY');\n\n  let diffX = startPointX - x;\n  let diffY = startPointY - y;\n\n  let item = scene.getIn(['layers', layerID, 'items', itemID]);\n  item = item.merge({\n    x: originalX - diffX,\n    y: originalY - diffY\n  });\n\n  return state.merge({\n    scene: scene.mergeIn(['layers', layerID, 'items', itemID], item)\n  });\n}\n\nfunction endDraggingItem(state, x, y) {\n  state = updateDraggingItem(state, x, y);\n  return state.merge({\n    mode: MODE_IDLE,\n    sceneHistory: state.sceneHistory.push(state.scene)\n  });\n}\n\nfunction beginRotatingItem(state, layerID, itemID, x, y) {\n\n  let item = state.getIn(['scene', 'layers', layerID, 'items', itemID]);\n\n  return state.merge({\n    mode: MODE_ROTATING_ITEM,\n    rotatingSupport: Map({\n      layerID,\n      itemID\n    })\n  });\n}\n\nfunction updateRotatingItem(state, x, y) {\n  let {rotatingSupport, scene} = state;\n\n  let layerID = rotatingSupport.get('layerID');\n  let itemID = rotatingSupport.get('itemID');\n  let item = state.getIn(['scene', 'layers', layerID, 'items', itemID]);\n\n  let deltaX = x - item.x;\n  let deltaY = y - item.y;\n  let rotation = Math.atan2(deltaY, deltaX) * 180 / Math.PI - 90;\n\n  if (-5 < rotation && rotation < 5) rotation = 0;\n  if (-95 < rotation && rotation < -85) rotation = -90;\n  if (-185 < rotation && rotation < -175) rotation = -180;\n  if (85 < rotation && rotation < 90) rotation = 90;\n  if (-270 < rotation && rotation < -265) rotation = 90;\n\n  item = item.merge({\n    rotation,\n  });\n\n  return state.merge({\n    scene: scene.mergeIn(['layers', layerID, 'items', itemID], item)\n  });\n}\n\nfunction endRotatingItem(state, x, y) {\n  state = updateRotatingItem(state, x, y);\n  return state.merge({\n    mode: MODE_IDLE,\n    sceneHistory: state.sceneHistory.push(state.scene)\n  });\n}\n\nfunction selectItem(state, layerID, itemID) {\n  let scene = state.scene;\n\n  scene = scene.merge({\n    layers: scene.layers.map(unselectAll),\n    selectedLayer: layerID\n  });\n\n  scene = scene.updateIn(['layers', layerID], layer => layer.withMutations(layer => {\n      select(layer, 'items', itemID);\n    })\n  );\n\n  return state.merge({\n    scene,\n    sceneHistory: state.sceneHistory.push(scene)\n  })\n}\n"]}