2023-04-25 19:59:54 -04:00
# include "FractoriumPch.h"
# include "Fractorium.h"
/// <summary>
/// Initialize the library tree UI.
/// </summary>
void Fractorium : : InitLibraryUI ( )
{
ui . LibraryTree - > SetMainWindow ( this ) ;
//Making the TreeItemChanged() events use a direct connection is absolutely critical.
2024-03-16 12:15:51 -04:00
connect ( ui . LibraryTree , SIGNAL ( itemChanged ( QTreeWidgetItem * , int ) ) , this , SLOT ( OnEmberTreeItemChanged ( QTreeWidgetItem * , int ) ) , Qt : : DirectConnection ) ;
connect ( ui . LibraryTree , SIGNAL ( itemDoubleClicked ( QTreeWidgetItem * , int ) ) , this , SLOT ( OnEmberTreeItemDoubleClicked ( QTreeWidgetItem * , int ) ) , Qt : : QueuedConnection ) ;
connect ( ui . LibraryTree , SIGNAL ( itemActivated ( QTreeWidgetItem * , int ) ) , this , SLOT ( OnEmberTreeItemDoubleClicked ( QTreeWidgetItem * , int ) ) , Qt : : QueuedConnection ) ;
connect ( ui . SequenceTree , SIGNAL ( itemChanged ( QTreeWidgetItem * , int ) ) , this , SLOT ( OnSequenceTreeItemChanged ( QTreeWidgetItem * , int ) ) , Qt : : DirectConnection ) ;
connect ( ui . SequenceStartPreviewsButton , SIGNAL ( clicked ( bool ) ) , this , SLOT ( OnSequenceStartPreviewsButtonClicked ( bool ) ) , Qt : : QueuedConnection ) ;
connect ( ui . SequenceStopPreviewsButton , SIGNAL ( clicked ( bool ) ) , this , SLOT ( OnSequenceStopPreviewsButtonClicked ( bool ) ) , Qt : : QueuedConnection ) ;
connect ( ui . SequenceAllButton , SIGNAL ( clicked ( bool ) ) , this , SLOT ( OnSequenceAllButtonClicked ( bool ) ) , Qt : : QueuedConnection ) ;
connect ( ui . SequenceGenerateButton , SIGNAL ( clicked ( bool ) ) , this , SLOT ( OnSequenceGenerateButtonClicked ( bool ) ) , Qt : : QueuedConnection ) ;
connect ( ui . SequenceRenderButton , SIGNAL ( clicked ( bool ) ) , this , SLOT ( OnSequenceRenderButtonClicked ( bool ) ) , Qt : : QueuedConnection ) ;
connect ( ui . SequenceAnimateButton , SIGNAL ( clicked ( bool ) ) , this , SLOT ( OnSequenceAnimateButtonClicked ( bool ) ) , Qt : : QueuedConnection ) ;
connect ( ui . SequenceClearButton , SIGNAL ( clicked ( bool ) ) , this , SLOT ( OnSequenceClearButtonClicked ( bool ) ) , Qt : : QueuedConnection ) ;
connect ( ui . SequenceSaveButton , SIGNAL ( clicked ( bool ) ) , this , SLOT ( OnSequenceSaveButtonClicked ( bool ) ) , Qt : : QueuedConnection ) ;
connect ( ui . SequenceOpenButton , SIGNAL ( clicked ( bool ) ) , this , SLOT ( OnSequenceOpenButtonClicked ( bool ) ) , Qt : : QueuedConnection ) ;
connect ( ui . SequenceStartFlameSpinBox , SIGNAL ( valueChanged ( int ) ) , this , SLOT ( OnSequenceStartFlameSpinBoxChanged ( int ) ) , Qt : : QueuedConnection ) ;
connect ( ui . SequenceStopFlameSpinBox , SIGNAL ( valueChanged ( int ) ) , this , SLOT ( OnSequenceStopFlameSpinBoxChanged ( int ) ) , Qt : : QueuedConnection ) ;
2023-04-25 19:59:54 -04:00
//Animation FPS.
ui . SequenceAnimationFpsSpinBox - > setValue ( m_Settings - > AnimationFps ( ) ) ;
}
/// <summary>
/// Select the item in the library tree specified by the passed in index.
/// </summary>
/// <param name="index">The 0-based index of the item in the library tree to select</param>
void Fractorium : : SelectLibraryItem ( size_t index )
{
2024-01-23 21:05:31 -05:00
EmberTreeWidgetItemBase * item = nullptr ;
2023-04-25 19:59:54 -04:00
if ( const auto top = ui . LibraryTree - > topLevelItem ( 0 ) )
{
for ( int i = 0 ; i < top - > childCount ( ) ; i + + )
{
if ( auto emberItem = dynamic_cast < EmberTreeWidgetItemBase * > ( top - > child ( i ) ) )
{
2024-01-23 21:05:31 -05:00
auto b = i = = index ;
if ( b )
item = emberItem ;
emberItem - > setSelected ( b ) ;
2024-05-08 09:31:55 -04:00
emberItem - > setCheckState ( NAME_COL , b ? Qt : : Checked : Qt : : Unchecked ) ;
2023-04-25 19:59:54 -04:00
}
}
2024-01-23 21:05:31 -05:00
if ( item )
ui . LibraryTree - > scrollToItem ( item , QAbstractItemView : : EnsureVisible ) ;
2023-04-25 19:59:54 -04:00
}
}
/// <summary>
/// Get the index of the currently selected ember in the library tree.
/// </summary>
/// <param name="isChecked">Whether to search for items that are checked or items that are only selected</param>
/// <returns>A pair containing the index of the item clicked and a pointer to the item</param>
vector < pair < size_t , QTreeWidgetItem * > > Fractorium : : GetCurrentEmberIndex ( bool isChecked )
{
int index = 0 ;
QTreeWidgetItem * item = nullptr ;
const auto tree = ui . LibraryTree ;
vector < pair < size_t , QTreeWidgetItem * > > v ;
if ( const auto top = tree - > topLevelItem ( 0 ) )
{
for ( int i = 0 ; i < top - > childCount ( ) ; i + + ) //Iterate through all of the children, which will represent the open embers.
{
item = top - > child ( index ) ;
if ( item & & item - > isSelected ( ) )
{
if ( isChecked )
{
2023-12-19 15:24:50 -05:00
if ( item - > checkState ( NAME_COL ) = = Qt : : Checked )
2023-04-25 19:59:54 -04:00
v . push_back ( make_pair ( index , item ) ) ;
}
else
v . push_back ( make_pair ( index , item ) ) ;
}
index + + ;
}
}
return v ;
}
/// <summary>
/// Slot function to be called via QMetaObject::invokeMethod() to update preview images in the preview thread.
/// </summary>
/// <param name="item">The item double clicked on</param>
/// <param name="v">The vector holding the RGBA bitmap</param>
/// <param name="w">The width of the bitmap</param>
/// <param name="h">The height of the bitmap</param>
void Fractorium : : SetTreeItemData ( EmberTreeWidgetItemBase * item , vv4F & v , uint w , uint h )
{
m_PreviewVec . resize ( size_t ( w ) * size_t ( h ) * 4 ) ;
Rgba32ToRgba8 ( v . data ( ) , m_PreviewVec . data ( ) , w , h , m_Settings - > Transparency ( ) ) ;
item - > SetImage ( m_PreviewVec , w , h ) ;
}
/// <summary>
/// Set all libary tree entries to the name of the corresponding ember they represent.
/// Set all libary tree entries to point to the underlying ember they represent.
/// </summary>
/// <param name="update">A bitfield representing the type of synchronizing to do. Update one or more of index, name or pointer.</param>
template < typename T >
void FractoriumEmberController < T > : : SyncLibrary ( eLibraryUpdate update )
{
auto it = m_EmberFile . m_Embers . begin ( ) ;
const auto tree = m_Fractorium - > ui . LibraryTree ;
if ( const auto top = tree - > topLevelItem ( 0 ) )
{
for ( int i = 0 ; i < top - > childCount ( ) & & it ! = m_EmberFile . m_Embers . end ( ) ; + + i , + + it ) //Iterate through all of the children, which will represent the open embers.
{
2023-12-19 15:24:50 -05:00
if ( auto emberItem = dynamic_cast < EmberTreeWidgetItem < T > * > ( top - > child ( i ) ) ) //Cast the child widget to the EmberTreeWidgetItem type.
2023-04-25 19:59:54 -04:00
{
if ( static_cast < uint > ( update ) & static_cast < uint > ( eLibraryUpdate : : INDEX ) )
it - > m_Index = i ;
if ( static_cast < uint > ( update ) & static_cast < uint > ( eLibraryUpdate : : NAME ) )
2023-12-19 15:24:50 -05:00
emberItem - > setText ( NAME_COL , QString : : fromStdString ( it - > m_Name ) ) ;
2023-04-25 19:59:54 -04:00
if ( static_cast < uint > ( update ) & static_cast < uint > ( eLibraryUpdate : : POINTER ) )
2023-12-19 15:24:50 -05:00
emberItem - > SetEmberPointer ( & ( * it ) ) ;
2023-04-25 19:59:54 -04:00
2023-12-19 15:24:50 -05:00
if ( emberItem - > checkState ( NAME_COL ) = = Qt : : Checked )
m_EmberFilePointer = emberItem - > GetEmber ( ) ;
emberItem - > setText ( INDEX_COL , ToString ( i ) ) ;
2023-04-25 19:59:54 -04:00
}
}
}
}
/// <summary>
/// Fill the library tree with the names of the embers in the
/// currently opened file.
/// Start preview render thread.
/// </summary>
/// <param name="selectIndex">After the tree is filled, select this index. Pass -1 to omit selecting an index.</param>
template < typename T >
void FractoriumEmberController < T > : : FillLibraryTree ( int selectIndex )
{
StopAllPreviewRenderers ( ) ;
const uint size = PREVIEW_SIZE ;
vector < unsigned char > empy_preview ( size * size * 4 ) ;
const auto tree = m_Fractorium - > ui . LibraryTree ;
tree - > clear ( ) ;
auto fileItem = new QTreeWidgetItem ( tree ) ;
QFileInfo info ( m_EmberFile . m_Filename ) ;
2023-12-19 15:24:50 -05:00
fileItem - > setText ( NAME_COL , info . fileName ( ) ) ;
fileItem - > setToolTip ( NAME_COL , m_EmberFile . m_Filename ) ;
2023-04-25 19:59:54 -04:00
fileItem - > setFlags ( Qt : : ItemIsEnabled | Qt : : ItemIsEditable | Qt : : ItemIsSelectable | Qt : : ItemIsDropEnabled ) ;
uint i = 0 ;
for ( auto & it : m_EmberFile . m_Embers )
{
2024-05-08 09:31:55 -04:00
it . m_Index = i ;
2023-04-25 19:59:54 -04:00
auto emberItem = new EmberTreeWidgetItem < T > ( & it , fileItem ) ;
2023-12-19 15:24:50 -05:00
auto istr = ToString ( i + + ) ;
emberItem - > setText ( INDEX_COL , istr ) ;
2023-04-25 19:59:54 -04:00
if ( it . m_Name . empty ( ) )
2023-12-19 15:24:50 -05:00
emberItem - > setText ( NAME_COL , istr ) ;
2023-04-25 19:59:54 -04:00
else
2023-12-19 15:24:50 -05:00
emberItem - > setText ( NAME_COL , it . m_Name . c_str ( ) ) ;
2023-04-25 19:59:54 -04:00
2023-12-19 15:24:50 -05:00
emberItem - > setToolTip ( NAME_COL , emberItem - > text ( NAME_COL ) ) ;
2023-04-25 19:59:54 -04:00
emberItem - > SetImage ( empy_preview , size , size ) ;
2024-05-08 09:31:55 -04:00
emberItem - > SetEmberPointer ( & it ) ;
2023-04-25 19:59:54 -04:00
}
2024-05-08 09:31:55 -04:00
//tree->update();
2023-04-25 19:59:54 -04:00
if ( selectIndex ! = - 1 )
m_Fractorium - > SelectLibraryItem ( selectIndex ) ;
m_Fractorium - > SyncFileCountToSequenceCount ( ) ;
RenderLibraryPreviews ( 0 , static_cast < uint > ( m_EmberFile . Size ( ) ) ) ;
tree - > expandAll ( ) ;
}
/// <summary>
/// Update the library tree with the newly added embers (most likely from pasting) and
/// only render previews for the new ones, without clearing the entire tree.
/// </summary>
template < typename T >
void FractoriumEmberController < T > : : UpdateLibraryTree ( )
{
const uint size = PREVIEW_SIZE ;
vector < unsigned char > empy_preview ( size * size * 4 ) ;
const auto tree = m_Fractorium - > ui . LibraryTree ;
if ( auto top = tree - > topLevelItem ( 0 ) )
{
const int origChildCount = top - > childCount ( ) ;
int i = origChildCount ;
for ( auto it = Advance ( m_EmberFile . m_Embers . begin ( ) , i ) ; it ! = m_EmberFile . m_Embers . end ( ) ; + + it )
{
auto emberItem = new EmberTreeWidgetItem < T > ( & ( * it ) , top ) ;
2023-12-19 15:24:50 -05:00
auto istr = ToString ( i + + ) ;
emberItem - > setText ( INDEX_COL , istr ) ;
2023-04-25 19:59:54 -04:00
if ( it - > m_Name . empty ( ) )
2023-12-19 15:24:50 -05:00
emberItem - > setText ( NAME_COL , istr ) ;
2023-04-25 19:59:54 -04:00
else
2023-12-19 15:24:50 -05:00
emberItem - > setText ( NAME_COL , it - > m_Name . c_str ( ) ) ;
2023-04-25 19:59:54 -04:00
2023-12-19 15:24:50 -05:00
emberItem - > setToolTip ( NAME_COL , emberItem - > text ( NAME_COL ) ) ;
2023-04-25 19:59:54 -04:00
emberItem - > SetImage ( empy_preview , size , size ) ;
}
//When adding elements, ensure all indices are sequential.
SyncLibrary ( eLibraryUpdate : : INDEX ) ;
m_Fractorium - > SyncFileCountToSequenceCount ( ) ;
RenderLibraryPreviews ( origChildCount , static_cast < uint > ( m_EmberFile . Size ( ) ) ) ;
}
}
/// <summary>
/// Copy the text of the item which was changed to the name of the current ember.
/// Ensure all names are unique in the opened file.
/// This seems to be called spuriously, so we do a check inside to make sure
/// the text was actually changed.
/// We also have to wrap the dynamic_cast call in a try/catch block because this can
/// be called on a widget that has already been deleted.
/// </summary>
/// <param name="item">The libary tree item changed</param>
/// <param name="col">The column clicked, ignored.</param>
template < typename T >
void FractoriumEmberController < T > : : EmberTreeItemChanged ( QTreeWidgetItem * item , int col )
{
try
{
const auto tree = m_Fractorium - > ui . LibraryTree ;
if ( auto emberItem = dynamic_cast < EmberTreeWidgetItem < T > * > ( item ) )
{
auto oldName = emberItem - > GetEmber ( ) - > m_Name ; //First preserve the previous name.
2023-12-19 15:24:50 -05:00
auto newName = emberItem - > text ( NAME_COL ) . toStdString ( ) ;
2023-04-25 19:59:54 -04:00
//Checking/unchecking other items shouldn't perform the processing below.
//If nothing changed, nothing to do.
if ( ! emberItem - > isSelected ( ) & & newName = = oldName )
return ;
if ( newName . empty ( ) ) //Prevent empty string.
{
emberItem - > UpdateEditText ( ) ;
return ;
}
emberItem - > UpdateEmberName ( ) ; //Copy edit text to the ember's name variable.
m_EmberFile . MakeNamesUnique ( ) ; //Ensure all names remain unique.
SyncLibrary ( eLibraryUpdate : : NAME ) ; //Copy all ember names to the tree items since some might have changed to be made unique.
newName = emberItem - > GetEmber ( ) - > m_Name ; //Get the new, final, unique name.
if ( m_EmberFilePointer & & m_EmberFilePointer = = emberItem - > GetEmber ( ) & & oldName ! = newName ) //If the ember edited was the current one, and the name was indeed changed, update the name of the current one.
{
m_Ember . m_Name = newName ;
m_LastSaveCurrent = " " ; //Reset will force the dialog to show on the next save current since the user probably wants a different name.
}
}
else if ( const auto parentItem = dynamic_cast < QTreeWidgetItem * > ( item ) )
{
2023-12-19 15:24:50 -05:00
const auto text = parentItem - > text ( NAME_COL ) ;
2023-04-25 19:59:54 -04:00
if ( text ! = " " )
{
m_EmberFile . m_Filename = text ;
2024-01-23 21:05:31 -05:00
//m_LastSaveAll = "";//Reset will force the dialog to show on the next save all since the user probably wants a different name.
2023-04-25 19:59:54 -04:00
}
}
}
catch ( const std : : exception & e )
{
qDebug ( ) < < " FractoriumEmberController<T>::EmberTreeItemChanged() : Exception thrown: " < < e . what ( ) ;
}
}
void Fractorium : : OnEmberTreeItemChanged ( QTreeWidgetItem * item , int col )
{
if ( item & & ui . LibraryTree - > topLevelItemCount ( ) ) //This can sometimes be spurriously called even when the tree is empty.
m_Controller - > EmberTreeItemChanged ( item , col ) ;
}
/// <summary>
/// Set the current ember to the selected item.
/// Clears the undo state.
/// Resets the rendering process.
/// Called when the user double clicks on a library tree item.
/// This will get called twice for some reason, so the check state is checked to prevent duplicate processing.
/// </summary>
/// <param name="item">The item double clicked on</param>
/// <param name="col">The column clicked</param>
template < typename T >
void FractoriumEmberController < T > : : EmberTreeItemDoubleClicked ( QTreeWidgetItem * item , int col )
{
if ( item - > checkState ( col ) = = Qt : : Unchecked )
SetEmber ( m_Fractorium - > ui . LibraryTree - > currentIndex ( ) . row ( ) , false ) ;
}
void Fractorium : : OnEmberTreeItemDoubleClicked ( QTreeWidgetItem * item , int col )
{
if ( ui . LibraryTree - > topLevelItemCount ( ) ) //This can sometimes be spurriously called even when the tree is empty.
m_Controller - > EmberTreeItemDoubleClicked ( item , col ) ;
}
/// <summary>
/// Move a possibly disjoint selection of library items from their indices,
/// to another index.
/// </summary>
/// <param name="items">The selected list of items to move</param>
/// <param name="destRow">The destination index to move the item to</param>
template < typename T >
void FractoriumEmberController < T > : : MoveLibraryItems ( const QModelIndexList & items , int destRow )
{
int i = 0 ;
const auto startRow = items [ 0 ] . row ( ) ;
const auto tree = m_Fractorium - > ui . LibraryTree ;
const auto top = tree - > topLevelItem ( 0 ) ;
list < string > names ;
for ( auto & item : items )
if ( auto temp = m_EmberFile . Get ( item . row ( ) ) )
names . push_back ( temp - > m_Name ) ;
auto b = m_EmberFile . m_Embers . begin ( ) ;
const auto result = Gather ( b , m_EmberFile . m_Embers . end ( ) , Advance ( b , destRow ) , [ & ] ( const Ember < T > & ember )
{
auto position = std : : find ( names . begin ( ) , names . end ( ) , ember . m_Name ) ;
if ( position ! = names . end ( ) )
{
names . erase ( position ) ;
return true ;
}
return false ;
} ) ;
tree - > update ( ) ;
2024-05-08 09:31:55 -04:00
SyncLibrary ( eLibraryUpdate ( static_cast < uint > ( eLibraryUpdate : : INDEX ) | static_cast < uint > ( eLibraryUpdate : : NAME ) | static_cast < uint > ( eLibraryUpdate : : POINTER ) ) ) ;
2023-04-25 19:59:54 -04:00
}
/// <summary>
/// Delete the currently selected items in the tree.
/// Note this is not necessarilly the current ember, it's just the items
/// in the tree that are selected.
/// </summary>
/// <param name="v">A vector of pairs, each containing the index of the item selected and a pointer to the item</param>
template < typename T >
void FractoriumEmberController < T > : : Delete ( const vector < pair < size_t , QTreeWidgetItem * > > & v )
{
size_t offset = 0 ;
uint last = 0 ;
for ( auto & p : v )
{
if ( p . second & & m_EmberFile . Delete ( p . first - offset ) )
{
last = uint ( p . first - offset ) ;
delete p . second ;
2024-05-08 09:31:55 -04:00
SyncLibrary ( eLibraryUpdate ( static_cast < uint > ( eLibraryUpdate : : INDEX ) | static_cast < uint > ( eLibraryUpdate : : NAME ) | static_cast < uint > ( eLibraryUpdate : : POINTER ) ) ) ;
2023-04-25 19:59:54 -04:00
m_Fractorium - > SyncFileCountToSequenceCount ( ) ;
}
offset + + ;
}
//Select the next item in the tree closest to the last one that was deleted.
if ( const auto top = m_Fractorium - > ui . LibraryTree - > topLevelItem ( 0 ) )
{
last = std : : min < uint > ( top - > childCount ( ) - 1 , last ) ;
if ( auto item = dynamic_cast < EmberTreeWidgetItem < T > * > ( top - > child ( last ) ) )
if ( item - > GetEmber ( ) - > m_Name ! = m_Ember . m_Name )
EmberTreeItemDoubleClicked ( item , 0 ) ;
}
}
/// <summary>
/// Called when the user presses and releases the delete key while the library tree has the focus,
/// and an item is selected.
/// </summary>
/// <param name="v">A vector of pairs, each containing the index of the item selected and a pointer to the item</param>
void Fractorium : : OnDelete ( const vector < pair < size_t , QTreeWidgetItem * > > & v )
{
m_Controller - > Delete ( v ) ;
}
/// <summary>
/// Stop the preview renderer if it's already running.
/// Clear all of the existing preview images, then start the preview rendering thread.
/// Optionally only render previews for a subset of all open embers.
/// </summary>
/// <param name="start">The 0-based index to start rendering previews for</param>
/// <param name="end">The 0-based index which is one beyond the last ember to render a preview for</param>
template < typename T >
void FractoriumEmberController < T > : : RenderPreviews ( QTreeWidget * tree , TreePreviewRenderer < T > * renderer , EmberFile < T > & file , uint start , uint end )
{
renderer - > Stop ( ) ;
if ( start = = UINT_MAX & & end = = UINT_MAX )
{
// Animated item might be at index 0, previews go in last item.
if ( const auto top = tree - > topLevelItem ( tree - > topLevelItemCount ( ) - 1 ) )
{
const auto childCount = top - > childCount ( ) ;
vector < unsigned char > emptyPreview ( PREVIEW_SIZE * PREVIEW_SIZE * 4 ) ;
for ( int i = 0 ; i < childCount ; i + + )
if ( auto treeItem = dynamic_cast < EmberTreeWidgetItemBase * > ( top - > child ( i ) ) )
treeItem - > SetImage ( emptyPreview , PREVIEW_SIZE , PREVIEW_SIZE ) ;
}
renderer - > Render ( 0 , uint ( file . Size ( ) ) ) ;
}
else
renderer - > Render ( start , end ) ;
}
/// <summary>
/// Wrapper around calling RenderPreviews with the appropriate values passed in for the previews in the main library tree.
/// </summary>
/// <param name="start">The 0-based index to start rendering previews for</param>
/// <param name="end">The 0-based index which is one beyond the last ember to render a preview for</param>
template < typename T >
void FractoriumEmberController < T > : : RenderLibraryPreviews ( uint start , uint end )
{
RenderPreviews ( m_Fractorium - > ui . LibraryTree , m_LibraryPreviewRenderer . get ( ) , m_EmberFile , start , end ) ;
}
template < typename T >
2023-11-23 16:52:34 -05:00
void FractoriumEmberController < T > : : StopLibraryPreviewRender ( )
{
m_LibraryPreviewRenderer - > Stop ( ) ;
QApplication : : processEvents ( ) ;
}
2023-04-25 19:59:54 -04:00
/// <summary>
/// Thing wrapper around StopLibraryPreviewRender() and StopSequencePreviewRender() to stop both preview renderers.
/// </summary>
template < typename T >
void FractoriumEmberController < T > : : StopAllPreviewRenderers ( )
{
StopLibraryPreviewRender ( ) ;
StopSequencePreviewRender ( ) ;
}
template < typename T >
void FractoriumEmberController < T > : : AddAnimationItem ( )
{
auto fileItem = new QTreeWidgetItem ( m_Fractorium - > ui . SequenceTree ) ;
2023-12-19 15:24:50 -05:00
fileItem - > setText ( NAME_COL , " Rendered Animation " ) ;
2023-04-25 19:59:54 -04:00
fileItem - > setFlags ( Qt : : ItemIsEnabled | Qt : : ItemIsSelectable ) ;
auto emberItem = new EmberTreeWidgetItemBase ( fileItem ) ;
emberItem - > setFlags ( Qt : : ItemIsEnabled | Qt : : ItemIsSelectable ) ;
2023-12-19 15:24:50 -05:00
emberItem - > setToolTip ( INDEX_COL , " Animated Frame " ) ;
2023-04-25 19:59:54 -04:00
const uint size = PREVIEW_SIZE ;
vector < unsigned char > empy_preview ( size * size * 4 ) ;
emberItem - > SetImage ( empy_preview , size , size ) ;
}
/// <summary>
/// Fill the sequence tree with the names of the embers in the
/// currently generated sequence.
/// Start the sequence preview render thread.
/// </summary>
template < typename T >
void FractoriumEmberController < T > : : FillSequenceTree ( )
{
StopAllPreviewRenderers ( ) ;
const uint size = PREVIEW_SIZE ;
vector < unsigned char > empy_preview ( size * size * 4 ) ;
const auto tree = m_Fractorium - > ui . SequenceTree ;
tree - > clear ( ) ;
2023-12-19 15:24:50 -05:00
//Add extra TreeWidget for animation at index 0.
2023-04-25 19:59:54 -04:00
AddAnimationItem ( ) ;
2023-12-19 15:24:50 -05:00
m_AnimateTimer - > stop ( ) ;
2023-04-25 19:59:54 -04:00
auto fileItem = new QTreeWidgetItem ( tree ) ;
QFileInfo info ( m_SequenceFile . m_Filename ) ;
2023-12-19 15:24:50 -05:00
fileItem - > setText ( NAME_COL , info . fileName ( ) ) ;
fileItem - > setToolTip ( NAME_COL , m_SequenceFile . m_Filename ) ;
2023-04-25 19:59:54 -04:00
fileItem - > setFlags ( Qt : : ItemIsEnabled | Qt : : ItemIsEditable | Qt : : ItemIsSelectable ) ;
uint i = 0 ;
for ( auto & it : m_SequenceFile . m_Embers )
{
auto emberItem = new EmberTreeWidgetItemBase ( fileItem ) ;
2023-12-19 15:24:50 -05:00
auto istr = ToString ( i + + ) ;
2023-04-25 19:59:54 -04:00
if ( it . m_Name . empty ( ) )
2023-12-19 15:24:50 -05:00
emberItem - > setText ( NAME_COL , istr ) ;
2023-04-25 19:59:54 -04:00
else
2023-12-19 15:24:50 -05:00
emberItem - > setText ( NAME_COL , it . m_Name . c_str ( ) ) ;
2023-04-25 19:59:54 -04:00
2023-12-19 15:24:50 -05:00
emberItem - > setText ( INDEX_COL , istr ) ;
2023-04-25 19:59:54 -04:00
emberItem - > setFlags ( Qt : : ItemIsEnabled | Qt : : ItemIsSelectable ) ;
2023-12-19 15:24:50 -05:00
emberItem - > setToolTip ( NAME_COL , emberItem - > text ( NAME_COL ) ) ;
2023-04-25 19:59:54 -04:00
emberItem - > SetImage ( empy_preview , size , size ) ;
}
tree - > expandAll ( ) ;
2023-12-19 15:24:50 -05:00
//Hide, then show the animation item.
2023-04-25 19:59:54 -04:00
tree - > collapseItem ( tree - > topLevelItem ( 0 ) ) ;
RenderSequencePreviews ( 0 , uint ( m_SequenceFile . Size ( ) ) ) ;
2023-12-19 15:24:50 -05:00
if ( const auto animation = tree - > topLevelItem ( 0 ) )
{
animation - > setExpanded ( true ) ;
m_AnimateFrame = 0 ;
m_AnimateTimer - > start ( 1000 / m_Fractorium - > ui . SequenceAnimationFpsSpinBox - > value ( ) ) ;
}
2023-04-25 19:59:54 -04:00
}
/// <summary>
/// Copy the text of the root item to the name of the sequence file.
/// Called whenever the text of the root item is changed.
/// </summary>
/// <param name="item">The root sequence tree item which changed</param>
/// <param name="col">The column clicked, ignored.</param>
template < typename T >
void FractoriumEmberController < T > : : SequenceTreeItemChanged ( QTreeWidgetItem * item , int col )
{
if ( item = = m_Fractorium - > ui . SequenceTree - > topLevelItem ( 1 ) )
{
2023-12-19 15:24:50 -05:00
auto text = item - > text ( NAME_COL ) ;
2023-04-25 19:59:54 -04:00
if ( text ! = " " )
m_SequenceFile . m_Filename = text ;
}
}
void Fractorium : : OnSequenceTreeItemChanged ( QTreeWidgetItem * item , int col )
{
if ( item & & ui . SequenceTree - > topLevelItemCount ( ) )
m_Controller - > SequenceTreeItemChanged ( item , col ) ;
}
/// <summary>
/// Wrapper around calling RenderPreviews with the appropriate values passed in for the previews in the sequence tree.
/// Called when Render Previews is clicked.
/// </summary>
/// <param name="start">Ignored, render all.</param>
/// <param name="end">Ignored, render all.</param>
template < typename T >
void FractoriumEmberController < T > : : RenderSequencePreviews ( uint start , uint end )
{
RenderPreviews ( m_Fractorium - > ui . SequenceTree , m_SequencePreviewRenderer . get ( ) , m_SequenceFile , start , end ) ;
}
2023-11-23 16:52:34 -05:00
void Fractorium : : OnSequenceStartPreviewsButtonClicked ( bool checked ) { m_Controller - > RenderSequencePreviews ( ) ; }
2023-04-25 19:59:54 -04:00
/// <summary>
/// Stop rendering the sequence previews.
/// Called when Stop Previews is clicked.
/// </summary>
template < typename T >
2023-11-23 16:52:34 -05:00
void FractoriumEmberController < T > : : StopSequencePreviewRender ( )
{
m_SequencePreviewRenderer - > Stop ( ) ;
QApplication : : processEvents ( ) ;
}
2023-04-25 19:59:54 -04:00
void Fractorium : : OnSequenceStopPreviewsButtonClicked ( bool checked ) { m_Controller - > StopSequencePreviewRender ( ) ; }
/// <summary>
/// Set the start and stop spin boxes to 0 and the length of the ember file minus 1, respectively.
/// Called whenever the count of the current file changes or when All is clicked.
/// </summary>
void Fractorium : : SyncFileCountToSequenceCount ( )
{
if ( const auto top = ui . LibraryTree - > topLevelItem ( 0 ) )
{
const int count = top - > childCount ( ) - 1 ;
2023-12-19 15:24:50 -05:00
ui . LibraryTree - > headerItem ( ) - > setText ( NAME_COL , " Current Flame File ( " + QString : : number ( top - > childCount ( ) ) + " ) " ) ;
2023-04-25 19:59:54 -04:00
ui . SequenceStartFlameSpinBox - > setMinimum ( 0 ) ;
ui . SequenceStartFlameSpinBox - > setMaximum ( count ) ;
ui . SequenceStartFlameSpinBox - > setValue ( 0 ) ;
ui . SequenceStopFlameSpinBox - > setMinimum ( 0 ) ;
ui . SequenceStopFlameSpinBox - > setMaximum ( count ) ;
ui . SequenceStopFlameSpinBox - > setValue ( count ) ;
}
}
void Fractorium : : OnSequenceAllButtonClicked ( bool checked ) { SyncFileCountToSequenceCount ( ) ; }
/// <summary>
/// Generate an animation sequence and place it in the sequence tree. This code is
/// mostly similar to that of EmberGenome.
/// It differs in a few ways:
/// The number of frames used in a rotation and in blending can differ. In EmberGenome, they are the same.
/// The number of rotations, frames used in rotations and frames used in blending can all be randomized.
/// Called when the Generate button is clicked.
/// </summary>
template < typename T >
void FractoriumEmberController < T > : : SequenceGenerateButtonClicked ( )
{
StopAllPreviewRenderers ( ) ;
SaveCurrentToOpenedFile ( false ) ;
Ember < T > result ;
auto & ui = m_Fractorium - > ui ;
auto s = m_Fractorium - > m_Settings ;
//Number of rotations performed during interpolation.
const size_t start = ui . SequenceStartFlameSpinBox - > value ( ) ;
const size_t stop = ui . SequenceStopFlameSpinBox - > value ( ) ;
const size_t startCount = ui . SequenceStartCountSpinBox - > value ( ) ;
const size_t keyFrames = ( stop - start ) + 1 ;
size_t frameCount = 0 ;
2024-03-16 12:15:51 -04:00
size_t fps = ui . SequenceAnimationFpsSpinBox - > value ( ) ;
2023-04-25 19:59:54 -04:00
double frames = 0 ;
vector < pair < size_t , size_t > > devices ; //Dummy.
EmberReport emberReport ;
ostringstream os ;
string palettePath = FindFirstDefaultPalette ( ) . toStdString ( ) ;
if ( palettePath . empty ( ) ) //This should never happen because the program requires a palette file to run this far.
{
QMessageBox : : warning ( nullptr , " Sequence " , " No flam3-palettes.xml file found, sequence will not be generated. " ) ;
return ;
}
2024-03-16 12:15:51 -04:00
SheepTools < T , float > tools ( palettePath , EmberCommon : : CreateRenderer < T > ( eRendererType : : CPU_RENDERER , devices , false , 0 , emberReport ) ) ;
auto it = Advance ( m_EmberFile . m_Embers . begin ( ) , start ) ;
for ( size_t i = start ; i < = stop & & it ! = m_EmberFile . m_Embers . end ( ) ; i + + , + + it )
2023-04-25 19:59:54 -04:00
{
2024-03-16 12:15:51 -04:00
if ( ( ! it - > m_Rotations | | ! it - > m_SecondsPerRotation ) & & ! it - > m_BlendSeconds )
2023-04-25 19:59:54 -04:00
{
QMessageBox : : critical ( m_Fractorium , " Animation sequence parameters error " ,
2024-03-16 12:15:51 -04:00
" Rotations and seconds per rotation, or blend seconds must be positive and non-zero " ) ;
2023-04-25 19:59:54 -04:00
return ;
}
2024-03-16 12:15:51 -04:00
frames + = fps * it - > m_SecondsPerRotation * it - > m_Rotations ;
2023-04-25 19:59:54 -04:00
2024-03-16 12:15:51 -04:00
if ( i < stop )
frames + = fps * it - > m_BlendSeconds ;
}
2023-04-25 19:59:54 -04:00
frames + = startCount ;
os < < setfill ( ' 0 ' ) < < setprecision ( 0 ) < < fixed ;
m_SequenceFile . Clear ( ) ;
m_SequenceFile . m_Filename = EmberFile < T > : : DefaultFilename ( " Sequence_ " ) ;
double blend ;
size_t frame ;
Ember < T > embers [ 2 ] ; //Spin needs contiguous array below, and this will also get modified, so a copy is needed to avoid modifying the embers in the original file.
const auto padding = streamsize ( std : : log10 ( frames ) ) + 1 ;
2024-03-16 12:15:51 -04:00
auto lastLoopsBlendCw = false ;
it = Advance ( m_EmberFile . m_Embers . begin ( ) , start ) ;
2023-04-25 19:59:54 -04:00
for ( size_t i = start ; i < = stop & & it ! = m_EmberFile . m_Embers . end ( ) ; i + + , + + it )
{
2024-03-16 12:15:51 -04:00
//The direction to rotate the loops.
const auto loopsCw = it - > m_RotateXformsCw ; //ui.SequenceRotationsCWCheckBox->isChecked();
const auto loopsBlendCw = it - > m_BlendRotateXformsCw ; //ui.SequenceRotationsPerBlendCWCheckBox->isChecked();
//Whether to stagger, default is 1 which means no stagger.
const auto stagger = it - > m_Stagger ; //ui.SequenceStaggerSpinBox->value();
//Rotations on keyframes.
const auto rotations = it - > m_Rotations ;
//Number of frames it takes to rotate a keyframe.
const auto rotFrames = fps * ( double ) it - > m_SecondsPerRotation ;
//Number of frames it takes to interpolate.
const auto framesBlend = fps * it - > m_BlendSeconds ;
const auto rotsPerBlend = it - > m_RotationsPerBlend ;
const auto linear = it - > m_Linear ;
tools . SetSpinParams ( ! linear ,
2024-05-08 09:31:55 -04:00
stagger ,
2024-03-16 12:15:51 -04:00
0 ,
0 ,
s - > Nick ( ) . toStdString ( ) ,
s - > Url ( ) . toStdString ( ) ,
s - > Id ( ) . toStdString ( ) ,
" " ,
0 ,
0 ) ;
2023-04-25 19:59:54 -04:00
embers [ 0 ] = * it ;
if ( rotations > 0 )
{
const auto roundFrames = size_t ( std : : round ( rotFrames * rotations ) ) ;
for ( frame = 0 ; frame < roundFrames ; frame + + )
{
blend = frame / rotFrames ;
tools . Spin ( embers [ 0 ] , nullptr , result , startCount + frameCount + + , blend , loopsCw ) ; //Result is cleared and reassigned each time inside of Spin().
FormatName ( result , os , padding ) ;
m_SequenceFile . m_Embers . push_back ( result ) ;
}
//The loop above will have rotated just shy of a complete rotation.
//Rotate the next step and save in result, but do not print.
//result will be the starting point for the interp phase below.
frame = roundFrames ;
blend = frame / rotFrames ;
tools . Spin ( embers [ 0 ] , nullptr , result , startCount + frameCount , blend , loopsCw ) ; //Do not increment frameCount here.
FormatName ( result , os , padding ) ;
}
if ( i < stop )
{
if ( rotations > 0 ) //Store the last result as the flame to interpolate from. This applies for whole or fractional values of opt.Loops().
embers [ 0 ] = result ;
auto it2 = it ; //Need a quick temporary to avoid modifying it, which is used in the loop.
embers [ 1 ] = * ( + + it2 ) ; //Get the next ember to be used with blending below.
2024-03-16 12:15:51 -04:00
const auto d = double ( rotsPerBlend ) ;
2023-04-25 19:59:54 -04:00
const auto rpb = size_t ( std : : round ( d ) ) ;
2024-03-16 12:15:51 -04:00
for ( frame = 0 ; frame < framesBlend ; frame + + )
2023-04-25 19:59:54 -04:00
{
2024-03-16 12:15:51 -04:00
const auto seqFlag = frame = = 0 | | ( frame = = framesBlend - 1 ) ;
blend = frame / framesBlend ;
2023-04-25 19:59:54 -04:00
result . Clear ( ) ;
tools . SpinInter ( & embers [ 0 ] , nullptr , result , startCount + frameCount + + , seqFlag , blend , rpb , loopsBlendCw ) ;
FormatName ( result , os , padding ) ;
m_SequenceFile . m_Embers . push_back ( result ) ;
}
}
2024-03-16 12:15:51 -04:00
lastLoopsBlendCw = loopsBlendCw ;
2023-04-25 19:59:54 -04:00
}
it = Advance ( m_EmberFile . m_Embers . begin ( ) , stop ) ;
2024-03-16 12:15:51 -04:00
tools . Spin ( * it , nullptr , result , startCount + frameCount , 0 , lastLoopsBlendCw ) ;
2023-04-25 19:59:54 -04:00
FormatName ( result , os , padding ) ;
m_SequenceFile . m_Embers . push_back ( result ) ;
FillSequenceTree ( ) ; //The sequence has been generated, now create preview thumbnails.
}
void Fractorium : : OnSequenceGenerateButtonClicked ( bool checked ) { m_Controller - > SequenceGenerateButtonClicked ( ) ; }
/// <summary>
/// Show the final render dialog and load the sequence into it.
/// This will automatically check the Render All and Render as Animation sequence checkboxes.
/// Called when the Render Sequence button is clicked.
/// </summary>
/// <param name="checked">Ignored.</param>
void Fractorium : : OnSequenceRenderButtonClicked ( bool checked )
{
if ( ui . SequenceTree - > topLevelItemCount ( ) > 0 )
{
//First completely stop what the current rendering process is doing.
m_Controller - > DeleteRenderer ( ) ; //Delete the renderer, but not the controller.
m_Controller - > StopAllPreviewRenderers ( ) ;
m_Controller - > SaveCurrentToOpenedFile ( false ) ; //Save whatever was edited back to the current open file.
m_RenderStatusLabel - > setText ( " Renderer stopped. " ) ;
SetupFinalRenderDialog ( ) ;
if ( m_FinalRenderDialog )
m_FinalRenderDialog - > Show ( true ) ; //Show with a bool specifying that it came from the sequence generator.
}
}
/// <summary>
/// Animate the sequence
/// </summary>
template < typename T >
void FractoriumEmberController < T > : : SequenceAnimateNextFrame ( )
{
const auto tree = m_Fractorium - > ui . SequenceTree ;
if ( const auto renders = tree - > topLevelItem ( 1 ) )
{
if ( renders - > childCount ( ) )
{
const auto animate = dynamic_cast < EmberTreeWidgetItemBase * > ( tree - > topLevelItem ( 0 ) - > child ( 0 ) ) ;
const auto frame = m_AnimateFrame + + % renders - > childCount ( ) ;
const auto nth = dynamic_cast < EmberTreeWidgetItemBase * > ( renders - > child ( frame ) ) ;
if ( animate & & nth )
{
if ( ! nth - > m_Rendered )
{
m_AnimateFrame = 0 ;
}
else
{
animate - > m_Pixmap = QPixmap ( nth - > m_Pixmap ) ;
2023-12-19 15:24:50 -05:00
animate - > setData ( NAME_COL , Qt : : DecorationRole , animate - > m_Pixmap ) ;
2023-04-25 19:59:54 -04:00
}
}
}
}
}
/// <summary>
/// Animate the sequence
/// </summary>
template < typename T >
void FractoriumEmberController < T > : : SequenceAnimateButtonClicked ( )
{
if ( const auto animation = m_Fractorium - > ui . SequenceTree - > topLevelItem ( 0 ) )
{
if ( animation - > isExpanded ( ) & & m_AnimateTimer - > isActive ( ) )
{
animation - > setExpanded ( false ) ;
m_AnimateTimer - > stop ( ) ;
}
else
{
animation - > setExpanded ( true ) ;
m_AnimateFrame = 0 ;
m_AnimateTimer - > start ( 1000 / m_Fractorium - > ui . SequenceAnimationFpsSpinBox - > value ( ) ) ;
}
}
}
void Fractorium : : OnSequenceAnimateButtonClicked ( bool checked ) { m_Controller - > SequenceAnimateButtonClicked ( ) ; }
/// <summary>
/// Clear the sequence.
/// </summary>
template < typename T >
void FractoriumEmberController < T > : : SequenceClearButtonClicked ( )
{
m_SequencePreviewRenderer - > Stop ( ) ;
m_Fractorium - > ui . SequenceTree - > clear ( ) ;
}
void Fractorium : : OnSequenceClearButtonClicked ( bool checked ) { m_Controller - > SequenceClearButtonClicked ( ) ; }
/// <summary>
/// Save the sequence to a file.
/// </summary>
template < typename T >
void FractoriumEmberController < T > : : SequenceSaveButtonClicked ( )
{
auto s = m_Fractorium - > m_Settings ;
QString filename = m_Fractorium - > SetupSaveXmlDialog ( m_SequenceFile . m_Filename ) ;
if ( filename ! = " " )
{
EmberToXml < T > writer ;
QFileInfo fileInfo ( filename ) ;
for ( auto & ember : m_SequenceFile . m_Embers )
ApplyXmlSavingTemplate ( ember ) ;
if ( writer . Save ( filename . toStdString ( ) . c_str ( ) , m_SequenceFile . m_Embers , 0 , true , true , false , false , false ) )
s - > SaveFolder ( fileInfo . canonicalPath ( ) ) ;
else
m_Fractorium - > ShowCritical ( " Save Failed " , " Could not save sequence file, try saving to a different folder. " ) ;
}
}
void Fractorium : : OnSequenceSaveButtonClicked ( bool checked ) { m_Controller - > SequenceSaveButtonClicked ( ) ; }
/// <summary>
/// Open one or more sequence file, concatenate them all, place them in the sequence
/// tree and begin rendering the sequence previews.
/// </summary>
template < typename T >
void FractoriumEmberController < T > : : SequenceOpenButtonClicked ( )
{
m_SequencePreviewRenderer - > Stop ( ) ;
auto filenames = m_Fractorium - > SetupOpenXmlDialog ( ) ;
if ( ! filenames . empty ( ) )
{
size_t i ;
EmberFile < T > emberFile ;
XmlToEmber < T > parser ;
vector < Ember < T > > embers ;
vector < string > errors ;
emberFile . m_Filename = filenames [ 0 ] ;
for ( auto & filename : filenames )
{
embers . clear ( ) ;
if ( parser . Parse ( filename . toStdString ( ) . c_str ( ) , embers , true ) & & ! embers . empty ( ) )
{
for ( i = 0 ; i < embers . size ( ) ; i + + )
if ( embers [ i ] . m_Name = = " " | | embers [ i ] . m_Name = = " No name " ) //Ensure it has a name.
embers [ i ] . m_Name = ToString < qulonglong > ( i ) . toStdString ( ) ;
emberFile . m_Embers . insert ( emberFile . m_Embers . end ( ) , embers . begin ( ) , embers . end ( ) ) ;
errors = parser . ErrorReport ( ) ;
}
else
{
errors = parser . ErrorReport ( ) ;
m_Fractorium - > ShowCritical ( " Open Failed " , " Could not open sequence file, see info tab for details. " ) ;
}
if ( ! errors . empty ( ) )
m_Fractorium - > ErrorReportToQTextEdit ( errors , m_Fractorium - > ui . InfoFileOpeningTextEdit , false ) ; //Concat errors from all files.
}
if ( emberFile . Size ( ) > 0 ) //Ensure at least something was read.
{
emberFile . MakeNamesUnique ( ) ;
m_SequenceFile = std : : move ( emberFile ) ; //Move the temp to avoid creating dupes because we no longer need it.
FillSequenceTree ( ) ;
}
}
}
void Fractorium : : OnSequenceOpenButtonClicked ( bool checked ) { m_Controller - > SequenceOpenButtonClicked ( ) ; }
/// <summary>
/// Constrain all min/max spinboxes.
/// </summary>
2024-03-16 12:15:51 -04:00
void Fractorium : : OnSequenceStartFlameSpinBoxChanged ( int d ) { ui . SequenceStopFlameSpinBox - > setMinimum ( d ) ; }
void Fractorium : : OnSequenceStopFlameSpinBoxChanged ( int d ) { if ( ui . SequenceStopFlameSpinBox - > hasFocus ( ) ) ui . SequenceStartFlameSpinBox - > setMaximum ( d ) ; }
2023-04-25 19:59:54 -04:00
/// <summary>
/// Save all sequence settings to match the values in the controls.
/// </summary>
void Fractorium : : SyncSequenceSettings ( )
{
m_Settings - > AnimationFps ( ui . SequenceAnimationFpsSpinBox - > value ( ) ) ;
}
template class FractoriumEmberController < float > ;
# ifdef DO_DOUBLE
template class FractoriumEmberController < double > ;
# endif