mirror of
https://bitbucket.org/mfeemster/fractorium.git
synced 2025-01-21 21:20:07 -05:00
8086cfa731
--User changes -Allow users to set the Exp value when using the Exp temporal filter type. -Set the default temporal filter type to be Box, which does not alter the palette values at all during animation. This is done to avoid confusion when using Gaussian or Exp which can produce darkened images. --Bug fixes -Sending a sequence to the final render dialog when the keyframes had non zero rotate and center Y values would produce off center animations when rendered. -Temporal filters were being unnecessarily recreated many times when rendering or generating sequences. -Exp filter was always treated like a Box filter. --Code changes -Add a new member function SaveCurrentAsXml(QString filename = "") to the controllers which is only used for testing. -Modernize some C++ code.
399 lines
14 KiB
C++
399 lines
14 KiB
C++
#include "FractoriumPch.h"
|
|
#include "Fractorium.h"
|
|
|
|
#define XAOS_PREC 6
|
|
|
|
/// <summary>
|
|
/// Initialize the xforms xaos UI.
|
|
/// </summary>
|
|
void Fractorium::InitXaosUI()
|
|
{
|
|
int spinHeight = 20;
|
|
ui.XaosTableView->verticalHeader()->setSectionsClickable(true);
|
|
ui.XaosTableView->horizontalHeader()->setSectionsClickable(true);
|
|
m_XaosSpinBox = new DoubleSpinBox(nullptr, spinHeight, 0.1, false);
|
|
m_XaosSpinBox->DoubleClick(true);
|
|
m_XaosSpinBox->DoubleClickZero(1);
|
|
m_XaosSpinBox->DoubleClickNonZero(0);
|
|
m_XaosSpinBox->setDecimals(XAOS_PREC);
|
|
m_XaosSpinBox->setObjectName("XaosSpinBox");
|
|
m_XaosTableModel = nullptr;
|
|
m_AppliedXaosTableModel = nullptr;
|
|
m_XaosTableItemDelegate = new DoubleSpinBoxTableItemDelegate(m_XaosSpinBox, this);
|
|
connect(m_XaosSpinBox, SIGNAL(valueChanged(double)), this, SLOT(OnXaosChanged(double)), Qt::QueuedConnection);
|
|
connect(ui.ClearXaosButton, SIGNAL(clicked(bool)), this, SLOT(OnClearXaosButtonClicked(bool)), Qt::QueuedConnection);
|
|
connect(ui.RandomXaosButton, SIGNAL(clicked(bool)), this, SLOT(OnRandomXaosButtonClicked(bool)), Qt::QueuedConnection);
|
|
connect(ui.TransposeXaosButton, SIGNAL(clicked(bool)), this, SLOT(OnTransposeXaosButtonClicked(bool)), Qt::QueuedConnection);
|
|
connect(ui.AddLayerButton, SIGNAL(clicked(bool)), this, SLOT(OnAddLayerButtonClicked(bool)), Qt::QueuedConnection);
|
|
connect(ui.XaosTableView->verticalHeader(), SIGNAL(sectionDoubleClicked(int)), this, SLOT(OnXaosRowDoubleClicked(int)), Qt::QueuedConnection);
|
|
connect(ui.XaosTableView->horizontalHeader(), SIGNAL(sectionDoubleClicked(int)), this, SLOT(OnXaosColDoubleClicked(int)), Qt::QueuedConnection);
|
|
connect(ui.XaosTableView->horizontalScrollBar(), SIGNAL(valueChanged(int)), this, SLOT(OnXaosHScrollValueChanged(int)), Qt::QueuedConnection);
|
|
connect(ui.XaosTableView->verticalScrollBar(), SIGNAL(valueChanged(int)), this, SLOT(OnXaosVScrollValueChanged(int)), Qt::QueuedConnection);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Fill the xaos table with the values from the ember.
|
|
/// </summary>
|
|
template <typename T>
|
|
void FractoriumEmberController<T>::FillXaos()
|
|
{
|
|
for (int i = 0, count = static_cast<int>(XformCount()); i < count; i++)//Column.
|
|
{
|
|
if (const auto xform = m_Ember.GetXform(i))
|
|
{
|
|
for (int j = 0; j < count; j++)//Row.
|
|
{
|
|
QModelIndex index = m_Fractorium->m_XaosTableModel->index(j, i, QModelIndex());//j and i are intentionally swapped here.
|
|
m_Fractorium->m_XaosTableModel->setData(index, xform->Xaos(j));
|
|
}
|
|
}
|
|
}
|
|
|
|
m_Fractorium->ui.XaosTableView->resizeRowsToContents();
|
|
m_Fractorium->ui.XaosTableView->resizeColumnsToContents();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Fill the xaos table with the xaos values applied to the xform weights from the ember.
|
|
/// </summary>
|
|
template <typename T>
|
|
void FractoriumEmberController<T>::FillAppliedXaos()
|
|
{
|
|
m_Ember.CalcNormalizedWeights(m_NormalizedWeights);
|
|
|
|
for (int i = 0, count = int(XformCount()); i < count; i++)//Column.
|
|
{
|
|
if (const auto xform = m_Ember.GetXform(i))
|
|
{
|
|
T norm = 0;
|
|
double start = 0, offset = 0;
|
|
auto tempweights = m_NormalizedWeights;
|
|
|
|
for (int j = 0; j < count; j++)//Row.
|
|
{
|
|
tempweights[j] *= xform->Xaos(j);
|
|
QModelIndex index = m_Fractorium->m_AppliedXaosTableModel->index(j, i, QModelIndex());//j and i are intentionally swapped here.
|
|
m_Fractorium->m_AppliedXaosTableModel->setData(index, TruncPrecision(xform->Xaos(j) * xform->m_Weight, 4));//Applied xaos is just a read only table for display purposes.
|
|
}
|
|
|
|
QPixmap pixmap(m_Fractorium->ui.XaosAppliedTableView->columnWidth(i) - 8, m_Fractorium->ui.XaosTableView->rowHeight(0) * count);
|
|
QPainter painter(&pixmap);
|
|
auto twi = std::make_unique<QTableWidgetItem>();
|
|
|
|
for (auto& w : tempweights) norm += w;
|
|
|
|
for (auto& w : tempweights) w = norm == static_cast<T>(0) ? static_cast<T>(0) : w / norm;
|
|
|
|
if (norm)
|
|
{
|
|
for (size_t i = 0; i < tempweights.size() && offset <= pixmap.height(); i++)
|
|
{
|
|
offset = std::min<T>(offset + tempweights[i] * pixmap.height(), pixmap.height());
|
|
painter.fillRect(0, start, pixmap.width(), offset, m_Fractorium->m_XformComboColors[i % XFORM_COLOR_COUNT]);
|
|
start = offset;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
painter.fillRect(0, 0, pixmap.width(), pixmap.height(), m_Fractorium->m_XformComboColors[0]);
|
|
}
|
|
|
|
twi->setData(Qt::DecorationRole, pixmap);
|
|
m_Fractorium->ui.XaosDistVizTableWidget->setItem(0, i, twi.release());
|
|
}
|
|
}
|
|
|
|
m_Fractorium->ui.XaosDistVizTableWidget->resizeRowsToContents();
|
|
m_Fractorium->ui.XaosDistVizTableWidget->resizeColumnsToContents();
|
|
m_Fractorium->ui.XaosAppliedTableView->resizeRowsToContents();
|
|
m_Fractorium->ui.XaosAppliedTableView->resizeColumnsToContents();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Set the xaos value.
|
|
/// Called when any xaos spinner is changed.
|
|
/// It actually gets called multiple times as the user clicks around the
|
|
/// xaos table due to how QTableView passes events to and from its model.
|
|
/// To filter out spurrious events, the value is checked against the existing
|
|
/// xaos value.
|
|
/// Resets the rendering process.
|
|
/// </summary>
|
|
/// <param name="x">The index of the xform whose xaos value was changed (column)</param>
|
|
/// <param name="y">The index of the to value that was changed for the xform (row)</param>
|
|
/// <param name="val">The changed value of the xaos element</param>
|
|
template <typename T>
|
|
void FractoriumEmberController<T>::XaosChanged(int x, int y, double val)
|
|
{
|
|
auto newVal = TruncPrecision(val, XAOS_PREC);//Sometimes 0 comes in as a very small number, so round.
|
|
|
|
if (auto xform = m_Ember.GetXform(x))
|
|
if (!IsClose<T>(newVal, xform->Xaos(y), T(1e-7)))
|
|
{
|
|
Update([&] { xform->SetXaos(y, newVal); });
|
|
FillAppliedXaos();
|
|
}
|
|
}
|
|
|
|
void Fractorium::OnXaosChanged(double d)
|
|
{
|
|
if (auto senderSpinBox = qobject_cast<DoubleSpinBox*>(sender()))
|
|
{
|
|
auto p = senderSpinBox->property("tableindex").toPoint();
|
|
m_Controller->XaosChanged(p.y(), p.x(), d);//Intentionally switched, column is the from xform, row is the to xform.
|
|
}
|
|
}
|
|
|
|
void Fractorium::OnXaosTableModelDataChanged(const QModelIndex& indexA, const QModelIndex& indexB)
|
|
{
|
|
m_Controller->XaosChanged(indexA.column(), indexA.row(), indexA.data().toDouble());//Intentionally switched, column is the from xform, row is the to xform.
|
|
}
|
|
|
|
/// <summary>
|
|
/// Clear xaos table, recreate all spinners based on the xaos used in the current ember.
|
|
/// </summary>
|
|
void Fractorium::FillXaosTable()
|
|
{
|
|
int count = static_cast<int>(m_Controller->XformCount());
|
|
QStringList hl, vl, blanks;
|
|
auto oldModel = std::make_unique<QStandardItemModel>(m_XaosTableModel);
|
|
hl.reserve(count);
|
|
vl.reserve(count);
|
|
blanks.push_back("");
|
|
m_XaosTableModel = new QStandardItemModel(count, count, this);
|
|
m_AppliedXaosTableModel = new QStandardItemModel(count, count, this);
|
|
connect(m_XaosTableModel, SIGNAL(dataChanged(QModelIndex, QModelIndex)), SLOT(OnXaosTableModelDataChanged(QModelIndex, QModelIndex)));
|
|
|
|
for (int i = 0; i < count; i++)
|
|
{
|
|
auto s = m_Controller->MakeXformCaption(i);
|
|
hl.push_back("F" + s);
|
|
vl.push_back("T" + s);
|
|
}
|
|
|
|
m_XaosTableModel->setHorizontalHeaderLabels(hl);
|
|
m_XaosTableModel->setVerticalHeaderLabels(vl);
|
|
m_AppliedXaosTableModel->setHorizontalHeaderLabels(hl);
|
|
m_AppliedXaosTableModel->setVerticalHeaderLabels(vl);
|
|
ui.XaosDistVizTableWidget->setRowCount(1);
|
|
ui.XaosDistVizTableWidget->setColumnCount(count);
|
|
ui.XaosDistVizTableWidget->setHorizontalHeaderLabels(hl);
|
|
ui.XaosDistVizTableWidget->setVerticalHeaderLabels(blanks);
|
|
ui.XaosDistVizTableWidget->verticalHeader()->setSectionsClickable(false);
|
|
ui.XaosDistVizTableWidget->horizontalHeader()->setSectionsClickable(false);
|
|
ui.XaosTableView->setModel(m_XaosTableModel);
|
|
ui.XaosAppliedTableView->setModel(m_AppliedXaosTableModel);
|
|
ui.XaosTableView->setItemDelegate(m_XaosTableItemDelegate);//No need for a delegate on the applied table because it's read-only.
|
|
ui.XaosDistVizTableWidget->verticalHeader()->setFixedWidth(ui.XaosTableView->verticalHeader()->width());
|
|
SetTabOrder(this, ui.ClearXaosButton, ui.RandomXaosButton);
|
|
ui.XaosDistVizTableWidget->setRowHeight(0, ui.XaosTableView->rowHeight(0) * count);
|
|
m_Controller->FillXaos();
|
|
m_Controller->FillAppliedXaos();
|
|
//Needed to get the dark stylesheet to correctly color the top left corner button.
|
|
auto widgetList = ui.XaosTableView->findChildren<QAbstractButton*>();
|
|
|
|
for (auto& it : widgetList)
|
|
it->setEnabled(true);
|
|
|
|
widgetList = ui.XaosAppliedTableView->findChildren<QAbstractButton*>();
|
|
|
|
for (auto& it : widgetList)
|
|
it->setEnabled(true);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Clear all xaos from the current ember.
|
|
/// </summary>
|
|
template <typename T>
|
|
void FractoriumEmberController<T>::ClearXaos()
|
|
{
|
|
UpdateAll([&](Ember<T>& ember, bool isMain)
|
|
{
|
|
ember.ClearXaos();
|
|
}, true, eProcessAction::FULL_RENDER, m_Fractorium->ApplyAll());
|
|
FillXaos();
|
|
FillAppliedXaos();
|
|
}
|
|
|
|
void Fractorium::OnClearXaosButtonClicked(bool checked) { m_Controller->ClearXaos(); }
|
|
|
|
/// <summary>
|
|
/// Set all xaos values to random numbers.
|
|
/// There is a 50% chance they're set to 0 or 1, and
|
|
/// 50% that they're 0-3.
|
|
/// Resets the rendering process.
|
|
/// </summary>
|
|
template <typename T>
|
|
void FractoriumEmberController<T>::RandomXaos()
|
|
{
|
|
bool ctrl = QGuiApplication::keyboardModifiers().testFlag(Qt::ControlModifier);
|
|
Update([&]
|
|
{
|
|
size_t i = 0;
|
|
|
|
while (auto xform = m_Ember.GetXform(i++))
|
|
{
|
|
for (size_t j = 0; j < m_Ember.XformCount(); j++)
|
|
{
|
|
if (!ctrl)
|
|
xform->SetXaos(j, static_cast<T>(m_Rand.RandBit()));
|
|
else if (m_Rand.RandBit())
|
|
xform->SetXaos(j, static_cast<T>(m_Rand.RandBit()));
|
|
else
|
|
xform->SetXaos(j, TruncPrecision(m_Rand.Frand<T>(0, 3), 3));
|
|
}
|
|
}
|
|
});
|
|
FillXaos();
|
|
FillAppliedXaos();
|
|
}
|
|
|
|
void Fractorium::OnRandomXaosButtonClicked(bool checked) { m_Controller->RandomXaos(); }
|
|
|
|
/// <summary>
|
|
/// Add a layer using the specified number of xforms.
|
|
/// A layer is defined as a new set of xforms whose xaos values are the following:
|
|
/// From existing to existing: unchanged.
|
|
/// From existing to new: 0.
|
|
/// From new to existing: 0.
|
|
/// From new to new: 1.
|
|
/// Resets the rendering process.
|
|
/// </summary>
|
|
/// <param name="xforms">The number of new xforms to add to create the layer</param>
|
|
template <typename T>
|
|
void FractoriumEmberController<T>::AddLayer(int xforms)
|
|
{
|
|
Update([&]
|
|
{
|
|
std::vector<std::pair<Xform<T>, size_t>> vec(xforms);
|
|
AddXformsWithXaos(m_Ember, vec, eXaosPasteStyle::ZERO_TO_ONE);
|
|
|
|
});
|
|
FillXforms();
|
|
FillSummary();
|
|
}
|
|
|
|
void Fractorium::OnAddLayerButtonClicked(bool checked) { m_Controller->AddLayer(ui.AddLayerSpinBox->value()); }
|
|
|
|
/// <summary>
|
|
/// Flip the row and column values of the xaos table.
|
|
/// Resets the rendering process.
|
|
/// </summary>
|
|
template <typename T>
|
|
void FractoriumEmberController<T>::TransposeXaos()
|
|
{
|
|
Update([&]
|
|
{
|
|
size_t i = 0, j = 0;
|
|
vector<vector<double>> tempxaos;
|
|
tempxaos.reserve(m_Ember.XformCount());
|
|
|
|
while (const auto xform = m_Ember.GetXform(i++))
|
|
{
|
|
vector<double> tempvec;
|
|
tempvec.reserve(m_Ember.XformCount());
|
|
|
|
for (j = 0; j < m_Ember.XformCount(); j++)
|
|
tempvec.push_back(xform->Xaos(j));
|
|
|
|
tempxaos.push_back(std::move(tempvec));
|
|
}
|
|
|
|
for (j = 0; j < tempxaos.size(); j++)
|
|
for (i = 0; i < tempxaos[j].size(); i++)
|
|
if (auto xform = m_Ember.GetXform(i))
|
|
xform->SetXaos(j, static_cast<T>(tempxaos[j][i]));
|
|
});
|
|
FillXaos();
|
|
FillAppliedXaos();
|
|
}
|
|
|
|
void Fractorium::OnTransposeXaosButtonClicked(bool checked) { m_Controller->TransposeXaos(); }
|
|
|
|
/// <summary>
|
|
/// Toggle all xaos values in one row on left mouse button double click and resize all cells to fit their data.
|
|
/// Skip toggling and only refit on right mouse button double click.
|
|
/// Resets the rendering process.
|
|
/// </summary>
|
|
/// <param name="logicalIndex">The index of the row that was double clicked</param>
|
|
void Fractorium::OnXaosRowDoubleClicked(int logicalIndex)
|
|
{
|
|
const auto btn = QApplication::mouseButtons();
|
|
|
|
if (!btn.testFlag(Qt::RightButton))
|
|
ToggleTableRow(ui.XaosTableView, logicalIndex);
|
|
|
|
ui.XaosTableView->resizeRowsToContents();
|
|
ui.XaosTableView->resizeColumnsToContents();
|
|
m_Controller->FillAppliedXaos();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Toggle all xaos values in one column on left mouse button double click and resize all cells to fit their data.
|
|
/// Skip toggling and only refit on right mouse button double click.
|
|
/// Resets the rendering process.
|
|
/// </summary>
|
|
/// <param name="logicalIndex">The index of the column that was double clicked</param>
|
|
void Fractorium::OnXaosColDoubleClicked(int logicalIndex)
|
|
{
|
|
const auto btn = QApplication::mouseButtons();
|
|
|
|
if (!btn.testFlag(Qt::RightButton))
|
|
ToggleTableCol(ui.XaosTableView, logicalIndex);
|
|
|
|
ui.XaosTableView->resizeRowsToContents();
|
|
ui.XaosTableView->resizeColumnsToContents();
|
|
m_Controller->FillAppliedXaos();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Take the value of the horizontal scrollbar on the xaos table and set the same
|
|
/// horizontal scroll bar position on XaosDistVizTableWidget and XaosAppliedTableView.
|
|
/// This allows them to easily see the same part of all three tables at the same time
|
|
/// when there are more xforms than can fit on the screen at once.
|
|
/// </summary>
|
|
/// <param name="value">The value of the xaos table horizontal scroll bar</param>
|
|
void Fractorium::OnXaosHScrollValueChanged(int value)
|
|
{
|
|
ui.XaosDistVizTableWidget->horizontalScrollBar()->setValue(value);
|
|
ui.XaosAppliedTableView->horizontalScrollBar()->setValue(value);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Take the value of the vertical scrollbar on the xaos table and set the same
|
|
/// vertical scroll bar position on XaosDistVizTableWidget and XaosAppliedTableView.
|
|
/// This allows them to easily see the same part of all three tables at the same time
|
|
/// when there are more xforms than can fit on the screen at once.
|
|
/// </summary>
|
|
/// <param name="value">The value of the xaos table vertical scroll bar</param>
|
|
void Fractorium::OnXaosVScrollValueChanged(int value)
|
|
{
|
|
ui.XaosDistVizTableWidget->verticalScrollBar()->setValue(value);
|
|
ui.XaosAppliedTableView->verticalScrollBar()->setValue(value);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Get an enum value corresponding to the currently selected xaos pasting mode.
|
|
/// </summary>
|
|
/// <returns>The xaos pasting mode enum</returns>
|
|
eXaosPasteStyle Fractorium::GetXaosPasteStyleType()
|
|
{
|
|
if (ui.XaosPasteNoneRadio->isChecked())
|
|
return eXaosPasteStyle::NONE;
|
|
else if (ui.XaosPaste0to1Radio->isChecked())
|
|
return eXaosPasteStyle::ZERO_TO_ONE;
|
|
else if (ui.XaosPaste0toValsRadio->isChecked())
|
|
return eXaosPasteStyle::ZERO_TO_VALS;
|
|
else if (ui.XaosPaste1toValsRadio->isChecked())
|
|
return eXaosPasteStyle::ONE_TO_VALS;
|
|
else if (ui.XaosPasteValsTo1Radio->isChecked())
|
|
return eXaosPasteStyle::VALS_TO_ONE;
|
|
else
|
|
return eXaosPasteStyle::NONE;
|
|
}
|
|
|
|
template class FractoriumEmberController<float>;
|
|
|
|
#ifdef DO_DOUBLE
|
|
template class FractoriumEmberController<double>;
|
|
#endif
|