summaryrefslogtreecommitdiffstats
path: root/starmath/source/ElementsDockingWindow.cxx
diff options
context:
space:
mode:
Diffstat (limited to '')
-rw-r--r--starmath/source/ElementsDockingWindow.cxx1230
1 files changed, 1230 insertions, 0 deletions
diff --git a/starmath/source/ElementsDockingWindow.cxx b/starmath/source/ElementsDockingWindow.cxx
new file mode 100644
index 000000000..c21933fe1
--- /dev/null
+++ b/starmath/source/ElementsDockingWindow.cxx
@@ -0,0 +1,1230 @@
+/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
+/*
+ * This file is part of the LibreOffice project.
+ *
+ * This Source Code Form is subject to the terms of the Mozilla Public
+ * License, v. 2.0. If a copy of the MPL was not distributed with this
+ * file, You can obtain one at http://mozilla.org/MPL/2.0/.
+ *
+ * This file incorporates work covered by the following license notice:
+ *
+ * Licensed to the Apache Software Foundation (ASF) under one or more
+ * contributor license agreements. See the NOTICE file distributed
+ * with this work for additional information regarding copyright
+ * ownership. The ASF licenses this file to you under the Apache
+ * License, Version 2.0 (the "License"); you may not use this file
+ * except in compliance with the License. You may obtain a copy of
+ * the License at http://www.apache.org/licenses/LICENSE-2.0 .
+ */
+
+#include <memory>
+#include <ElementsDockingWindow.hxx>
+
+#include <starmath.hrc>
+#include <strings.hrc>
+#include <smmod.hxx>
+#include <view.hxx>
+#include <visitors.hxx>
+#include <document.hxx>
+#include <node.hxx>
+#include "uiobject.hxx"
+#include <strings.hxx>
+
+#include <sfx2/dispatch.hxx>
+#include <sfx2/sfxmodelfactory.hxx>
+#include <svl/stritem.hxx>
+#include <svtools/colorcfg.hxx>
+#include <vcl/event.hxx>
+#include <vcl/help.hxx>
+#include <vcl/settings.hxx>
+#include <vcl/uitest/eventdescription.hxx>
+#include <vcl/uitest/logger.hxx>
+
+SmElement::SmElement(std::unique_ptr<SmNode>&& pNode, const OUString& aText, const OUString& aHelpText) :
+ mpNode(std::move(pNode)),
+ maText(aText),
+ maHelpText(aHelpText)
+{}
+
+SmElement::~SmElement()
+{}
+
+const std::unique_ptr<SmNode>& SmElement::getNode() const { return mpNode; }
+
+SmElementSeparator::SmElementSeparator() :
+ SmElement(std::unique_ptr<SmNode>(), OUString(), OUString())
+{}
+
+const SmElementDescr SmElementsControl::m_aUnaryBinaryOperatorsList[] =
+{
+ {RID_PLUSX, RID_PLUSX_HELP}, {RID_MINUSX, RID_MINUSX_HELP},
+ {RID_PLUSMINUSX, RID_PLUSMINUSX_HELP}, {RID_MINUSPLUSX, RID_MINUSPLUSX_HELP},
+ {nullptr, nullptr},
+ {RID_XPLUSY, RID_XPLUSY_HELP}, {RID_XMINUSY, RID_XMINUSY_HELP},
+ {RID_XCDOTY, RID_XCDOTY_HELP}, {RID_XTIMESY, RID_XTIMESY_HELP},
+ {RID_XSYMTIMESY, RID_XSYMTIMESY_HELP}, {RID_XOVERY, RID_XOVERY_HELP},
+ {RID_XDIVY, RID_XDIVY_HELP}, {RID_XSYMDIVIDEY, RID_XSYMDIVIDEY_HELP},
+ {RID_XOPLUSY, RID_XOPLUSY_HELP}, {RID_XOMINUSY, RID_XOMINUSY_HELP},
+ {RID_XODOTY, RID_XODOTY_HELP}, {RID_XOTIMESY, RID_XOTIMESY_HELP},
+ {RID_XODIVIDEY, RID_XODIVIDEY_HELP}, {RID_XCIRCY, RID_XCIRCY_HELP},
+ {RID_XWIDESLASHY, RID_XWIDESLASHY_HELP}, {RID_XWIDEBSLASHY, RID_XWIDEBSLASHY_HELP},
+ {nullptr, nullptr},
+ {RID_NEGX, RID_NEGX_HELP}, {RID_XANDY, RID_XANDY_HELP}, {RID_XORY, RID_XORY_HELP},
+};
+
+const SmElementDescr SmElementsControl::m_aRelationsList[] =
+{
+ {RID_XEQY, RID_XEQY_HELP}, {RID_XNEQY, RID_XNEQY_HELP}, {RID_XLTY, RID_XLTY_HELP},
+ {RID_XLEY, RID_XLEY_HELP}, {RID_XLESLANTY, RID_XLESLANTY_HELP}, {RID_XGTY, RID_XGTY_HELP},
+ {RID_XGEY, RID_XGEY_HELP}, {RID_XGESLANTY, RID_XGESLANTY_HELP},
+ {RID_XLLY, RID_XLLY_HELP}, {RID_XGGY, RID_XGGY_HELP},
+ {nullptr, nullptr},
+ {RID_XAPPROXY, RID_XAPPROXY_HELP}, {RID_XSIMY, RID_XSIMY_HELP}, {RID_XSIMEQY, RID_XSIMEQY_HELP},
+ {RID_XEQUIVY, RID_XEQUIVY_HELP}, {RID_XPROPY, RID_XPROPY_HELP}, {RID_XPARALLELY, RID_XPARALLELY_HELP},
+ {RID_XORTHOY, RID_XORTHOY_HELP}, {RID_XDIVIDESY, RID_XDIVIDESY_HELP}, {RID_XNDIVIDESY, RID_XNDIVIDESY_HELP},
+ {RID_XTOWARDY, RID_XTOWARDY_HELP}, {RID_XTRANSLY, RID_XTRANSLY_HELP}, {RID_XTRANSRY, RID_XTRANSRY_HELP},
+ {RID_XDEFY, RID_XDEFY_HELP},
+ {nullptr, nullptr},
+ {RID_DLARROW, RID_DLARROW_HELP}, {RID_DLRARROW, RID_DLRARROW_HELP}, {RID_DRARROW, RID_DRARROW_HELP},
+ {nullptr, nullptr},
+ {RID_XPRECEDESY, RID_XPRECEDESY_HELP}, {RID_XSUCCEEDSY, RID_XSUCCEEDSY_HELP},
+ {RID_XPRECEDESEQUALY, RID_XPRECEDESEQUALY_HELP}, {RID_XSUCCEEDSEQUALY, RID_XSUCCEEDSEQUALY_HELP},
+ {RID_XPRECEDESEQUIVY, RID_XPRECEDESEQUIVY_HELP}, {RID_XSUCCEEDSEQUIVY, RID_XSUCCEEDSEQUIVY_HELP},
+ {RID_XNOTPRECEDESY, RID_XNOTPRECEDESY_HELP}, {RID_XNOTSUCCEEDSY, RID_XNOTSUCCEEDSY_HELP},
+};
+
+const SmElementDescr SmElementsControl::m_aSetOperationsList[] =
+{
+ {RID_XINY, RID_XINY_HELP}, {RID_XNOTINY, RID_XNOTINY_HELP}, {RID_XOWNSY, RID_XOWNSY_HELP},
+ {nullptr, nullptr},
+ {RID_XINTERSECTIONY, RID_XINTERSECTIONY_HELP}, {RID_XUNIONY, RID_XUNIONY_HELP},
+ {RID_XSETMINUSY, RID_XSETMINUSY_HELP}, {RID_XSLASHY, RID_XSLASHY_HELP},
+ {RID_XSUBSETY, RID_XSUBSETY_HELP}, {RID_XSUBSETEQY, RID_XSUBSETEQY_HELP},
+ {RID_XSUPSETY, RID_XSUPSETY_HELP}, {RID_XSUPSETEQY, RID_XSUPSETEQY_HELP},
+ {RID_XNSUBSETY, RID_XNSUBSETY_HELP}, {RID_XNSUBSETEQY, RID_XNSUBSETEQY_HELP},
+ {RID_XNSUPSETY, RID_XNSUPSETY_HELP}, {RID_XNSUPSETEQY, RID_XNSUPSETEQY_HELP},
+ {nullptr, nullptr},
+ {RID_EMPTYSET, RID_EMPTYSET_HELP}, {RID_ALEPH, RID_ALEPH_HELP}, {RID_SETN, RID_SETN_HELP},
+ {RID_SETZ, RID_SETZ_HELP}, {RID_SETQ, RID_SETQ_HELP}, {RID_SETR, RID_SETR_HELP}, {RID_SETC, RID_SETC_HELP}
+};
+
+const SmElementDescr SmElementsControl::m_aFunctionsList[] =
+{
+ {RID_ABSX, RID_ABSX_HELP}, {RID_FACTX, RID_FACTX_HELP}, {RID_SQRTX, RID_SQRTX_HELP},
+ {RID_NROOTXY, RID_NROOTXY_HELP}, {RID_RSUPX, RID_RSUPX_HELP}, {RID_EX, RID_EX_HELP},
+ {RID_LNX, RID_LNX_HELP}, {RID_EXPX, RID_EXPX_HELP}, {RID_LOGX, RID_LOGX_HELP},
+ {nullptr, nullptr},
+ {RID_SINX, RID_SINX_HELP}, {RID_COSX, RID_COSX_HELP}, {RID_TANX, RID_TANX_HELP}, {RID_COTX, RID_COTX_HELP},
+ {RID_SINHX, RID_SINHX_HELP}, {RID_COSHX, RID_COSHX_HELP}, {RID_TANHX, RID_TANHX_HELP},
+ {RID_COTHX, RID_COTHX_HELP},
+ {nullptr, nullptr},
+ {RID_ARCSINX, RID_ARCSINX_HELP}, {RID_ARCCOSX, RID_ARCCOSX_HELP}, {RID_ARCTANX, RID_ARCTANX_HELP},
+ {RID_ARCCOTX, RID_ARCCOTX_HELP}, {RID_ARSINHX, RID_ARSINHX_HELP}, {RID_ARCOSHX, RID_ARCOSHX_HELP},
+ {RID_ARTANHX, RID_ARTANHX_HELP}, {RID_ARCOTHX, RID_ARCOTHX_HELP}
+};
+
+const SmElementDescr SmElementsControl::m_aOperatorsList[] =
+{
+ {RID_LIMX, RID_LIMX_HELP}, {RID_LIM_FROMX, RID_LIM_FROMX_HELP},
+ {RID_LIM_TOX, RID_LIM_TOX_HELP}, {RID_LIM_FROMTOX, RID_LIM_FROMTOX_HELP},
+ {nullptr, nullptr},
+ {RID_LIMINFX, RID_LIMINFX_HELP}, {RID_LIMINF_FROMX, RID_LIMINF_FROMX_HELP},
+ {RID_LIMINF_TOX, RID_LIMINF_TOX_HELP}, {RID_LIMINF_FROMTOX, RID_LIMINF_FROMTOX_HELP},
+ {nullptr, nullptr},
+ {RID_LIMSUPX, RID_LIMSUPX_HELP}, {RID_LIMSUP_FROMX, RID_LIMSUP_FROMX_HELP},
+ {RID_LIMSUP_TOX, RID_LIMSUP_TOX_HELP}, {RID_LIMSUP_FROMTOX, RID_LIMSUP_FROMTOX_HELP},
+ {nullptr, nullptr},
+ {RID_SUMX, RID_SUMX_HELP}, {RID_SUM_FROMX, RID_SUM_FROMX_HELP},
+ {RID_SUM_TOX, RID_SUM_TOX_HELP}, {RID_SUM_FROMTOX, RID_SUM_FROMTOX_HELP},
+ {nullptr, nullptr},
+ {RID_PRODX, RID_PRODX_HELP}, {RID_PROD_FROMX, RID_PROD_FROMX_HELP},
+ {RID_PROD_TOX, RID_PROD_TOX_HELP}, {RID_PROD_FROMTOX, RID_PROD_FROMTOX_HELP},
+ {nullptr, nullptr},
+ {RID_COPRODX, RID_COPRODX_HELP}, {RID_COPROD_FROMX, RID_COPROD_FROMX_HELP},
+ {RID_COPROD_TOX, RID_COPROD_TOX_HELP}, {RID_COPROD_FROMTOX, RID_COPROD_FROMTOX_HELP},
+ {nullptr, nullptr},
+ {RID_INTX, RID_INTX_HELP}, {RID_INT_FROMX, RID_INT_FROMX_HELP},
+ {RID_INT_TOX, RID_INT_TOX_HELP}, {RID_INT_FROMTOX, RID_INT_FROMTOX_HELP},
+ {nullptr, nullptr},
+ {RID_IINTX, RID_IINTX_HELP}, {RID_IINT_FROMX, RID_IINT_FROMX_HELP},
+ {RID_IINT_TOX, RID_IINT_TOX_HELP}, {RID_IINT_FROMTOX, RID_IINT_FROMTOX_HELP},
+ {nullptr, nullptr},
+ {RID_IIINTX, RID_IIINTX_HELP}, {RID_IIINT_FROMX, RID_IIINT_FROMX_HELP},
+ {RID_IIINT_TOX, RID_IIINT_TOX_HELP}, {RID_IIINT_FROMTOX, RID_IIINT_FROMTOX_HELP},
+ {nullptr, nullptr},
+ {RID_LINTX, RID_LINTX_HELP}, {RID_LINT_FROMX, RID_LINT_FROMX_HELP},
+ {RID_LINT_TOX, RID_LINT_TOX_HELP}, {RID_LINT_FROMTOX, RID_LINT_FROMTOX_HELP},
+ {nullptr, nullptr},
+ {RID_LLINTX, RID_LLINTX_HELP}, {RID_LLINT_FROMX, RID_LLINT_FROMX_HELP},
+ {RID_LLINT_TOX, RID_LLINT_TOX_HELP}, {RID_LLINT_FROMTOX, RID_LLINT_FROMTOX_HELP},
+ {nullptr, nullptr},
+ {RID_LLLINTX, RID_LLLINTX_HELP}, {RID_LLLINT_FROMX, RID_LLLINT_FROMX_HELP},
+ {RID_LLLINT_TOX, RID_LLLINT_TOX_HELP}, {RID_LLLINT_FROMTOX, RID_LLLINT_FROMTOX_HELP},
+};
+
+const SmElementDescr SmElementsControl::m_aAttributesList[] =
+{
+ {RID_ACUTEX, RID_ACUTEX_HELP}, {RID_GRAVEX, RID_GRAVEX_HELP}, {RID_BREVEX, RID_BREVEX_HELP},
+ {RID_CIRCLEX, RID_CIRCLEX_HELP}, {RID_DOTX, RID_DOTX_HELP}, {RID_DDOTX, RID_DDOTX_HELP},
+ {RID_DDDOTX, RID_DDDOTX_HELP}, {RID_BARX, RID_BARX_HELP}, {RID_VECX, RID_VECX_HELP},
+ {RID_HARPOONX, RID_HARPOONX_HELP},
+ {RID_TILDEX, RID_TILDEX_HELP}, {RID_HATX, RID_HATX_HELP}, {RID_CHECKX, RID_CHECKX_HELP},
+ {nullptr, nullptr},
+ {RID_WIDEVECX, RID_WIDEVECX_HELP}, {RID_WIDEHARPOONX, RID_WIDEHARPOONX_HELP},
+ {RID_WIDETILDEX, RID_WIDETILDEX_HELP}, {RID_WIDEHATX, RID_WIDEHATX_HELP},
+ {RID_OVERLINEX, RID_OVERLINEX_HELP}, {RID_UNDERLINEX, RID_UNDERLINEX_HELP}, {RID_OVERSTRIKEX, RID_OVERSTRIKEX_HELP},
+ {nullptr, nullptr},
+ {RID_PHANTOMX, RID_PHANTOMX_HELP}, {RID_BOLDX, RID_BOLDX_HELP}, {RID_ITALX, RID_ITALX_HELP},
+ {RID_SIZEXY, RID_SIZEXY_HELP}, {RID_FONTXY, RID_FONTXY_HELP},
+ {nullptr, nullptr},
+ {RID_COLORX_BLACK, RID_COLORX_BLACK_HELP}, {RID_COLORX_BLUE, RID_COLORX_BLUE_HELP},
+ {RID_COLORX_GREEN, RID_COLORX_GREEN_HELP}, {RID_COLORX_RED, RID_COLORX_RED_HELP},
+ {RID_COLORX_CYAN, RID_COLORX_CYAN_HELP}, {RID_COLORX_MAGENTA, RID_COLORX_MAGENTA_HELP},
+ {RID_COLORX_YELLOW, RID_COLORX_YELLOW_HELP}, {RID_COLORX_GRAY, RID_COLORX_GRAY_HELP},
+ {RID_COLORX_LIME, RID_COLORX_LIME_HELP}, {RID_COLORX_MAROON, RID_COLORX_MAROON_HELP},
+ {RID_COLORX_NAVY, RID_COLORX_NAVY_HELP}, {RID_COLORX_OLIVE, RID_COLORX_OLIVE_HELP},
+ {RID_COLORX_PURPLE, RID_COLORX_PURPLE_HELP}, {RID_COLORX_SILVER, RID_COLORX_SILVER_HELP},
+ {RID_COLORX_TEAL, RID_COLORX_TEAL_HELP},{RID_COLORX_RGB, RID_COLORX_RGB_HELP}
+};
+
+const SmElementDescr SmElementsControl::m_aBracketsList[] =
+{
+ {RID_LRGROUPX, RID_LRGROUPX_HELP},
+ {nullptr, nullptr},
+ {RID_LRPARENTX, RID_LRPARENTX_HELP}, {RID_LRBRACKETX, RID_LRBRACKETX_HELP},
+ {RID_LRDBRACKETX, RID_LRDBRACKETX_HELP}, {RID_LRBRACEX, RID_LRBRACEX_HELP},
+ {RID_LRANGLEX, RID_LRANGLEX_HELP}, {RID_LMRANGLEXY, RID_LMRANGLEXY_HELP},
+ {RID_LRCEILX, RID_LRCEILX_HELP}, {RID_LRFLOORX, RID_LRFLOORX_HELP},
+ {RID_LRLINEX, RID_LRLINEX_HELP}, {RID_LRDLINEX, RID_LRDLINEX_HELP},
+ {nullptr, nullptr},
+ {RID_SLRPARENTX, RID_SLRPARENTX_HELP}, {RID_SLRBRACKETX, RID_SLRBRACKETX_HELP},
+ {RID_SLRDBRACKETX, RID_SLRDBRACKETX_HELP}, {RID_SLRBRACEX, RID_SLRBRACEX_HELP},
+ {RID_SLRANGLEX, RID_SLRANGLEX_HELP}, {RID_SLMRANGLEXY, RID_SLMRANGLEXY_HELP},
+ {RID_SLRCEILX, RID_SLRCEILX_HELP}, {RID_SLRFLOORX, RID_SLRFLOORX_HELP},
+ {RID_SLRLINEX, RID_SLRLINEX_HELP}, {RID_SLRDLINEX, RID_SLRDLINEX_HELP},
+ {RID_XEVALUATEDATY, RID_XEVALUATEDATY_HELP},
+ {nullptr, nullptr},
+ {RID_XOVERBRACEY, RID_XOVERBRACEY_HELP}, {RID_XUNDERBRACEY, RID_XUNDERBRACEY_HELP},
+};
+
+const SmElementDescr SmElementsControl::m_aFormatsList[] =
+{
+ {RID_RSUPX, RID_RSUPX_HELP}, {RID_RSUBX, RID_RSUBX_HELP}, {RID_LSUPX, RID_LSUPX_HELP},
+ {RID_LSUBX, RID_LSUBX_HELP}, {RID_CSUPX, RID_CSUPX_HELP}, {RID_CSUBX, RID_CSUBX_HELP},
+ {nullptr, nullptr},
+ {RID_NEWLINE, RID_NEWLINE_HELP}, {RID_SBLANK, RID_SBLANK_HELP}, {RID_BLANK, RID_BLANK_HELP},
+ {RID_NOSPACE, RID_NOSPACE_HELP},
+ {RID_ALIGNLX, RID_ALIGNLX_HELP}, {RID_ALIGNCX, RID_ALIGNCX_HELP}, {RID_ALIGNRX, RID_ALIGNRX_HELP},
+ {nullptr, nullptr},
+ {RID_BINOMXY, RID_BINOMXY_HELP}, {RID_STACK, RID_STACK_HELP},
+ {RID_MATRIX, RID_MATRIX_HELP},
+};
+
+const SmElementDescr SmElementsControl::m_aOthersList[] =
+{
+ {RID_INFINITY, RID_INFINITY_HELP}, {RID_PARTIAL, RID_PARTIAL_HELP}, {RID_NABLA, RID_NABLA_HELP},
+ {RID_EXISTS, RID_EXISTS_HELP}, {RID_NOTEXISTS, RID_NOTEXISTS_HELP}, {RID_FORALL, RID_FORALL_HELP},
+ {RID_HBAR, RID_HBAR_HELP}, {RID_LAMBDABAR, RID_LAMBDABAR_HELP}, {RID_RE, RID_RE_HELP},
+ {RID_IM, RID_IM_HELP}, {RID_WP, RID_WP_HELP}, {RID_LAPLACE, RID_LAPLACE_HELP},
+ {nullptr, nullptr},
+ {RID_LEFTARROW, RID_LEFTARROW_HELP}, {RID_RIGHTARROW, RID_RIGHTARROW_HELP}, {RID_UPARROW, RID_UPARROW_HELP},
+ {RID_DOWNARROW, RID_DOWNARROW_HELP},
+ {nullptr, nullptr},
+ {RID_DOTSLOW, RID_DOTSLOW_HELP}, {RID_DOTSAXIS, RID_DOTSAXIS_HELP}, {RID_DOTSVERT, RID_DOTSVERT_HELP},
+ {RID_DOTSUP, RID_DOTSUP_HELP}, {RID_DOTSDOWN, RID_DOTSDOWN_HELP}
+};
+
+const SmElementDescr SmElementsControl::m_aExamplesList[] =
+{
+ {"C=%pi cdot d = 2 cdot %pi cdot r", RID_EXAMPLE_CIRCUMFERENCE_HELP},
+ {"E=mc^2", RID_EXAMPLE_MASS_ENERGY_EQUIV_HELP},
+ {"a^2 + b^2 = c^2", RID_EXAMPLE_PYTHAGOREAN_THEO_HELP},
+ {"f ( x ) = sum from { { i = 0 } } to { infinity } { {f^{(i)}(0)} over {i!} x^i}", RID_EXAMPLE_A_SIMPLE_SERIES_HELP},
+ {"f ( x ) = {1} over {%sigma sqrt{2%pi} }func e^-{{(x-%mu)^2} over {2%sigma^2}}", RID_EXAMPLE_GAUSS_DISTRIBUTION_HELP},
+};
+
+#define AS_PAIR(a) a, SAL_N_ELEMENTS(a)
+const std::tuple<const char*, const SmElementDescr*, size_t> SmElementsControl::m_aCategories[] =
+{
+ {RID_CATEGORY_UNARY_BINARY_OPERATORS, AS_PAIR(m_aUnaryBinaryOperatorsList)},
+ {RID_CATEGORY_RELATIONS, AS_PAIR(m_aRelationsList)},
+ {RID_CATEGORY_SET_OPERATIONS, AS_PAIR(m_aSetOperationsList)},
+ {RID_CATEGORY_FUNCTIONS, AS_PAIR(m_aFunctionsList)},
+ {RID_CATEGORY_OPERATORS, AS_PAIR(m_aOperatorsList)},
+ {RID_CATEGORY_ATTRIBUTES, AS_PAIR(m_aAttributesList)},
+ {RID_CATEGORY_BRACKETS, AS_PAIR(m_aBracketsList)},
+ {RID_CATEGORY_FORMATS, AS_PAIR(m_aFormatsList)},
+ {RID_CATEGORY_OTHERS, AS_PAIR(m_aOthersList)},
+ {RID_CATEGORY_EXAMPLES, AS_PAIR(m_aExamplesList)},
+};
+
+const size_t SmElementsControl::m_aCategoriesSize = SAL_N_ELEMENTS(m_aCategories);
+
+SmElementsControl::SmElementsControl(std::unique_ptr<weld::ScrolledWindow> xScrolledWindow)
+ : mpDocShell(new SmDocShell(SfxModelFlags::EMBEDDED_OBJECT))
+ , m_nCurrentElement(SAL_MAX_UINT16)
+ , m_nCurrentRolloverElement(SAL_MAX_UINT16)
+ , m_nCurrentOffset(1) // Default offset of 1 due to the ScrollBar child
+ , mbVerticalMode(true)
+ , mxScroll(std::move(xScrolledWindow))
+ , m_bFirstPaintAfterLayout(false)
+{
+ mxScroll->set_user_managed_scrolling();
+ mxScroll->connect_hadjustment_changed( LINK(this, SmElementsControl, ScrollHdl) );
+ mxScroll->connect_vadjustment_changed( LINK(this, SmElementsControl, ScrollHdl) );
+}
+
+SmElementsControl::~SmElementsControl()
+{
+ mpDocShell->DoClose();
+}
+
+void SmElementsControl::setVerticalMode(bool bVerticalMode)
+{
+ if (mbVerticalMode == bVerticalMode)
+ return;
+ mbVerticalMode = bVerticalMode;
+ // turn off scrollbars, LayoutOrPaintContents will enable whichever one
+ // might be needed
+ mxScroll->set_vpolicy(VclPolicyType::NEVER);
+ mxScroll->set_hpolicy(VclPolicyType::NEVER);
+ LayoutOrPaintContents(GetDrawingArea()->get_ref_device(), false);
+ Invalidate();
+}
+
+SmElement* SmElementsControl::current() const
+{
+ sal_uInt16 nCur = (m_nCurrentRolloverElement != SAL_MAX_UINT16)
+ ? m_nCurrentRolloverElement
+ : (HasFocus() ? m_nCurrentElement : SAL_MAX_UINT16);
+ return (nCur < maElementList.size()) ? maElementList[nCur].get() : nullptr;
+}
+
+void SmElementsControl::setCurrentElement(sal_uInt16 nPos)
+{
+ if (m_nCurrentElement == nPos)
+ return;
+ if (nPos != SAL_MAX_UINT16 && nPos >= maElementList.size())
+ return;
+ if (m_xAccessible.is() && m_nCurrentElement != SAL_MAX_UINT16)
+ m_xAccessible->ReleaseFocus(m_nCurrentElement);
+ m_nCurrentElement = nPos;
+ if (m_xAccessible.is() && m_nCurrentElement != SAL_MAX_UINT16)
+ m_xAccessible->AcquireFocus();
+}
+
+Color SmElementsControl::GetTextColor()
+{
+ const StyleSettings& rStyleSettings = Application::GetSettings().GetStyleSettings();
+ return rStyleSettings.GetFieldTextColor();
+}
+
+Color SmElementsControl::GetControlBackground()
+{
+ const StyleSettings& rStyleSettings = Application::GetSettings().GetStyleSettings();
+ return rStyleSettings.GetFieldColor();
+}
+
+/**
+ * !bDraw => layout only
+ *
+ * Layouting is always done without a scrollbar and will show or hide it.
+ * The first paint (m_bFirstPaintAfterLayout) therefore needs to update a
+ * visible scrollbar, because the layouting was wrong.
+ **/
+void SmElementsControl::LayoutOrPaintContents(vcl::RenderContext& rContext, bool bDraw)
+{
+ rContext.Push();
+
+ rContext.SetMapMode( MapMode(MapUnit::Map100thMM) );
+ rContext.SetDrawMode( DrawModeFlags::Default );
+ rContext.SetLayoutMode( ComplexTextLayoutFlags::Default );
+ rContext.SetDigitLanguage( LANGUAGE_ENGLISH );
+ if (bDraw)
+ {
+ rContext.SetBackground(GetControlBackground());
+ rContext.SetTextColor(GetTextColor());
+ rContext.Erase();
+ }
+
+ const sal_Int32 nControlHeight = GetOutputSizePixel().Height();
+ const sal_Int32 nControlWidth = GetOutputSizePixel().Width();
+
+ sal_Int32 boxX = maMaxElementDimensions.Width() + 10;
+ sal_Int32 boxY = maMaxElementDimensions.Height() + 10;
+
+ sal_Int32 x = mbVerticalMode ? -mxScroll->hadjustment_get_value() : 0;
+ sal_Int32 y = !mbVerticalMode ? -mxScroll->vadjustment_get_value() : 0;
+
+ sal_Int32 perLine = 0;
+
+ if (mbVerticalMode)
+ perLine = nControlHeight / boxY;
+ else
+ perLine = nControlWidth / boxX;
+ if (perLine <= 0)
+ perLine = 1;
+
+ if (mbVerticalMode)
+ boxY = nControlHeight / perLine;
+ else
+ boxX = nControlWidth / perLine;
+
+ const SmElement* pCurrentElement = current();
+ for (const std::unique_ptr<SmElement> & i : maElementList)
+ {
+ SmElement* element = i.get();
+ if (element->isSeparator())
+ {
+ if (mbVerticalMode)
+ {
+ x += boxX;
+ y = 0;
+
+ element->mBoxLocation = Point(x, y);
+ element->mBoxSize = Size(10, nControlHeight);
+
+ tools::Rectangle aSelectionRectangle(x + 5 - 1, y + 5,
+ x + 5 + 1, nControlHeight - 5);
+
+ if (bDraw)
+ rContext.DrawRect(rContext.PixelToLogic(aSelectionRectangle));
+ x += 10;
+ }
+ else
+ {
+ x = 0;
+ y += boxY;
+
+ element->mBoxLocation = Point(x, y);
+ element->mBoxSize = Size(nControlWidth, 10);
+
+ tools::Rectangle aSelectionRectangle(x + 5, y + 5 - 1,
+ nControlWidth - 5, y + 5 + 1);
+
+ if (bDraw)
+ rContext.DrawRect(rContext.PixelToLogic(aSelectionRectangle));
+ y += 10;
+ }
+ }
+ else
+ {
+ if (mbVerticalMode)
+ {
+ if (y + boxY > nControlHeight)
+ {
+ x += boxX;
+ y = 0;
+ }
+ }
+ else
+ {
+ if ( x + boxX > nControlWidth)
+ {
+ x = 0;
+ y += boxY;
+ }
+ }
+
+ element->mBoxLocation = Point(x,y);
+ element->mBoxSize = Size(boxX, boxY);
+
+ if (bDraw)
+ {
+ if (pCurrentElement == element)
+ {
+ rContext.Push(PushFlags::FILLCOLOR | PushFlags::LINECOLOR);
+ const StyleSettings& rStyleSettings = rContext.GetSettings().GetStyleSettings();
+ rContext.SetLineColor(rStyleSettings.GetHighlightColor());
+ rContext.SetFillColor(COL_TRANSPARENT);
+ rContext.DrawRect(rContext.PixelToLogic(tools::Rectangle(x + 1, y + 1, x + boxX - 1, y + boxY - 1)));
+ rContext.DrawRect(rContext.PixelToLogic(tools::Rectangle(x + 2, y + 2, x + boxX - 2, y + boxY - 2)));
+ rContext.Pop();
+ }
+
+ Size aSizePixel = rContext.LogicToPixel(Size(element->getNode()->GetWidth(),
+ element->getNode()->GetHeight()));
+ Point location(x + ((boxX - aSizePixel.Width()) / 2),
+ y + ((boxY - aSizePixel.Height()) / 2));
+ SmDrawingVisitor(rContext, rContext.PixelToLogic(location), element->getNode().get());
+ }
+
+ if (mbVerticalMode)
+ y += boxY;
+ else
+ x += boxX;
+ }
+ }
+
+ if (bDraw)
+ {
+ if (!m_bFirstPaintAfterLayout)
+ {
+ rContext.Pop();
+ return;
+ }
+ m_bFirstPaintAfterLayout = false;
+ }
+ else
+ m_bFirstPaintAfterLayout = true;
+
+ if (mbVerticalMode)
+ {
+ sal_Int32 nTotalControlWidth = x + boxX + mxScroll->hadjustment_get_value();
+ if (nTotalControlWidth > GetOutputSizePixel().Width())
+ {
+ mxScroll->hadjustment_set_upper(nTotalControlWidth);
+ mxScroll->hadjustment_set_page_size(nControlWidth);
+ mxScroll->hadjustment_set_page_increment(nControlWidth);
+ mxScroll->set_hpolicy(VclPolicyType::ALWAYS);
+ }
+ else
+ {
+ mxScroll->hadjustment_set_value(0);
+ mxScroll->set_hpolicy(VclPolicyType::NEVER);
+ }
+ }
+ else
+ {
+ sal_Int32 nTotalControlHeight = y + boxY + mxScroll->vadjustment_get_value();
+ if (nTotalControlHeight > GetOutputSizePixel().Height())
+ {
+ mxScroll->vadjustment_set_upper(nTotalControlHeight);
+ mxScroll->vadjustment_set_page_size(nControlHeight);
+ mxScroll->vadjustment_set_page_increment(nControlHeight);
+ mxScroll->set_vpolicy(VclPolicyType::ALWAYS);
+ }
+ else
+ {
+ mxScroll->vadjustment_set_value(0);
+ mxScroll->set_vpolicy(VclPolicyType::NEVER);
+ }
+ }
+ rContext.Pop();
+}
+
+void SmElementsControl::Resize()
+{
+ CustomWidgetController::Resize();
+ LayoutOrPaintContents(GetDrawingArea()->get_ref_device(), false);
+}
+
+void SmElementsControl::Paint(vcl::RenderContext& rRenderContext, const tools::Rectangle&)
+{
+ LayoutOrPaintContents(rRenderContext, true);
+}
+
+OUString SmElementsControl::RequestHelp(tools::Rectangle& rRect)
+{
+ if (!hasRollover())
+ return OUString();
+
+ const SmElement* pHelpElement = current();
+ if (!pHelpElement)
+ return OUString();
+
+ rRect = tools::Rectangle(pHelpElement->mBoxLocation, pHelpElement->mBoxSize);
+
+ // get text and display it
+ return pHelpElement->getHelpText();
+}
+
+bool SmElementsControl::MouseMove( const MouseEvent& rMouseEvent )
+{
+ if (rMouseEvent.IsLeaveWindow())
+ {
+ m_nCurrentRolloverElement = SAL_MAX_UINT16;
+ Invalidate();
+ return false;
+ }
+
+ if (tools::Rectangle(Point(0, 0), GetOutputSizePixel()).IsInside(rMouseEvent.GetPosPixel()))
+ {
+ const SmElement* pPrevElement = current();
+ if (pPrevElement)
+ {
+ const tools::Rectangle rect(pPrevElement->mBoxLocation, pPrevElement->mBoxSize);
+ if (rect.IsInside(rMouseEvent.GetPosPixel()))
+ return true;
+ }
+
+ const sal_uInt16 nElementCount = maElementList.size();
+ for (sal_uInt16 n = 0; n < nElementCount; n++)
+ {
+ const SmElement* element = maElementList[n].get();
+ if (pPrevElement == element)
+ continue;
+
+ const tools::Rectangle rect(element->mBoxLocation, element->mBoxSize);
+ if (rect.IsInside(rMouseEvent.GetPosPixel()))
+ {
+ m_nCurrentRolloverElement = n;
+ Invalidate();
+ return true;
+ }
+ }
+ if (pPrevElement && hasRollover())
+ Invalidate();
+ m_nCurrentRolloverElement = SAL_MAX_UINT16;
+ return true;
+ }
+
+ return false;
+}
+
+namespace {
+
+void collectUIInformation(const OUString& aID)
+{
+ EventDescription aDescription;
+ aDescription.aID = aID;
+ aDescription.aParent = "element_selector";
+ aDescription.aAction = "SELECT";
+ aDescription.aKeyWord = "ElementUIObject";
+ UITestLogger::getInstance().logEvent(aDescription);
+}
+
+}
+
+bool SmElementsControl::MouseButtonDown(const MouseEvent& rMouseEvent)
+{
+ GrabFocus();
+
+ if (rMouseEvent.IsLeft() && tools::Rectangle(Point(0, 0), GetOutputSizePixel()).IsInside(rMouseEvent.GetPosPixel()) && maSelectHdlLink.IsSet())
+ {
+ const SmElement* pPrevElement = hasRollover() ? current() : nullptr;
+ if (pPrevElement)
+ {
+ tools::Rectangle rect(pPrevElement->mBoxLocation, pPrevElement->mBoxSize);
+ if (rect.IsInside(rMouseEvent.GetPosPixel()))
+ {
+ setCurrentElement(m_nCurrentRolloverElement);
+ maSelectHdlLink.Call(*const_cast<SmElement*>(pPrevElement));
+ collectUIInformation(OUString::number(m_nCurrentRolloverElement));
+ return true;
+ }
+ }
+
+ const sal_uInt16 nElementCount = maElementList.size();
+ for (sal_uInt16 n = 0; n < nElementCount; n++)
+ {
+ SmElement* element = maElementList[n].get();
+ tools::Rectangle rect(element->mBoxLocation, element->mBoxSize);
+ if (rect.IsInside(rMouseEvent.GetPosPixel()))
+ {
+ setCurrentElement(n);
+ maSelectHdlLink.Call(*element);
+ collectUIInformation(OUString::number(n));
+ return true;
+ }
+ }
+
+ return true;
+ }
+ return false;
+}
+
+void SmElementsControl::GetFocus()
+{
+ CustomWidgetController::GetFocus();
+ Invalidate();
+}
+
+void SmElementsControl::LoseFocus()
+{
+ CustomWidgetController::LoseFocus();
+ Invalidate();
+}
+
+sal_uInt16 SmElementsControl::nextElement(const bool bBackward, const sal_uInt16 nStartPos, const sal_uInt16 nLastElement)
+{
+ sal_uInt16 nPos = nStartPos;
+
+ while (true)
+ {
+ if (bBackward)
+ {
+ if (nPos == 0)
+ break;
+ nPos--;
+ }
+ else
+ {
+ if (nPos == nLastElement)
+ break;
+ nPos++;
+ }
+
+ if (nStartPos == nPos)
+ break;
+ if (!maElementList[nPos]->isSeparator())
+ break;
+ }
+
+ return nPos;
+}
+
+void SmElementsControl::scrollToElement(const bool bBackward, const SmElement *pCur)
+{
+ if (mbVerticalMode)
+ {
+ auto nScrollPos = mxScroll->hadjustment_get_value();
+ nScrollPos += pCur->mBoxLocation.X();
+ if (!bBackward)
+ nScrollPos += pCur->mBoxSize.Width() - GetOutputSizePixel().Width();
+ mxScroll->hadjustment_set_value(nScrollPos);
+ }
+ else
+ {
+ auto nScrollPos = mxScroll->vadjustment_get_value();
+ nScrollPos += pCur->mBoxLocation.Y();
+ if (!bBackward)
+ nScrollPos += pCur->mBoxSize.Height() - GetOutputSizePixel().Height();
+ mxScroll->vadjustment_set_value(nScrollPos);
+ }
+}
+
+void SmElementsControl::stepFocus(const bool bBackward)
+{
+ const sal_uInt16 nStartPos = m_nCurrentElement;
+ const sal_uInt16 nLastElement = (maElementList.size() ? maElementList.size() - 1 : 0);
+ assert(nStartPos <= nLastElement);
+
+ sal_uInt16 nPos = nextElement(bBackward, nStartPos, nLastElement);
+ if (nStartPos != nPos)
+ {
+ m_nCurrentRolloverElement = SAL_MAX_UINT16;
+ setCurrentElement(nPos);
+
+ const tools::Rectangle outputRect(Point(0,0), GetOutputSizePixel());
+ const SmElement *pCur = maElementList[nPos].get();
+ tools::Rectangle elementRect(pCur->mBoxLocation, pCur->mBoxSize);
+ if (!outputRect.IsInside(elementRect))
+ scrollToElement(bBackward, pCur);
+ Invalidate();
+ }
+}
+
+void SmElementsControl::pageFocus(const bool bBackward)
+{
+ const sal_uInt16 nStartPos = m_nCurrentElement;
+ const sal_uInt16 nLastElement = (maElementList.size() ? maElementList.size() - 1 : 0);
+ assert(nStartPos <= nLastElement);
+ tools::Rectangle outputRect(Point(0,0), GetOutputSizePixel());
+ sal_uInt16 nPrevPos = nStartPos;
+ sal_uInt16 nPos = nPrevPos;
+
+ bool bMoved = false;
+ while (true)
+ {
+ nPrevPos = nPos;
+ nPos = nextElement(bBackward, nPrevPos, nLastElement);
+ if (nPrevPos == nPos)
+ break;
+
+ m_nCurrentRolloverElement = SAL_MAX_UINT16;
+
+ SmElement *pCur = maElementList[nPos].get();
+ tools::Rectangle elementRect(pCur->mBoxLocation, pCur->mBoxSize);
+ if (!outputRect.IsInside(elementRect))
+ {
+ if (nPrevPos != nStartPos)
+ {
+ nPos = nPrevPos;
+ break;
+ }
+ if (bMoved)
+ break;
+ pCur = maElementList[nPrevPos].get();
+
+ elementRect = tools::Rectangle(pCur->mBoxLocation, pCur->mBoxSize);
+ if (mbVerticalMode)
+ outputRect.Move(bBackward ? -outputRect.GetWidth() + elementRect.Right() : elementRect.Left(), 0);
+ else
+ outputRect.Move(0, bBackward ? -outputRect.GetHeight() + elementRect.Bottom() : elementRect.Top());
+ bMoved = true;
+ }
+ }
+
+ if (nStartPos != nPos)
+ {
+ setCurrentElement(nPos);
+ if (bMoved)
+ scrollToElement(bBackward, maElementList[nPos].get());
+ Invalidate();
+ }
+}
+
+bool SmElementsControl::KeyInput(const KeyEvent& rKEvt)
+{
+ vcl::KeyCode aKeyCode = rKEvt.GetKeyCode();
+
+ if (aKeyCode.GetModifier())
+ {
+ return false;
+ }
+
+ switch(aKeyCode.GetCode())
+ {
+ case KEY_RETURN:
+ [[fallthrough]];
+ case KEY_SPACE:
+ assert(m_nCurrentElement < maElementList.size());
+ assert(maSelectHdlLink.IsSet());
+ maSelectHdlLink.Call(*maElementList[m_nCurrentElement]);
+ collectUIInformation(OUString::number(m_nCurrentElement));
+ break;
+
+ case KEY_DOWN:
+ [[fallthrough]];
+ case KEY_RIGHT:
+ stepFocus(false);
+ break;
+
+ case KEY_LEFT:
+ [[fallthrough]];
+ case KEY_UP:
+ stepFocus(true);
+ break;
+
+ case KEY_HOME:
+ if (!maElementList.empty())
+ {
+ setCurrentElement(0);
+ mxScroll->vadjustment_set_value(0);
+ }
+ break;
+ case KEY_END:
+ if (!maElementList.empty())
+ {
+ setCurrentElement(maElementList.size() - 1);
+ mxScroll->vadjustment_set_value(mxScroll->vadjustment_get_upper());
+ }
+ break;
+
+ case KEY_PAGEUP:
+ pageFocus(true);
+ break;
+ case KEY_PAGEDOWN:
+ pageFocus(false);
+ break;
+
+ default:
+ return false;
+ break;
+ }
+ return true;
+}
+
+IMPL_LINK_NOARG( SmElementsControl, ScrollHdl, weld::ScrolledWindow&, void )
+{
+ Invalidate();
+}
+
+void SmElementsControl::addElement(SmParser &rParser, const OUString& aElementVisual, const OUString& aElementSource, const OUString& aHelpText)
+{
+ // SAL_MAX_UINT16 is invalid, zero is the scrollbar
+ assert(maElementList.size() < SAL_MAX_UINT16 - 2);
+ auto pNode = rParser.ParseExpression(aElementVisual);
+
+ OutputDevice& rDevice = GetDrawingArea()->get_ref_device();
+ rDevice.Push(PushFlags::MAPMODE);
+ rDevice.SetMapMode( MapMode(MapUnit::Map100thMM) );
+
+ pNode->Prepare(maFormat, *mpDocShell, 0);
+ pNode->SetSize(Fraction(10,8));
+ pNode->Arrange(rDevice, maFormat);
+
+ Size aSizePixel = rDevice.LogicToPixel(Size(pNode->GetWidth(), pNode->GetHeight()), MapMode(MapUnit::Map100thMM));
+ if (aSizePixel.Width() > maMaxElementDimensions.Width()) {
+ maMaxElementDimensions.setWidth( aSizePixel.Width() );
+ }
+
+ if (aSizePixel.Height() > maMaxElementDimensions.Height()) {
+ maMaxElementDimensions.setHeight( aSizePixel.Height() );
+ }
+
+ maElementList.push_back(std::make_unique<SmElement>(std::move(pNode), aElementSource, aHelpText));
+
+ rDevice.Pop();
+}
+
+void SmElementsControl::setElementSetId(const char* pSetId)
+{
+ if (msCurrentSetId == pSetId)
+ return;
+ msCurrentSetId = pSetId;
+ maMaxElementDimensions = Size();
+ build();
+}
+
+void SmElementsControl::addElements(const SmElementDescr aElementsArray[], sal_uInt16 aElementsArraySize)
+{
+ SmParser aParser;
+ aParser.SetImportSymbolNames(true);
+
+ for (sal_uInt16 i = 0; i < aElementsArraySize ; i++)
+ {
+ const char* pElement = aElementsArray[i].first;
+ const char* pElementHelp = aElementsArray[i].second;
+ if (!pElement) {
+ maElementList.push_back(std::make_unique<SmElementSeparator>());
+ } else {
+ OUString aElement(OUString::createFromAscii(pElement));
+ if (aElement == RID_NEWLINE)
+ addElement(aParser, OUString(u"\u21B5"), aElement, SmResId(pElementHelp));
+ else if (aElement == RID_SBLANK)
+ addElement(aParser, "\"`\"", aElement, SmResId(pElementHelp));
+ else if (aElement == RID_BLANK)
+ addElement(aParser, "\"~\"", aElement, SmResId(pElementHelp));
+ else if (aElement == RID_PHANTOMX)
+ addElement(aParser, "\"" + SmResId(STR_HIDE) +"\"", aElement, SmResId(pElementHelp));
+ else if (aElement == RID_BOLDX)
+ addElement(aParser, "bold B", aElement, SmResId(pElementHelp));
+ else if (aElement == RID_ITALX)
+ addElement(aParser, "ital I", aElement, SmResId(pElementHelp));
+ else if (aElement == RID_SIZEXY)
+ addElement(aParser, "\"" + SmResId(STR_SIZE) + "\"", aElement, SmResId(pElementHelp));
+ else if (aElement == RID_FONTXY)
+ addElement(aParser, "\"" + SmResId(STR_FONT) + "\"", aElement, SmResId(pElementHelp));
+ else if (aElement == RID_COLORX_BLACK)
+ addElement(aParser, "color black { \"" + SmResId(STR_BLACK) + "\" }", aElement, SmResId(pElementHelp));
+ else if (aElement == RID_COLORX_BLUE)
+ addElement(aParser, "color blue { \"" + SmResId(STR_BLUE) + "\" }", aElement, SmResId(pElementHelp));
+ else if (aElement == RID_COLORX_GREEN)
+ addElement(aParser, "color green { \"" + SmResId(STR_GREEN) + "\" }", aElement, SmResId(pElementHelp));
+ else if (aElement == RID_COLORX_RED)
+ addElement(aParser, "color red { \"" + SmResId(STR_RED) + "\" }", aElement, SmResId(pElementHelp));
+ else if (aElement == RID_COLORX_CYAN)
+ addElement(aParser, "color cyan { \"" + SmResId(STR_CYAN) + "\" }", aElement, SmResId(pElementHelp));
+ else if (aElement == RID_COLORX_MAGENTA)
+ addElement(aParser, "color magenta { \"" + SmResId(STR_MAGENTA) + "\" }", aElement, SmResId(pElementHelp));
+ else if (aElement == RID_COLORX_YELLOW)
+ addElement(aParser, "color yellow { \"" + SmResId(STR_YELLOW) + "\" }", aElement, SmResId(pElementHelp));
+ else if (aElement == RID_COLORX_GRAY)
+ addElement(aParser, "color gray { \"" + SmResId(STR_GRAY) + "\" }", aElement, SmResId(pElementHelp));
+ else if (aElement == RID_COLORX_LIME)
+ addElement(aParser, "color lime { \"" + SmResId(STR_LIME) + "\" }", aElement, SmResId(pElementHelp));
+ else if (aElement == RID_COLORX_MAROON)
+ addElement(aParser, "color maroon { \"" + SmResId(STR_MAROON) + "\" }", aElement, SmResId(pElementHelp));
+ else if (aElement == RID_COLORX_NAVY)
+ addElement(aParser, "color navy { \"" + SmResId(STR_NAVY) + "\" }", aElement, SmResId(pElementHelp));
+ else if (aElement == RID_COLORX_OLIVE)
+ addElement(aParser, "color olive { \"" + SmResId(STR_OLIVE) + "\" }", aElement, SmResId(pElementHelp));
+ else if (aElement == RID_COLORX_PURPLE)
+ addElement(aParser, "color purple { \"" + SmResId(STR_PURPLE) + "\" }", aElement, SmResId(pElementHelp));
+ else if (aElement == RID_COLORX_SILVER)
+ addElement(aParser, "color silver { \"" + SmResId(STR_SILVER) + "\" }", aElement, SmResId(pElementHelp));
+ else if (aElement == RID_COLORX_TEAL)
+ addElement(aParser, "color teal { \"" + SmResId(STR_TEAL) + "\" }", aElement, SmResId(pElementHelp));
+ else if (aElement == RID_COLORX_RGB)
+ addElement(aParser, "color rgb 0 0 0 { \"" + SmResId(STR_RGB) + "\" }", aElement, SmResId(pElementHelp));
+ else if (aElement == RID_ALIGNLX)
+ addElement(aParser, "\"" + SmResId(STR_ALIGN_LEFT) + "\"", aElement, SmResId(pElementHelp));
+ else if (aElement == RID_ALIGNCX)
+ addElement(aParser, "\"" + SmResId(STR_ALIGN_CENTER) + "\"", aElement, SmResId(pElementHelp));
+ else if (aElement == RID_ALIGNRX)
+ addElement(aParser, "\"" + SmResId(STR_ALIGN_RIGHT) + "\"", aElement, SmResId(pElementHelp));
+
+ else if (aElement == RID_SLRPARENTX)
+ addElement(aParser, "left ( binom{<?>}{<?>} right ) ", aElement, SmResId(pElementHelp));
+ else if (aElement == RID_SLRBRACKETX)
+ addElement(aParser, "left [ binom{<?>}{<?>} right ] ", aElement, SmResId(pElementHelp));
+ else if (aElement == RID_SLRDBRACKETX)
+ addElement(aParser, "left ldbracket binom{<?>}{<?>} right rdbracket ", aElement, SmResId(pElementHelp));
+ else if (aElement == RID_SLRBRACEX)
+ addElement(aParser, "left lbrace binom{<?>}{<?>} right rbrace ", aElement, SmResId(pElementHelp));
+ else if (aElement == RID_SLRANGLEX)
+ addElement(aParser, "left langle binom{<?>}{<?>} right rangle ", aElement, SmResId(pElementHelp));
+ else if (aElement == RID_SLRCEILX)
+ addElement(aParser, "left lceil binom{<?>}{<?>} right rceil ", aElement, SmResId(pElementHelp));
+ else if (aElement == RID_SLRFLOORX)
+ addElement(aParser, "left lfloor binom{<?>}{<?>} right rfloor ", aElement, SmResId(pElementHelp));
+
+ else if (aElement == RID_SLRLINEX)
+ addElement(aParser, "left lline binom{<?>}{<?>} right rline ", aElement, SmResId(pElementHelp));
+ else if (aElement == RID_SLRDLINEX)
+ addElement(aParser, "left ldline binom{<?>}{<?>} right rdline ", aElement, SmResId(pElementHelp));
+ else if (aElement == RID_SLMRANGLEXY)
+ addElement(aParser, "left langle binom{<?>}{<?>} mline binom{<?>}{<?>} right rangle ", aElement, SmResId(pElementHelp));
+
+ else if (aElement == RID_XOVERBRACEY)
+ addElement(aParser, "{<?><?><?>} overbrace {<?>} ", aElement, SmResId(pElementHelp));
+ else if (aElement == RID_XUNDERBRACEY)
+ addElement(aParser, "{<?><?><?>} underbrace {<?>} ", aElement, SmResId(pElementHelp));
+ else
+ addElement(aParser, aElement, aElement, pElementHelp ? SmResId(pElementHelp) : "");
+ }
+ }
+}
+
+void SmElementsControl::build()
+{
+ // The order is important!
+ // 1. Ensure there are no items left, including the default scrollbar!
+ // 2. Release all the current accessible items.
+ // This will check for new items after releasing them!
+ // 3. Set the cursor element
+ maElementList.clear();
+ mxScroll->hadjustment_set_value(0);
+ mxScroll->vadjustment_set_value(0);
+ mxScroll->set_hpolicy(VclPolicyType::NEVER);
+ mxScroll->set_vpolicy(VclPolicyType::NEVER);
+
+ if (m_xAccessible.is())
+ m_xAccessible->ReleaseAllItems();
+
+ setCurrentElement(SAL_MAX_UINT16);
+
+ // The first element is the scrollbar. We can't change its indexInParent
+ // value, as this is set by being a child of the SmElementsControl.
+ m_nCurrentOffset = 1;
+ for (sal_uInt16 n = 0; n < SAL_N_ELEMENTS(m_aCategories); ++n)
+ {
+ if (msCurrentSetId == std::get<0>(m_aCategories[n]))
+ {
+ addElements(std::get<1>(m_aCategories[n]), std::get<2>(m_aCategories[n]));
+ break;
+ }
+ else
+ m_nCurrentOffset += std::get<2>(m_aCategories[n]);
+ }
+
+ m_nCurrentRolloverElement = SAL_MAX_UINT16;
+ LayoutOrPaintContents(GetDrawingArea()->get_ref_device(), false);
+
+ if (m_xAccessible.is())
+ m_xAccessible->AddAllItems();
+
+ setCurrentElement(0);
+ Invalidate();
+}
+
+void SmElementsControl::SetDrawingArea(weld::DrawingArea* pDrawingArea)
+{
+ CustomWidgetController::SetDrawingArea(pDrawingArea);
+ OutputDevice& rDevice = pDrawingArea->get_ref_device();
+ maFormat.SetBaseSize(rDevice.PixelToLogic(Size(0, SmPtsTo100th_mm(12))));
+ Size aSize(rDevice.LogicToPixel(Size(10, 100), MapMode(MapUnit::MapAppFont)));
+ // give it an arbitrary small width request so it can shrink in the sidebar
+ pDrawingArea->set_size_request(42, aSize.Height());
+ SetOutputSizePixel(aSize);
+}
+
+FactoryFunction SmElementsControl::GetUITestFactory() const
+{
+ return ElementSelectorUIObject::create;
+}
+
+bool SmElementsControl::itemIsSeparator(sal_uInt16 nPos) const
+{
+ if (nPos < m_nCurrentOffset)
+ return true;
+ nPos -= m_nCurrentOffset;
+ if (nPos >= maElementList.size())
+ return true;
+ return maElementList[nPos]->isSeparator();
+}
+
+css::uno::Reference<css::accessibility::XAccessible> SmElementsControl::CreateAccessible()
+{
+ if (!m_xAccessible.is())
+ {
+ m_xAccessible = new AccessibleSmElementsControl(*this);
+ m_xAccessible->AddAllItems();
+ }
+ return m_xAccessible.get();
+}
+
+bool SmElementsControl::itemTrigger(sal_uInt16 nPos)
+{
+ if (nPos < m_nCurrentOffset)
+ return false;
+ nPos -= m_nCurrentOffset;
+ if (nPos >= maElementList.size())
+ return false;
+
+ maSelectHdlLink.Call(*maElementList[nPos]);
+ collectUIInformation(OUString::number(nPos));
+ return true;
+}
+
+tools::Rectangle SmElementsControl::itemPosRect(sal_uInt16 nPos) const
+{
+ if (nPos < m_nCurrentOffset)
+ return tools::Rectangle();
+ nPos -= m_nCurrentOffset;
+ if (nPos >= maElementList.size())
+ return tools::Rectangle();
+
+ SmElement* pItem = maElementList[nPos].get();
+ return tools::Rectangle(pItem->mBoxLocation, pItem->mBoxSize);
+}
+
+bool SmElementsControl::itemIsVisible(sal_uInt16 nPos) const
+{
+ tools::Rectangle elementRect = itemPosRect(nPos);
+ if (elementRect.IsEmpty())
+ return false;
+
+ tools::Rectangle outputRect(Point(0, 0), GetOutputSizePixel());
+ return outputRect.IsInside(elementRect);
+}
+
+sal_uInt16 SmElementsControl::itemCount() const { return maElementList.size(); }
+
+sal_uInt16 SmElementsControl::itemHighlighted() const { return m_nCurrentElement; }
+
+void SmElementsControl::setItemHighlighted(sal_uInt16 nPos)
+{
+ if (m_nCurrentRolloverElement == nPos)
+ return;
+ if (nPos != SAL_MAX_UINT16 && nPos >= maElementList.size())
+ return;
+
+ if (maElementList[nPos]->isSeparator())
+ m_nCurrentRolloverElement = SAL_MAX_UINT16;
+ else
+ m_nCurrentRolloverElement = nPos;
+ Invalidate();
+}
+
+OUString SmElementsControl::itemName(sal_uInt16 nPos) const
+{
+ if (nPos < m_nCurrentOffset)
+ return OUString();
+ nPos -= m_nCurrentOffset;
+ if (nPos >= maElementList.size())
+ return OUString();
+
+ return maElementList[nPos]->getHelpText();
+}
+
+sal_uInt16 SmElementsControl::itemAtPos(const Point& rPoint) const
+{
+ sal_uInt16 nElementCount = maElementList.size();
+ for (sal_uInt16 n = 0; n < nElementCount; n++)
+ {
+ const SmElement* pItem = maElementList[n].get();
+ tools::Rectangle elementRect(pItem->mBoxLocation, pItem->mBoxSize);
+ if (elementRect.IsInside(rPoint))
+ return n;
+ }
+ return SAL_MAX_UINT16;
+}
+
+SmElementsDockingWindow::SmElementsDockingWindow(SfxBindings* pInputBindings, SfxChildWindow* pChildWindow, vcl::Window* pParent)
+ : SfxDockingWindow(pInputBindings, pChildWindow, pParent, "DockingElements",
+ "modules/smath/ui/dockingelements.ui")
+ , mxElementsControl(new SmElementsControl(m_xBuilder->weld_scrolled_window("scrolledwindow")))
+ , mxElementsControlWin(new weld::CustomWeld(*m_xBuilder, "element_selector", *mxElementsControl))
+ , mxElementListBox(m_xBuilder->weld_combo_box("listbox"))
+{
+ // give it an arbitrary small width request so it can shrink in the sidebar
+ mxElementListBox->set_size_request(42, -1);
+
+ for (size_t i = 0; i < SmElementsControl::categoriesSize(); ++i)
+ mxElementListBox->append_text(SmResId(std::get<0>(SmElementsControl::categories()[i])));
+
+ mxElementListBox->connect_changed(LINK(this, SmElementsDockingWindow, ElementSelectedHandle));
+ mxElementListBox->set_active_text(SmResId(RID_CATEGORY_UNARY_BINARY_OPERATORS));
+
+ mxElementsControl->setElementSetId(RID_CATEGORY_UNARY_BINARY_OPERATORS);
+ mxElementsControl->SetSelectHdl(LINK(this, SmElementsDockingWindow, SelectClickHandler));
+}
+
+SmElementsDockingWindow::~SmElementsDockingWindow ()
+{
+ disposeOnce();
+}
+
+void SmElementsDockingWindow::dispose()
+{
+ mxElementsControlWin.reset();
+ mxElementsControl.reset();
+ mxElementListBox.reset();
+ SfxDockingWindow::dispose();
+}
+
+void SmElementsDockingWindow::ToggleFloatingMode()
+{
+ SfxDockingWindow::ToggleFloatingMode();
+
+ if (GetFloatingWindow())
+ GetFloatingWindow()->SetMinOutputSizePixel( Size(100, 100) );
+
+ Invalidate();
+}
+
+void SmElementsDockingWindow::EndDocking( const tools::Rectangle& rReactangle, bool bFloatMode)
+{
+ SfxDockingWindow::EndDocking(rReactangle, bFloatMode);
+ bool bVertical = ( GetAlignment() == SfxChildAlignment::TOP || GetAlignment() == SfxChildAlignment::BOTTOM );
+ mxElementsControl->setVerticalMode(bVertical);
+}
+
+IMPL_LINK(SmElementsDockingWindow, SelectClickHandler, SmElement&, rElement, void)
+{
+ SmViewShell* pViewSh = GetView();
+
+ if (pViewSh)
+ {
+ std::unique_ptr<SfxStringItem> pInsertCommand = std::make_unique<SfxStringItem>(SID_INSERTCOMMANDTEXT, rElement.getText());
+ pViewSh->GetViewFrame()->GetDispatcher()->ExecuteList(
+ SID_INSERTCOMMANDTEXT, SfxCallMode::RECORD,
+ { pInsertCommand.get() });
+ }
+}
+
+IMPL_LINK( SmElementsDockingWindow, ElementSelectedHandle, weld::ComboBox&, rList, void)
+{
+ for (size_t i = 0; i < SmElementsControl::categoriesSize(); ++i)
+ {
+ const char *pCurrentCategory = std::get<0>(SmElementsControl::categories()[i]);
+ OUString aCurrentCategoryString = SmResId(pCurrentCategory);
+ if (aCurrentCategoryString == rList.get_active_text())
+ {
+ mxElementsControl->setElementSetId(pCurrentCategory);
+ return;
+ }
+ }
+}
+
+SmViewShell* SmElementsDockingWindow::GetView()
+{
+ SfxViewShell* pView = GetBindings().GetDispatcher()->GetFrame()->GetViewShell();
+ return dynamic_cast<SmViewShell*>( pView);
+}
+
+void SmElementsDockingWindow::Resize()
+{
+ bool bVertical = ( GetAlignment() == SfxChildAlignment::TOP || GetAlignment() == SfxChildAlignment::BOTTOM );
+ mxElementsControl->setVerticalMode(bVertical);
+
+ SfxDockingWindow::Resize();
+ Invalidate();
+}
+
+SFX_IMPL_DOCKINGWINDOW_WITHID(SmElementsDockingWindowWrapper, SID_ELEMENTSDOCKINGWINDOW);
+
+SmElementsDockingWindowWrapper::SmElementsDockingWindowWrapper(
+ vcl::Window *pParentWindow, sal_uInt16 nId,
+ SfxBindings *pBindings, SfxChildWinInfo *pInfo) :
+ SfxChildWindow(pParentWindow, nId)
+{
+ VclPtrInstance<SmElementsDockingWindow> pDialog(pBindings, this, pParentWindow);
+ SetWindow(pDialog);
+ pDialog->setDeferredProperties();
+ pDialog->SetPosSizePixel(Point(0, 0), Size(300, 0));
+ pDialog->Show();
+
+ SetAlignment(SfxChildAlignment::LEFT);
+
+ pDialog->Initialize( pInfo );
+}
+
+SmElementsDockingWindowWrapper::~SmElementsDockingWindowWrapper()
+{
+}
+
+/* vim:set shiftwidth=4 softtabstop=4 expandtab: */