diff --git a/src/slic3r/GUI/KBShortcutsDialog.cpp b/src/slic3r/GUI/KBShortcutsDialog.cpp
index 113638105..7d4211b24 100644
--- a/src/slic3r/GUI/KBShortcutsDialog.cpp
+++ b/src/slic3r/GUI/KBShortcutsDialog.cpp
@@ -7,250 +7,297 @@
 #include "GUI_App.hpp"
 #include "wxExtensions.hpp"
 
+#define NOTEBOOK_TOP 1
+#define NOTEBOOK_LEFT 2
+#define LISTBOOK_TOP 3
+#define LISTBOOK_LEFT 4
+#define TOOLBOOK 5
+#define CHOICEBOOK 6
+#define BOOK_TYPE NOTEBOOK_TOP
+
+#if (BOOK_TYPE == NOTEBOOK_TOP) || (BOOK_TYPE == NOTEBOOK_LEFT)
+#include <wx/notebook.h>
+#elif (BOOK_TYPE == LISTBOOK_TOP) || (BOOK_TYPE == LISTBOOK_LEFT)
+#include <wx/listbook.h>
+#elif BOOK_TYPE == TOOLBOOK
+#include <wx/toolbook.h>
+#elif BOOK_TYPE == CHOICEBOOK
+#include <wx/choicebk.h>
+#endif // BOOK_TYPE 
+
+#if ENABLE_SCROLLABLE
+static wxSize get_screen_size(wxWindow* window)
+{
+    const auto idx = wxDisplay::GetFromWindow(window);
+    wxDisplay display(idx != wxNOT_FOUND ? idx : 0u);
+    return display.GetClientArea().GetSize();
+}
+#endif // ENABLE_SCROLLABLE
+
 namespace Slic3r {
 namespace GUI {
 
 KBShortcutsDialog::KBShortcutsDialog()
     : DPIDialog(NULL, wxID_ANY, wxString(SLIC3R_APP_NAME) + " - " + _(L("Keyboard Shortcuts")),
-     wxDefaultPosition, wxDefaultSize, wxDEFAULT_DIALOG_STYLE | wxRESIZE_BORDER)
+#if ENABLE_SCROLLABLE
+    wxDefaultPosition, wxDefaultSize, wxDEFAULT_DIALOG_STYLE | wxRESIZE_BORDER)
+#else
+    wxDefaultPosition, wxDefaultSize, wxDEFAULT_DIALOG_STYLE)
+#endif // ENABLE_SCROLLABLE
 {
     SetBackgroundColour(wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOW));
 
-    auto main_sizer = new wxBoxSizer(wxVERTICAL);
-
-    // logo
-    m_logo_bmp = ScalableBitmap(this, "PrusaSlicer_32px.png", 32);
-
     // fonts
     const wxFont& font = wxGetApp().normal_font();
     const wxFont& bold_font = wxGetApp().bold_font();
     SetFont(font);
 
-    wxFont head_font = bold_font;
-#ifdef __WXOSX__
-    head_font.SetPointSize(14);
-#else
-    head_font.SetPointSize(bold_font.GetPointSize() + 2);
-#endif // __WXOSX__
+    auto main_sizer = new wxBoxSizer(wxVERTICAL);
+
+    main_sizer->Add(create_header(this, bold_font), 0, wxEXPAND | wxALL, 10);
+
+#if BOOK_TYPE == NOTEBOOK_TOP
+    wxNotebook* book = new wxNotebook(this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxNB_TOP);
+#elif BOOK_TYPE == NOTEBOOK_LEFT
+    wxNotebook* book = new wxNotebook(this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxNB_LEFT);
+#elif BOOK_TYPE == LISTBOOK_TOP
+    wxListbook* book = new wxListbook(this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLB_TOP);
+#elif BOOK_TYPE == LISTBOOK_LEFT
+    wxListbook* book = new wxListbook(this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLB_LEFT);
+#elif BOOK_TYPE == TOOLBOOK
+    wxToolbook* book = new wxToolbook(this, wxID_ANY);
+#elif BOOK_TYPE == CHOICEBOOK
+    wxChoicebook* book = new wxChoicebook(this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxCHB_TOP);
+#endif // BOOK_TYPE 
+main_sizer->Add(book, 1, wxEXPAND | wxALL, 10);
 
     fill_shortcuts();
