// Copyright 1998-2019 Epic Games, Inc. All Rights Reserved. #include "LevelViewportLayoutFourPanes.h" #include "Framework/Docking/LayoutService.h" #include "Editor.h" #include "Misc/ConfigCacheIni.h" #include "Modules/ModuleManager.h" #include "Framework/Application/SlateApplication.h" #include "LevelEditor.h" namespace ViewportLayoutFourPanesDefs { /** Default main splitter to equal 50/50 split */ static const float DefaultPrimarySplitterPercentage = 0.5f; /** Default secondary splitter to equal three-way split */ static const float DefaultSecondarySplitterPercentage = 0.333f; } // FLevelViewportLayoutFourPanes ///////////////////////////// TSharedRef FLevelViewportLayoutFourPanes::MakeViewportLayout(const FString& LayoutString) { FString SpecificLayoutString = GetTypeSpecificLayoutString(LayoutString); FString ViewportKey0, ViewportKey1, ViewportKey2, ViewportKey3; FString ViewportType0 = TEXT("Default"), ViewportType1 = TEXT("Default"), ViewportType2 = TEXT("Default"), ViewportType3 = TEXT("Default"); float PrimarySplitterPercentage = ViewportLayoutFourPanesDefs::DefaultPrimarySplitterPercentage; float SecondarySplitterPercentage0 = ViewportLayoutFourPanesDefs::DefaultSecondarySplitterPercentage; float SecondarySplitterPercentage1 = ViewportLayoutFourPanesDefs::DefaultSecondarySplitterPercentage; if (!SpecificLayoutString.IsEmpty()) { // The Layout String only holds the unique ID of the Additional Layout Configs to use const FString& IniSection = FLayoutSaveRestore::GetAdditionalLayoutConfigIni(); ViewportKey0 = SpecificLayoutString + TEXT(".Viewport0"); ViewportKey1 = SpecificLayoutString + TEXT(".Viewport1"); ViewportKey2 = SpecificLayoutString + TEXT(".Viewport2"); ViewportKey3 = SpecificLayoutString + TEXT(".Viewport3"); GConfig->GetString(*IniSection, *(ViewportKey0 + TEXT(".TypeWithinLayout")), ViewportType0, GEditorPerProjectIni); GConfig->GetString(*IniSection, *(ViewportKey1 + TEXT(".TypeWithinLayout")), ViewportType1, GEditorPerProjectIni); GConfig->GetString(*IniSection, *(ViewportKey2 + TEXT(".TypeWithinLayout")), ViewportType2, GEditorPerProjectIni); GConfig->GetString(*IniSection, *(ViewportKey3 + TEXT(".TypeWithinLayout")), ViewportType3, GEditorPerProjectIni); FString PercentageString; if(GConfig->GetString(*IniSection, *(SpecificLayoutString + TEXT(".Percentage0")), PercentageString, GEditorPerProjectIni)) { TTypeFromString::FromString(PrimarySplitterPercentage, *PercentageString); } if(GConfig->GetString(*IniSection, *(SpecificLayoutString + TEXT(".Percentage1")), PercentageString, GEditorPerProjectIni)) { TTypeFromString::FromString(SecondarySplitterPercentage0, *PercentageString); } if(GConfig->GetString(*IniSection, *(SpecificLayoutString + TEXT(".Percentage2")), PercentageString, GEditorPerProjectIni)) { TTypeFromString::FromString(SecondarySplitterPercentage1, *PercentageString); } } FLevelEditorModule& LevelEditor = FModuleManager::LoadModuleChecked("LevelEditor"); // Set up the viewports FViewportConstructionArgs Args; Args.ParentLayout = AsShared(); Args.ParentLevelEditor = ParentLevelEditor; Args.IsEnabled = FSlateApplication::Get().GetNormalExecutionAttribute(); Args.bRealtime = !FPlatformMisc::IsRemoteSession(); Args.ConfigKey = *ViewportKey0; Args.ViewportType = LVT_Perspective; TSharedRef Viewport0 = LevelEditor.FactoryViewport(*ViewportType0, Args); Args.bRealtime = false; Args.ConfigKey = *ViewportKey1; Args.ViewportType = LVT_OrthoXY; TSharedRef Viewport1 = LevelEditor.FactoryViewport(*ViewportType1, Args); // Front viewport Args.bRealtime = false; Args.ConfigKey = *ViewportKey2; Args.ViewportType = LVT_OrthoXZ; TSharedRef Viewport2 = LevelEditor.FactoryViewport(*ViewportType2, Args); // Top Viewport Args.bRealtime = false; Args.ConfigKey = *ViewportKey3; Args.ViewportType = LVT_OrthoYZ; TSharedRef Viewport3 = LevelEditor.FactoryViewport(*ViewportType2, Args); Viewports.Add( *ViewportKey0, Viewport0 ); Viewports.Add( *ViewportKey1, Viewport1 ); Viewports.Add( *ViewportKey2, Viewport2 ); Viewports.Add( *ViewportKey3, Viewport3 ); // Make newly-created perspective viewports active by default GCurrentLevelEditingViewportClient = &Viewport0->GetLevelViewportClient(); TSharedRef LayoutWidget = MakeFourPanelWidget( Viewports, Viewport0->AsWidget(), Viewport1->AsWidget(), Viewport2->AsWidget(), Viewport3->AsWidget(), PrimarySplitterPercentage, SecondarySplitterPercentage0, SecondarySplitterPercentage1); InitCommonLayoutFromString(SpecificLayoutString, *ViewportKey1); return LayoutWidget; } void FLevelViewportLayoutFourPanes::SaveLayoutString(const FString& LayoutString) const { if (!bIsTransitioning) { FString SpecificLayoutString = GetTypeSpecificLayoutString(LayoutString); const FString& IniSection = FLayoutSaveRestore::GetAdditionalLayoutConfigIni(); check (PrimarySplitterWidget->GetChildren()->Num() == 2); float PrimaryPercentage = PrimarySplitterWidget->SlotAt(0).SizeValue.Get(); check (SecondarySplitterWidget->GetChildren()->Num() == 3); float SecondaryPercentage0 = SecondarySplitterWidget->SlotAt(0).SizeValue.Get(); float SecondaryPercentage1 = SecondarySplitterWidget->SlotAt(1).SizeValue.Get(); GConfig->SetString(*IniSection, *(SpecificLayoutString + TEXT(".Percentage0")), *TTypeToString::ToString(PrimaryPercentage), GEditorPerProjectIni); GConfig->SetString(*IniSection, *(SpecificLayoutString + TEXT(".Percentage1")), *TTypeToString::ToString(SecondaryPercentage0), GEditorPerProjectIni); GConfig->SetString(*IniSection, *(SpecificLayoutString + TEXT(".Percentage2")), *TTypeToString::ToString(SecondaryPercentage1), GEditorPerProjectIni); SaveCommonLayoutString(SpecificLayoutString); } } void FLevelViewportLayoutFourPanes::ReplaceWidget( TSharedRef< SWidget > Source, TSharedRef< SWidget > Replacement ) { bool bWasFound = false; for (int32 SlotIdx = 0; SlotIdx < PrimarySplitterWidget->GetChildren()->Num(); SlotIdx++) { if (PrimarySplitterWidget->GetChildren()->GetChildAt(SlotIdx) == Source) { PrimarySplitterWidget->SlotAt(SlotIdx) [ Replacement ]; bWasFound = true; break; } } for (int32 SlotIdx = 0; SlotIdx < SecondarySplitterWidget->GetChildren()->Num(); SlotIdx++) { if (SecondarySplitterWidget->GetChildren()->GetChildAt(SlotIdx) == Source) { SecondarySplitterWidget->SlotAt(SlotIdx) [ Replacement ]; bWasFound = true; break; } } // Source widget should have already been a content widget for the splitter check( bWasFound ); } // FLevelViewportLayoutFourPanesLeft ///////////////////////////// TSharedRef FLevelViewportLayoutFourPanesLeft::MakeFourPanelWidget( TMap>& ViewportWidgets, TSharedRef Viewport0, TSharedRef Viewport1, TSharedRef Viewport2, TSharedRef Viewport3, float PrimarySplitterPercentage, float SecondarySplitterPercentage0, float SecondarySplitterPercentage1) { TSharedRef Widget = SAssignNew( PrimarySplitterWidget, SSplitter ) .Orientation(EOrientation::Orient_Horizontal) +SSplitter::Slot() .Value(PrimarySplitterPercentage) [ Viewport0 ] +SSplitter::Slot() .Value(1.0f - PrimarySplitterPercentage) [ SAssignNew( SecondarySplitterWidget, SSplitter ) .Orientation(EOrientation::Orient_Vertical) +SSplitter::Slot() .Value(SecondarySplitterPercentage0) [ Viewport1 ] +SSplitter::Slot() .Value(SecondarySplitterPercentage1) [ Viewport2 ] +SSplitter::Slot() .Value(1.0f - SecondarySplitterPercentage0 - SecondarySplitterPercentage1) [ Viewport3 ] ]; return Widget; } // FLevelViewportLayoutFourPanesRight ///////////////////////////// TSharedRef FLevelViewportLayoutFourPanesRight::MakeFourPanelWidget( TMap>& ViewportWidgets, TSharedRef Viewport0, TSharedRef Viewport1, TSharedRef Viewport2, TSharedRef Viewport3, float PrimarySplitterPercentage, float SecondarySplitterPercentage0, float SecondarySplitterPercentage1) { TSharedRef Widget = SAssignNew( PrimarySplitterWidget, SSplitter ) .Orientation(EOrientation::Orient_Horizontal) +SSplitter::Slot() .Value(PrimarySplitterPercentage) [ SAssignNew( SecondarySplitterWidget, SSplitter ) .Orientation(EOrientation::Orient_Vertical) +SSplitter::Slot() .Value(SecondarySplitterPercentage0) [ Viewport1 ] +SSplitter::Slot() .Value(SecondarySplitterPercentage1) [ Viewport2 ] +SSplitter::Slot() .Value(1.0f - SecondarySplitterPercentage0 - SecondarySplitterPercentage1) [ Viewport3 ] ] +SSplitter::Slot() .Value(1.0f - PrimarySplitterPercentage) [ Viewport0 ]; return Widget; } // FLevelViewportLayoutFourPanesTop ///////////////////////////// TSharedRef FLevelViewportLayoutFourPanesTop::MakeFourPanelWidget( TMap>& ViewportWidgets, TSharedRef Viewport0, TSharedRef Viewport1, TSharedRef Viewport2, TSharedRef Viewport3, float PrimarySplitterPercentage, float SecondarySplitterPercentage0, float SecondarySplitterPercentage1) { TSharedRef Widget = SAssignNew( PrimarySplitterWidget, SSplitter ) .Orientation(EOrientation::Orient_Vertical) +SSplitter::Slot() .Value(PrimarySplitterPercentage) [ Viewport0 ] +SSplitter::Slot() .Value(1.0f - PrimarySplitterPercentage) [ SAssignNew( SecondarySplitterWidget, SSplitter ) .Orientation(EOrientation::Orient_Horizontal) +SSplitter::Slot() .Value(SecondarySplitterPercentage0) [ Viewport1 ] +SSplitter::Slot() .Value(SecondarySplitterPercentage1) [ Viewport2 ] +SSplitter::Slot() .Value(1.0f - SecondarySplitterPercentage0 - SecondarySplitterPercentage1) [ Viewport3 ] ]; return Widget; } // FLevelViewportLayoutFourPanesBottom ///////////////////////////// TSharedRef FLevelViewportLayoutFourPanesBottom::MakeFourPanelWidget( TMap>& ViewportWidgets, TSharedRef Viewport0, TSharedRef Viewport1, TSharedRef Viewport2, TSharedRef Viewport3, float PrimarySplitterPercentage, float SecondarySplitterPercentage0, float SecondarySplitterPercentage1) { TSharedRef Widget = SAssignNew( PrimarySplitterWidget, SSplitter ) .Orientation(EOrientation::Orient_Vertical) +SSplitter::Slot() .Value(PrimarySplitterPercentage) [ SAssignNew( SecondarySplitterWidget, SSplitter ) .Orientation(EOrientation::Orient_Horizontal) +SSplitter::Slot() .Value(SecondarySplitterPercentage0) [ Viewport1 ] +SSplitter::Slot() .Value(SecondarySplitterPercentage1) [ Viewport2 ] +SSplitter::Slot() .Value(1.0f - SecondarySplitterPercentage0 - SecondarySplitterPercentage1) [ Viewport3 ] ] +SSplitter::Slot() .Value(1.0f - PrimarySplitterPercentage) [ Viewport0 ]; return Widget; }