#pragma once #include "EmberFile.h" #include "DoubleSpinBox.h" #include "GLEmberController.h" /// /// FractoriumEmberControllerBase and FractoriumEmberController classes. /// /// /// An enum representing the type of edit being done. /// enum class eEditUndoState : et { REGULAR_EDIT, UNDO_REDO, EDIT_UNDO }; /// /// An enum representing which xforms an update should be applied to. /// enum class eXformUpdate : et { UPDATE_SPECIFIC, UPDATE_CURRENT, UPDATE_SELECTED, UPDATE_CURRENT_AND_SELECTED, UPDATE_SELECTED_EXCEPT_FINAL, UPDATE_ALL, UPDATE_ALL_EXCEPT_FINAL }; /// /// An enum representing the type of synchronizing to do between the list of Embers kept in memory /// and the widgets in the library tree. /// enum class eLibraryUpdate { INDEX = 1, NAME = 2, POINTER = 4 }; /// /// FractoriumEmberController and Fractorium need each other, but each can't include the other. /// So Fractorium includes this file, and Fractorium is declared as a forward declaration here. /// class Fractorium; template class PreviewRenderer; template class TreePreviewRenderer; #define PREVIEW_SIZE 128 #define UNDO_SIZE 512 /// /// FractoriumEmberControllerBase serves as a non-templated base class with virtual /// functions which will be overridden in a derived class that takes a template parameter. /// The controller serves as a way to access both the Fractorium GUI and the underlying ember /// objects through an interface that doesn't require template argument, but does allow /// templated objects to be used underneath. /// Note that there are a few functions which access a templated object, so for those both /// versions for float and double must be provided, then overridden in the templated derived /// class. It's definitely a design flaw, but C++ doesn't offer any alternative since /// templated virtual functions are not supported. /// The functions not implemented in this file can be found in the GUI files which use them. /// class FractoriumEmberControllerBase : public RenderCallback { friend Fractorium; public: FractoriumEmberControllerBase(Fractorium* fractorium); FractoriumEmberControllerBase(const FractoriumEmberControllerBase& controller) = delete; virtual ~FractoriumEmberControllerBase(); //Embers. virtual void SetEmber(const Ember& ember, bool verbatim, bool updatePointer) { } virtual void CopyEmber(Ember& ember, std::function& ember)> perEmberOperation/* = [&](Ember& ember) { }*/) { }//Uncomment default lambdas once LLVM fixes a crash in their compiler with default lambda parameters.//TODO virtual void SetEmberFile(const EmberFile& emberFile, bool move) { } virtual void CopyEmberFile(EmberFile& emberFile, bool sequence, std::function& ember)> perEmberOperation/* = [&](Ember& ember) { }*/) { } virtual void SetTempPalette(const Palette& palette) { } virtual void CopyTempPalette(Palette& palette) { } #ifdef DO_DOUBLE virtual void SetEmber(const Ember& ember, bool verbatim, bool updatePointer) { } virtual void CopyEmber(Ember& ember, std::function& ember)> perEmberOperation/* = [&](Ember& ember) { }*/) { } virtual void SetEmberFile(const EmberFile& emberFile, bool move) { } virtual void CopyEmberFile(EmberFile& emberFile, bool sequence, std::function& ember)> perEmberOperation/* = [&](Ember& ember) { }*/) { } virtual void SetTempPalette(const Palette& palette) { } virtual void CopyTempPalette(Palette& palette) { } #endif virtual void SetEmber(size_t index, bool verbatim) { } virtual void AddXform() { } virtual void AddLinkedXform() { } virtual void DuplicateXform() { } virtual void ClearXform() { } virtual void DeleteXforms() { } virtual void AddFinalXform() { } virtual bool UseFinalXform() const noexcept { return false; } virtual size_t XformCount() const noexcept { return 0; } virtual size_t TotalXformCount() const noexcept { return 0; } virtual QString Name() const { return ""; } virtual void Name(const string& s) { } virtual size_t FinalRasW() const noexcept { return 0; } virtual void FinalRasW(size_t w) noexcept { } virtual size_t FinalRasH() const noexcept { return 0; } virtual void FinalRasH(size_t h) noexcept { } virtual size_t Index() const noexcept { return 0; } virtual void AddSymmetry(int sym, QTIsaac& rand) { } virtual void CalcNormalizedWeights() { } //Menu. virtual void NewFlock(size_t count) { }//File. virtual void NewEmptyFlameInCurrentFile() { } virtual void NewRandomFlameInCurrentFile() { } virtual void CopyFlameInCurrentFile() { } virtual void CreateReferenceFile() { } virtual void OpenAndPrepFiles(const QStringList& filenames, bool append) { } virtual void SaveCurrentAsXml(QString filename = "") { } virtual void SaveEntireFileAsXml() { } virtual uint SaveCurrentToOpenedFile(bool render = true) { return 0; } virtual void SaveCurrentFileOnShutdown() { } virtual void Undo() { }//Edit. virtual void Redo() { } virtual void CopyXml() { } virtual void CopyAllXml() { } virtual void PasteXmlAppend() { } virtual void PasteXmlOver() { } virtual void CopySelectedXforms() { } virtual void PasteSelectedXforms() { } virtual void CopyKernel() { } virtual void AddReflectiveSymmetry() { }//Tools. virtual void AddRotationalSymmetry() { } virtual void AddBothSymmetry() { } virtual void Flatten() { } virtual void Unflatten() { } virtual void ClearFlame() { } //Toolbar. //Library. virtual void SyncLibrary(eLibraryUpdate update) { } virtual void FillLibraryTree(int selectIndex = -1) { } virtual void UpdateLibraryTree() { } virtual void EmberTreeItemChanged(QTreeWidgetItem* item, int col) { } virtual void EmberTreeItemDoubleClicked(QTreeWidgetItem* item, int col) { } virtual void RenderLibraryPreviews(uint start = UINT_MAX, uint end = UINT_MAX) { } virtual void RenderSequencePreviews(uint start = UINT_MAX, uint end = UINT_MAX) { } virtual void SequenceTreeItemChanged(QTreeWidgetItem* item, int col) { } virtual void StopLibraryPreviewRender() { } virtual void StopSequencePreviewRender() { } virtual void StopAllPreviewRenderers() { } virtual void MoveLibraryItems(const QModelIndexList& items, int destRow) { } virtual void Delete(const vector>& v) { } virtual void FillSequenceTree() { } virtual void AddAnimationItem() { } virtual void SequenceGenerateButtonClicked() { } virtual void SequenceSaveButtonClicked() { } virtual void SequenceOpenButtonClicked() { } virtual void SequenceAnimateButtonClicked() { } virtual void SequenceAnimateNextFrame() { } virtual void SequenceClearButtonClicked() { } //Params. virtual void ParamsToEmber(Ember& ember, bool imageParamsOnly = false) { }; #ifdef DO_DOUBLE virtual void ParamsToEmber(Ember& ember, bool imageParamsOnly = false) { }; #endif virtual void SetCenter(double x, double y) { } virtual void FillParamTablesAndPalette() { } virtual void BrightnessChanged(double d) { } virtual void GammaChanged(double d) { } virtual void GammaThresholdChanged(double d) { } virtual void VibrancyChanged(double d) { } virtual void HighlightPowerChanged(double d) { } virtual void K2Changed(double d) { } virtual void PaletteModeChanged(uint i) { } virtual void WidthChanged(uint i) { } virtual void HeightChanged(uint i) { } virtual void ResizeAndScale(int width, int height, eScaleType scaleType) { } virtual void CenterXChanged(double d) { } virtual void CenterYChanged(double d) { } virtual void ScaleChanged(double d) { } virtual void ZoomChanged(double d) { } virtual void RotateChanged(double d) { } virtual void ZPosChanged(double d) { } virtual void PerspectiveChanged(double d) { } virtual void PitchChanged(double d) { } virtual void YawChanged(double d) { } virtual void DepthBlurChanged(double d) { } virtual void BlurCurveChanged(double d) { } virtual void SpatialFilterWidthChanged(double d) { } virtual void SpatialFilterTypeChanged(const QString& text) { } virtual void TemporalFilterWidthChanged(double d) { } virtual void TemporalFilterTypeChanged(const QString& text) { } virtual void DEFilterMinRadiusWidthChanged(double d) { } virtual void DEFilterMaxRadiusWidthChanged(double d) { } virtual void DEFilterCurveWidthChanged(double d) { } virtual void SbsChanged(int d) { } virtual void FuseChanged(int d) { } virtual void RandRangeChanged(double d) { } virtual void QualityChanged(double d) { } virtual void SupersampleChanged(int d) { } virtual void AffineInterpTypeChanged(int i) { } virtual void InterpTypeChanged(int i) { } virtual void BackgroundChanged(const QColor& color) { } virtual void ClearColorCurves(int i) { } virtual void ColorCurveChanged(int curveIndex, int pointInxed, const QPointF& point) { } virtual void ColorCurvesPointAdded(size_t curveIndex, const QPointF& point) { } virtual void ColorCurvesPointRemoved(size_t curveIndex, int pointIndex) { } virtual void ExpChanged(double d) { } //Xforms. virtual void CurrentXformComboChanged(int index) { } virtual void XformWeightChanged(double d) { } virtual void EqualizeWeights() { } virtual void XformNameChanged(const QString& s) { } virtual void XformAnimateChanged(int state) { } virtual void FillXforms(int index = 0) { } virtual void UpdateXformName(int index) { } //Xforms Affine. virtual void AffineSetHelper(double d, int index, bool pre) { } virtual void FlipXforms(bool horizontal, bool vertical, bool pre) { } virtual void RotateXformsByAngle(double angle, bool pre) { } virtual void MoveXforms(double x, double y, bool pre) { } virtual void ScaleXforms(double scale, bool pre) { } virtual void ResetXformsAffine(bool pre) { } virtual void CopyXformsAffine(bool pre) { } virtual void PasteXformsAffine(bool pre) { } virtual void RandomXformsAffine(bool pre) { } virtual void FillBothAffines() { } virtual void SwapAffines() { } double LockedScale() const noexcept { return m_LockedScale; } double LockedX() const noexcept { return m_LockedX; } double LockedY() const noexcept { return m_LockedY; } void LockedScale(double scale) noexcept { m_LockedScale = scale; } virtual void InitLockedScale() noexcept { } virtual double AffineScaleCurrentToLocked() noexcept { return 0; }; virtual double AffineScaleLockedToCurrent() noexcept { return 0; }; //Xforms Color. virtual void XformColorIndexChanged(double d, bool updateRender, bool updateSpinner, bool updateScroll, eXformUpdate update = eXformUpdate::UPDATE_SELECTED, size_t index = 0) { } virtual void RandomColorIndicesButtonClicked() { } virtual void ToggleColorIndicesButtonClicked() { } virtual void RandomColorSpeedButtonClicked() { } virtual void ToggleColorSpeedsButtonClicked() { } virtual void XformColorSpeedChanged(double d) { } virtual void XformOpacityChanged(double d) { } virtual void XformDirectColorChanged(double d) { } virtual void SoloXformCheckBoxStateChanged(int state, int index) { } virtual QColor ColorIndexToQColor(double d) { return QColor(); } //Xforms Variations. virtual void Filter(const QString& text) { } virtual void SetupVariationsTree() { } virtual void ClearVariationsTree() { } virtual void VariationSpinBoxValueChanged(double d) { } virtual void FilteredVariations() { } virtual void FillVariationTreeWithCurrentXform() { } //Xforms Selection. virtual QString MakeXformCaption(size_t i) { return ""; } //Xaos. virtual void FillXaos() { } virtual void FillAppliedXaos() { } virtual void XaosChanged(int x, int y, double val) { } virtual void ClearXaos() { } virtual void RandomXaos() { } virtual void AddLayer(int xforms) { } virtual void TransposeXaos() { } //Palette. virtual size_t InitPaletteList(const QString& s) { return 0; } virtual bool FillPaletteTable(const string& s) { return false; } virtual void ApplyPaletteToEmber() { } virtual void PaletteAdjust() { } virtual void PaletteCellClicked(int row, int col) { } virtual void SetBasePaletteAndAdjust(const Palette& palette) { } virtual void PaletteEditorButtonClicked() { } virtual void PaletteEditorColorChanged() { } virtual void SyncPalette(bool accepted) { } QImage& FinalPaletteImage() { return m_FinalPaletteImage; } //Info. virtual void FillSummary() { } virtual void ReorderVariations(QTreeWidgetItem* item) { } //Rendering/progress. virtual bool Render() { return false; } virtual bool CreateRenderer(eRendererType renderType, const vector>& devices, bool updatePreviews, bool shared = true) { return false; } virtual uint SizeOfT() const noexcept { return 0; } virtual void ClearUndo() { } virtual void DeleteRenderer() { } virtual GLEmberControllerBase* GLController() { return nullptr; } virtual void Pause(bool pause); virtual bool Paused(); bool RenderTimerRunning(); void StartRenderTimer(); void DelayedStartRenderTimer(); void StopRenderTimer(bool wait); void ClearFinalImages(); void Shutdown(); void UpdateRender(eProcessAction action = eProcessAction::FULL_RENDER); bool SaveCurrentRender(const QString& filename, const EmberImageComments& comments, vector& pixels, size_t width, size_t height, bool png16Bit, bool transparency); RendererBase* Renderer() { return m_Renderer.get(); } vector* FinalImage() { return &(m_FinalImage); } vector* PreviewFinalImage() { return &m_PreviewFinalImage; } EmberStats Stats() { return m_Stats; } eProcessState ProcessState() { return m_Renderer.get() ? m_Renderer->ProcessState() : eProcessState::NONE; } protected: //Rendering/progress. void AddProcessAction(eProcessAction action); eProcessAction CondenseAndClearProcessActions(); //Non-templated members. bool m_Rendering = false; bool m_LastEditWasUndoRedo; vector> m_Devices; size_t m_SubBatchCount = 1;//Will be ovewritten by the options on first render. uint m_FailedRenders = 0; size_t m_UndoIndex = 0; double m_LockedScale = 1; double m_LockedX = 0; double m_LockedY = 0; eRendererType m_RenderType = eRendererType::CPU_RENDERER; eEditUndoState m_EditState; GLuint m_OutputTexID = 0; Timing m_RenderElapsedTimer; EmberStats m_Stats; QImage m_FinalPaletteImage; QString m_LastSaveAll; QString m_LastSaveCurrent; string m_CurrentPaletteFilePath; std::recursive_mutex m_Cs; std::thread m_WriteThread; vector m_FinalImage; vector m_PreviewFinalImage; vector m_ProcessActions; vector m_FilteredVariations; unique_ptr m_Renderer; QTIsaac m_Rand; Fractorium* m_Fractorium; Palette m_TempPalette, m_PreviousTempPalette; std::unique_ptr m_RenderTimer; std::unique_ptr m_RenderRestartTimer; std::unique_ptr m_AnimateTimer; shared_ptr> m_PaletteList; shared_ptr m_Info = OpenCLInfo::Instance(); int m_AnimateFrame = 0; }; /// /// Templated derived class which implements all interaction functionality between the embers /// of a specific template type and the GUI. /// Switching between template arguments requires complete re-creation of the controller and the /// underlying renderer. Switching between CPU and OpenCL only requires re-creation of the renderer. /// template class FractoriumEmberController : public FractoriumEmberControllerBase { friend PreviewRenderer; friend TreePreviewRenderer; public: FractoriumEmberController(Fractorium* fractorium); FractoriumEmberController(const FractoriumEmberController& controller) = delete; virtual ~FractoriumEmberController(); //Embers. void SetEmber(const Ember& ember, bool verbatim, bool updatePointer) override; void CopyEmber(Ember& ember, std::function& ember)> perEmberOperation/* = [&](Ember& ember) { }*/) override; void SetEmberFile(const EmberFile& emberFile, bool move) override; void CopyEmberFile(EmberFile& emberFile, bool sequence, std::function& ember)> perEmberOperation/* = [&](Ember& ember) { }*/) override; void SetTempPalette(const Palette& palette) override; void CopyTempPalette(Palette& palette) override; #ifdef DO_DOUBLE void SetEmber(const Ember& ember, bool verbatim, bool updatePointer) override; void CopyEmber(Ember& ember, std::function& ember)> perEmberOperation/* = [&](Ember& ember) { }*/) override; void SetEmberFile(const EmberFile& emberFile, bool move) override; void CopyEmberFile(EmberFile& emberFile, bool sequence, std::function& ember)> perEmberOperation/* = [&](Ember& ember) { }*/) override; void SetTempPalette(const Palette& palette) override; void CopyTempPalette(Palette& palette) override; #endif void SetEmber(size_t index, bool verbatim) override; void AddXform() override; void AddLinkedXform() override; void DuplicateXform() override; void ClearXform() override; void DeleteXforms() override; void AddFinalXform() override; bool UseFinalXform() const noexcept override { return m_Ember.UseFinalXform(); } size_t XformCount() const noexcept override { return m_Ember.XformCount(); } size_t TotalXformCount() const noexcept override { return m_Ember.TotalXformCount(); } QString Name() const override { return QString::fromStdString(m_Ember.m_Name); } void Name(const string& s) override { m_Ember.m_Name = s; } size_t FinalRasW() const noexcept override { return m_Ember.m_FinalRasW; } void FinalRasW(size_t w) noexcept override { m_Ember.m_FinalRasW = w; } size_t FinalRasH() const noexcept override { return m_Ember.m_FinalRasH; } void FinalRasH(size_t h) noexcept override { m_Ember.m_FinalRasH = h; } size_t Index() const noexcept override { return m_Ember.m_Index; } void AddSymmetry(int sym, QTIsaac& rand) override { m_Ember.AddSymmetry(sym, rand); } void CalcNormalizedWeights() override { m_Ember.CalcNormalizedWeights(m_NormalizedWeights); } void ConstrainDimensions(Ember& ember); Ember* CurrentEmber(); //Menu. void NewFlock(size_t count) override; void NewEmptyFlameInCurrentFile() override; void NewRandomFlameInCurrentFile() override; void CopyFlameInCurrentFile() override; void CreateReferenceFile() override; void OpenAndPrepFiles(const QStringList& filenames, bool append) override; void SaveCurrentAsXml(QString filename = "") override; void SaveEntireFileAsXml() override; uint SaveCurrentToOpenedFile(bool render = true) override; void SaveCurrentFileOnShutdown() override; void Undo() override; void Redo() override; void CopyXml() override; void CopyAllXml() override; void PasteXmlAppend() override; void PasteXmlOver() override; void CopySelectedXforms() override; void PasteSelectedXforms() override; void CopyKernel() override; void AddReflectiveSymmetry() override; void AddRotationalSymmetry() override; void AddBothSymmetry() override; void Flatten() override; void Unflatten() override; void ClearFlame() override; //Toolbar. //Library. void SyncLibrary(eLibraryUpdate update) override; void FillLibraryTree(int selectIndex = -1) override; void UpdateLibraryTree() override; void MoveLibraryItems(const QModelIndexList& items, int destRow) override; void Delete(const vector>& v) override; void EmberTreeItemChanged(QTreeWidgetItem* item, int col) override; void EmberTreeItemDoubleClicked(QTreeWidgetItem* item, int col) override; void RenderPreviews(QTreeWidget* tree, TreePreviewRenderer* renderer, EmberFile& file, uint start = UINT_MAX, uint end = UINT_MAX); void RenderLibraryPreviews(uint start = UINT_MAX, uint end = UINT_MAX) override; void RenderSequencePreviews(uint start = UINT_MAX, uint end = UINT_MAX) override; void SequenceTreeItemChanged(QTreeWidgetItem* item, int col) override; void StopLibraryPreviewRender() override; void StopSequencePreviewRender() override; void StopAllPreviewRenderers() override; void FillSequenceTree() override; void AddAnimationItem() override; void SequenceGenerateButtonClicked() override; void SequenceSaveButtonClicked() override; void SequenceOpenButtonClicked() override; void SequenceAnimateButtonClicked() override; void SequenceAnimateNextFrame() override; void SequenceClearButtonClicked() override; //Params. void ParamsToEmber(Ember& ember, bool imageParamsOnly = false) override; #ifdef DO_DOUBLE void ParamsToEmber(Ember& ember, bool imageParamsOnly = false) override; #endif void SetCenter(double x, double y) override; void FillParamTablesAndPalette() override; void BrightnessChanged(double d) override; void GammaChanged(double d) override; void GammaThresholdChanged(double d) override; void VibrancyChanged(double d) override; void HighlightPowerChanged(double d) override; void K2Changed(double d) override; void PaletteModeChanged(uint i) override; void WidthChanged(uint i) override; void HeightChanged(uint i) override; void ResizeAndScale(int width, int height, eScaleType scaleType) override; void CenterXChanged(double d) override; void CenterYChanged(double d) override; void ScaleChanged(double d) override; void ZoomChanged(double d) override; void RotateChanged(double d) override; void ZPosChanged(double d) override; void PerspectiveChanged(double d) override; void PitchChanged(double d) override; void YawChanged(double d) override; void DepthBlurChanged(double d) override; void BlurCurveChanged(double d) override; void SpatialFilterWidthChanged(double d) override; void SpatialFilterTypeChanged(const QString& text) override; void TemporalFilterWidthChanged(double d) override; void TemporalFilterTypeChanged(const QString& text) override; void DEFilterMinRadiusWidthChanged(double d) override; void DEFilterMaxRadiusWidthChanged(double d) override; void DEFilterCurveWidthChanged(double d) override; void SbsChanged(int d) override; void FuseChanged(int d) override; void RandRangeChanged(double d) override; void QualityChanged(double d) override; void SupersampleChanged(int d) override; void AffineInterpTypeChanged(int index) override; void InterpTypeChanged(int index) override; void BackgroundChanged(const QColor& col) override; void ClearColorCurves(int i) override; void ColorCurveChanged(int curveIndex, int pointInxed, const QPointF& point) override; void ColorCurvesPointAdded(size_t curveIndex, const QPointF& point) override; void ColorCurvesPointRemoved(size_t curveIndex, int pointIndex) override; void ExpChanged(double d) override; //Xforms. void CurrentXformComboChanged(int index) override; void XformWeightChanged(double d) override; void EqualizeWeights() override; void XformNameChanged(const QString& s) override; void XformAnimateChanged(int state) override; void FillXforms(int index = 0) override; void UpdateXformName(int index) override; void FillWithXform(Xform* xform); Xform* CurrentXform(); void UpdateXform(std::function*, size_t, size_t)> func, eXformUpdate updateType = eXformUpdate::UPDATE_CURRENT, bool updateRender = true, eProcessAction action = eProcessAction::FULL_RENDER, size_t index = 0); static void AddXformsWithXaos(Ember& ember, std::vector, size_t>>& xforms, eXaosPasteStyle pastestyle); //Xforms Affine. void AffineSetHelper(double d, int index, bool pre) override; void FlipXforms(bool horizontal, bool vertical, bool pre) override; void RotateXformsByAngle(double angle, bool pre) override; void MoveXforms(double x, double y, bool pre) override; void ScaleXforms(double scale, bool pre) override; void ResetXformsAffine(bool pre) override; void CopyXformsAffine(bool pre) override; void PasteXformsAffine(bool pre) override; void RandomXformsAffine(bool pre) override; void FillBothAffines() override; void SwapAffines() override; void InitLockedScale() noexcept override; double AffineScaleCurrentToLocked() noexcept override; double AffineScaleLockedToCurrent() noexcept override; void FillAffineWithXform(Xform* xform, bool pre); void ChangeLockedScale(T value); //Xforms Color. void XformColorIndexChanged(double d, bool updateRender, bool updateSpinner, bool updateScroll, eXformUpdate update = eXformUpdate::UPDATE_SELECTED, size_t index = 0) override; void RandomColorIndicesButtonClicked() override; void ToggleColorIndicesButtonClicked() override; void RandomColorSpeedButtonClicked() override; void ToggleColorSpeedsButtonClicked() override; void XformColorSpeedChanged(double d) override; void XformOpacityChanged(double d) override; void XformDirectColorChanged(double d) override; void SoloXformCheckBoxStateChanged(int state, int index) override; QColor ColorIndexToQColor(double d) override; void FillColorWithXform(Xform* xform); //Xforms Variations. void Filter(const QString& text) override; void SetupVariationsTree() override; void ClearVariationsTree() override; void VariationSpinBoxValueChanged(double d) override; void FilteredVariations() override; void FillVariationTreeWithCurrentXform() override; void FillVariationTreeWithXform(Xform* xform); //Xforms Xaos. void FillXaos() override; void FillAppliedXaos() override; void XaosChanged(int x, int y, double val) override; void ClearXaos() override; void RandomXaos() override; void AddLayer(int xforms) override; void TransposeXaos() override; //Xforms Selection. virtual QString MakeXformCaption(size_t i) override; bool XformCheckboxAt(int i, std::function func); bool XformCheckboxAt(Xform* xform, std::function func); //Palette. size_t InitPaletteList(const QString& s) override; bool FillPaletteTable(const string& s) override; void ApplyPaletteToEmber() override; void PaletteAdjust() override; void PaletteCellClicked(int row, int col) override; void SetBasePaletteAndAdjust(const Palette& palette) override; void PaletteEditorButtonClicked() override; void PaletteEditorColorChanged() override; void SyncPalette(bool accepted) override; //Info. void FillSummary() override; void ReorderVariations(QTreeWidgetItem* item) override; //Rendering/progress. bool Render() override; bool CreateRenderer(eRendererType renderType, const vector>& devices, bool updatePreviews, bool shared = true) override; uint SizeOfT() const noexcept override { return sizeof(T); } int ProgressFunc(Ember& ember, void* foo, double fraction, int stage, double etaMs) override; void ClearUndo() override; GLEmberControllerBase* GLController() override { return m_GLController.get(); } void DeleteRenderer() override; private: //Embers. void ApplyXmlSavingTemplate(Ember& ember); template void SetEmberPrivate(const Ember& ember, bool verbatim, bool updatePointer); //Params. template void ParamsToEmberPrivate(Ember& ember, bool imageParamsOnly); //Xforms. void SetNormalizedWeightText(Xform* xform); bool IsFinal(Xform* xform); //Xforms Color. void FillCurvesControl(); //Palette. void UpdateAdjustedPaletteGUI(Palette& palette); //Rendering/progress. void Update(std::function func, bool updateRender = true, eProcessAction action = eProcessAction::FULL_RENDER); void UpdateAll(std::function&, bool)> func, bool updateRender = true, eProcessAction action = eProcessAction::FULL_RENDER, bool applyAll = false); bool SyncSizes(); //Templated members. bool m_PreviewRunning = false; vector m_TempOpacities; vector m_NormalizedWeights; Ember m_Ember; Ember* m_EmberFilePointer = nullptr; EmberFile m_EmberFile; EmberFile m_SequenceFile; deque> m_UndoList; vector, size_t>> m_CopiedXforms; Xform m_CopiedFinalXform; Affine2D m_CopiedAffine; shared_ptr> m_VariationList; unique_ptr> m_SheepTools; unique_ptr> m_GLController; unique_ptr> m_LibraryPreviewRenderer; unique_ptr> m_SequencePreviewRenderer; }; /// /// Base class for encapsulating a preview renderer which will be used /// in such places as the main library tree, the sequence tree and the /// single preview thumbnail shown in the final render dialog. /// Derived classes will implement PreviewRenderFunc() to handle the rendering /// functionality specific to their previews. /// template class PreviewRenderer { public: PreviewRenderer() { } virtual ~PreviewRenderer() { } void Render(uint start, uint end) { Stop(); m_PreviewResult = QtConcurrent::run([&](uint s, uint e) { rlg l(m_PreviewCs); m_PreviewRun = true; PreviewRenderFunc(s, e); m_PreviewRun = false; }, start, end); } void Stop() { m_PreviewRun = false; m_PreviewRenderer.Abort(); m_PreviewResult.waitForFinished(); } bool EarlyClip() { return m_PreviewRenderer.EarlyClip(); } bool YAxisUp() { return m_PreviewRenderer.YAxisUp(); } bool Running() { return m_PreviewRun || m_PreviewResult.isRunning(); } virtual void PreviewRenderFunc(uint start, uint end) {} protected: volatile bool m_PreviewRun = false; Ember m_PreviewEmber; vector m_PreviewVec; vv4F m_PreviewFinalImage; EmberNs::Renderer m_PreviewRenderer; private: QFuture m_PreviewResult; std::recursive_mutex m_PreviewCs; }; /// /// Thin derivation to handle preview rendering multiple embers previews to a tree. /// template class TreePreviewRenderer : public PreviewRenderer { public: using PreviewRenderer::m_PreviewRun; using PreviewRenderer::m_PreviewEmber; using PreviewRenderer::m_PreviewVec; using PreviewRenderer::m_PreviewRenderer; using PreviewRenderer::m_PreviewFinalImage; /// /// Initializes a new instance of the class. /// /// A pointer to the controller this instance is a member of /// A pointer to the tree to render to /// A reference to the ember file to render TreePreviewRenderer(FractoriumEmberController* controller, QTreeWidget* tree, EmberFile& emberFile) : m_Controller(controller), m_Tree(tree), m_EmberFile(emberFile) { const auto f = m_Controller->m_Fractorium; m_PreviewRenderer.Callback(nullptr); m_PreviewRenderer.EarlyClip(f->m_Settings->EarlyClip()); m_PreviewRenderer.YAxisUp(f->m_Settings->YAxisUp()); } void PreviewRenderFunc(uint start, uint end) override; protected: FractoriumEmberController* m_Controller; QTreeWidget* m_Tree; EmberFile& m_EmberFile; };