-
-    panel = new wxScrolledWindow(this, wxID_ANY, wxDefaultPosition, get_size());
-    panel->SetScrollbars(1, 20, 1, 2);
-
-    auto main_grid_sizer = new wxFlexGridSizer(2, 10, 10);
-    panel->SetSizer(main_grid_sizer);
-    main_sizer->Add(panel, 1, wxEXPAND | wxALL, 0);
-
-    wxBoxSizer* l_sizer = new wxBoxSizer(wxVERTICAL);
-    main_grid_sizer->Add(l_sizer, 0);
-
-    wxBoxSizer* r_sizer = new wxBoxSizer(wxVERTICAL);
-    main_grid_sizer->Add(r_sizer, 0);
-
-    m_head_bitmaps.reserve(m_full_shortcuts.size());
-
-    for (auto& shortcut : m_full_shortcuts)
+    for (size_t i = 0; i < m_full_shortcuts.size(); ++i)
     {
-        auto sizer = shortcut.second.second == szLeft ? l_sizer : r_sizer;
-        wxBoxSizer* hsizer = new wxBoxSizer(wxHORIZONTAL);
-        sizer->Add(hsizer, 0, wxEXPAND | wxTOP | wxBOTTOM, 10);
-
-        // logo
-        m_head_bitmaps.push_back(new wxStaticBitmap(panel, wxID_ANY, m_logo_bmp.bmp()));
-        hsizer->Add(m_head_bitmaps.back(), 0, wxEXPAND | wxLEFT | wxRIGHT, 15);
-
-        // head
-        wxStaticText* head = new wxStaticText(panel, wxID_ANY, shortcut.first);
-        head->SetFont(head_font);
-        hsizer->Add(head, 0, wxALIGN_CENTER_VERTICAL);
-
-
-        // Shortcuts list
-        auto grid_sizer = new wxFlexGridSizer(2, 5, 15);
-        sizer->Add(grid_sizer, 0, wxEXPAND | wxLEFT| wxRIGHT, 15);
-
-        for (auto pair : shortcut.second.first)
-        {
-            auto shortcut = new wxStaticText(panel, wxID_ANY, _(pair.first));
-            shortcut->SetFont(bold_font);
-            grid_sizer->Add(shortcut, -1, wxALIGN_CENTRE_VERTICAL);
-
-            auto description = new wxStaticText(panel, wxID_ANY, _(pair.second));
-            description->SetFont(font);
-            grid_sizer->Add(description, -1, wxALIGN_CENTRE_VERTICAL);
-        }
+#if ENABLE_SCROLLABLE
+        wxPanel* page = create_page(book, m_full_shortcuts[i], font, bold_font);
+        m_pages.push_back(page);
+        book->AddPage(page, m_full_shortcuts[i].first, i == 0);
+#else
+        book->AddPage(create_page(book, m_full_shortcuts[i], font, bold_font), m_full_shortcuts[i].first, i == 0);
+#endif // ENABLE_SCROLLABLE
     }
 
     wxStdDialogButtonSizer* buttons = this->CreateStdDialogButtonSizer(wxOK);
-
     this->SetEscapeId(wxID_OK);
-    this->Bind(wxEVT_BUTTON, &KBShortcutsDialog::onCloseDialog, this, wxID_OK);
-    main_sizer->Add(buttons, 0, wxEXPAND | wxRIGHT | wxBOTTOM, 15);
-
-    this->Bind(wxEVT_LEFT_DOWN, &KBShortcutsDialog::onCloseDialog, this);
+    main_sizer->Add(buttons, 0, wxEXPAND | wxALL, 5);
 
     SetSizer(main_sizer);
     main_sizer->SetSizeHints(this);
 }
 
