/* -*- 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 . */ #undef SC_DLLIMPLEMENTATION #include #include #include #include #include #include #include #include #include "calcoptionsdlg.hxx" #include ScTpFormulaOptions::ScTpFormulaOptions(weld::Container* pPage, weld::DialogController* pController, const SfxItemSet& rCoreAttrs) : SfxTabPage(pPage, pController, "modules/scalc/ui/optformula.ui", "OptFormula", &rCoreAttrs) , mnDecSep(0) , mxLbFormulaSyntax(m_xBuilder->weld_combo_box("formulasyntax")) , mxCbEnglishFuncName(m_xBuilder->weld_check_button("englishfuncname")) , mxBtnCustomCalcDefault(m_xBuilder->weld_radio_button("calcdefault")) , mxBtnCustomCalcCustom(m_xBuilder->weld_radio_button("calccustom")) , mxBtnCustomCalcDetails(m_xBuilder->weld_button("details")) , mxEdSepFuncArg(m_xBuilder->weld_entry("function")) , mxEdSepArrayCol(m_xBuilder->weld_entry("arraycolumn")) , mxEdSepArrayRow(m_xBuilder->weld_entry("arrayrow")) , mxBtnSepReset(m_xBuilder->weld_button("reset")) , mxLbOOXMLRecalcOptions(m_xBuilder->weld_combo_box("ooxmlrecalc")) , mxLbODFRecalcOptions(m_xBuilder->weld_combo_box("odfrecalc")) { mxLbFormulaSyntax->append_text(ScResId(SCSTR_FORMULA_SYNTAX_CALC_A1)); mxLbFormulaSyntax->append_text(ScResId(SCSTR_FORMULA_SYNTAX_XL_A1)); mxLbFormulaSyntax->append_text(ScResId(SCSTR_FORMULA_SYNTAX_XL_R1C1)); Link aLink2 = LINK( this, ScTpFormulaOptions, ButtonHdl ); mxBtnSepReset->connect_clicked(aLink2); mxBtnCustomCalcDefault->connect_clicked(aLink2); mxBtnCustomCalcCustom->connect_clicked(aLink2); mxBtnCustomCalcDetails->connect_clicked(aLink2); mxEdSepFuncArg->connect_insert_text(LINK( this, ScTpFormulaOptions, SepInsertTextHdl )); mxEdSepArrayCol->connect_insert_text(LINK( this, ScTpFormulaOptions, ColSepInsertTextHdl )); mxEdSepArrayRow->connect_insert_text(LINK( this, ScTpFormulaOptions, RowSepInsertTextHdl )); Link aLink = LINK( this, ScTpFormulaOptions, SepModifyHdl ); mxEdSepFuncArg->connect_changed(aLink); mxEdSepArrayCol->connect_changed(aLink); mxEdSepArrayRow->connect_changed(aLink); Link aLink3 = LINK( this, ScTpFormulaOptions, SepEditOnFocusHdl ); mxEdSepFuncArg->connect_focus_in(aLink3); mxEdSepArrayCol->connect_focus_in(aLink3); mxEdSepArrayRow->connect_focus_in(aLink3); // Get the decimal separator for current locale. OUString aSep = ScGlobal::getLocaleDataPtr()->getNumDecimalSep(); mnDecSep = aSep.isEmpty() ? u'.' : aSep[0]; maSavedDocOptions = static_cast(rCoreAttrs.Get( GetWhich(SID_SCDOCOPTIONS))).GetDocOptions(); } ScTpFormulaOptions::~ScTpFormulaOptions() { } void ScTpFormulaOptions::ResetSeparators() { OUString aFuncArg, aArrayCol, aArrayRow; ScFormulaOptions::GetDefaultFormulaSeparators(aFuncArg, aArrayCol, aArrayRow); mxEdSepFuncArg->set_text(aFuncArg); mxEdSepArrayCol->set_text(aArrayCol); mxEdSepArrayRow->set_text(aArrayRow); } void ScTpFormulaOptions::OnFocusSeparatorInput(weld::Entry* pEdit) { if (!pEdit) return; // Make sure the entire text is selected. pEdit->select_region(0, -1); OUString sSepValue = pEdit->get_text(); if (!sSepValue.isEmpty()) maOldSepValue = sSepValue; } void ScTpFormulaOptions::UpdateCustomCalcRadioButtons(bool bDefault) { if (bDefault) { mxBtnCustomCalcDefault->set_active(true); mxBtnCustomCalcCustom->set_active(false); mxBtnCustomCalcDetails->set_sensitive(false); } else { mxBtnCustomCalcDefault->set_active(false); mxBtnCustomCalcCustom->set_active(true); mxBtnCustomCalcDetails->set_sensitive(true); } } void ScTpFormulaOptions::LaunchCustomCalcSettings() { ScCalcOptionsDialog aDlg(GetFrameWeld(), maCurrentConfig, maCurrentDocOptions.IsWriteCalcConfig()); if (aDlg.run() == RET_OK) { maCurrentConfig = aDlg.GetConfig(); maCurrentDocOptions.SetWriteCalcConfig(aDlg.GetWriteCalcConfig()); } } bool ScTpFormulaOptions::IsValidSeparator(const OUString& rSep) const { if (rSep.getLength() != 1) // Must be one-character long. return false; if (rSep.compareToAscii("a") >= 0 && rSep.compareToAscii("z") <= 0) return false; if (rSep.compareToAscii("A") >= 0 && rSep.compareToAscii("Z") <= 0) return false; sal_Unicode c = rSep[0]; switch (c) { case '+': case '-': case '/': case '*': case '<': case '>': case '[': case ']': case '(': case ')': case '"': case '\'': // Disallowed characters. Anything else we want to disallow ? return false; } if (c == mnDecSep) // decimal separator is not allowed. return false; return true; } IMPL_LINK( ScTpFormulaOptions, ButtonHdl, weld::Button&, rBtn, void ) { if (&rBtn == mxBtnSepReset.get()) ResetSeparators(); else if (&rBtn == mxBtnCustomCalcDefault.get()) UpdateCustomCalcRadioButtons(true); else if (&rBtn == mxBtnCustomCalcCustom.get()) UpdateCustomCalcRadioButtons(false); else if (&rBtn == mxBtnCustomCalcDetails.get()) LaunchCustomCalcSettings(); } IMPL_LINK(ScTpFormulaOptions, SepInsertTextHdl, OUString&, rTest, bool) { if (!IsValidSeparator(rTest) && !maOldSepValue.isEmpty()) // Invalid separator. Restore the old value. rTest = maOldSepValue; return true; } IMPL_LINK(ScTpFormulaOptions, RowSepInsertTextHdl, OUString&, rTest, bool) { // Invalid separator or same as ColStr - Restore the old value. if ((!IsValidSeparator(rTest) || rTest == mxEdSepArrayCol->get_text()) && !maOldSepValue.isEmpty()) rTest = maOldSepValue; return true; } IMPL_LINK(ScTpFormulaOptions, ColSepInsertTextHdl, OUString&, rTest, bool) { // Invalid separator or same as RowStr - Restore the old value. if ((!IsValidSeparator(rTest) || rTest == mxEdSepArrayRow->get_text()) && !maOldSepValue.isEmpty()) rTest = maOldSepValue; return true; } IMPL_LINK( ScTpFormulaOptions, SepModifyHdl, weld::Entry&, rEdit, void ) { OnFocusSeparatorInput(&rEdit); } IMPL_LINK( ScTpFormulaOptions, SepEditOnFocusHdl, weld::Widget&, rControl, void ) { OnFocusSeparatorInput(dynamic_cast(&rControl)); } std::unique_ptr ScTpFormulaOptions::Create(weld::Container* pPage, weld::DialogController* pController, const SfxItemSet* rCoreSet) { return std::make_unique(pPage, pController, *rCoreSet); } bool ScTpFormulaOptions::FillItemSet(SfxItemSet* rCoreSet) { bool bRet = false; ScFormulaOptions aOpt; bool bEnglishFuncName = mxCbEnglishFuncName->get_active(); sal_Int16 aSyntaxPos = mxLbFormulaSyntax->get_active(); OUString aSep = mxEdSepFuncArg->get_text(); OUString aSepArrayCol = mxEdSepArrayCol->get_text(); OUString aSepArrayRow = mxEdSepArrayRow->get_text(); sal_Int16 nOOXMLRecalcMode = mxLbOOXMLRecalcOptions->get_active(); sal_Int16 nODFRecalcMode = mxLbODFRecalcOptions->get_active(); if (mxBtnCustomCalcDefault->get_active()) { // When Default is selected, reset all the calc config settings to default. maCurrentConfig.reset(); } if ( mxLbFormulaSyntax->get_saved_value() != mxLbFormulaSyntax->get_text(aSyntaxPos) || mxCbEnglishFuncName->get_saved_state() != (bEnglishFuncName ? 1 : 0) || mxEdSepFuncArg->get_saved_value() != aSep || mxEdSepArrayCol->get_saved_value() != aSepArrayCol || mxEdSepArrayRow->get_saved_value() != aSepArrayRow || mxLbOOXMLRecalcOptions->get_saved_value() != mxLbOOXMLRecalcOptions->get_text(nOOXMLRecalcMode) || mxLbODFRecalcOptions->get_saved_value() != mxLbODFRecalcOptions->get_text(nODFRecalcMode) || maSavedConfig != maCurrentConfig || maSavedDocOptions != maCurrentDocOptions ) { ::formula::FormulaGrammar::Grammar eGram = ::formula::FormulaGrammar::GRAM_DEFAULT; switch (aSyntaxPos) { case 0: eGram = ::formula::FormulaGrammar::GRAM_NATIVE; break; case 1: eGram = ::formula::FormulaGrammar::GRAM_NATIVE_XL_A1; break; case 2: eGram = ::formula::FormulaGrammar::GRAM_NATIVE_XL_R1C1; break; } ScRecalcOptions eOOXMLRecalc = static_cast(nOOXMLRecalcMode); ScRecalcOptions eODFRecalc = static_cast(nODFRecalcMode); aOpt.SetFormulaSyntax(eGram); aOpt.SetUseEnglishFuncName(bEnglishFuncName); aOpt.SetFormulaSepArg(aSep); aOpt.SetFormulaSepArrayCol(aSepArrayCol); aOpt.SetFormulaSepArrayRow(aSepArrayRow); aOpt.SetCalcConfig(maCurrentConfig); aOpt.SetOOXMLRecalcOptions(eOOXMLRecalc); aOpt.SetODFRecalcOptions(eODFRecalc); aOpt.SetWriteCalcConfig( maCurrentDocOptions.IsWriteCalcConfig()); rCoreSet->Put( ScTpFormulaItem( aOpt ) ); rCoreSet->Put( ScTpCalcItem( SID_SCDOCOPTIONS, maCurrentDocOptions ) ); bRet = true; } return bRet; } void ScTpFormulaOptions::Reset(const SfxItemSet* rCoreSet) { ScFormulaOptions aOpt; const SfxPoolItem* pItem = nullptr; if(SfxItemState::SET == rCoreSet->GetItemState(SID_SCFORMULAOPTIONS, false , &pItem)) aOpt = static_cast(pItem)->GetFormulaOptions(); // formula grammar. ::formula::FormulaGrammar::Grammar eGram = aOpt.GetFormulaSyntax(); switch (eGram) { case ::formula::FormulaGrammar::GRAM_NATIVE: mxLbFormulaSyntax->set_active(0); break; case ::formula::FormulaGrammar::GRAM_NATIVE_XL_A1: mxLbFormulaSyntax->set_active(1); break; case ::formula::FormulaGrammar::GRAM_NATIVE_XL_R1C1: mxLbFormulaSyntax->set_active(2); break; default: mxLbFormulaSyntax->set_active(0); } mxLbFormulaSyntax->save_value(); mxLbFormulaSyntax->set_sensitive( !officecfg::Office::Calc::Formula::Syntax::Grammar::isReadOnly() ); ScRecalcOptions eOOXMLRecalc = aOpt.GetOOXMLRecalcOptions(); mxLbOOXMLRecalcOptions->set_active(static_cast(eOOXMLRecalc)); mxLbOOXMLRecalcOptions->save_value(); mxLbOOXMLRecalcOptions->set_sensitive( !officecfg::Office::Calc::Formula::Load::OOXMLRecalcMode::isReadOnly() ); ScRecalcOptions eODFRecalc = aOpt.GetODFRecalcOptions(); mxLbODFRecalcOptions->set_active(static_cast(eODFRecalc)); mxLbODFRecalcOptions->save_value(); mxLbODFRecalcOptions->set_sensitive( !officecfg::Office::Calc::Formula::Load::ODFRecalcMode::isReadOnly() ); // english function name. mxCbEnglishFuncName->set_active( aOpt.GetUseEnglishFuncName() ); mxCbEnglishFuncName->save_state(); mxCbEnglishFuncName->set_sensitive( !officecfg::Office::Calc::Formula::Syntax::EnglishFunctionName::isReadOnly() ); // Separators OUString aSep = aOpt.GetFormulaSepArg(); OUString aSepArrayRow = aOpt.GetFormulaSepArrayRow(); OUString aSepArrayCol = aOpt.GetFormulaSepArrayCol(); if (aSep.getLength() == 1 && aSepArrayRow.getLength() == 1 && aSepArrayCol.getLength() == 1) { // Each separator must be one character long. mxEdSepFuncArg->set_text(aSep); mxEdSepArrayCol->set_text(aSepArrayCol); mxEdSepArrayRow->set_text(aSepArrayRow); mxEdSepFuncArg->save_value(); mxEdSepArrayCol->save_value(); mxEdSepArrayRow->save_value(); } else ResetSeparators(); mxEdSepFuncArg->set_sensitive( !officecfg::Office::Calc::Formula::Syntax::SeparatorArg::isReadOnly() ); mxEdSepArrayCol->set_sensitive( !officecfg::Office::Calc::Formula::Syntax::SeparatorArrayCol::isReadOnly() ); mxEdSepArrayRow->set_sensitive( !officecfg::Office::Calc::Formula::Syntax::SeparatorArrayRow::isReadOnly() ); mxBtnSepReset->set_sensitive ( !officecfg::Office::Calc::Formula::Syntax::SeparatorArg::isReadOnly() && !officecfg::Office::Calc::Formula::Syntax::SeparatorArrayCol::isReadOnly() && !officecfg::Office::Calc::Formula::Syntax::SeparatorArrayRow::isReadOnly() ); // detailed calc settings. ScFormulaOptions aDefaults; maSavedConfig = aOpt.GetCalcConfig(); bool bDefault = aDefaults.GetCalcConfig() == maSavedConfig; UpdateCustomCalcRadioButtons(bDefault); maCurrentConfig = maSavedConfig; maCurrentDocOptions = maSavedDocOptions; } DeactivateRC ScTpFormulaOptions::DeactivatePage(SfxItemSet* /*pSet*/) { // What's this method for ? return DeactivateRC::KeepPage; } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */