Files
create/src/Base/Tools.h

339 lines
8.1 KiB
C++

/***************************************************************************
* Copyright (c) 2009 Werner Mayer <wmayer[at]users.sourceforge.net> *
* *
* This file is part of the FreeCAD CAx development system. *
* *
* This library is free software; you can redistribute it and/or *
* modify it under the terms of the GNU Library General Public *
* License as published by the Free Software Foundation; either *
* version 2 of the License, or (at your option) any later version. *
* *
* This library is distributed in the hope that it will be useful, *
* but WITHOUT ANY WARRANTY; without even the implied warranty of *
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
* GNU Library General Public License for more details. *
* *
* You should have received a copy of the GNU Library General Public *
* License along with this library; see the file COPYING.LIB. If not, *
* write to the Free Software Foundation, Inc., 59 Temple Place, *
* Suite 330, Boston, MA 02111-1307, USA *
* *
***************************************************************************/
#ifndef SRC_BASE_TOOLS_H_
#define SRC_BASE_TOOLS_H_
#ifndef FC_GLOBAL_H
#include <FCGlobal.h>
#endif
#include <cmath>
#include <numbers>
#include <ostream>
#include <string>
#include <vector>
#include <boost/signals2/shared_connection_block.hpp>
namespace boost
{
namespace signals2
{
class connection;
}
} // namespace boost
class QString;
namespace Base
{
template<class T>
struct iotaGen
{
public:
T operator()()
{
return n++;
}
explicit iotaGen(T v)
: n(v)
{}
private:
T n;
};
// ----------------------------------------------------------------------------
template<class T>
class manipulator
{
T i_;
std::ostream& (*f_)(std::ostream&, T);
public:
manipulator(std::ostream& (*f)(std::ostream&, T), T i)
: i_(i)
, f_(f)
{}
friend std::ostream& operator<<(std::ostream& os, manipulator m)
{
return m.f_(os, m.i_);
}
};
inline std::ostream& tabsN(std::ostream& os, int n)
{
for (int i = 0; i < n; i++) {
os << "\t";
}
return os;
}
inline std::ostream& blanksN(std::ostream& os, int n)
{
for (int i = 0; i < n; i++) {
os << " ";
}
return os;
}
inline manipulator<int> tabs(int n)
{
return {&tabsN, n};
}
inline manipulator<int> blanks(int n)
{
return {&blanksN, n};
}
// ----------------------------------------------------------------------------
template<class T>
inline T clamp(T num, T lower, T upper)
{
return std::max<T>(std::min<T>(upper, num), lower);
}
template<class T>
inline T sgn(T t)
{
if (t == 0) {
return T(0);
}
return (t > 0) ? T(1) : T(-1);
}
template<class T>
inline T toRadians(T d)
{
return static_cast<T>((d * std::numbers::pi) / 180.0);
}
template<class T>
inline T toDegrees(T r)
{
return static_cast<T>((r / std::numbers::pi) * 180.0);
}
inline float fromPercent(const long value)
{
return std::roundf(static_cast<float>(value)) / 100.0F;
}
inline long toPercent(float value)
{
return std::lround(100.0 * value);
}
template<class T>
inline T fmod(T numerator, T denominator)
{
T modulo = std::fmod(numerator, denominator);
return (modulo >= T(0)) ? modulo : modulo + denominator;
}
// ----------------------------------------------------------------------------
// NOLINTBEGIN
template<typename Flag = bool>
struct FlagToggler
{
Flag& flag;
bool toggled;
FlagToggler(Flag& _flag)
: flag(_flag)
, toggled(true)
{
flag = !flag;
}
FlagToggler(Flag& _flag, Flag check)
: flag(_flag)
, toggled(check == _flag)
{
if (toggled) {
flag = !flag;
}
}
~FlagToggler()
{
if (toggled) {
flag = !flag;
}
}
};
// ----------------------------------------------------------------------------
template<typename Status, class Object>
class ObjectStatusLocker
{
public:
ObjectStatusLocker(Status s, Object* o, bool value = true)
: status(s)
, obj(o)
{
old_value = obj->testStatus(status);
obj->setStatus(status, value);
}
~ObjectStatusLocker()
{
obj->setStatus(status, old_value);
}
private:
Status status;
Object* obj;
bool old_value;
};
// ----------------------------------------------------------------------------
class StateLocker
{
public:
StateLocker(bool& flag, bool value = true)
: lock(flag)
{
old_value = lock;
lock = value;
} // NOLINT
~StateLocker()
{
lock = old_value;
}
private:
bool& lock;
bool old_value;
};
// ----------------------------------------------------------------------------
template<typename T>
class BitsetLocker
{
public:
BitsetLocker(T& flags, std::size_t flag, bool value = true)
: flags(flags)
, flag(flag)
{
oldValue = flags.test(flag);
flags.set(flag, value);
}
~BitsetLocker()
{
flags.set(flag, oldValue);
}
private:
T& flags;
std::size_t flag;
bool oldValue;
};
// ----------------------------------------------------------------------------
class ConnectionBlocker
{
using Connection = boost::signals2::connection;
using ConnectionBlock = boost::signals2::shared_connection_block;
ConnectionBlock blocker;
public:
ConnectionBlocker(Connection& c)
: blocker(c)
{}
~ConnectionBlocker() = default;
};
// NOLINTEND
// ----------------------------------------------------------------------------
struct BaseExport Tools
{
static std::string getIdentifier(const std::string&);
static std::wstring widen(const std::string& str);
static std::string narrow(const std::wstring& str);
static std::string escapedUnicodeFromUtf8(const char* s);
static std::string escapedUnicodeToUtf8(const std::string& s);
static std::string escapeQuotesFromString(const std::string& s);
static QString escapeEncodeString(const QString& s);
static std::string escapeEncodeString(const std::string& s);
static QString escapeEncodeFilename(const QString& s);
static std::string escapeEncodeFilename(const std::string& s);
/**
* @brief quoted Creates a quoted string.
* @param String to be quoted.
* @return A quoted std::string.
*/
static std::string quoted(const char*);
/**
* @brief quoted Creates a quoted string.
* @param String to be quoted.
* @return A quoted std::string.
*/
static std::string quoted(const std::string&);
static constexpr bool isNullOrEmpty(const char* str)
{
return !str || str[0] == '\0';
}
/**
* @brief joinList
* Join the vector of strings \a vec using the separator \a sep
* @param vec
* @param sep
* @return
*/
static std::string joinList(const std::vector<std::string>& vec, const std::string& sep = ", ");
static std::string currentDateTimeString();
static std::vector<std::string> splitSubName(const std::string& subname);
};
struct BaseExport ZipTools
{
/**
* @brief rewrite Rewrite a zip file under a new name.
*/
static void rewrite(const std::string& source, const std::string& target);
};
} // namespace Base
#endif // SRC_BASE_TOOLS_H_