-void KBShortcutsDialog::fill_shortcuts()
-{
-    const std::string &ctrl = GUI::shortkey_ctrl_prefix();
-    const std::string &alt  = GUI::shortkey_alt_prefix();
-
-    m_full_shortcuts.reserve(4);
-
-    Shortcuts main_shortcuts;
-    main_shortcuts.reserve(25);
-
-    main_shortcuts.push_back(Shortcut(ctrl+"O"          ,L("Open project STL/OBJ/AMF/3MF with config, delete bed")));
-    main_shortcuts.push_back(Shortcut(ctrl+"I"          ,L("Import STL/OBJ/AMF/3MF without config, keep bed")));
-    main_shortcuts.push_back(Shortcut(ctrl+"L"          ,L("Load Config from .ini/amf/3mf/gcode")));
-    main_shortcuts.push_back(Shortcut(ctrl+"G"          ,L("Export G-code")));
-    main_shortcuts.push_back(Shortcut(ctrl+"S"          ,L("Save project (3MF)")));
-    main_shortcuts.push_back(Shortcut(ctrl+alt+"L"      ,L("Load Config from .ini/amf/3mf/gcode and merge")));
-    main_shortcuts.push_back(Shortcut(ctrl+"R"          ,L("(Re)slice")));
-//    main_shortcuts.push_back(Shortcut(ctrl+"U"          ,L("Quick slice")));
-//    main_shortcuts.push_back(Shortcut(ctrl+"Shift+U"    ,L("Repeat last quick slice")));
-    main_shortcuts.push_back(Shortcut(ctrl+"1"          ,L("Select Plater Tab")));
-//    main_shortcuts.push_back(Shortcut(ctrl+alt+"U"      ,L("Quick slice and Save as")));
-    main_shortcuts.push_back(Shortcut(ctrl+"2"          ,L("Select Print Settings Tab")));
-    main_shortcuts.push_back(Shortcut(ctrl+"3"          ,L("Select Filament Settings Tab")));
-    main_shortcuts.push_back(Shortcut(ctrl+"4"          ,L("Select Printer Settings Tab")));
-    main_shortcuts.push_back(Shortcut(ctrl+"5"          ,L("Switch to 3D")));
-    main_shortcuts.push_back(Shortcut(ctrl+"6"          ,L("Switch to Preview")));
-    main_shortcuts.push_back(Shortcut(ctrl+"P"          ,L("Preferences")));
-    main_shortcuts.push_back(Shortcut(ctrl+"J"          ,L("Print host upload queue")));
-    main_shortcuts.push_back(Shortcut("0-6"             ,L("Camera view")));
-    main_shortcuts.push_back(Shortcut("+"               ,L("Add Instance of the selected object")));
-    main_shortcuts.push_back(Shortcut("-"               ,L("Remove Instance of the selected object")));
-    main_shortcuts.push_back(Shortcut("?"               ,L("Show keyboard shortcuts list")));
-    main_shortcuts.push_back(Shortcut(ctrl/*+"LeftMouse"*/,L("Press to select multiple object or move multiple object with mouse")));
-
-    m_full_shortcuts.push_back(std::make_pair(_(L("Main Shortcuts")), std::make_pair(main_shortcuts, szLeft)));
-
-
-    Shortcuts plater_shortcuts;
-    plater_shortcuts.reserve(20);
-
-    plater_shortcuts.push_back(Shortcut("A",        L("Arrange")));
-    plater_shortcuts.push_back(Shortcut("Shift+A",  L("Arrange selection")));
-    plater_shortcuts.push_back(Shortcut(ctrl+"A",   L("Select All objects")));
-    plater_shortcuts.push_back(Shortcut("Del",      L("Delete selected")));
-    plater_shortcuts.push_back(Shortcut(ctrl+"Del", L("Delete All")));
-    plater_shortcuts.push_back(Shortcut(ctrl+"C",   L("Copy to clipboard")));
-    plater_shortcuts.push_back(Shortcut(ctrl+"V",   L("Paste from clipboard")));
-    plater_shortcuts.push_back(Shortcut("M",        L("Gizmo move")));
-    plater_shortcuts.push_back(Shortcut("S",        L("Gizmo scale")));
-    plater_shortcuts.push_back(Shortcut("R",        L("Gizmo rotate")));
-    plater_shortcuts.push_back(Shortcut("C",        L("Gizmo cut")));
-    plater_shortcuts.push_back(Shortcut("F",        L("Gizmo Place face on bed")));
-    plater_shortcuts.push_back(Shortcut("L",        L("Gizmo SLA support points")));
-    plater_shortcuts.push_back(Shortcut("Shift+",   L("Press to activate selection rectangle\nor to snap by 5% in Gizmo scale\nor to snap by 1mm in Gizmo move")));
-    plater_shortcuts.push_back(Shortcut("F",        L("Press to scale selection to fit print volume\nin Gizmo scale")));
-    plater_shortcuts.push_back(Shortcut(alt,        L("Press to activate deselection rectangle\nor to scale or rotate selected objects\naround their own center")));
-    plater_shortcuts.push_back(Shortcut(ctrl,       L("Press to activate one direction scaling in Gizmo scale")));
-    plater_shortcuts.push_back(Shortcut("K",        L("Change camera type (perspective, orthographic)")));
-    plater_shortcuts.push_back(Shortcut("B",        L("Zoom to Bed")));
-    plater_shortcuts.push_back(Shortcut("Z",        L("Zoom to all objects in scene, if none selected")));
-    plater_shortcuts.push_back(Shortcut("Z",        L("Zoom to selected object")));
-    plater_shortcuts.push_back(Shortcut("I",        L("Zoom in")));
-    plater_shortcuts.push_back(Shortcut("O",        L("Zoom out")));
-    plater_shortcuts.push_back(Shortcut("E",        L("Show/Hide object/instance labels")));
-    plater_shortcuts.push_back(Shortcut(ctrl+"M",   L("Show/Hide 3Dconnexion devices settings dialog")));
-    plater_shortcuts.push_back(Shortcut("ESC",      L("Unselect gizmo / Clear selection")));
-#if ENABLE_RENDER_PICKING_PASS
-    // Don't localize debugging texts.
-    plater_shortcuts.push_back(Shortcut("T",        "Toggle picking pass texture rendering on/off"));
-#endif // ENABLE_RENDER_PICKING_PASS
-
-    m_full_shortcuts.push_back(std::make_pair(_(L("Plater Shortcuts")), std::make_pair(plater_shortcuts, szRight)));
-
-
-//     Shortcuts gizmo_shortcuts;
-//     gizmo_shortcuts.reserve(2);
-//
-//     gizmo_shortcuts.push_back(Shortcut("Shift+",    L("Press to snap by 5% in Gizmo Scale\n or by 1mm in Gizmo Move")));
-//     gizmo_shortcuts.push_back(Shortcut(alt,         L("Press to scale or rotate selected objects around their own center")));
-//
-//     m_full_shortcuts.push_back(std::make_pair(_(L("Gizmo Shortcuts")), std::make_pair(gizmo_shortcuts, 1)));
-
-
-    Shortcuts preview_shortcuts;
-    preview_shortcuts.reserve(4);
-
-    preview_shortcuts.push_back(Shortcut(L("Arrow Up"),     L("Upper Layer")));
-    preview_shortcuts.push_back(Shortcut(L("Arrow Down"),   L("Lower Layer")));
-    preview_shortcuts.push_back(Shortcut("U",               L("Upper Layer")));
-    preview_shortcuts.push_back(Shortcut("D",               L("Lower Layer")));
-    preview_shortcuts.push_back(Shortcut("L",               L("Show/Hide (L)egend")));
-
-    m_full_shortcuts.push_back(std::make_pair(_(L("Preview Shortcuts")), std::make_pair(preview_shortcuts, szLeft)));
-
-
-    Shortcuts layers_slider_shortcuts;
-    layers_slider_shortcuts.reserve(6);
-
-    layers_slider_shortcuts.push_back(Shortcut(L("Arrow Up"),   L("Move current slider thumb Up")));
-    layers_slider_shortcuts.push_back(Shortcut(L("Arrow Down"), L("Move current slider thumb Down")));
-    layers_slider_shortcuts.push_back(Shortcut(L("Arrow Left"), L("Set upper thumb to current slider thumb")));
-    layers_slider_shortcuts.push_back(Shortcut(L("Arrow Right"),L("Set lower thumb to current slider thumb")));
-    layers_slider_shortcuts.push_back(Shortcut("+",             L("Add color change marker for current layer")));
-    layers_slider_shortcuts.push_back(Shortcut("-",             L("Delete color change marker for current layer")));
-
-    m_full_shortcuts.push_back(std::make_pair(_(L("Layers Slider Shortcuts")), std::make_pair(layers_slider_shortcuts, szRight)));
-}
-
-void KBShortcutsDialog::on_dpi_changed(const wxRect &suggested_rect)
+void KBShortcutsDialog::on_dpi_changed(const wxRect& suggested_rect)
 {
     m_logo_bmp.msw_rescale();
+    m_header_bitmap->SetBitmap(m_logo_bmp.bmp());
+    msw_buttons_rescale(this, em_unit(), { wxID_OK });
 
-    for (wxStaticBitmap* bmp : m_head_bitmaps)
-        bmp->SetBitmap(m_logo_bmp.bmp());
-
-    const int em = em_unit();
-
-    msw_buttons_rescale(this, em, { wxID_OK });
-
-    wxSize size = get_size();
-
-    panel->SetMinSize(size);
-
-    SetMinSize(size);
+    Layout();
     Fit();
-
     Refresh();
 }
 
