diff --git a/b_asic/scheduler-gui/graphics_component_item.py b/b_asic/scheduler-gui/graphics_component_item.py
index 8f34295071cf3c6d79fa5912e9c59a3cb45a999c..81bc4a9d9262ccb05fb14bbd4022dc791a87be95 100644
--- a/b_asic/scheduler-gui/graphics_component_item.py
+++ b/b_asic/scheduler-gui/graphics_component_item.py
@@ -22,10 +22,9 @@ class GraphicsComponentItem(QGraphicsItemGroup):
     """A class to represent an component in a graph."""
     _scale:                 float = 1.0
     """Static, changed from MainWindow."""
-    _op_id:                 str
+    _operation:             GraphComponent
     _height:                float
     _ports:                 Dict[str, Dict[str, Union[float, QPointF]]]     # ['port-id']['latency/pos']
-    _execution_time:        Union[int, None]
     _end_time:              int
     _component_item:        QGraphicsPathItem
     _execution_time_item:   QGraphicsRectItem
diff --git a/b_asic/scheduler-gui/graphics_graph_event.py b/b_asic/scheduler-gui/graphics_graph_event.py
index f1b1cfdba859a6c4a8e7231c180dbb1ab4e13356..f277fdc49c3f8e72815cb96faa5956c23f7a1bfd 100644
--- a/b_asic/scheduler-gui/graphics_graph_event.py
+++ b/b_asic/scheduler-gui/graphics_graph_event.py
@@ -38,7 +38,7 @@ class GraphicsGraphEvent:                                       # PyQt5
 
 
     #@overload
-    def is_component_valid_pos(self, pos: float, end_time: int) -> bool: ...
+    def is_component_valid_pos(self, item: GraphicsComponentItem, pos: float) -> bool: ...
     #@overload
     def is_valid_delta_time(self, delta_time: int) -> bool: ...
     #@overload
@@ -156,13 +156,13 @@ class GraphicsGraphEvent:                                       # PyQt5
         dx = (item.mapToParent(event.pos()) - self._current_pos).x()
         if dx > 0.505:
             pos = item.x() + 1.0
-            if self.is_component_valid_pos(pos, item.end_time):
+            if self.is_component_valid_pos(item, pos):
                 # self.prepareGeometryChange()
                 item.setX(pos)
                 self._current_pos.setX(self._current_pos.x() + 1.0)
         elif dx < -0.505:
             pos = item.x() - 1.0
-            if self.is_component_valid_pos(pos, item.end_time):
+            if self.is_component_valid_pos(item, pos):
                 # self.prepareGeometryChange()
                 item.setX(pos)
                 self._current_pos.setX(self._current_pos.x() - 1.0)
@@ -183,6 +183,9 @@ class GraphicsGraphEvent:                                       # PyQt5
         """Changes the cursor to OpenHandCursor when releasing an object."""
         item: GraphicsComponentItem = self.scene().mouseGrabberItem()
         item.setCursor(QCursor(Qt.OpenHandCursor))
+        dx = (item.mapToParent(event.pos()) - self._current_pos).x()
+        pos = item.x()
+        self.set_new_starttime(item, pos)
 
     def comp_mouseDoubleClickEvent(self, event: QGraphicsSceneMouseEvent) -> None: ...
     def comp_wheelEvent(self, event: QGraphicsSceneWheelEvent) -> None: ...
diff --git a/b_asic/scheduler-gui/graphics_graph_item.py b/b_asic/scheduler-gui/graphics_graph_item.py
index a15f3f0c046d614ba9cc612136a42798dc120440..173aa8bff95ccb3199b7b4c6ca550b08b254b207 100644
--- a/b_asic/scheduler-gui/graphics_graph_item.py
+++ b/b_asic/scheduler-gui/graphics_graph_item.py
@@ -60,23 +60,36 @@ class GraphicsGraphItem(GraphicsGraphEvent, QGraphicsItemGroup):    # PySide2 /
             item.setParentItem(None)
             del item
 
-    def is_component_valid_pos(self, pos: float, end_time: int) -> bool:
+    def is_component_valid_pos(self, item: GraphicsComponentItem, pos: float) -> bool:
         """Takes in a component position and returns true if the component's new
         position is valid, false otherwise."""
         # TODO: implement
         assert self.schedule is not None , "No schedule installed."
-        start_time = floor(pos) - floor(self._x_axis_indent)
+        end_time = item.end_time
+        new_start_time = floor(pos) - floor(self._x_axis_indent)
+        slacks = self.schedule.slacks(item.op_id)
+        op_start_time = self.schedule.start_time_of_operation(item.op_id)
+        if not -slacks[0] <=  new_start_time - op_start_time <= slacks[1]:
+            return False
         if pos < 0:
             return False
         elif (self.schedule.cyclic
-              and start_time > self.schedule.schedule_time):
+              and new_start_time > self.schedule.schedule_time):
             return False
         elif (not self.schedule.cyclic
-              and start_time + end_time > self.schedule.schedule_time):
+              and new_start_time + end_time > self.schedule.schedule_time):
             return False
 
         return True
 
+    def set_new_starttime(self, item: GraphicsComponentItem, pos: float) -> None:
+        op_start_time = self.schedule.start_time_of_operation(item.op_id)
+        new_start_time = floor(pos) - floor(self._x_axis_indent)
+        move_time = new_start_time - op_start_time
+        if move_time:
+            self.schedule.move_operation(item.op_id, move_time)
+
+
     def is_valid_delta_time(self, delta_time: int) -> bool:
         """Takes in a delta time and returns true if the new schedule time is
         valid, false otherwise."""