UI Integration
This chapter covers how plugins can integrate with OpenCPN’s user interface. Plugins can add toolbar buttons, context menu items, preferences panels, and dockable windows. This allows your plugin to provide a seamless user experience that integrates well with OpenCPN’s existing interface.
Toolbar Integration
Plugins can add buttons to OpenCPN’s main toolbar. These buttons can be used to toggle features, open dialogs, or perform actions.
Declaring Toolbar Capabilities
To add toolbar buttons, your plugin must declare the appropriate capabilities in the Init()
method:
int MyPlugin::Init(void) {
// Initialize resources
// ...
// Return capabilities
return INSTALLS_TOOLBAR_TOOL |
WANTS_TOOLBAR_CALLBACK;
}
Implementing Toolbar Methods
You need to implement several methods to support toolbar integration:
int MyPlugin::GetToolbarToolCount(void) {
return 1; // Number of toolbar buttons to add
}
void MyPlugin::OnToolbarToolCallback(int id) {
if (id == m_toolbar_item_id) {
// Handle toolbar button click
// Toggle feature, open dialog, etc.
}
}
For API version 1.13 and later, you can also implement these methods for more advanced toolbar interactions:
void MyPlugin::OnToolbarToolDownCallback(int id) {
// Called when toolbar button is pressed down
}
void MyPlugin::OnToolbarToolUpCallback(int id) {
// Called when toolbar button is released
}
Adding Toolbar Buttons
To actually add a button to the toolbar, use the InsertPlugInTool()
function. This is typically done in your Init()
method:
m_toolbar_item_id = InsertPlugInTool(
"MyPlugin", // Label
_img_plugin_normal, // Normal bitmap
_img_plugin_rollover, // Rollover bitmap
wxITEM_CHECK, // Kind (normal, check, radio)
_("My Plugin"), // Short help text
_("Long help text for my plugin"), // Long help text
NULL, // User data
TOOL_POSITION_ANY, // Position
0, // Tool selection
this // Plugin pointer
);
OpenCPN assigns a unique ID to each toolbar button. Store this ID to identify which button was clicked in the callback methods. |
Using SVG Icons
For better scaling across different display densities, you can use SVG icons for toolbar buttons:
m_toolbar_item_id = InsertPlugInToolSVG(
"MyPlugin", // Label
"icons/my_plugin_normal.svg", // Normal SVG
"icons/my_plugin_rollover.svg",// Rollover SVG
"icons/my_plugin_toggled.svg", // Toggled SVG
wxITEM_CHECK, // Kind (normal, check, radio)
_("My Plugin"), // Short help text
_("Long help text for my plugin"), // Long help text
NULL, // User data
TOOL_POSITION_ANY, // Position
0, // Tool selection
this // Plugin pointer
);
Managing Toolbar Buttons
You can control toolbar button state and visibility:
// Toggle button state (pressed/unpressed)
SetToolbarItemState(m_toolbar_item_id, true); // Set to pressed
SetToolbarItemState(m_toolbar_item_id, false); // Set to unpressed
// Hide/show button
SetToolbarToolViz(m_toolbar_item_id, false); // Hide
SetToolbarToolViz(m_toolbar_item_id, true); // Show
// Update button icons
SetToolbarToolBitmaps(m_toolbar_item_id, new_bitmap, new_rollover);
SetToolbarToolBitmapsSVG(m_toolbar_item_id,
"icons/new_normal.svg",
"icons/new_rollover.svg",
"icons/new_toggled.svg");
Context Menu Integration
Plugins can add items to the chart context menu (right-click menu). This allows users to access plugin functionality directly from the chart display.
Declaring Context Menu Capabilities
To add context menu items, your plugin must declare the appropriate capabilities:
int MyPlugin::Init(void) {
// Initialize resources
// ...
// Return capabilities
return INSTALLS_CONTEXTMENU_ITEMS;
}
Adding Context Menu Items
Add menu items using the AddCanvasContextMenuItem()
function:
wxMenuItem *pmi = new wxMenuItem(NULL, -1, _("My Plugin Action"));
m_menu_item_id = AddCanvasContextMenuItem(pmi, this);
For a submenu with multiple items:
wxMenu *submenu = new wxMenu();
submenu->Append(-1, _("Submenu Item 1"));
submenu->Append(-1, _("Submenu Item 2"));
wxMenuItem *pmi = new wxMenuItem(NULL, -1, _("My Plugin"), wxEmptyString, wxITEM_NORMAL, submenu);
m_menu_item_id = AddCanvasContextMenuItem(pmi, this);
Handling Menu Item Selection
Implement the OnContextMenuItemCallback()
method to handle menu item clicks:
void MyPlugin::OnContextMenuItemCallback(int id) {
if (id == m_menu_item_id) {
// Handle context menu item click
// Perform action, open dialog, etc.
}
}
Managing Context Menu Items
Control context menu item state and visibility:
// Hide/show menu item
SetCanvasContextMenuItemViz(m_menu_item_id, false); // Hide
SetCanvasContextMenuItemViz(m_menu_item_id, true); // Show
// Enable/disable (grey out) menu item
SetCanvasContextMenuItemGrey(m_menu_item_id, true); // Disable (grey out)
SetCanvasContextMenuItemGrey(m_menu_item_id, false); // Enable
Multi-Canvas Support
For API version 1.16 and later, implement the PrepareContextMenu()
method to update menu items based on the current canvas:
void MyPlugin::PrepareContextMenu(int canvasIndex) {
// Update context menu items based on canvas index
// For example, show/hide or enable/disable based on canvas features
bool relevant_for_this_canvas = IsFeatureRelevantForCanvas(canvasIndex);
SetCanvasContextMenuItemViz(m_menu_item_id, relevant_for_this_canvas);
}
Preferences Panels
Plugins can add configuration panels to OpenCPN’s settings dialog. This allows users to configure your plugin through OpenCPN’s standard interface.
Toolbox Pages
The traditional approach is to add pages to the "toolbox" (settings dialog):
Declaring Toolbox Capabilities
int MyPlugin::Init(void) {
// Initialize resources
// ...
// Return capabilities
return INSTALLS_TOOLBOX_PAGE;
}
Implementing Toolbox Methods
int MyPlugin::GetToolboxPanelCount(void) {
return 1; // Number of pages to add
}
void MyPlugin::SetupToolboxPanel(int page_sel, wxNotebook *notebook) {
if (page_sel == 0) {
// Create first page
wxPanel *panel = new wxPanel(notebook);
// Add controls to panel
wxBoxSizer *sizer = new wxBoxSizer(wxVERTICAL);
sizer->Add(new wxStaticText(panel, wxID_ANY, _("My Plugin Settings")),
0, wxALL, 5);
sizer->Add(new wxCheckBox(panel, wxID_ANY, _("Enable feature")),
0, wxALL, 5);
// Add more controls as needed
panel->SetSizer(sizer);
notebook->AddPage(panel, _("My Plugin"));
}
}
void MyPlugin::OnCloseToolboxPanel(int page_sel, int ok_apply_cancel) {
if (page_sel == 0) {
// Save settings based on ok_apply_cancel
if (ok_apply_cancel == 0 || ok_apply_cancel == 1) { // OK or Apply
// Save settings
}
}
}
Preferences Dialog
For API version 1.9 and later, plugins can also use the newer preferences approach:
Declaring Preferences Capabilities
int MyPlugin::Init(void) {
// Initialize resources
// ...
// Return capabilities
return WANTS_PREFERENCES;
}
Implementing Preferences Methods
void MyPlugin::ShowPreferencesDialog(wxWindow *parent) {
// Create and show a dialog
wxDialog *dialog = new wxDialog(parent, wxID_ANY, _("My Plugin Preferences"),
wxDefaultPosition, wxDefaultSize,
wxDEFAULT_DIALOG_STYLE | wxRESIZE_BORDER);
// Add controls to dialog
wxBoxSizer *sizer = new wxBoxSizer(wxVERTICAL);
sizer->Add(new wxStaticText(dialog, wxID_ANY, _("My Plugin Settings")),
0, wxALL, 5);
sizer->Add(new wxCheckBox(dialog, wxID_ANY, _("Enable feature")),
0, wxALL, 5);
// Add more controls as needed
// Add standard buttons
sizer->Add(dialog->CreateButtonSizer(wxOK | wxCANCEL),
0, wxEXPAND | wxALL, 5);
dialog->SetSizer(sizer);
sizer->Fit(dialog);
// Show dialog modally and process result
if (dialog->ShowModal() == wxID_OK) {
// Save settings
}
dialog->Destroy();
}
Options Pages (API 1.13+)
For API version 1.13 and later, plugins can add pages to specific sections of the main options dialog:
void MyPlugin::OnSetupOptions(void) {
// Add a page to the "Charts" section
AddOptionsPage(PI_OPTIONS_PARENT_CHARTS, _("My Plugin Charts"));
// Add a page to the "Connections" section
AddOptionsPage(PI_OPTIONS_PARENT_CONNECTIONS, _("My Plugin Connections"));
// Add widget to page
wxCheckBox *check = AddOptionsPageCheckbox(_("Enable feature"), _("tooltip"), false);
// Add text control with label
AddOptionsPageText(_("Label:"), _("tooltip"), _("default value"));
// Add control groups
wxFlexGridSizer *group = AddOptionsPageSizer(2, 2, 2); // rows, cols, vgap
// Add controls to group
}
Dockable Windows and Panels
Plugins can create dockable windows and panels that integrate with OpenCPN’s window management system.
Declaring AUI Manager Capability
To use the AUI manager, declare the appropriate capability:
int MyPlugin::Init(void) {
// Initialize resources
// ...
// Return capabilities
return USES_AUI_MANAGER;
}
Creating Dockable Windows
void MyPlugin::CreateDockableWindow() {
// Get AUI manager
wxAuiManager *aui_mgr = GetFrameAuiManager();
// Create panel
wxPanel *panel = new wxPanel(GetOCPNCanvasWindow(), wxID_ANY);
// Add content to panel
wxBoxSizer *sizer = new wxBoxSizer(wxVERTICAL);
sizer->Add(new wxStaticText(panel, wxID_ANY, _("My Plugin Panel")),
0, wxALL, 5);
// Add more controls as needed
panel->SetSizer(sizer);
// Add panel to AUI manager
wxAuiPaneInfo pane;
pane.Name("MyPluginPane");
pane.Caption(_("My Plugin"));
pane.Float();
pane.FloatingPosition(100, 100);
pane.FloatingSize(300, 200);
pane.Dockable(true);
aui_mgr->AddPane(panel, pane);
aui_mgr->Update();
// Store panel reference
m_panel = panel;
}
Handling AUI Updates
Implement the UpdateAuiStatus()
method to handle AUI manager updates:
void MyPlugin::UpdateAuiStatus(void) {
// Update panel content or layout if needed
// This is called when the AUI manager updates
}
Removing Dockable Windows
Clean up dockable windows when your plugin is disabled or unloaded:
bool MyPlugin::DeInit(void) {
// Get AUI manager
wxAuiManager *aui_mgr = GetFrameAuiManager();
// Remove panel
if (m_panel) {
aui_mgr->DetachPane(m_panel);
m_panel->Destroy();
m_panel = NULL;
aui_mgr->Update();
}
return true;
}
Color Schemes and Styling
Plugins should respect OpenCPN’s color schemes to provide a consistent user experience.
Implementing Color Scheme Support
Implement the SetColorScheme()
method to update your plugin’s UI colors:
void MyPlugin::SetColorScheme(PI_ColorScheme cs) {
// Store current color scheme
m_color_scheme = cs;
// Update colors of UI elements
switch (cs) {
case PI_GLOBAL_COLOR_SCHEME_DAY:
// Set day colors
m_bg_color = wxColour(255, 255, 255);
m_text_color = wxColour(0, 0, 0);
break;
case PI_GLOBAL_COLOR_SCHEME_DUSK:
// Set dusk colors
m_bg_color = wxColour(80, 80, 80);
m_text_color = wxColour(200, 200, 200);
break;
case PI_GLOBAL_COLOR_SCHEME_NIGHT:
// Set night colors
m_bg_color = wxColour(30, 30, 30);
m_text_color = wxColour(150, 150, 150);
break;
default:
// Use day colors as default
m_bg_color = wxColour(255, 255, 255);
m_text_color = wxColour(0, 0, 0);
break;
}
// Apply colors to UI elements
if (m_panel) {
m_panel->SetBackgroundColour(m_bg_color);
// Update other controls
m_panel->Refresh();
}
}
Getting System Colors
Use the GetGlobalColor()
function to get colors from OpenCPN’s color scheme:
wxColour color;
if (GetGlobalColor("DILG0", &color)) {
// Use color for dialog background
dialog->SetBackgroundColour(color);
}
Best Practices for UI Integration
-
Respect OpenCPN’s UI guidelines: Follow existing patterns for consistency
-
Support all color schemes: Test your UI in day, dusk, and night modes
-
Handle window resizes: Make your UI elements responsive
-
Clean up UI elements: Remove all UI elements in
DeInit()
-
Use translations: Wrap user-visible strings in
_()
for translation -
Provide clear feedback: Use tooltips, status messages, and visual cues
-
Consider performance: Don’t create excessive UI updates
-
Test on different platforms: Ensure your UI works on Windows, macOS, and Linux