diff --git a/CHANGELOG b/CHANGELOG
index 38c549d2e98b878c372fbc845de32cfadddd1ee3..53f55a8601161a064625d27da90e623af5abada7 100644
--- a/CHANGELOG
+++ b/CHANGELOG
@@ -1,6 +1,7 @@
 minerva-front (19.0.0~alpha.0) stable; urgency=medium
   * Feature: support for matomo (#289)
   * Feature: allow plugin to not have a panel (#306)
+  * Feature: allow plugin to add menu option to context menu (#307)
 
  -- Piotr Gawron <piotr.gawron@uni.lu>  Fri, 18 Oct 2024 13:00:00 +0200
 
diff --git a/docs/plugins/data/bioentities.md b/docs/plugins/data/bioentities.md
index e6d8fd8c611ea2997be3d141025ca60821a74268..52b7eaf5be8ade9bd1a5e5d05b52a28ee4ed073e 100644
--- a/docs/plugins/data/bioentities.md
+++ b/docs/plugins/data/bioentities.md
@@ -19,7 +19,7 @@ Below is a description of the methods, as well as the types they return. A descr
   - gets list of added markers
   - returns array of `Marker`
 - `getShownElements`
-  - gets list of all currently shown content/chemicals/drugs bioentities + markers
+  - gets list of all currently shown content/chemicals/drugs BioEntities + markers
   - returns object of
     ```
     {
@@ -68,12 +68,12 @@ Below is a description of the methods, as well as the types they return. A descr
   - **opacity** - should be a float between `0` and `1` (example: `0.54`)
   - **x** - x coord on the map [surface/pin marker only]
   - **y** - y coord on the map [surface/pin marker only]
-  - **width** - width of surface [surface marker only]
-  - **height** - width of height [surface marker only]
-  - **number** - number presented on the pin [pin marker only]
+  - **width** - width of surface
+  - **height** - height of surface
+  - **number** - number presented on the pin
   - **modelId** - if marker should be visible only on single map, modelId should be provided
-  - **start** - start point of the line [line marker only]
-  - **end** - end point of the line [line marker only]
+  - **start** - start point of the line
+  - **end** - end point of the line
 - adds one marker to markers list
 - returns created `Marker`
 - examples:
diff --git a/docs/plugins/errors.md b/docs/plugins/errors.md
index 9958a801c236704ab1ea5157068b2d0e5b988857..d8ddee6f9d668d4cf5de1ebc9d7204c8f5eb3031 100644
--- a/docs/plugins/errors.md
+++ b/docs/plugins/errors.md
@@ -36,7 +36,7 @@
 
 ## Zoom errors
 
-- **Provided zoom value exeeds max zoom of ...**: This error occurs when `zoom` param of `setZoom` exeeds max zoom value of the selected map
+- **Provided zoom value exceeds max zoom of ...**: This error occurs when `zoom` param of `setZoom` exceeds max zoom value of the selected map
 
 - **Provided zoom value exceeds min zoom of ...**: This error occurs when `zoom` param of `setZoom` exceeds min zoom value of the selected map
 
diff --git a/docs/plugins/events.md b/docs/plugins/events.md
index b996c0c5e5ed98aa9551e8fe2e4d1d2f56b87f4e..b89b725a3cf388f84c75f5144f147b71be08940a 100644
--- a/docs/plugins/events.md
+++ b/docs/plugins/events.md
@@ -69,7 +69,7 @@ To listen for specific events, plugins can use the `addListener` method in `even
 
 - onSearch - triggered after completing a search; the elements returned by the search are passed as arguments. Three separate events 'onSearch' are triggered, each with a different searched category type. Category types include: bioEntity, drugs, chemicals, reaction. Example argument:
 
-```javascript
+```
 {
   type: 'drugs',
   searchValues: ['PRKN'],
@@ -171,7 +171,7 @@ To listen for specific events, plugins can use the `addListener` method in `even
 
 - onZoomChanged - triggered after changing the zoom level on the map; the zoom level and the map ID are passed as argument. Example argument:
 
-```javascript
+```json
 {
   "modelId": 52,
   "zoom": 9.033753064925367
@@ -180,7 +180,7 @@ To listen for specific events, plugins can use the `addListener` method in `even
 
 - onCenterChanged - triggered after the coordinates of the map center change; the coordinates of the center and map ID are passed as argument. Example argument:
 
-```javascript
+```json
 {
   "modelId": 52,
   "x": 8557,
@@ -190,7 +190,7 @@ To listen for specific events, plugins can use the `addListener` method in `even
 
 - onBioEntityClick - triggered when someone clicks on a pin; the element to which the pin is attached is passed as an argument. Example argument:
 
-```javascript
+```json
 {
   "id": 40072,
   "modelId": 52,
@@ -200,29 +200,33 @@ To listen for specific events, plugins can use the `addListener` method in `even
 
 - onPinIconClick - triggered when someone clicks on a pin icon; the element to which the pin is attached is passed as an argument. Example argument:
 
-```javascript
+```json
 {
-  "id": 40072,
+  "id": 40072
 }
 ```
 
-```javascript
+Marker pin:
+
+```json
 {
-  "id": "b0a478ad-7e7a-47f5-8130-e96cbeaa0cfe", // marker pin
+  "id": "b0a478ad-7e7a-47f5-8130-e96cbeaa0cfe"
 }
 ```
 
-- onSurfaceClick - triggered when someone clicks on a overlay surface; the element to which the pin is attached is passed as an argument. Example argument:
+- onSurfaceClick - triggered when someone clicks on an overlay surface; the element to which the pin is attached is passed as an argument. Example argument:
 
-```javascript
+```json
 {
-  "id": 18,
+  "id": 18
 }
 ```
 
-```javascript
+Surface marker overlay:
+
+```json
 {
-  "id": "a3a5305f-acfa-47ff-bf77-a26d017c6eb3", // surface marker overlay
+  "id": "a3a5305f-acfa-47ff-bf77-a26d017c6eb3"
 }
 ```
 
diff --git a/docs/plugins/map/position.md b/docs/plugins/map/position.md
index a08872c6e6c00f67b041ef4b97641ef3ff84580e..67a397d3180a062a725e12e83bf74e1aeaebd1c2 100644
--- a/docs/plugins/map/position.md
+++ b/docs/plugins/map/position.md
@@ -1,4 +1,4 @@
-### Map positon
+### Map position
 
 With use of the methods below plugins can access and modify user position data.
 
@@ -33,7 +33,7 @@ window.minerva.map.setZoom(-14);
 
 #### Get center
 
-User position is defined as center coordinate. It's value is defined as x/y/z points of current viewport center translated to map position. Plugins can access center value and modify it.
+User position is defined as center coordinate. Its value is defined as x/y/z points of current viewport center translated to map position. Plugins can access center value and modify it.
 
 To get current position value, plugins can use the `window.minerva.map.getCenter()` method which returns current position value as an object containing `x`, `y` and `z` fields. All of them are non-negative numbers but `z` is an optional field and it defines current zoom value. If argument is invalid, `getCenter` method throws an error.
 
diff --git a/public/config.js b/public/config.js
index 486d01447b0a63fc61cf907a5f10c709b56f9832..4d245aa0d429a56c8c6eb2e3cf44078f4edef59e 100644
--- a/public/config.js
+++ b/public/config.js
@@ -1,4 +1,6 @@
 // const root = 'https://minerva-dev.lcsb.uni.lu';
+// const root = 'https://scimap.lcsb.uni.lu';
+
 const root = 'https://lux1.atcomp.pl';
 window.config = {
   BASE_API_URL: `${root}/minerva/api`,
diff --git a/src/components/FunctionalArea/ContextMenu/ContextMenu.component.test.tsx b/src/components/FunctionalArea/ContextMenu/ContextMenu.component.test.tsx
index 90661a29cff805007733f3ed98c0fa7950cd8b32..0e83701bc5e0d32ad44f6d980693168e645a5254 100644
--- a/src/components/FunctionalArea/ContextMenu/ContextMenu.component.test.tsx
+++ b/src/components/FunctionalArea/ContextMenu/ContextMenu.component.test.tsx
@@ -7,6 +7,7 @@ import {
 import { act, render, screen } from '@testing-library/react';
 import { CONTEXT_MENU_INITIAL_STATE } from '@/redux/contextMenu/contextMenu.constants';
 import { bioEntityContentFixture } from '@/models/fixtures/bioEntityContentsFixture';
+import { PluginsContextMenu } from '@/services/pluginsManager/pluginContextMenu/pluginsContextMenu';
 import { ContextMenu } from './ContextMenu.component';
 
 const renderComponent = (initialStore?: InitialStoreState): { store: StoreType } => {
@@ -24,6 +25,13 @@ const renderComponent = (initialStore?: InitialStoreState): { store: StoreType }
 };
 
 describe('ContextMenu - Component', () => {
+  beforeEach(() => {
+    PluginsContextMenu.menuItems = [];
+  });
+  afterEach(() => {
+    PluginsContextMenu.menuItems = [];
+  });
+
   describe('when context menu is hidden', () => {
     beforeEach(() => {
       renderComponent({
@@ -183,4 +191,23 @@ describe('ContextMenu - Component', () => {
       expect(modal.modalName).toBe('mol-art');
     });
   });
+
+  it('should render context menu', () => {
+    const callback = jest.fn();
+    PluginsContextMenu.addMenu('1324235432', 'Click me', '', true, callback);
+
+    renderComponent({
+      contextMenu: {
+        ...CONTEXT_MENU_INITIAL_STATE,
+        isOpen: true,
+        coordinates: [0, 0],
+        uniprot: '',
+      },
+    });
+
+    expect(screen.getByTestId('context-modal')).toBeInTheDocument();
+    expect(screen.getByTestId('context-modal')).not.toHaveClass('hidden');
+
+    expect(screen.getByText('Click me')).toBeInTheDocument();
+  });
 });
diff --git a/src/components/FunctionalArea/ContextMenu/ContextMenu.component.tsx b/src/components/FunctionalArea/ContextMenu/ContextMenu.component.tsx
index 59bb7f07e6d101242a35b68793a472901c532eb1..873cdf573c813ff7e0b09adffe05dcd3e5a1133a 100644
--- a/src/components/FunctionalArea/ContextMenu/ContextMenu.component.tsx
+++ b/src/components/FunctionalArea/ContextMenu/ContextMenu.component.tsx
@@ -7,9 +7,18 @@ import { openAddCommentModal, openMolArtModalById } from '@/redux/modal/modal.sl
 import React from 'react';
 import { twMerge } from 'tailwind-merge';
 
-import { FIRST_ARRAY_ELEMENT, SECOND_ARRAY_ELEMENT } from '@/constants/common';
+import { FIRST_ARRAY_ELEMENT, SECOND_ARRAY_ELEMENT, ZERO } from '@/constants/common';
+import { PluginsContextMenu } from '@/services/pluginsManager/pluginContextMenu/pluginsContextMenu';
+import { BioEntity, Reaction } from '@/types/models';
+import { ClickCoordinates } from '@/services/pluginsManager/pluginContextMenu/pluginsContextMenu.types';
+import { currentModelSelector } from '@/redux/models/models.selectors';
+import { mapDataLastPositionSelector } from '@/redux/map/map.selectors';
+import { DEFAULT_ZOOM } from '@/constants/map';
 
 export const ContextMenu = (): React.ReactNode => {
+  const pluginContextMenu = PluginsContextMenu.menuItems;
+  const model = useAppSelector(currentModelSelector);
+  const lastPosition = useAppSelector(mapDataLastPositionSelector);
   const dispatch = useAppDispatch();
   const { isOpen, coordinates } = useAppSelector(contextMenuSelector);
   const unitProtId = useAppSelector(searchedBioEntityElementUniProtIdSelector);
@@ -32,6 +41,25 @@ export const ContextMenu = (): React.ReactNode => {
     dispatch(openAddCommentModal());
   };
 
+  const modelId = model ? model.idObject : ZERO;
+
+  const handleCallback = (
+    callback: (coordinates: ClickCoordinates, element: BioEntity | Reaction | undefined) => void,
+  ) => {
+    return () => {
+      dispatch(closeContextMenu());
+      return callback(
+        {
+          modelId,
+          x: coordinates[FIRST_ARRAY_ELEMENT],
+          y: coordinates[SECOND_ARRAY_ELEMENT],
+          zoom: lastPosition.z ? lastPosition.z : DEFAULT_ZOOM,
+        },
+        undefined,
+      );
+    };
+  };
+
   return (
     <div
       className={twMerge(
@@ -46,7 +74,7 @@ export const ContextMenu = (): React.ReactNode => {
     >
       <button
         className={twMerge(
-          'cursor-pointer text-xs font-normal',
+          'w-full cursor-pointer text-left text-xs font-normal',
           !isUnitProtIdAvailable() ? 'cursor-not-allowed text-greyscale-700' : '',
         )}
         onClick={handleOpenMolArtClick}
@@ -57,13 +85,31 @@ export const ContextMenu = (): React.ReactNode => {
       </button>
       <hr />
       <button
-        className={twMerge('cursor-pointer text-xs font-normal')}
+        className={twMerge('w-full cursor-pointer text-left text-xs font-normal')}
         onClick={handleAddCommentClick}
         type="button"
         data-testid="add-comment"
       >
         Add comment
       </button>
+      {pluginContextMenu.length && <hr />}
+
+      {pluginContextMenu.map(contextMenuEntry => (
+        <button
+          key={contextMenuEntry.id}
+          id={contextMenuEntry.id}
+          className={twMerge(
+            'cursor-pointer text-xs font-normal',
+            contextMenuEntry.style,
+            !contextMenuEntry.enabled ? 'cursor-not-allowed text-greyscale-700' : '',
+          )}
+          onClick={handleCallback(contextMenuEntry.callback)}
+          type="button"
+          data-testid={contextMenuEntry.id}
+        >
+          {contextMenuEntry.name}
+        </button>
+      ))}
     </div>
   );
 };
diff --git a/src/services/pluginsManager/pluginContextMenu/pluginsContextMenu.test.ts b/src/services/pluginsManager/pluginContextMenu/pluginsContextMenu.test.ts
new file mode 100644
index 0000000000000000000000000000000000000000..9264dcdb4bbc274da290d699de833dc160349949
--- /dev/null
+++ b/src/services/pluginsManager/pluginContextMenu/pluginsContextMenu.test.ts
@@ -0,0 +1,58 @@
+/* eslint-disable no-magic-numbers */
+
+import { PLUGINS_MOCK } from '@/models/mocks/pluginsMock';
+import { FIRST_ARRAY_ELEMENT } from '@/constants/common';
+import { PluginsContextMenu } from '@/services/pluginsManager/pluginContextMenu/pluginsContextMenu';
+
+const plugin = PLUGINS_MOCK[FIRST_ARRAY_ELEMENT];
+
+jest.mock('../../../utils/showToast');
+
+describe('PluginsContextMenu', () => {
+  beforeEach(() => {
+    PluginsContextMenu.menuItems = [];
+  });
+  afterEach(() => {
+    PluginsContextMenu.menuItems = [];
+  });
+  describe('addMenu', () => {
+    it('add store context menu', () => {
+      const callback = jest.fn();
+      const id = PluginsContextMenu.addMenu(plugin.hash, 'Click me', '', true, callback);
+
+      expect(PluginsContextMenu.menuItems).toEqual([
+        {
+          hash: plugin.hash,
+          style: '',
+          name: 'Click me',
+          enabled: true,
+          id,
+          callback,
+        },
+      ]);
+    });
+    it('update store context menu', () => {
+      const callback = jest.fn();
+      const id = PluginsContextMenu.addMenu(plugin.hash, 'Click me', '', true, callback);
+      PluginsContextMenu.updateMenu(plugin.hash, id, 'New name', '.stop-me', false);
+
+      expect(PluginsContextMenu.menuItems).toEqual([
+        {
+          hash: plugin.hash,
+          style: '.stop-me',
+          name: 'New name',
+          enabled: false,
+          id,
+          callback,
+        },
+      ]);
+    });
+    it('remove from store context menu', () => {
+      const callback = jest.fn();
+      const id = PluginsContextMenu.addMenu(plugin.hash, 'Click me', '', true, callback);
+      PluginsContextMenu.removeMenu(plugin.hash, id);
+
+      expect(PluginsContextMenu.menuItems).toEqual([]);
+    });
+  });
+});
diff --git a/src/services/pluginsManager/pluginContextMenu/pluginsContextMenu.ts b/src/services/pluginsManager/pluginContextMenu/pluginsContextMenu.ts
new file mode 100644
index 0000000000000000000000000000000000000000..eaed61039e1ada2478e630d4f5a9d33092e8751f
--- /dev/null
+++ b/src/services/pluginsManager/pluginContextMenu/pluginsContextMenu.ts
@@ -0,0 +1,36 @@
+import { PluginsContextMenuType } from '@/services/pluginsManager/pluginContextMenu/pluginsContextMenu.types';
+import { v4 as uuidv4 } from 'uuid';
+import { ZERO } from '@/constants/common';
+
+export const PluginsContextMenu: PluginsContextMenuType = {
+  menuItems: [],
+  addMenu: (hash, name, style, enabled, callback) => {
+    const uuid = uuidv4();
+    PluginsContextMenu.menuItems.push({
+      hash,
+      callback,
+      enabled,
+      name,
+      style,
+      id: uuid,
+    });
+    return uuid;
+  },
+  removeMenu: (hash, id) => {
+    PluginsContextMenu.menuItems = PluginsContextMenu.menuItems.filter(
+      item => item.hash !== hash || item.id !== id,
+    );
+  },
+  updateMenu: (hash, id, name, style, enabled) => {
+    const originalItems = PluginsContextMenu.menuItems.filter(
+      item => item.hash === hash && item.id === id,
+    );
+    if (originalItems.length > ZERO) {
+      originalItems[ZERO].name = name;
+      originalItems[ZERO].style = style;
+      originalItems[ZERO].enabled = enabled;
+    } else {
+      throw new Error(`Cannot find menu item with id=${id}`);
+    }
+  },
+};
diff --git a/src/services/pluginsManager/pluginContextMenu/pluginsContextMenu.types.ts b/src/services/pluginsManager/pluginContextMenu/pluginsContextMenu.types.ts
new file mode 100644
index 0000000000000000000000000000000000000000..0a54586b39e154501e2118311e0abbfeef7c1dcf
--- /dev/null
+++ b/src/services/pluginsManager/pluginContextMenu/pluginsContextMenu.types.ts
@@ -0,0 +1,30 @@
+import { BioEntity, Reaction } from '@/types/models';
+
+export type ClickCoordinates = {
+  modelId: number;
+  x: number;
+  y: number;
+  zoom: number;
+};
+
+export type PluginContextMenuItemType = {
+  id: string;
+  hash: string;
+  name: string;
+  style: string;
+  enabled: boolean;
+  callback: (coordinates: ClickCoordinates, element: BioEntity | Reaction | undefined) => void;
+};
+
+export type PluginsContextMenuType = {
+  menuItems: PluginContextMenuItemType[];
+  addMenu: (
+    hash: string,
+    name: string,
+    style: string,
+    enabled: boolean,
+    callback: (coordinates: ClickCoordinates, element: BioEntity | Reaction | undefined) => void,
+  ) => string;
+  removeMenu: (hash: string, id: string) => void;
+  updateMenu: (hash: string, id: string, name: string, style: string, enabled: boolean) => void;
+};
diff --git a/src/services/pluginsManager/pluginsManager.ts b/src/services/pluginsManager/pluginsManager.ts
index 78f09d96cda37ed216aeafc68f0690221b97cf50..60d8942a1547067ea4201faaac7e4ea3786d2987 100644
--- a/src/services/pluginsManager/pluginsManager.ts
+++ b/src/services/pluginsManager/pluginsManager.ts
@@ -8,6 +8,7 @@ import { isPluginHashWithPrefix } from '@/utils/plugins/isPluginHashWithPrefix';
 import { getPluginHashWithoutPrefix } from '@/utils/plugins/getPluginHashWithoutPrefix';
 import { ONE, ZERO } from '@/constants/common';
 import { minervaDefine } from '@/services/pluginsManager/map/minervaDefine';
+import { PluginsContextMenu } from '@/services/pluginsManager/pluginContextMenu/pluginsContextMenu';
 import { bioEntitiesMethods } from './bioEntities';
 import { getModels } from './map/models/getModels';
 import { openMap } from './map/openMap';
@@ -56,16 +57,16 @@ export const PluginsManager: PluginsManagerType = {
   pluginsOccurrences: {},
 
   unloadActivePlugin: hash => {
-    const hashWihtoutPrefix = getPluginHashWithoutPrefix(hash);
+    const hashWithoutPrefix = getPluginHashWithoutPrefix(hash);
 
-    PluginsManager.activePlugins[hashWihtoutPrefix] =
-      PluginsManager.activePlugins[hashWihtoutPrefix]?.filter(el => el !== hash) || [];
+    PluginsManager.activePlugins[hashWithoutPrefix] =
+      PluginsManager.activePlugins[hashWithoutPrefix]?.filter(el => el !== hash) || [];
 
     if (
-      PluginsManager.activePlugins[hashWihtoutPrefix].length === ZERO &&
-      hashWihtoutPrefix in PluginsManager.pluginsOccurrences
+      PluginsManager.activePlugins[hashWithoutPrefix].length === ZERO &&
+      hashWithoutPrefix in PluginsManager.pluginsOccurrences
     ) {
-      PluginsManager.pluginsOccurrences[hashWihtoutPrefix] = ZERO;
+      PluginsManager.pluginsOccurrences[hashWithoutPrefix] = ZERO;
     }
   },
   init() {
@@ -200,6 +201,11 @@ export const PluginsManager: PluginsManagerType = {
         removeListener: PluginsEventBus.removeListener.bind(this, extendedHash),
         removeAllListeners: PluginsEventBus.removeAllListeners.bind(this, extendedHash),
       },
+      contextMenu: {
+        addMenu: PluginsContextMenu.addMenu.bind(this, extendedHash),
+        updateMenu: PluginsContextMenu.updateMenu.bind(this, extendedHash),
+        removeMenu: PluginsContextMenu.removeMenu.bind(this, extendedHash),
+      },
       legend: {
         setLegend: setLegend.bind(this, extendedHash),
         removeLegend: removeLegend.bind(this, extendedHash),