1
0
mirror of https://github.com/QtExcel/QXlsx.git synced 2025-01-16 04:42:53 +08:00
QXlsx/Copycat/MainWindow.cpp

357 lines
8.5 KiB
C++
Raw Normal View History

// MainWindow.cpp
2018-09-27 17:41:21 +09:00
#include <QtGlobal>
#include <QString>
#include <QFileDialog>
#include <QMessageBox>
2018-09-28 18:08:40 +09:00
#include <QVBoxLayout>
2018-10-04 13:24:28 +09:00
#include <QPrinter>
#include <QPrintPreviewDialog>
#include <QPainter>
#include <QDebug>
#include <QVector>
#include <QList>
#include <QSharedPointer>
2018-10-05 13:03:21 +09:00
#include <QInputDialog>
#include <QStringList>
#include <QVarLengthArray>
2018-10-04 13:24:28 +09:00
#include "xlsxcelllocation.h"
#include "xlsxcell.h"
2018-10-05 13:03:21 +09:00
#include "XlsxTableModel2.h"
2018-10-04 13:24:28 +09:00
#include "tableprinter.h"
2018-09-27 17:41:21 +09:00
#include "MainWindow.h"
#include "ui_MainWindow.h"
MainWindow::MainWindow(QWidget *parent) :
QMainWindow(parent),
ui(new Ui::MainWindow)
{
xlsxDoc = NULL;
tabWidget = NULL;
ui->setupUi(this);
tabWidget = new QTabWidget(this);
2018-09-28 18:08:40 +09:00
setCentralWidget(tabWidget);
2018-09-27 17:41:21 +09:00
2018-09-28 21:20:29 +09:00
this->setWindowTitle(QString("Copycat"));
2018-09-27 17:41:21 +09:00
}
MainWindow::~MainWindow()
{
delete ui;
if ( NULL != xlsxDoc )
{
delete xlsxDoc;
}
}
void MainWindow::on_action_Quit_triggered()
{
2018-09-28 21:20:29 +09:00
// quit
2018-09-27 17:41:21 +09:00
this->close();
}
void MainWindow::on_action_Open_triggered()
{
// open file dialog
2018-09-27 17:41:21 +09:00
QString fileName = QFileDialog::getOpenFileName(this,
tr("Open Excel"), ".", tr("Excel Files (*.xlsx)"));
2018-09-28 18:08:40 +09:00
if ( ! loadXlsx(fileName) ) // load xlsx
2018-09-27 17:41:21 +09:00
{
QMessageBox msgBox;
QString alertMsg = QString("Failed to load file.\n %1").arg(fileName);
2018-10-05 13:03:21 +09:00
msgBox.setIcon( QMessageBox::Critical );
msgBox.setText( alertMsg );
2018-09-27 17:41:21 +09:00
msgBox.exec();
return;
}
2018-09-28 18:08:40 +09:00
this->setWindowTitle(fileName);
}
2018-09-27 17:41:21 +09:00
bool MainWindow::loadXlsx(QString fileName)
{
2018-09-28 18:08:40 +09:00
// tried to load xlsx using temporary document
2018-10-05 13:03:21 +09:00
QXlsx::Document xlsxTmp( fileName );
if ( !xlsxTmp.isLoadPackage() )
2018-09-28 18:08:40 +09:00
{
2018-09-28 21:20:29 +09:00
return false; // failed to load
2018-09-28 18:08:40 +09:00
}
2018-09-27 17:41:21 +09:00
// clear xlsxDoc
if ( NULL != xlsxDoc )
{
delete xlsxDoc;
xlsxDoc = NULL;
}
2018-09-28 18:08:40 +09:00
// load new xlsx using new document
2018-10-05 13:03:21 +09:00
xlsxDoc = new QXlsx::Document( fileName );
2018-09-27 17:41:21 +09:00
xlsxDoc->isLoadPackage();
// clear tab widget
tabWidget->clear();
// Removes all the pages, but does not delete them.
2018-09-28 18:08:40 +09:00
// Calling this function is equivalent to calling removeTab()
// until the tab widget is empty.
2018-09-27 17:41:21 +09:00
2018-09-28 18:08:40 +09:00
// clear sub-items of every tab
2018-09-28 21:20:29 +09:00
foreach ( XlsxTab* ptrTab, xlsxTabList )
{
2018-09-28 18:08:40 +09:00
if ( NULL == ptrTab )
continue;
delete ptrTab;
}
xlsxTabList.clear();
2018-09-28 18:08:40 +09:00
int sheetIndexNumber = 0;
2018-10-04 17:35:49 +09:00
int activeSheetNumber = -1;
AbstractSheet* activeSheet = xlsxDoc->workbook()->activeSheet();
// NOTICE: active sheet is lastest sheet. It's not focused sheet.
2018-09-28 18:08:40 +09:00
foreach( QString curretnSheetName, xlsxDoc->sheetNames() )
{
QXlsx::AbstractSheet* currentSheet = xlsxDoc->sheet( curretnSheetName );
if ( NULL == currentSheet )
continue;
2018-09-27 17:41:21 +09:00
2018-10-04 17:35:49 +09:00
if ( activeSheet == currentSheet )
{
// current sheet is active sheet.
activeSheetNumber = sheetIndexNumber;
}
2018-12-18 18:20:27 +09:00
XlsxTab* newSheet = new XlsxTab( this, xlsxDoc, currentSheet, sheetIndexNumber ); // create new tab
2018-09-28 18:08:40 +09:00
xlsxTabList.push_back( newSheet ); // append to xlsx pointer list
2018-10-03 20:04:33 +09:00
tabWidget->addTab( newSheet, curretnSheetName ); // add tab widget
sheetIndexNumber++; // increase sheet index number
2018-09-27 17:41:21 +09:00
}
2018-10-04 17:35:49 +09:00
if ( (-1) != activeSheetNumber )
tabWidget->setCurrentIndex(activeSheetNumber);
2018-09-28 18:08:40 +09:00
return true;
}
void MainWindow::on_action_About_triggered()
{
2018-09-30 22:08:38 +09:00
QString text = "QXlsx<br />"
"<a href=\"https://github.com/j2doll/QXlsx\">https://github.com/j2doll/QXlsx</a><br />"
"MIT License<br />" ;
2018-10-04 13:24:28 +09:00
QMessageBox::about(this, "QXlsx", text);
2018-09-28 21:20:29 +09:00
}
void MainWindow::on_action_New_triggered()
{
2018-10-04 13:24:28 +09:00
// TODO: new document
2018-09-28 21:20:29 +09:00
QMessageBox msgBox;
msgBox.setText( "New" );
2018-09-28 18:08:40 +09:00
msgBox.exec();
2018-09-27 17:41:21 +09:00
}
2018-09-28 21:20:29 +09:00
void MainWindow::on_action_Save_triggered()
{
2018-10-04 13:24:28 +09:00
// TODO: save document
2018-09-28 21:20:29 +09:00
QMessageBox msgBox;
msgBox.setText( "Save" );
msgBox.exec();
}
2018-10-04 13:24:28 +09:00
void MainWindow::on_action_Print_triggered()
{
if ( NULL == xlsxDoc )
return;
QPrintPreviewDialog dialog;
connect(&dialog, SIGNAL(paintRequested(QPrinter*)), this, SLOT(print(QPrinter*)));
dialog.exec();
}
void MainWindow::print(QPrinter *printer)
{
if ( NULL == xlsxDoc )
2018-10-05 13:03:21 +09:00
{
QMessageBox msgBox;
msgBox.setText( "xlsx document is empty." );
msgBox.setIcon( QMessageBox::Warning );
msgBox.exec();
2018-10-04 13:24:28 +09:00
return;
2018-10-05 13:03:21 +09:00
}
2018-10-04 13:24:28 +09:00
2018-10-04 17:35:49 +09:00
Worksheet* wsheet = (Worksheet*) xlsxDoc->workbook()->activeSheet();
if ( NULL == wsheet )
2018-10-05 13:03:21 +09:00
{
QMessageBox msgBox;
msgBox.setText( "worksheet is set." );
msgBox.setIcon( QMessageBox::Warning );
msgBox.exec();
return;
}
QStringList items;
foreach( QString curretnSheetName, xlsxDoc->sheetNames() )
{
items << curretnSheetName;
}
bool ok = false;
QString selectedItem
= QInputDialog::getItem( this, tr("QXlsx"), tr("Select sheet:"), items, 0, false, &ok );
if (!ok)
{
QMessageBox msgBox;
msgBox.setText( "Please select printable sheet" );
msgBox.setIcon( QMessageBox::Warning );
msgBox.exec();
2018-10-04 17:35:49 +09:00
return;
2018-10-05 13:03:21 +09:00
}
int idx = 0;
foreach (const QString& varList, items)
{
if ( varList == selectedItem )
{
break;
}
idx++;
}
xlsxDoc->workbook()->setActiveSheet( idx );
wsheet = (Worksheet*) xlsxDoc->workbook()->activeSheet(); // set active sheet
2018-10-04 13:24:28 +09:00
2018-10-04 17:35:49 +09:00
QList<QString> colTitle;
2018-10-04 13:24:28 +09:00
QList<VLIST> xlsxData;
QVector<int> printColumnStretch;
int sheetIndexNumber = 0;
2018-10-04 17:35:49 +09:00
int maxRow = -1;
int maxCol = -1;
QVector<CellLocation> clList = wsheet->getFullCells( &maxRow, &maxCol );
// Fixed for Visual C++, cause VC++ does not support C99.
// https://docs.microsoft.com/en-us/cpp/c-language/ansi-conformance?view=vs-2017
// QString arr[maxRow][maxCol];
QVector <QVector <QString> > arr;
for (int i = 0; i < maxRow; i++)
{
QVector<QString> tempVector;
for (int j = 0; j < maxCol; j++)
{
tempVector.push_back(QString(""));
}
arr.push_back(tempVector);
}
2018-10-04 13:24:28 +09:00
2018-10-04 17:35:49 +09:00
for ( int ic = 0; ic < clList.size(); ++ic )
{
CellLocation cl = clList.at(ic);
2018-10-04 13:24:28 +09:00
2018-10-05 13:03:21 +09:00
int row = cl.row - 1;
int col = cl.col - 1;
2018-10-04 13:24:28 +09:00
2018-10-04 17:35:49 +09:00
QSharedPointer<Cell> ptrCell = cl.cell; // cell pointer
2018-10-04 13:24:28 +09:00
2018-10-04 17:35:49 +09:00
QString strValue = ptrCell->value().toString();
2018-10-04 13:24:28 +09:00
2018-10-05 13:03:21 +09:00
arr[row][col] = strValue;
2018-10-04 17:35:49 +09:00
}
for (int ir = 0 ; ir < maxRow; ir++)
{
VLIST vl;
2018-10-04 13:24:28 +09:00
2018-10-04 17:35:49 +09:00
for (int ic = 0 ; ic < maxCol; ic++)
{
QString strValue = arr[ir][ic];
if ( strValue.isNull() )
strValue = QString("");
2018-10-04 13:24:28 +09:00
2018-10-04 17:35:49 +09:00
vl.append( strValue );
2018-10-04 13:24:28 +09:00
}
2018-10-04 17:35:49 +09:00
xlsxData.append( vl );
}
2018-10-04 13:24:28 +09:00
2018-10-04 17:35:49 +09:00
QVector<QString> printHeaders;
2018-10-04 13:24:28 +09:00
2018-10-05 13:03:21 +09:00
for ( int ic = 0 ; ic < maxCol; ic++ )
2018-10-04 17:35:49 +09:00
{
std::string colString = convertFromNumberToExcelColumn( ( ic + 1 ) );
QString strCol = QString::fromStdString( colString );
2018-10-04 13:24:28 +09:00
colTitle.append( strCol );
2018-10-04 17:35:49 +09:00
printHeaders.append( strCol );
2018-10-04 13:24:28 +09:00
2018-10-04 17:35:49 +09:00
printColumnStretch.append( wsheet->columnWidth( (ic + 1) ) ); // TODO: check this code
2018-10-04 13:24:28 +09:00
}
2018-10-05 13:03:21 +09:00
XlsxTableModel2 xlsxTableModel2(colTitle, xlsxData); // model
2018-10-04 17:35:49 +09:00
2018-10-04 13:24:28 +09:00
QPainter painter;
if ( !painter.begin(printer) )
{
QMessageBox msgBox;
msgBox.setText( "Can't start printer" );
2018-10-04 17:35:49 +09:00
msgBox.setIcon( QMessageBox::Critical );
2018-10-04 13:24:28 +09:00
msgBox.exec();
return;
}
// print table
TablePrinter tablePrinter(&painter, printer);
2018-10-05 13:03:21 +09:00
if(!tablePrinter.printTable( &xlsxTableModel2, printColumnStretch, printHeaders ))
2018-10-04 13:24:28 +09:00
{
QMessageBox msgBox;
msgBox.setText( tablePrinter.lastError() );
2018-10-04 17:35:49 +09:00
msgBox.setIcon( QMessageBox::Warning );
2018-10-04 13:24:28 +09:00
msgBox.exec();
}
painter.end();
}
std::string MainWindow::convertFromNumberToExcelColumn(int n)
{
// main code from https://www.geeksforgeeks.org/find-excel-column-name-given-number/
// Function to print Excel column name for a given column number
std::string stdString;
char str[1000]; // To store result (Excel column name)
int i = 0; // To store current index in str which is result
while ( n > 0 )
{
// Find remainder
int rem = n % 26;
// If remainder is 0, then a 'Z' must be there in output
if ( rem == 0 )
{
str[i++] = 'Z';
n = (n/26) - 1;
}
else // If remainder is non-zero
{
str[i++] = (rem-1) + 'A';
n = n / 26;
}
}
str[i] = '\0';
// Reverse the string and print result
std::reverse( str, str + strlen(str) );
stdString = str;
return stdString;
}