-void KBShortcutsDialog::onCloseDialog(wxEvent &)
+void KBShortcutsDialog::fill_shortcuts()
 {
-    this->EndModal(wxID_CLOSE);
+    const std::string& ctrl = GUI::shortkey_ctrl_prefix();
+    const std::string& alt = GUI::shortkey_alt_prefix();
+
+    Shortcuts commands_shortcuts = {
+        // File
+        { ctrl + "N", L("New project, clear plater") },
+        { ctrl + "O", L("Open project STL/OBJ/AMF/3MF with config, clear plater") },
+        { ctrl + "S", L("Save project (3mf)") },
+        { ctrl + alt + "S", L("Save project as (3mf)") },
+        { ctrl + "R", L("(Re)slice") },
+        // File>Import
+        { ctrl + "I", L("Import STL/OBJ/AMF/3MF without config, keep plater") },
+        { ctrl + "L", L("Import Config from ini/amf/3mf/gcode") },
+        { ctrl + alt + "L", L("Load Config from ini/amf/3mf/gcode and merge") },
+        // File>Export
+        { ctrl + "G", L("Export G-code") },
+        { ctrl + "Shift+" + "G", L("Send G-code") },
+        { ctrl + "E", L("Export config") },
+        // Edit
+        { ctrl + "A", L("Select all objects") },
+        { "Esc", L("Deselect all") },
+        { "Del", L("Delete selected") },
+        { ctrl + "Del", L("Delete all") },
+        { ctrl + "Z", L("Undo") },
+        { ctrl + "Y", L("Redo") },
+        { ctrl + "C", L("Copy to clipboard") },
+        { ctrl + "V", L("Paste from clipboard") },
+        { "F5", L("Reload plater from disk") },
+        // Window
+        { ctrl + "1", L("Select Plater Tab") },
+        { ctrl + "2", L("Select Print Settings Tab") },
+        { ctrl + "3", L("Select Filament Settings Tab") },
+        { ctrl + "4", L("Select Printer Settings Tab") },
+        { ctrl + "5", L("Switch to 3D") },
+        { ctrl + "6", L("Switch to Preview") },
+        { ctrl + "J", L("Print host upload queue") },
+        // View
+        { "0-6", L("Camera view") },
+#if ENABLE_SHOW_SCENE_LABELS
+        { "E", L("Show/Hide object/instance labels") },
+#endif // ENABLE_SHOW_SCENE_LABELS
+        // Configuration
+        { ctrl + "P", L("Preferences") },
+        // Help
+        { "?", L("Show keyboard shortcuts list") }
+    };
+
+    m_full_shortcuts.push_back(std::make_pair(_(L("Commands")), commands_shortcuts));
+
+    Shortcuts plater_shortcuts = {
+        { "A", L("Arrange") },
+        { "Shift+A", L("Arrange selection") },
+        { "+", L("Add Instance of the selected object") },
+        { "-", L("Remove Instance of the selected object") },
+        { ctrl, L("Press to select multiple object\nor move multiple object with mouse") },
+        { "Shift+", L("Press to activate selection rectangle") },
+        { alt, L("Press to activate deselection rectangle") },
+        { L("Arrow Up"), L("Move selection 10 mm in positive Y direction") },
+        { L("Arrow Down"), L("Move selection 10 mm in negative Y direction") },
+        { L("Arrow Left"), L("Move selection 10 mm in negative X direction") },
+        { L("Arrow Right"), L("Move selection 10 mm in positive X direction") },
+        { std::string("Shift+") + L("Any arrow"), L("Movement step set to 1 mm") },
+        { ctrl + L("Any arrow"), L("Movement in camera space") },
+        { L("Page Up"), L("Rotate selection 45 degrees CCW") },
+        { L("Page Down"), L("Rotate selection 45 degrees CW") },
+        { "M", L("Gizmo move") },
+        { "S", L("Gizmo scale") },
+        { "R", L("Gizmo rotate") },
+        { "C", L("Gizmo cut") },
+        { "F", L("Gizmo Place face on bed") },
+        { "H", L("Gizmo SLA hollow") },
+        { "L", L("Gizmo SLA support points") },
+        { "Esc", L("Unselect gizmo or clear selection") },
+        { "K", L("Change camera type (perspective, orthographic)") },
+        { "B", L("Zoom to Bed") },
+        { "Z", L("Zoom to selected object\nor all objects in scene, if none selected") },
+        { "I", L("Zoom in") },
+        { "O", L("Zoom out") },
+        { ctrl + "M", L("Show/Hide 3Dconnexion devices settings dialog") }
+#if ENABLE_RENDER_PICKING_PASS
+        // Don't localize debugging texts.
+        , { "T", "Toggle picking pass texture rendering on/off" }
+#endif // ENABLE_RENDER_PICKING_PASS
+    };
+
+    m_full_shortcuts.push_back(std::make_pair(_(L("Plater")), plater_shortcuts));
+
+    Shortcuts gizmos_shortcuts = {
+        { "Shift+", L("Press to to snap by 5% in Gizmo scale\nor to snap by 1mm in Gizmo move") },
+        { "F", L("Scale selection to fit print volume\nin Gizmo scale") },
+        { ctrl, L("Press to activate one direction scaling in Gizmo scale") },
+        { alt, L("Press to scale (in Gizmo scale) or rotate (in Gizmo rotate)\nselected objects around their own center") },
+    };
+
+    m_full_shortcuts.push_back(std::make_pair(_(L("Gizmos")), gizmos_shortcuts));
+
+    Shortcuts preview_shortcuts = {
+        { L("Arrow Up"), L("Upper Layer") },
+        { L("Arrow Down"), L("Lower Layer") },
+        { "U", L("Upper Layer") },
+        { "D", L("Lower Layer") },
+        { "L", L("Show/Hide Legend") }
+    };
+
+    m_full_shortcuts.push_back(std::make_pair(_(L("Preview")), preview_shortcuts));
+
+    Shortcuts layers_slider_shortcuts = {
+        { L("Arrow Up"), L("Move current slider thumb Up") },
+        { L("Arrow Down"), L("Move current slider thumb Down") },
+        { L("Arrow Left"), L("Set upper thumb to current slider thumb") },
+        { L("Arrow Right"), L("Set lower thumb to current slider thumb") },
+        { "+", L("Add color change marker for current layer") },
+        { "-", L("Delete color change marker for current layer") }
+    };
+
+    m_full_shortcuts.push_back(std::make_pair(_(L("Layers Slider")), layers_slider_shortcuts));
 }
 
