/**************************************************************************** ** Copyright (c) 2013-2014 Debao Zhang ** All right reserved. ** ** Permission is hereby granted, free of charge, to any person obtaining ** a copy of this software and associated documentation files (the ** "Software"), to deal in the Software without restriction, including ** without limitation the rights to use, copy, modify, merge, publish, ** distribute, sublicense, and/or sell copies of the Software, and to ** permit persons to whom the Software is furnished to do so, subject to ** the following conditions: ** ** The above copyright notice and this permission notice shall be ** included in all copies or substantial portions of the Software. ** ** THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, ** EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF ** MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND ** NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE ** LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION ** OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION ** WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. ** ****************************************************************************/ #include "xlsxcellreference.h" #include #include #include QT_BEGIN_NAMESPACE_XLSX namespace { int intPow(int x, int p) { if (p == 0) return 1; if (p == 1) return x; int tmp = intPow(x, p / 2); if (p % 2 == 0) return tmp * tmp; else return x * tmp * tmp; } QString col_to_name(int col_num) { static QMap col_cache; if (!col_cache.contains(col_num)) { QString col_str; int remainder; while (col_num) { remainder = col_num % 26; if (remainder == 0) remainder = 26; col_str.prepend(QChar('A' + remainder - 1)); col_num = (col_num - 1) / 26; } col_cache.insert(col_num, col_str); } return col_cache[col_num]; } int col_from_name(const QString &col_str) { int col = 0; int expn = 0; for (int i = col_str.size() - 1; i > -1; --i) { col += (col_str[i].unicode() - 'A' + 1) * intPow(26, expn); expn++; } return col; } } // namespace /*! \class CellReference \brief For one single cell such as "A1" \inmodule QtXlsx The CellReference class stores the cell location in a worksheet. */ /*! Constructs an invalid Cell Reference */ CellReference::CellReference() : _row(-1) , _column(-1) { } /*! Constructs the Reference from the given \a row, and \a column. */ CellReference::CellReference(int row, int column) : _row(row) , _column(column) { } /*! \overload Constructs the Reference form the given \a cell string. */ CellReference::CellReference(const QString &cell) { init(cell); } /*! \overload Constructs the Reference form the given \a cell string. */ CellReference::CellReference(const char *cell) { init(QString::fromLatin1(cell)); } void CellReference::init(const QString &cell_str) { static QRegularExpression re(QStringLiteral("^\\$?([A-Z]{1,3})\\$?(\\d+)$")); QRegularExpressionMatch match = re.match(cell_str); if (match.hasMatch()) { const QString col_str = match.captured(1); const QString row_str = match.captured(2); _row = row_str.toInt(); _column = col_from_name(col_str); } } /*! Constructs a Reference by copying the given \a other Reference. */ CellReference::CellReference(const CellReference &other) : _row(other._row) , _column(other._column) { } /*! Destroys the Reference. */ CellReference::~CellReference() { } /*! Convert the Reference to string notation, such as "A1" or "$A$1". If current object is invalid, an empty string will be returned. */ QString CellReference::toString(bool row_abs, bool col_abs) const { if (!isValid()) return QString(); QString cell_str; if (col_abs) cell_str.append(QLatin1Char('$')); cell_str.append(col_to_name(_column)); if (row_abs) cell_str.append(QLatin1Char('$')); cell_str.append(QString::number(_row)); return cell_str; } /*! * Returns true if the Reference is valid. */ bool CellReference::isValid() const { return _row > 0 && _column > 0; } QT_END_NAMESPACE_XLSX