-wxSize KBShortcutsDialog::get_size()
+wxPanel* KBShortcutsDialog::create_header(wxWindow* parent, const wxFont& bold_font)
 {
-    wxTopLevelWindow* window = Slic3r::GUI::find_toplevel_parent(this);
-    const int display_idx = wxDisplay::GetFromWindow(window);
-    wxRect display;
-    if (display_idx == wxNOT_FOUND) {
-        display = wxDisplay(0u).GetClientArea();
-        window->Move(display.GetTopLeft());
-    }
-    else {
-        display = wxDisplay(display_idx).GetClientArea();
+    wxPanel* panel = new wxPanel(parent);
+    wxBoxSizer* sizer = new wxBoxSizer(wxHORIZONTAL);
+
+    wxFont header_font = bold_font;
+#ifdef __WXOSX__
+    header_font.SetPointSize(14);
+#else
+    header_font.SetPointSize(bold_font.GetPointSize() + 2);
+#endif // __WXOSX__
+
+    sizer->AddStretchSpacer();
+
+    // logo
+    m_logo_bmp = ScalableBitmap(this, "PrusaSlicer_32px.png", 32);
+    m_header_bitmap = new wxStaticBitmap(panel, wxID_ANY, m_logo_bmp.bmp());
+    sizer->Add(m_header_bitmap, 0, wxEXPAND | wxLEFT | wxRIGHT, 10);
+
+    // text
+    wxStaticText* text = new wxStaticText(panel, wxID_ANY, _(L("Keyboard shortcuts")));
+    text->SetFont(header_font);
+    sizer->Add(text, 0, wxALIGN_CENTER_VERTICAL);
+
+    sizer->AddStretchSpacer();
+
+    panel->SetSizer(sizer);
+    return panel;
+}
+
+wxPanel* KBShortcutsDialog::create_page(wxWindow* parent, const std::pair<wxString, Shortcuts>& shortcuts, const wxFont& font, const wxFont& bold_font)
+{
+    static const int max_items_per_column = 20;
+    int columns_count = 1 + (int)shortcuts.second.size() / max_items_per_column;
+
+#if ENABLE_SCROLLABLE
+    wxScrolledWindow* page = new wxScrolledWindow(parent);
+    page->SetScrollbars(20, 20, 50, 50);
+    page->SetInitialSize(wxSize(850, 450));
+#else
+    wxPanel* page = new wxPanel(parent);
+#endif // ENABLE_SCROLLABLE
+
+#if (BOOK_TYPE == LISTBOOK_TOP) || (BOOK_TYPE == LISTBOOK_LEFT)
+    wxStaticBoxSizer* sizer = new wxStaticBoxSizer(wxVERTICAL, page, " " + shortcuts.first + " ");
+    sizer->GetStaticBox()->SetFont(bold_font);
+#else
+    wxBoxSizer* sizer = new wxBoxSizer(wxVERTICAL);
+#endif // BOOK_TYPE
+
+    wxFlexGridSizer* grid_sizer = new wxFlexGridSizer(2 * columns_count, 5, 15);
+
+    int items_count = (int)shortcuts.second.size();
+    for (int i = 0; i < max_items_per_column; ++i)
+    {
+        for (int j = 0; j < columns_count; ++j)
+        {
+            int id = j * max_items_per_column + i;
+            if (id >= items_count)
+                break;
+
+            const auto& [shortcut, description] = shortcuts.second[id];
+            auto key = new wxStaticText(page, wxID_ANY, _(shortcut));
+            key->SetFont(bold_font);
+            grid_sizer->Add(key, 0, wxALIGN_CENTRE_VERTICAL);
+
+            auto desc = new wxStaticText(page, wxID_ANY, _(description));
+            desc->SetFont(font);
+            grid_sizer->Add(desc, 0, wxALIGN_CENTRE_VERTICAL);
+        }
     }
 
-    const int em = em_unit();
-    wxSize dialog_size = wxSize(90 * em, 85 * em);
+    sizer->Add(grid_sizer, 1, wxEXPAND | wxALL, 10);
 
-    const int margin = 10 * em;
-    if (dialog_size.x > display.GetWidth())
-        dialog_size.x = display.GetWidth() - margin;
-    if (dialog_size.y > display.GetHeight())
-        dialog_size.y = display.GetHeight() - margin;
-
-    return dialog_size;
+    page->SetSizer(sizer);
+    return page;
 }
 
 } // namespace GUI
diff --git a/src/slic3r/GUI/KBShortcutsDialog.hpp b/src/slic3r/GUI/KBShortcutsDialog.hpp
index 7ac28778b..70820ae77 100644
--- a/src/slic3r/GUI/KBShortcutsDialog.hpp
+++ b/src/slic3r/GUI/KBShortcutsDialog.hpp
@@ -8,38 +8,36 @@
 #include "GUI_Utils.hpp"
 #include "wxExtensions.hpp"
 
+#define ENABLE_SCROLLABLE 1
+
 namespace Slic3r { 
 namespace GUI {
 
 class KBShortcutsDialog : public DPIDialog
 {
-    enum PLACED_SIZER_ID
-    {
-        szLeft = 0,
-        szRight
-    };
-
     typedef std::pair<std::string, std::string> Shortcut;
-    typedef std::vector< Shortcut >             Shortcuts;
-    typedef std::vector< std::pair<wxString, std::pair<Shortcuts, PLACED_SIZER_ID>> >   ShortcutsVec;
+    typedef std::vector<Shortcut> Shortcuts;
+    typedef std::vector<std::pair<wxString, Shortcuts>> ShortcutsVec;
 
-    wxScrolledWindow*               panel;
-
-    ShortcutsVec                    m_full_shortcuts;
-    ScalableBitmap                  m_logo_bmp;
-    std::vector<wxStaticBitmap*>    m_head_bitmaps;
+    ShortcutsVec    m_full_shortcuts;
+    ScalableBitmap  m_logo_bmp;
+    wxStaticBitmap* m_header_bitmap;
+#if ENABLE_SCROLLABLE
+    std::vector<wxPanel*> m_pages;
+#endif // ENABLE_SCROLLABLE
 
 public:
     KBShortcutsDialog();
     
-    void fill_shortcuts();
-
 protected:
     void on_dpi_changed(const wxRect &suggested_rect) override;
 
 private:
-    void onCloseDialog(wxEvent &);
-    wxSize get_size();
+    void fill_shortcuts();
+
+    wxPanel* create_header(wxWindow* parent, const wxFont& bold_font);
+    wxPanel* create_page(wxWindow* parent, const std::pair<wxString, Shortcuts>& shortcuts, const wxFont& font, const wxFont& bold_font);
+
 };
 
 } // namespace GUI