mirror of
https://github.com/vczh-libraries/Release.git
synced 2026-02-06 03:42:11 +08:00
40683 lines
1.1 MiB
40683 lines
1.1 MiB
/***********************************************************************
|
|
THIS FILE IS AUTOMATICALLY GENERATED. DO NOT MODIFY
|
|
DEVELOPER: Zihan Chen(vczh)
|
|
***********************************************************************/
|
|
#include "GacUI.h"
|
|
#ifndef VCZH_DEBUG_NO_REFLECTION
|
|
#include "GacUIReflection.h"
|
|
#endif
|
|
|
|
/***********************************************************************
|
|
.\GACUIREFLECTIONHELPER.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace reflection
|
|
{
|
|
namespace description
|
|
{
|
|
using namespace parsing;
|
|
using namespace parsing::tabling;
|
|
using namespace parsing::xml;
|
|
using namespace stream;
|
|
using namespace collections;
|
|
using namespace presentation;
|
|
using namespace presentation::elements;
|
|
using namespace presentation::compositions;
|
|
using namespace presentation::controls;
|
|
using namespace presentation::theme;
|
|
using namespace presentation::templates;
|
|
|
|
/***********************************************************************
|
|
Serialization (Color)
|
|
***********************************************************************/
|
|
|
|
Color TypedValueSerializerProvider<Color>::GetDefaultValue()
|
|
{
|
|
return Color();
|
|
}
|
|
|
|
bool TypedValueSerializerProvider<Color>::Serialize(const Color& input, WString& output)
|
|
{
|
|
output = input.ToString();
|
|
return true;
|
|
}
|
|
|
|
bool TypedValueSerializerProvider<Color>::Deserialize(const WString& input, Color& output)
|
|
{
|
|
output = Color::Parse(input);
|
|
return true;
|
|
}
|
|
|
|
IBoxedValue::CompareResult TypedValueSerializerProvider<Color>::Compare(const presentation::Color& a, const presentation::Color& b)
|
|
{
|
|
return TypedValueSerializerProvider<vuint32_t>::Compare(a.value, b.value);
|
|
}
|
|
|
|
/***********************************************************************
|
|
Serialization (DocumentFontSize)
|
|
***********************************************************************/
|
|
|
|
DocumentFontSize TypedValueSerializerProvider<DocumentFontSize>::GetDefaultValue()
|
|
{
|
|
return DocumentFontSize();
|
|
}
|
|
|
|
bool TypedValueSerializerProvider<DocumentFontSize>::Serialize(const DocumentFontSize& input, WString& output)
|
|
{
|
|
output = input.ToString();
|
|
return true;
|
|
}
|
|
|
|
bool TypedValueSerializerProvider<DocumentFontSize>::Deserialize(const WString& input, DocumentFontSize& output)
|
|
{
|
|
output = DocumentFontSize::Parse(input);
|
|
return true;
|
|
}
|
|
|
|
IBoxedValue::CompareResult TypedValueSerializerProvider<DocumentFontSize>::Compare(const presentation::DocumentFontSize& a, const presentation::DocumentFontSize& b)
|
|
{
|
|
return TypedValueSerializerProvider<WString>::Compare(a.ToString(), b.ToString());
|
|
}
|
|
|
|
/***********************************************************************
|
|
Serialization (GlobalStringKey)
|
|
***********************************************************************/
|
|
|
|
GlobalStringKey TypedValueSerializerProvider<GlobalStringKey>::GetDefaultValue()
|
|
{
|
|
return GlobalStringKey();
|
|
}
|
|
|
|
bool TypedValueSerializerProvider<GlobalStringKey>::Serialize(const GlobalStringKey& input, WString& output)
|
|
{
|
|
output = input.ToString();
|
|
return true;
|
|
}
|
|
|
|
bool TypedValueSerializerProvider<GlobalStringKey>::Deserialize(const WString& input, GlobalStringKey& output)
|
|
{
|
|
output = GlobalStringKey::Get(input);
|
|
return true;
|
|
}
|
|
|
|
IBoxedValue::CompareResult TypedValueSerializerProvider<GlobalStringKey>::Compare(const presentation::GlobalStringKey& a, const presentation::GlobalStringKey& b)
|
|
{
|
|
return TypedValueSerializerProvider<WString>::Compare(a.ToString(), b.ToString());
|
|
}
|
|
|
|
/***********************************************************************
|
|
External Functions (Basic)
|
|
***********************************************************************/
|
|
|
|
Ptr<INativeImage> INativeImage_Constructor(const WString& path)
|
|
{
|
|
return GetCurrentController()->ImageService()->CreateImageFromFile(path);
|
|
}
|
|
|
|
INativeCursor* INativeCursor_Constructor1()
|
|
{
|
|
return GetCurrentController()->ResourceService()->GetDefaultSystemCursor();
|
|
}
|
|
|
|
INativeCursor* INativeCursor_Constructor2(INativeCursor::SystemCursorType type)
|
|
{
|
|
return GetCurrentController()->ResourceService()->GetSystemCursor(type);
|
|
}
|
|
|
|
/***********************************************************************
|
|
External Functions (Elements)
|
|
***********************************************************************/
|
|
|
|
text::TextLines* GuiColorizedTextElement_GetLines(GuiColorizedTextElement* thisObject)
|
|
{
|
|
return &thisObject->GetLines();
|
|
}
|
|
|
|
/***********************************************************************
|
|
External Functions (Compositions)
|
|
***********************************************************************/
|
|
|
|
void GuiTableComposition_SetRows(GuiTableComposition* thisObject, vint value)
|
|
{
|
|
vint columns = thisObject->GetColumns();
|
|
if (columns <= 0) columns = 1;
|
|
thisObject->SetRowsAndColumns(value, columns);
|
|
}
|
|
|
|
void GuiTableComposition_SetColumns(GuiTableComposition* thisObject, vint value)
|
|
{
|
|
vint row = thisObject->GetRows();
|
|
if (row <= 0) row = 1;
|
|
thisObject->SetRowsAndColumns(row, value);
|
|
}
|
|
|
|
void IGuiAltActionHost_CollectAltActions(IGuiAltActionHost* host, List<IGuiAltAction*>& actions)
|
|
{
|
|
Group<WString, IGuiAltAction*> group;
|
|
host->CollectAltActions(group);
|
|
for (vint i = 0; i < group.Count(); i++)
|
|
{
|
|
CopyFrom(actions, group.GetByIndex(i), true);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\CONTROLS\GUIAPPLICATION.CPP
|
|
***********************************************************************/
|
|
|
|
extern void GuiMain();
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace controls
|
|
{
|
|
using namespace collections;
|
|
using namespace compositions;
|
|
using namespace theme;
|
|
using namespace description;
|
|
|
|
/***********************************************************************
|
|
GuiApplication
|
|
***********************************************************************/
|
|
|
|
void GuiApplication::InvokeClipboardNotify(compositions::GuiGraphicsComposition* composition, compositions::GuiEventArgs& arguments)
|
|
{
|
|
if(composition->HasEventReceiver())
|
|
{
|
|
composition->GetEventReceiver()->clipboardNotify.Execute(arguments);
|
|
}
|
|
FOREACH(GuiGraphicsComposition*, subComposition, composition->Children())
|
|
{
|
|
InvokeClipboardNotify(subComposition, arguments);
|
|
}
|
|
}
|
|
|
|
void GuiApplication::LeftButtonDown(NativePoint position)
|
|
{
|
|
OnMouseDown(position);
|
|
}
|
|
|
|
void GuiApplication::LeftButtonUp(NativePoint position)
|
|
{
|
|
}
|
|
|
|
void GuiApplication::RightButtonDown(NativePoint position)
|
|
{
|
|
OnMouseDown(position);
|
|
}
|
|
|
|
void GuiApplication::RightButtonUp(NativePoint position)
|
|
{
|
|
}
|
|
|
|
void GuiApplication::ClipboardUpdated()
|
|
{
|
|
for(vint i=0;i<windows.Count();i++)
|
|
{
|
|
GuiEventArgs arguments=windows[i]->GetNotifyEventArguments();
|
|
windows[i]->ClipboardUpdated.Execute(arguments);
|
|
InvokeClipboardNotify(windows[i]->GetBoundsComposition(), arguments);
|
|
}
|
|
}
|
|
|
|
GuiApplication::GuiApplication()
|
|
:locale(Locale::UserDefault())
|
|
{
|
|
GetCurrentController()->CallbackService()->InstallListener(this);
|
|
}
|
|
|
|
GuiApplication::~GuiApplication()
|
|
{
|
|
if(sharedTooltipControl)
|
|
{
|
|
delete sharedTooltipControl;
|
|
sharedTooltipControl=0;
|
|
}
|
|
GetCurrentController()->CallbackService()->UninstallListener(this);
|
|
}
|
|
|
|
INativeWindow* GuiApplication::GetThreadContextNativeWindow(GuiControlHost* controlHost)
|
|
{
|
|
return nullptr;
|
|
}
|
|
|
|
void GuiApplication::RegisterWindow(GuiWindow* window)
|
|
{
|
|
windows.Add(window);
|
|
}
|
|
|
|
void GuiApplication::UnregisterWindow(GuiWindow* window)
|
|
{
|
|
windows.Remove(window);
|
|
}
|
|
|
|
void GuiApplication::RegisterPopupOpened(GuiPopup* popup)
|
|
{
|
|
vint index=openingPopups.IndexOf(popup);
|
|
if(index==-1)
|
|
{
|
|
openingPopups.Add(popup);
|
|
if(openingPopups.Count()==1)
|
|
{
|
|
GetCurrentController()->InputService()->StartHookMouse();
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiApplication::RegisterPopupClosed(GuiPopup* popup)
|
|
{
|
|
if(openingPopups.Remove(popup))
|
|
{
|
|
if(openingPopups.Count()==0)
|
|
{
|
|
GetCurrentController()->InputService()->StopHookMouse();
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiApplication::OnMouseDown(NativePoint location)
|
|
{
|
|
GuiWindow* window=GetWindow(location);
|
|
for(vint i=0;i<windows.Count();i++)
|
|
{
|
|
if(windows[i]!=window)
|
|
{
|
|
windows[i]->MouseClickedOnOtherWindow(window);
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiApplication::TooltipMouseEnter(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
sharedTooltipHovering=true;
|
|
}
|
|
|
|
void GuiApplication::TooltipMouseLeave(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
sharedTooltipHovering=false;
|
|
if(sharedTooltipClosing)
|
|
{
|
|
CloseTooltip();
|
|
}
|
|
}
|
|
|
|
Locale GuiApplication::GetLocale()
|
|
{
|
|
return locale;
|
|
}
|
|
|
|
void GuiApplication::SetLocale(Locale value)
|
|
{
|
|
if (locale != value)
|
|
{
|
|
locale = value;
|
|
LocaleChanged();
|
|
}
|
|
}
|
|
|
|
void GuiApplication::Run(GuiWindow* _mainWindow)
|
|
{
|
|
if (!mainWindow)
|
|
{
|
|
mainWindow = _mainWindow;
|
|
GetCurrentController()->WindowService()->Run(mainWindow->GetNativeWindow());
|
|
mainWindow = nullptr;
|
|
}
|
|
}
|
|
|
|
GuiWindow* GuiApplication::GetMainWindow()
|
|
{
|
|
return mainWindow;
|
|
}
|
|
|
|
const collections::List<GuiWindow*>& GuiApplication::GetWindows()
|
|
{
|
|
return windows;
|
|
}
|
|
|
|
GuiWindow* GuiApplication::GetWindow(NativePoint location)
|
|
{
|
|
INativeWindow* nativeWindow = GetCurrentController()->WindowService()->GetWindow(location);
|
|
if (nativeWindow)
|
|
{
|
|
for (vint i = 0; i < windows.Count(); i++)
|
|
{
|
|
GuiWindow* window = windows[i];
|
|
if (window->GetNativeWindow() == nativeWindow)
|
|
{
|
|
return window;
|
|
}
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
void GuiApplication::ShowTooltip(GuiControl* owner, GuiControl* tooltip, vint preferredContentWidth, Point location)
|
|
{
|
|
GuiWindow* ownerWindow = dynamic_cast<GuiWindow*>(owner->GetRelatedControlHost());
|
|
if (sharedTooltipOwnerWindow != ownerWindow)
|
|
{
|
|
delete sharedTooltipControl;
|
|
sharedTooltipControl = 0;
|
|
}
|
|
|
|
if(!sharedTooltipControl)
|
|
{
|
|
sharedTooltipControl = new GuiTooltip(theme::ThemeName::Tooltip);
|
|
if (ownerWindow)
|
|
{
|
|
if (auto tooltipStyle = ownerWindow->GetControlTemplateObject(true)->GetTooltipTemplate())
|
|
{
|
|
sharedTooltipControl->SetControlTemplate(tooltipStyle);
|
|
}
|
|
}
|
|
sharedTooltipControl->GetBoundsComposition()->GetEventReceiver()->mouseEnter.AttachMethod(this, &GuiApplication::TooltipMouseEnter);
|
|
sharedTooltipControl->GetBoundsComposition()->GetEventReceiver()->mouseLeave.AttachMethod(this, &GuiApplication::TooltipMouseLeave);
|
|
}
|
|
|
|
sharedTooltipHovering=false;
|
|
sharedTooltipClosing=false;
|
|
sharedTooltipOwnerWindow = ownerWindow;
|
|
sharedTooltipOwner=owner;
|
|
sharedTooltipControl->SetTemporaryContentControl(tooltip);
|
|
sharedTooltipControl->SetPreferredContentWidth(preferredContentWidth);
|
|
sharedTooltipControl->SetClientSize(Size(10, 10));
|
|
sharedTooltipControl->ShowPopup(owner, location);
|
|
}
|
|
|
|
void GuiApplication::CloseTooltip()
|
|
{
|
|
if(sharedTooltipControl)
|
|
{
|
|
if(sharedTooltipHovering)
|
|
{
|
|
sharedTooltipClosing=true;
|
|
}
|
|
else
|
|
{
|
|
sharedTooltipClosing=false;
|
|
sharedTooltipControl->Close();
|
|
}
|
|
}
|
|
}
|
|
|
|
GuiControl* GuiApplication::GetTooltipOwner()
|
|
{
|
|
if(!sharedTooltipControl) return 0;
|
|
if(!sharedTooltipControl->GetTemporaryContentControl()) return 0;
|
|
return sharedTooltipOwner;
|
|
}
|
|
|
|
WString GuiApplication::GetExecutablePath()
|
|
{
|
|
return GetCurrentController()->GetExecutablePath();
|
|
}
|
|
|
|
WString GuiApplication::GetExecutableFolder()
|
|
{
|
|
WString path=GetExecutablePath();
|
|
for(vint i=path.Length()-1;i>=0;i--)
|
|
{
|
|
if(path[i]==L'\\' || path[i]==L'/')
|
|
{
|
|
return path.Sub(0, i+1);
|
|
}
|
|
}
|
|
return L"";
|
|
}
|
|
|
|
bool GuiApplication::IsInMainThread(GuiControlHost* controlHost)
|
|
{
|
|
return GetCurrentController()->AsyncService()->IsInMainThread(GetThreadContextNativeWindow(controlHost));
|
|
}
|
|
|
|
void GuiApplication::InvokeAsync(const Func<void()>& proc)
|
|
{
|
|
GetCurrentController()->AsyncService()->InvokeAsync(proc);
|
|
}
|
|
|
|
void GuiApplication::InvokeInMainThread(GuiControlHost* controlHost, const Func<void()>& proc)
|
|
{
|
|
GetCurrentController()->AsyncService()->InvokeInMainThread(GetThreadContextNativeWindow(controlHost), proc);
|
|
}
|
|
|
|
bool GuiApplication::InvokeInMainThreadAndWait(GuiControlHost* controlHost, const Func<void()>& proc, vint milliseconds)
|
|
{
|
|
CHECK_ERROR(!IsInMainThread(controlHost), L"GuiApplication::InvokeInMainThreadAndWait(GuiControlHost*, const Func<void()>&, vint)#This function cannot be called in UI thread.");
|
|
return GetCurrentController()->AsyncService()->InvokeInMainThreadAndWait(GetThreadContextNativeWindow(controlHost), proc, milliseconds);
|
|
}
|
|
|
|
Ptr<INativeDelay> GuiApplication::DelayExecute(const Func<void()>& proc, vint milliseconds)
|
|
{
|
|
return GetCurrentController()->AsyncService()->DelayExecute(proc, milliseconds);
|
|
}
|
|
|
|
Ptr<INativeDelay> GuiApplication::DelayExecuteInMainThread(const Func<void()>& proc, vint milliseconds)
|
|
{
|
|
return GetCurrentController()->AsyncService()->DelayExecuteInMainThread(proc, milliseconds);
|
|
}
|
|
|
|
void GuiApplication::RunGuiTask(GuiControlHost* controlHost, const Func<void()>& proc)
|
|
{
|
|
if(IsInMainThread(controlHost))
|
|
{
|
|
return proc();
|
|
}
|
|
else
|
|
{
|
|
InvokeInMainThreadAndWait(controlHost, [&proc]()
|
|
{
|
|
proc();
|
|
});
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiPluginManager
|
|
***********************************************************************/
|
|
|
|
class GuiPluginManager : public Object, public IGuiPluginManager
|
|
{
|
|
protected:
|
|
List<Ptr<IGuiPlugin>> plugins;
|
|
bool loaded;
|
|
public:
|
|
GuiPluginManager()
|
|
:loaded(false)
|
|
{
|
|
}
|
|
|
|
~GuiPluginManager()
|
|
{
|
|
Unload();
|
|
}
|
|
|
|
void AddPlugin(Ptr<IGuiPlugin> plugin)override
|
|
{
|
|
CHECK_ERROR(!loaded, L"GuiPluginManager::AddPlugin(Ptr<IGuiPlugin>)#Load function has already been executed.");
|
|
auto name = plugin->GetName();
|
|
if (name != L"")
|
|
{
|
|
FOREACH(Ptr<IGuiPlugin>, plugin, plugins)
|
|
{
|
|
CHECK_ERROR(plugin->GetName() != name, L"GuiPluginManager::AddPlugin(Ptr<IGuiPlugin>)#Duplicated plugin name.");
|
|
}
|
|
}
|
|
plugins.Add(plugin);
|
|
}
|
|
|
|
void Load()override
|
|
{
|
|
CHECK_ERROR(!loaded, L"GuiPluginManager::AddPlugin(Ptr<IGuiPlugin>)#Load function has already been executed.");
|
|
loaded=true;
|
|
|
|
SortedList<WString> loaded;
|
|
Group<WString, WString> loading;
|
|
Dictionary<WString, Ptr<IGuiPlugin>> pluginsToLoad;
|
|
FOREACH(Ptr<IGuiPlugin>, plugin, plugins)
|
|
{
|
|
auto name = plugin->GetName();
|
|
pluginsToLoad.Add(name, plugin);
|
|
List<WString> dependencies;
|
|
plugin->GetDependencies(dependencies);
|
|
FOREACH(WString, dependency, dependencies)
|
|
{
|
|
loading.Add(name, dependency);
|
|
}
|
|
}
|
|
|
|
while (pluginsToLoad.Count() > 0)
|
|
{
|
|
vint count = pluginsToLoad.Count();
|
|
{
|
|
FOREACH_INDEXER(WString, name, index, pluginsToLoad.Keys())
|
|
{
|
|
if (!loading.Keys().Contains(name))
|
|
{
|
|
for (vint i = loading.Count() - 1; i >= 0; i--)
|
|
{
|
|
loading.Remove(loading.Keys()[i], name);
|
|
}
|
|
loaded.Add(name);
|
|
|
|
auto plugin = pluginsToLoad.Values()[index];
|
|
pluginsToLoad.Remove(name);
|
|
plugin->Load();
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
if (count == pluginsToLoad.Count())
|
|
{
|
|
WString message;
|
|
FOREACH(Ptr<IGuiPlugin>, plugin, pluginsToLoad.Values())
|
|
{
|
|
message += L"Cannot load plugin \"" + plugin->GetName() + L"\" because part of its dependencies are not ready:";
|
|
List<WString> dependencies;
|
|
plugin->GetDependencies(dependencies);
|
|
bool first = true;
|
|
FOREACH(WString, dependency, dependencies)
|
|
{
|
|
if (!loaded.Contains(dependency))
|
|
{
|
|
message += L" \"" + dependency + L"\";";
|
|
}
|
|
}
|
|
message += L"\r\n";
|
|
}
|
|
throw Exception(message);
|
|
}
|
|
}
|
|
}
|
|
|
|
void Unload()override
|
|
{
|
|
CHECK_ERROR(loaded, L"GuiPluginManager::AddPlugin(Ptr<IGuiPlugin>)#Load function has not been executed.");
|
|
loaded=false;
|
|
FOREACH(Ptr<IGuiPlugin>, plugin, plugins)
|
|
{
|
|
plugin->Unload();
|
|
}
|
|
}
|
|
|
|
bool IsLoaded()override
|
|
{
|
|
return loaded;
|
|
}
|
|
};
|
|
|
|
/***********************************************************************
|
|
Helpers
|
|
***********************************************************************/
|
|
|
|
GuiApplication* application=0;
|
|
IGuiPluginManager* pluginManager=0;
|
|
|
|
GuiApplication* GetApplication()
|
|
{
|
|
return application;
|
|
}
|
|
|
|
IGuiPluginManager* GetPluginManager()
|
|
{
|
|
if(!pluginManager)
|
|
{
|
|
pluginManager=new GuiPluginManager;
|
|
}
|
|
return pluginManager;
|
|
}
|
|
|
|
void DestroyPluginManager()
|
|
{
|
|
if(pluginManager)
|
|
{
|
|
delete pluginManager;
|
|
pluginManager=0;
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiApplicationMain
|
|
***********************************************************************/
|
|
|
|
class UIThreadAsyncScheduler :public Object, public IAsyncScheduler, public Description<UIThreadAsyncScheduler>
|
|
{
|
|
public:
|
|
void Execute(const Func<void()>& callback)override
|
|
{
|
|
GetApplication()->InvokeInMainThread(GetApplication()->GetMainWindow(), callback);
|
|
}
|
|
|
|
void ExecuteInBackground(const Func<void()>& callback)override
|
|
{
|
|
GetApplication()->InvokeAsync(callback);
|
|
}
|
|
|
|
void DelayExecute(const Func<void()>& callback, vint milliseconds)override
|
|
{
|
|
GetApplication()->DelayExecuteInMainThread(callback, milliseconds);
|
|
}
|
|
};
|
|
|
|
class OtherThreadAsyncScheduler :public Object, public IAsyncScheduler, public Description<UIThreadAsyncScheduler>
|
|
{
|
|
public:
|
|
void Execute(const Func<void()>& callback)override
|
|
{
|
|
GetApplication()->InvokeAsync(callback);
|
|
}
|
|
|
|
void ExecuteInBackground(const Func<void()>& callback)override
|
|
{
|
|
GetApplication()->InvokeAsync(callback);
|
|
}
|
|
|
|
void DelayExecute(const Func<void()>& callback, vint milliseconds)override
|
|
{
|
|
GetApplication()->DelayExecute(callback, milliseconds);
|
|
}
|
|
};
|
|
|
|
void GuiApplicationInitialize()
|
|
{
|
|
GetCurrentController()->InputService()->StartTimer();
|
|
theme::InitializeTheme();
|
|
|
|
#ifndef VCZH_DEBUG_NO_REFLECTION
|
|
GetGlobalTypeManager()->Load();
|
|
#endif
|
|
GetPluginManager()->Load();
|
|
|
|
{
|
|
GuiApplication app;
|
|
application = &app;
|
|
IAsyncScheduler::RegisterSchedulerForCurrentThread(new UIThreadAsyncScheduler);
|
|
IAsyncScheduler::RegisterDefaultScheduler(new OtherThreadAsyncScheduler);
|
|
GuiMain();
|
|
IAsyncScheduler::UnregisterDefaultScheduler();
|
|
IAsyncScheduler::UnregisterSchedulerForCurrentThread();
|
|
}
|
|
application = nullptr;
|
|
|
|
DestroyPluginManager();
|
|
theme::FinalizeTheme();
|
|
ThreadLocalStorage::DisposeStorages();
|
|
FinalizeGlobalStorage();
|
|
#ifndef VCZH_DEBUG_NO_REFLECTION
|
|
DestroyGlobalTypeManager();
|
|
#endif
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiApplicationMain()
|
|
{
|
|
vl::presentation::controls::GuiApplicationInitialize();
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\CONTROLS\GUIBASICCONTROLS.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace controls
|
|
{
|
|
using namespace elements;
|
|
using namespace compositions;
|
|
using namespace collections;
|
|
using namespace reflection::description;
|
|
|
|
/***********************************************************************
|
|
GuiDisposedFlag
|
|
***********************************************************************/
|
|
|
|
void GuiDisposedFlag::SetDisposed()
|
|
{
|
|
disposed = true;
|
|
}
|
|
|
|
GuiDisposedFlag::GuiDisposedFlag(GuiControl* _owner)
|
|
:owner(_owner)
|
|
{
|
|
}
|
|
|
|
GuiDisposedFlag::~GuiDisposedFlag()
|
|
{
|
|
}
|
|
|
|
bool GuiDisposedFlag::IsDisposed()
|
|
{
|
|
return disposed;
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiControl
|
|
***********************************************************************/
|
|
|
|
Ptr<GuiDisposedFlag> GuiControl::GetDisposedFlag()
|
|
{
|
|
if (!disposedFlag)
|
|
{
|
|
disposedFlag = new GuiDisposedFlag(this);
|
|
}
|
|
return disposedFlag;
|
|
}
|
|
|
|
void GuiControl::BeforeControlTemplateUninstalled()
|
|
{
|
|
}
|
|
|
|
void GuiControl::AfterControlTemplateInstalled(bool initialize)
|
|
{
|
|
controlTemplateObject->SetText(text);
|
|
controlTemplateObject->SetFont(displayFont);
|
|
controlTemplateObject->SetContext(context);
|
|
controlTemplateObject->SetVisuallyEnabled(isVisuallyEnabled);
|
|
controlTemplateObject->SetFocusableComposition(focusableComposition);
|
|
controlTemplateObject->SetFocused(isFocused);
|
|
}
|
|
|
|
void GuiControl::CheckAndStoreControlTemplate(templates::GuiControlTemplate* value)
|
|
{
|
|
controlTemplateObject = value;
|
|
}
|
|
|
|
void GuiControl::EnsureControlTemplateExists()
|
|
{
|
|
if (!controlTemplateObject)
|
|
{
|
|
RebuildControlTemplate();
|
|
}
|
|
}
|
|
|
|
void GuiControl::RebuildControlTemplate()
|
|
{
|
|
bool initialize = controlTemplateObject == nullptr;
|
|
if (controlTemplateObject)
|
|
{
|
|
BeforeControlTemplateUninstalled();
|
|
containerComposition->GetParent()->RemoveChild(containerComposition);
|
|
boundsComposition->AddChild(containerComposition);
|
|
SafeDeleteComposition(controlTemplateObject);
|
|
controlTemplateObject = nullptr;
|
|
}
|
|
|
|
if (controlTemplate)
|
|
{
|
|
CheckAndStoreControlTemplate(controlTemplate({}));
|
|
}
|
|
else
|
|
{
|
|
CheckAndStoreControlTemplate(theme::GetCurrentTheme()->CreateStyle(controlThemeName)({}));
|
|
}
|
|
|
|
if (controlTemplateObject)
|
|
{
|
|
controlTemplateObject->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
|
|
containerComposition->GetParent()->RemoveChild(containerComposition);
|
|
boundsComposition->AddChild(controlTemplateObject);
|
|
controlTemplateObject->GetContainerComposition()->AddChild(containerComposition);
|
|
AfterControlTemplateInstalled(initialize);
|
|
}
|
|
}
|
|
|
|
void GuiControl::OnChildInserted(GuiControl* control)
|
|
{
|
|
GuiControl* oldParent=control->parent;
|
|
children.Add(control);
|
|
control->parent=this;
|
|
control->OnParentChanged(oldParent, control->parent);
|
|
control->UpdateVisuallyEnabled();
|
|
control->UpdateDisplayFont();
|
|
|
|
if (auto host = boundsComposition->GetRelatedGraphicsHost())
|
|
{
|
|
host->InvalidateTabOrderCache();
|
|
}
|
|
}
|
|
|
|
void GuiControl::OnChildRemoved(GuiControl* control)
|
|
{
|
|
GuiControl* oldParent=control->parent;
|
|
control->parent=0;
|
|
children.Remove(control);
|
|
control->OnParentChanged(oldParent, control->parent);
|
|
|
|
if (auto host = boundsComposition->GetRelatedGraphicsHost())
|
|
{
|
|
host->InvalidateTabOrderCache();
|
|
}
|
|
}
|
|
|
|
void GuiControl::OnParentChanged(GuiControl* oldParent, GuiControl* newParent)
|
|
{
|
|
OnParentLineChanged();
|
|
}
|
|
|
|
void GuiControl::OnParentLineChanged()
|
|
{
|
|
{
|
|
GuiControlSignalEventArgs arguments(boundsComposition);
|
|
arguments.controlSignal = ControlSignal::ParentLineChanged;
|
|
ControlSignalTrigerred.Execute(arguments);
|
|
}
|
|
for(vint i=0;i<children.Count();i++)
|
|
{
|
|
children[i]->OnParentLineChanged();
|
|
}
|
|
}
|
|
|
|
void GuiControl::OnServiceAdded()
|
|
{
|
|
{
|
|
GuiControlSignalEventArgs arguments(boundsComposition);
|
|
arguments.controlSignal = ControlSignal::ServiceAdded;
|
|
ControlSignalTrigerred.Execute(arguments);
|
|
}
|
|
for(vint i=0;i<children.Count();i++)
|
|
{
|
|
children[i]->OnParentLineChanged();
|
|
}
|
|
}
|
|
|
|
void GuiControl::OnRenderTargetChanged(elements::IGuiGraphicsRenderTarget* renderTarget)
|
|
{
|
|
GuiControlSignalEventArgs arguments(boundsComposition);
|
|
arguments.controlSignal = ControlSignal::RenderTargetChanged;
|
|
ControlSignalTrigerred.Execute(arguments);
|
|
}
|
|
|
|
void GuiControl::OnBeforeReleaseGraphicsHost()
|
|
{
|
|
for(vint i=0;i<children.Count();i++)
|
|
{
|
|
children[i]->OnBeforeReleaseGraphicsHost();
|
|
}
|
|
}
|
|
|
|
void GuiControl::UpdateVisuallyEnabled()
|
|
{
|
|
bool newValue = isEnabled && (parent == 0 ? true : parent->GetVisuallyEnabled());
|
|
if (isVisuallyEnabled != newValue)
|
|
{
|
|
isVisuallyEnabled = newValue;
|
|
if (controlTemplateObject)
|
|
{
|
|
controlTemplateObject->SetVisuallyEnabled(isVisuallyEnabled);
|
|
}
|
|
VisuallyEnabledChanged.Execute(GetNotifyEventArguments());
|
|
|
|
for (vint i = 0; i < children.Count(); i++)
|
|
{
|
|
children[i]->UpdateVisuallyEnabled();
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiControl::UpdateDisplayFont()
|
|
{
|
|
auto newValue =
|
|
font ? font.Value() :
|
|
parent ? parent->GetDisplayFont() :
|
|
GetCurrentController()->ResourceService()->GetDefaultFont();
|
|
|
|
if (displayFont != newValue)
|
|
{
|
|
displayFont = newValue;
|
|
if (controlTemplateObject)
|
|
{
|
|
controlTemplateObject->SetFont(displayFont);
|
|
}
|
|
DisplayFontChanged.Execute(GetNotifyEventArguments());
|
|
|
|
for (vint i = 0; i < children.Count(); i++)
|
|
{
|
|
children[i]->UpdateDisplayFont();
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiControl::OnGotFocus(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
if (!isFocused)
|
|
{
|
|
isFocused = true;
|
|
if (controlTemplateObject)
|
|
{
|
|
controlTemplateObject->SetFocused(true);
|
|
}
|
|
FocusedChanged.Execute(GetNotifyEventArguments());
|
|
}
|
|
}
|
|
|
|
void GuiControl::OnLostFocus(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
if (isFocused)
|
|
{
|
|
isFocused = false;
|
|
if (controlTemplateObject)
|
|
{
|
|
controlTemplateObject->SetFocused(false);
|
|
}
|
|
FocusedChanged.Execute(GetNotifyEventArguments());
|
|
}
|
|
}
|
|
|
|
void GuiControl::SetFocusableComposition(compositions::GuiGraphicsComposition* value)
|
|
{
|
|
if (focusableComposition != value)
|
|
{
|
|
if (focusableComposition)
|
|
{
|
|
focusableComposition->GetEventReceiver()->gotFocus.Detach(gotFocusHandler);
|
|
focusableComposition->GetEventReceiver()->lostFocus.Detach(lostFocusHandler);
|
|
|
|
gotFocusHandler = nullptr;
|
|
lostFocusHandler = nullptr;
|
|
}
|
|
|
|
focusableComposition = value;
|
|
if (controlTemplateObject)
|
|
{
|
|
controlTemplateObject->SetFocusableComposition(focusableComposition);
|
|
}
|
|
|
|
if (focusableComposition)
|
|
{
|
|
gotFocusHandler = focusableComposition->GetEventReceiver()->gotFocus.AttachMethod(this, &GuiControl::OnGotFocus);
|
|
lostFocusHandler = focusableComposition->GetEventReceiver()->lostFocus.AttachMethod(this, &GuiControl::OnLostFocus);
|
|
}
|
|
else
|
|
{
|
|
GuiEventArgs arguments(boundsComposition);
|
|
OnLostFocus(boundsComposition, arguments);
|
|
}
|
|
}
|
|
}
|
|
|
|
bool GuiControl::IsControlVisibleAndEnabled()
|
|
{
|
|
GuiControl* control = this;
|
|
while (control)
|
|
{
|
|
if (!control->GetVisible() || !control->GetEnabled())
|
|
{
|
|
return false;
|
|
}
|
|
control = control->GetParent();
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool GuiControl::IsAltEnabled()
|
|
{
|
|
return IsControlVisibleAndEnabled();
|
|
}
|
|
|
|
bool GuiControl::IsAltAvailable()
|
|
{
|
|
return focusableComposition != nullptr && alt != L"";
|
|
}
|
|
|
|
compositions::GuiGraphicsComposition* GuiControl::GetAltComposition()
|
|
{
|
|
return boundsComposition;
|
|
}
|
|
|
|
compositions::IGuiAltActionHost* GuiControl::GetActivatingAltHost()
|
|
{
|
|
return activatingAltHost;
|
|
}
|
|
|
|
void GuiControl::OnActiveAlt()
|
|
{
|
|
SetFocus();
|
|
}
|
|
|
|
bool GuiControl::IsTabEnabled()
|
|
{
|
|
return IsControlVisibleAndEnabled();
|
|
}
|
|
|
|
bool GuiControl::IsTabAvailable()
|
|
{
|
|
return focusableComposition != nullptr;
|
|
}
|
|
|
|
bool GuiControl::SharedPtrDestructorProc(DescriptableObject* obj, bool forceDisposing)
|
|
{
|
|
GuiControl* value=dynamic_cast<GuiControl*>(obj);
|
|
if(value->GetBoundsComposition()->GetParent())
|
|
{
|
|
if (!forceDisposing) return false;
|
|
}
|
|
SafeDeleteControl(value);
|
|
return true;
|
|
}
|
|
|
|
GuiControl::GuiControl(theme::ThemeName themeName)
|
|
:controlThemeName(themeName)
|
|
, displayFont(GetCurrentController()->ResourceService()->GetDefaultFont())
|
|
{
|
|
{
|
|
boundsComposition = new GuiBoundsComposition;
|
|
boundsComposition->SetAssociatedControl(this);
|
|
boundsComposition->SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
|
|
containerComposition = new GuiBoundsComposition;
|
|
containerComposition->SetTransparentToMouse(true);
|
|
containerComposition->SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
containerComposition->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
|
|
boundsComposition->AddChild(containerComposition);
|
|
}
|
|
{
|
|
ControlThemeNameChanged.SetAssociatedComposition(boundsComposition);
|
|
ControlTemplateChanged.SetAssociatedComposition(boundsComposition);
|
|
ControlSignalTrigerred.SetAssociatedComposition(boundsComposition);
|
|
VisibleChanged.SetAssociatedComposition(boundsComposition);
|
|
EnabledChanged.SetAssociatedComposition(boundsComposition);
|
|
FocusedChanged.SetAssociatedComposition(boundsComposition);
|
|
VisuallyEnabledChanged.SetAssociatedComposition(boundsComposition);
|
|
DisplayFontChanged.SetAssociatedComposition(boundsComposition);
|
|
AltChanged.SetAssociatedComposition(boundsComposition);
|
|
TextChanged.SetAssociatedComposition(boundsComposition);
|
|
FontChanged.SetAssociatedComposition(boundsComposition);
|
|
ContextChanged.SetAssociatedComposition(boundsComposition);
|
|
}
|
|
sharedPtrDestructorProc = &GuiControl::SharedPtrDestructorProc;
|
|
}
|
|
|
|
GuiControl::~GuiControl()
|
|
{
|
|
if (disposedFlag)
|
|
{
|
|
disposedFlag->SetDisposed();
|
|
}
|
|
// prevent a root bounds composition from notifying its dead controls
|
|
if (!parent)
|
|
{
|
|
NotifyFinalizeInstance(boundsComposition);
|
|
}
|
|
|
|
if (tooltipControl)
|
|
{
|
|
// the only legal parent is the GuiApplication::sharedTooltipWindow
|
|
if (tooltipControl->GetBoundsComposition()->GetParent())
|
|
{
|
|
tooltipControl->GetBoundsComposition()->GetParent()->RemoveChild(tooltipControl->GetBoundsComposition());
|
|
}
|
|
delete tooltipControl;
|
|
}
|
|
|
|
for (vint i = 0; i < children.Count(); i++)
|
|
{
|
|
delete children[i];
|
|
}
|
|
children.Clear();
|
|
|
|
// let the root control of a control tree delete the whole composition tree
|
|
if (!parent)
|
|
{
|
|
delete boundsComposition;
|
|
}
|
|
}
|
|
|
|
void GuiControl::InvokeOrDelayIfRendering(Func<void()> proc)
|
|
{
|
|
auto controlHost = GetRelatedControlHost();
|
|
if (controlHost && boundsComposition->IsRendering())
|
|
{
|
|
auto flag = GetDisposedFlag();
|
|
GetApplication()->InvokeInMainThread(controlHost, [=]()
|
|
{
|
|
if (!flag->IsDisposed())
|
|
{
|
|
proc();
|
|
}
|
|
});
|
|
}
|
|
else
|
|
{
|
|
proc();
|
|
}
|
|
}
|
|
|
|
compositions::GuiEventArgs GuiControl::GetNotifyEventArguments()
|
|
{
|
|
return GuiEventArgs(boundsComposition);
|
|
}
|
|
|
|
theme::ThemeName GuiControl::GetControlThemeName()
|
|
{
|
|
return controlThemeName;
|
|
}
|
|
|
|
void GuiControl::SetControlThemeName(theme::ThemeName value)
|
|
{
|
|
SetControlThemeNameAndTemplate(value, controlTemplate);
|
|
}
|
|
|
|
GuiControl::ControlTemplatePropertyType GuiControl::GetControlTemplate()
|
|
{
|
|
return controlTemplate;
|
|
}
|
|
|
|
void GuiControl::SetControlTemplate(const ControlTemplatePropertyType& value)
|
|
{
|
|
SetControlThemeNameAndTemplate(controlThemeName, value);
|
|
}
|
|
|
|
void GuiControl::SetControlThemeNameAndTemplate(theme::ThemeName themeNameValue, const ControlTemplatePropertyType& controlTemplateValue)
|
|
{
|
|
bool themeChanged = (controlThemeName != themeNameValue);
|
|
bool templateChanged = (controlTemplate || controlTemplateValue);
|
|
if (themeChanged || templateChanged)
|
|
{
|
|
controlThemeName = themeNameValue;
|
|
controlTemplate = controlTemplateValue;
|
|
RebuildControlTemplate();
|
|
|
|
if (themeChanged)
|
|
{
|
|
ControlThemeNameChanged.Execute(GetNotifyEventArguments());
|
|
}
|
|
if (templateChanged)
|
|
{
|
|
ControlTemplateChanged.Execute(GetNotifyEventArguments());
|
|
}
|
|
}
|
|
}
|
|
|
|
templates::GuiControlTemplate* GuiControl::GetControlTemplateObject()
|
|
{
|
|
EnsureControlTemplateExists();
|
|
return controlTemplateObject;
|
|
}
|
|
|
|
compositions::GuiBoundsComposition* GuiControl::GetBoundsComposition()
|
|
{
|
|
EnsureControlTemplateExists();
|
|
return boundsComposition;
|
|
}
|
|
|
|
compositions::GuiGraphicsComposition* GuiControl::GetContainerComposition()
|
|
{
|
|
EnsureControlTemplateExists();
|
|
return containerComposition;
|
|
}
|
|
|
|
compositions::GuiGraphicsComposition* GuiControl::GetFocusableComposition()
|
|
{
|
|
EnsureControlTemplateExists();
|
|
return focusableComposition;
|
|
}
|
|
|
|
GuiControl* GuiControl::GetParent()
|
|
{
|
|
return parent;
|
|
}
|
|
|
|
vint GuiControl::GetChildrenCount()
|
|
{
|
|
return children.Count();
|
|
}
|
|
|
|
GuiControl* GuiControl::GetChild(vint index)
|
|
{
|
|
return children[index];
|
|
}
|
|
|
|
bool GuiControl::AddChild(GuiControl* control)
|
|
{
|
|
return GetContainerComposition()->AddChild(control->GetBoundsComposition());
|
|
}
|
|
|
|
bool GuiControl::HasChild(GuiControl* control)
|
|
{
|
|
return children.Contains(control);
|
|
}
|
|
|
|
GuiControlHost* GuiControl::GetRelatedControlHost()
|
|
{
|
|
return parent?parent->GetRelatedControlHost():0;
|
|
}
|
|
|
|
bool GuiControl::GetVisuallyEnabled()
|
|
{
|
|
return isVisuallyEnabled;
|
|
}
|
|
|
|
bool GuiControl::GetFocused()
|
|
{
|
|
return isFocused;
|
|
}
|
|
|
|
bool GuiControl::GetAcceptTabInput()
|
|
{
|
|
return acceptTabInput;
|
|
}
|
|
|
|
void GuiControl::SetAcceptTabInput(bool value)
|
|
{
|
|
acceptTabInput = value;
|
|
}
|
|
|
|
vint GuiControl::GetTabPriority()
|
|
{
|
|
return tabPriority;
|
|
}
|
|
|
|
void GuiControl::SetTabPriority(vint value)
|
|
{
|
|
vint newTabPriority = value < 0 ? -1 : value;
|
|
if (tabPriority != newTabPriority)
|
|
{
|
|
tabPriority = newTabPriority;
|
|
if (auto host = boundsComposition->GetRelatedGraphicsHost())
|
|
{
|
|
host->InvalidateTabOrderCache();
|
|
}
|
|
}
|
|
}
|
|
|
|
bool GuiControl::GetEnabled()
|
|
{
|
|
return isEnabled;
|
|
}
|
|
|
|
void GuiControl::SetEnabled(bool value)
|
|
{
|
|
if(isEnabled!=value)
|
|
{
|
|
isEnabled=value;
|
|
EnabledChanged.Execute(GetNotifyEventArguments());
|
|
UpdateVisuallyEnabled();
|
|
}
|
|
}
|
|
|
|
bool GuiControl::GetVisible()
|
|
{
|
|
return isVisible;
|
|
}
|
|
|
|
void GuiControl::SetVisible(bool value)
|
|
{
|
|
boundsComposition->SetVisible(value);
|
|
if(isVisible!=value)
|
|
{
|
|
isVisible=value;
|
|
VisibleChanged.Execute(GetNotifyEventArguments());
|
|
}
|
|
}
|
|
|
|
const WString& GuiControl::GetAlt()
|
|
{
|
|
return alt;
|
|
}
|
|
|
|
bool GuiControl::SetAlt(const WString& value)
|
|
{
|
|
if (!IGuiAltAction::IsLegalAlt(value)) return false;
|
|
if (alt != value)
|
|
{
|
|
alt = value;
|
|
AltChanged.Execute(GetNotifyEventArguments());
|
|
}
|
|
return true;
|
|
}
|
|
|
|
void GuiControl::SetActivatingAltHost(compositions::IGuiAltActionHost* host)
|
|
{
|
|
activatingAltHost = host;
|
|
}
|
|
|
|
const WString& GuiControl::GetText()
|
|
{
|
|
return text;
|
|
}
|
|
|
|
void GuiControl::SetText(const WString& value)
|
|
{
|
|
if (text != value)
|
|
{
|
|
text = value;
|
|
if (controlTemplateObject)
|
|
{
|
|
controlTemplateObject->SetText(text);
|
|
}
|
|
TextChanged.Execute(GetNotifyEventArguments());
|
|
}
|
|
}
|
|
|
|
const Nullable<FontProperties>& GuiControl::GetFont()
|
|
{
|
|
return font;
|
|
}
|
|
|
|
void GuiControl::SetFont(const Nullable<FontProperties>& value)
|
|
{
|
|
if (font != value)
|
|
{
|
|
font = value;
|
|
FontChanged.Execute(GetNotifyEventArguments());
|
|
UpdateDisplayFont();
|
|
}
|
|
}
|
|
|
|
const FontProperties& GuiControl::GetDisplayFont()
|
|
{
|
|
return displayFont;
|
|
}
|
|
|
|
description::Value GuiControl::GetContext()
|
|
{
|
|
return context;
|
|
}
|
|
|
|
void GuiControl::SetContext(const description::Value& value)
|
|
{
|
|
if (context != value)
|
|
{
|
|
context = value;
|
|
if (controlTemplateObject)
|
|
{
|
|
controlTemplateObject->SetContext(context);
|
|
}
|
|
ContextChanged.Execute(GetNotifyEventArguments());
|
|
}
|
|
}
|
|
|
|
void GuiControl::SetFocus()
|
|
{
|
|
if (focusableComposition)
|
|
{
|
|
if (auto host = focusableComposition->GetRelatedGraphicsHost())
|
|
{
|
|
host->SetFocus(focusableComposition);
|
|
}
|
|
}
|
|
}
|
|
|
|
description::Value GuiControl::GetTag()
|
|
{
|
|
return tag;
|
|
}
|
|
|
|
void GuiControl::SetTag(const description::Value& value)
|
|
{
|
|
tag=value;
|
|
}
|
|
|
|
GuiControl* GuiControl::GetTooltipControl()
|
|
{
|
|
return tooltipControl;
|
|
}
|
|
|
|
GuiControl* GuiControl::SetTooltipControl(GuiControl* value)
|
|
{
|
|
GuiControl* oldTooltipControl=tooltipControl;
|
|
tooltipControl=value;
|
|
return oldTooltipControl;
|
|
}
|
|
|
|
vint GuiControl::GetTooltipWidth()
|
|
{
|
|
return tooltipWidth;
|
|
}
|
|
|
|
void GuiControl::SetTooltipWidth(vint value)
|
|
{
|
|
tooltipWidth=value;
|
|
}
|
|
|
|
bool GuiControl::DisplayTooltip(Point location)
|
|
{
|
|
if(!tooltipControl) return false;
|
|
GetApplication()->ShowTooltip(this, tooltipControl, tooltipWidth, location);
|
|
return true;
|
|
}
|
|
|
|
void GuiControl::CloseTooltip()
|
|
{
|
|
if(GetApplication()->GetTooltipOwner()==this)
|
|
{
|
|
GetApplication()->CloseTooltip();
|
|
}
|
|
}
|
|
|
|
IDescriptable* GuiControl::QueryService(const WString& identifier)
|
|
{
|
|
if (identifier == IGuiAltAction::Identifier)
|
|
{
|
|
return (IGuiAltAction*)this;
|
|
}
|
|
else if (identifier == IGuiAltActionContainer::Identifier)
|
|
{
|
|
return nullptr;
|
|
}
|
|
else if (identifier == IGuiTabAction::Identifier)
|
|
{
|
|
return (IGuiTabAction*)this;
|
|
}
|
|
else
|
|
{
|
|
vint index = controlServices.Keys().IndexOf(identifier);
|
|
if (index != -1)
|
|
{
|
|
return controlServices.Values()[index].Obj();
|
|
}
|
|
|
|
if (parent)
|
|
{
|
|
return parent->QueryService(identifier);
|
|
}
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
bool GuiControl::AddService(const WString& identifier, Ptr<IDescriptable> value)
|
|
{
|
|
if (controlServices.Keys().Contains(identifier))
|
|
{
|
|
return false;
|
|
}
|
|
|
|
controlServices.Add(identifier, value);
|
|
OnServiceAdded();
|
|
return true;
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiCustomControl
|
|
***********************************************************************/
|
|
|
|
controls::GuiControlHost* GuiCustomControl::GetControlHostForInstance()
|
|
{
|
|
return GetRelatedControlHost();
|
|
}
|
|
|
|
void GuiCustomControl::OnParentLineChanged()
|
|
{
|
|
GuiControl::OnParentLineChanged();
|
|
OnControlHostForInstanceChanged();
|
|
}
|
|
|
|
GuiCustomControl::GuiCustomControl(theme::ThemeName themeName)
|
|
:GuiControl(themeName)
|
|
{
|
|
}
|
|
|
|
GuiCustomControl::~GuiCustomControl()
|
|
{
|
|
FinalizeAggregation();
|
|
FinalizeInstanceRecursively(this);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\CONTROLS\GUIBUTTONCONTROLS.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace controls
|
|
{
|
|
using namespace elements;
|
|
using namespace compositions;
|
|
using namespace collections;
|
|
using namespace reflection::description;
|
|
|
|
/***********************************************************************
|
|
GuiButton
|
|
***********************************************************************/
|
|
|
|
void GuiButton::BeforeControlTemplateUninstalled_()
|
|
{
|
|
}
|
|
|
|
void GuiButton::AfterControlTemplateInstalled_(bool initialize)
|
|
{
|
|
auto ct = GetControlTemplateObject(true);
|
|
GetControlTemplateObject(true)->SetState(controlState);
|
|
}
|
|
|
|
void GuiButton::OnParentLineChanged()
|
|
{
|
|
GuiControl::OnParentLineChanged();
|
|
if(GetRelatedControlHost()==0)
|
|
{
|
|
mousePressing=false;
|
|
mouseHoving=false;
|
|
UpdateControlState();
|
|
}
|
|
}
|
|
|
|
void GuiButton::OnActiveAlt()
|
|
{
|
|
if (autoFocus)
|
|
{
|
|
GuiControl::OnActiveAlt();
|
|
}
|
|
Clicked.Execute(GetNotifyEventArguments());
|
|
}
|
|
|
|
bool GuiButton::IsTabAvailable()
|
|
{
|
|
return autoFocus && GuiControl::IsTabAvailable();
|
|
}
|
|
|
|
void GuiButton::UpdateControlState()
|
|
{
|
|
auto newControlState = ButtonState::Normal;
|
|
if (keyPressing)
|
|
{
|
|
newControlState = ButtonState::Pressed;
|
|
}
|
|
else if (mousePressing)
|
|
{
|
|
if (mouseHoving)
|
|
{
|
|
newControlState = ButtonState::Pressed;
|
|
}
|
|
else
|
|
{
|
|
newControlState = ButtonState::Active;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (mouseHoving)
|
|
{
|
|
newControlState = ButtonState::Active;
|
|
}
|
|
else
|
|
{
|
|
newControlState = ButtonState::Normal;
|
|
}
|
|
}
|
|
if (controlState != newControlState)
|
|
{
|
|
controlState = newControlState;
|
|
GetControlTemplateObject(true)->SetState(controlState);
|
|
}
|
|
}
|
|
|
|
void GuiButton::CheckAndClick(compositions::GuiEventArgs& arguments)
|
|
{
|
|
auto eventSource = arguments.eventSource->GetAssociatedControl();
|
|
while (eventSource && eventSource != this)
|
|
{
|
|
if (eventSource->GetFocusableComposition())
|
|
{
|
|
return;
|
|
}
|
|
eventSource = eventSource->GetParent();
|
|
}
|
|
Clicked.Execute(GetNotifyEventArguments());
|
|
}
|
|
|
|
void GuiButton::OnLeftButtonDown(compositions::GuiGraphicsComposition* sender, compositions::GuiMouseEventArgs& arguments)
|
|
{
|
|
if(arguments.eventSource==boundsComposition)
|
|
{
|
|
mousePressing=true;
|
|
if (autoFocus)
|
|
{
|
|
SetFocus();
|
|
}
|
|
UpdateControlState();
|
|
if(!clickOnMouseUp)
|
|
{
|
|
CheckAndClick(arguments);
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiButton::OnLeftButtonUp(compositions::GuiGraphicsComposition* sender, compositions::GuiMouseEventArgs& arguments)
|
|
{
|
|
if(arguments.eventSource==boundsComposition)
|
|
{
|
|
mousePressing=false;
|
|
UpdateControlState();
|
|
}
|
|
if(GetVisuallyEnabled())
|
|
{
|
|
if(mouseHoving && clickOnMouseUp)
|
|
{
|
|
CheckAndClick(arguments);
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiButton::OnMouseEnter(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
if(arguments.eventSource==boundsComposition)
|
|
{
|
|
mouseHoving=true;
|
|
UpdateControlState();
|
|
}
|
|
}
|
|
|
|
void GuiButton::OnMouseLeave(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
if(arguments.eventSource==boundsComposition)
|
|
{
|
|
mouseHoving=false;
|
|
UpdateControlState();
|
|
}
|
|
}
|
|
|
|
void GuiButton::OnKeyDown(compositions::GuiGraphicsComposition* sender, compositions::GuiKeyEventArgs& arguments)
|
|
{
|
|
if (arguments.eventSource == focusableComposition && !arguments.ctrl && !arguments.shift && !arguments.alt)
|
|
{
|
|
switch (arguments.code)
|
|
{
|
|
case VKEY::_RETURN:
|
|
CheckAndClick(arguments);
|
|
arguments.handled = true;
|
|
break;
|
|
case VKEY::_SPACE:
|
|
if (!arguments.autoRepeatKeyDown)
|
|
{
|
|
keyPressing = true;
|
|
UpdateControlState();
|
|
}
|
|
arguments.handled = true;
|
|
break;
|
|
default:;
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiButton::OnKeyUp(compositions::GuiGraphicsComposition* sender, compositions::GuiKeyEventArgs& arguments)
|
|
{
|
|
if (arguments.eventSource == focusableComposition && !arguments.ctrl && !arguments.shift && !arguments.alt)
|
|
{
|
|
switch (arguments.code)
|
|
{
|
|
case VKEY::_SPACE:
|
|
if (keyPressing)
|
|
{
|
|
keyPressing = false;
|
|
UpdateControlState();
|
|
CheckAndClick(arguments);
|
|
}
|
|
arguments.handled = true;
|
|
break;
|
|
default:;
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiButton::OnLostFocus(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
if (keyPressing)
|
|
{
|
|
keyPressing = false;
|
|
UpdateControlState();
|
|
}
|
|
}
|
|
|
|
GuiButton::GuiButton(theme::ThemeName themeName)
|
|
:GuiControl(themeName)
|
|
{
|
|
Clicked.SetAssociatedComposition(boundsComposition);
|
|
SetFocusableComposition(boundsComposition);
|
|
|
|
boundsComposition->GetEventReceiver()->leftButtonDown.AttachMethod(this, &GuiButton::OnLeftButtonDown);
|
|
boundsComposition->GetEventReceiver()->leftButtonUp.AttachMethod(this, &GuiButton::OnLeftButtonUp);
|
|
boundsComposition->GetEventReceiver()->mouseEnter.AttachMethod(this, &GuiButton::OnMouseEnter);
|
|
boundsComposition->GetEventReceiver()->mouseLeave.AttachMethod(this, &GuiButton::OnMouseLeave);
|
|
boundsComposition->GetEventReceiver()->keyDown.AttachMethod(this, &GuiButton::OnKeyDown);
|
|
boundsComposition->GetEventReceiver()->keyUp.AttachMethod(this, &GuiButton::OnKeyUp);
|
|
boundsComposition->GetEventReceiver()->lostFocus.AttachMethod(this, &GuiButton::OnLostFocus);
|
|
}
|
|
|
|
GuiButton::~GuiButton()
|
|
{
|
|
}
|
|
|
|
bool GuiButton::GetClickOnMouseUp()
|
|
{
|
|
return clickOnMouseUp;
|
|
}
|
|
|
|
void GuiButton::SetClickOnMouseUp(bool value)
|
|
{
|
|
clickOnMouseUp=value;
|
|
}
|
|
|
|
bool GuiButton::GetAutoFocus()
|
|
{
|
|
return autoFocus;
|
|
}
|
|
|
|
void GuiButton::SetAutoFocus(bool value)
|
|
{
|
|
autoFocus = value;
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiSelectableButton::GroupController
|
|
***********************************************************************/
|
|
|
|
GuiSelectableButton::GroupController::GroupController()
|
|
{
|
|
}
|
|
|
|
GuiSelectableButton::GroupController::~GroupController()
|
|
{
|
|
for(vint i=buttons.Count()-1;i>=0;i--)
|
|
{
|
|
buttons[i]->SetGroupController(0);
|
|
}
|
|
}
|
|
|
|
void GuiSelectableButton::GroupController::Attach(GuiSelectableButton* button)
|
|
{
|
|
if(!buttons.Contains(button))
|
|
{
|
|
buttons.Add(button);
|
|
}
|
|
}
|
|
|
|
void GuiSelectableButton::GroupController::Detach(GuiSelectableButton* button)
|
|
{
|
|
buttons.Remove(button);
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiSelectableButton::MutexGroupController
|
|
***********************************************************************/
|
|
|
|
GuiSelectableButton::MutexGroupController::MutexGroupController()
|
|
:suppress(false)
|
|
{
|
|
}
|
|
|
|
GuiSelectableButton::MutexGroupController::~MutexGroupController()
|
|
{
|
|
}
|
|
|
|
void GuiSelectableButton::MutexGroupController::OnSelectedChanged(GuiSelectableButton* button)
|
|
{
|
|
if(!suppress)
|
|
{
|
|
suppress=true;
|
|
for(vint i=0;i<buttons.Count();i++)
|
|
{
|
|
buttons[i]->SetSelected(buttons[i]==button);
|
|
}
|
|
suppress=false;
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiSelectableButton
|
|
***********************************************************************/
|
|
|
|
void GuiSelectableButton::BeforeControlTemplateUninstalled_()
|
|
{
|
|
}
|
|
|
|
void GuiSelectableButton::AfterControlTemplateInstalled_(bool initialize)
|
|
{
|
|
GetControlTemplateObject(true)->SetSelected(isSelected);
|
|
}
|
|
|
|
void GuiSelectableButton::OnClicked(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
if(autoSelection)
|
|
{
|
|
SetSelected(!GetSelected());
|
|
}
|
|
}
|
|
|
|
GuiSelectableButton::GuiSelectableButton(theme::ThemeName themeName)
|
|
:GuiButton(themeName)
|
|
{
|
|
GroupControllerChanged.SetAssociatedComposition(boundsComposition);
|
|
AutoSelectionChanged.SetAssociatedComposition(boundsComposition);
|
|
SelectedChanged.SetAssociatedComposition(boundsComposition);
|
|
|
|
Clicked.AttachMethod(this, &GuiSelectableButton::OnClicked);
|
|
}
|
|
|
|
GuiSelectableButton::~GuiSelectableButton()
|
|
{
|
|
if(groupController)
|
|
{
|
|
groupController->Detach(this);
|
|
}
|
|
}
|
|
|
|
GuiSelectableButton::GroupController* GuiSelectableButton::GetGroupController()
|
|
{
|
|
return groupController;
|
|
}
|
|
|
|
void GuiSelectableButton::SetGroupController(GroupController* value)
|
|
{
|
|
if(groupController)
|
|
{
|
|
groupController->Detach(this);
|
|
}
|
|
groupController=value;
|
|
if(groupController)
|
|
{
|
|
groupController->Attach(this);
|
|
}
|
|
GroupControllerChanged.Execute(GetNotifyEventArguments());
|
|
}
|
|
|
|
bool GuiSelectableButton::GetAutoSelection()
|
|
{
|
|
return autoSelection;
|
|
}
|
|
|
|
void GuiSelectableButton::SetAutoSelection(bool value)
|
|
{
|
|
if(autoSelection!=value)
|
|
{
|
|
autoSelection=value;
|
|
AutoSelectionChanged.Execute(GetNotifyEventArguments());
|
|
}
|
|
}
|
|
|
|
bool GuiSelectableButton::GetSelected()
|
|
{
|
|
return isSelected;
|
|
}
|
|
|
|
void GuiSelectableButton::SetSelected(bool value)
|
|
{
|
|
if (isSelected != value)
|
|
{
|
|
isSelected = value;
|
|
GetControlTemplateObject(true)->SetSelected(isSelected);
|
|
if (groupController)
|
|
{
|
|
groupController->OnSelectedChanged(this);
|
|
}
|
|
SelectedChanged.Execute(GetNotifyEventArguments());
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\CONTROLS\GUICONTAINERCONTROLS.CPP
|
|
***********************************************************************/
|
|
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
using namespace compositions;
|
|
|
|
namespace controls
|
|
{
|
|
|
|
/***********************************************************************
|
|
GuiTabPage
|
|
***********************************************************************/
|
|
|
|
bool GuiTabPage::IsAltAvailable()
|
|
{
|
|
return false;
|
|
}
|
|
|
|
GuiTabPage::GuiTabPage(theme::ThemeName themeName)
|
|
:GuiCustomControl(themeName)
|
|
{
|
|
}
|
|
|
|
GuiTabPage::~GuiTabPage()
|
|
{
|
|
FinalizeAggregation();
|
|
}
|
|
|
|
GuiTab* GuiTabPage::GetOwnerTab()
|
|
{
|
|
return tab;
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiTabPageList
|
|
***********************************************************************/
|
|
|
|
bool GuiTabPageList::QueryInsert(vint index, GuiTabPage* const& value)
|
|
{
|
|
return !items.Contains(value) && value->tab == nullptr;
|
|
}
|
|
|
|
void GuiTabPageList::AfterInsert(vint index, GuiTabPage* const& value)
|
|
{
|
|
value->tab = tab;
|
|
value->SetVisible(false);
|
|
value->boundsComposition->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
tab->containerComposition->AddChild(value->boundsComposition);
|
|
|
|
if (!tab->selectedPage)
|
|
{
|
|
tab->SetSelectedPage(value);
|
|
}
|
|
}
|
|
|
|
void GuiTabPageList::BeforeRemove(vint index, GuiTabPage* const& value)
|
|
{
|
|
tab->containerComposition->RemoveChild(value->boundsComposition);
|
|
value->tab = nullptr;
|
|
|
|
if (items.Count() == 0)
|
|
{
|
|
tab->SetSelectedPage(nullptr);
|
|
}
|
|
else if (tab->selectedPage == value)
|
|
{
|
|
tab->SetSelectedPage(items[0]);
|
|
}
|
|
}
|
|
|
|
GuiTabPageList::GuiTabPageList(GuiTab* _tab)
|
|
:tab(_tab)
|
|
{
|
|
}
|
|
|
|
GuiTabPageList::~GuiTabPageList()
|
|
{
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiTab::CommandExecutor
|
|
***********************************************************************/
|
|
|
|
GuiTab::CommandExecutor::CommandExecutor(GuiTab* _tab)
|
|
:tab(_tab)
|
|
{
|
|
}
|
|
|
|
GuiTab::CommandExecutor::~CommandExecutor()
|
|
{
|
|
}
|
|
|
|
void GuiTab::CommandExecutor::ShowTab(vint index, bool setFocus)
|
|
{
|
|
tab->SetSelectedPage(tab->GetPages().Get(index));
|
|
if (setFocus)
|
|
{
|
|
tab->SetFocus();
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiTab
|
|
***********************************************************************/
|
|
|
|
void GuiTab::BeforeControlTemplateUninstalled_()
|
|
{
|
|
auto ct = GetControlTemplateObject(false);
|
|
if (!ct) return;
|
|
|
|
ct->SetCommands(nullptr);
|
|
ct->SetTabPages(nullptr);
|
|
ct->SetSelectedTabPage(nullptr);
|
|
}
|
|
|
|
void GuiTab::AfterControlTemplateInstalled_(bool initialize)
|
|
{
|
|
auto ct = GetControlTemplateObject(true);
|
|
ct->SetCommands(commandExecutor.Obj());
|
|
ct->SetTabPages(tabPages.GetWrapper());
|
|
ct->SetSelectedTabPage(selectedPage);
|
|
}
|
|
|
|
void GuiTab::OnKeyDown(compositions::GuiGraphicsComposition* sender, compositions::GuiKeyEventArgs& arguments)
|
|
{
|
|
if (arguments.eventSource == focusableComposition)
|
|
{
|
|
if (auto ct = GetControlTemplateObject(false))
|
|
{
|
|
vint index = tabPages.IndexOf(selectedPage);
|
|
if (index != -1)
|
|
{
|
|
auto hint = ct->GetTabOrder();
|
|
vint tabOffset = 0;
|
|
switch (hint)
|
|
{
|
|
case TabPageOrder::LeftToRight:
|
|
if (arguments.code == VKEY::_LEFT) tabOffset = -1;
|
|
else if (arguments.code == VKEY::_RIGHT) tabOffset = 1;
|
|
break;
|
|
case TabPageOrder::RightToLeft:
|
|
if (arguments.code == VKEY::_LEFT) tabOffset = 1;
|
|
else if (arguments.code == VKEY::_RIGHT) tabOffset = -1;
|
|
break;
|
|
case TabPageOrder::TopToBottom:
|
|
if (arguments.code == VKEY::_UP) tabOffset = -1;
|
|
else if (arguments.code == VKEY::_DOWN) tabOffset = 1;
|
|
break;
|
|
case TabPageOrder::BottomToTop:
|
|
if (arguments.code == VKEY::_UP) tabOffset = 1;
|
|
else if (arguments.code == VKEY::_DOWN) tabOffset = -1;
|
|
break;
|
|
default:;
|
|
}
|
|
|
|
if (tabOffset != 0)
|
|
{
|
|
arguments.handled = true;
|
|
index += tabOffset;
|
|
if (index < 0) index = 0;
|
|
else if (index >= tabPages.Count()) index = tabPages.Count() - 1;
|
|
|
|
SetSelectedPage(tabPages[index]);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
GuiTab::GuiTab(theme::ThemeName themeName)
|
|
:GuiControl(themeName)
|
|
, tabPages(this)
|
|
{
|
|
commandExecutor = new CommandExecutor(this);
|
|
SetFocusableComposition(boundsComposition);
|
|
|
|
boundsComposition->GetEventReceiver()->keyDown.AttachMethod(this, &GuiTab::OnKeyDown);
|
|
}
|
|
|
|
GuiTab::~GuiTab()
|
|
{
|
|
}
|
|
|
|
collections::ObservableList<GuiTabPage*>& GuiTab::GetPages()
|
|
{
|
|
return tabPages;
|
|
}
|
|
|
|
GuiTabPage* GuiTab::GetSelectedPage()
|
|
{
|
|
return selectedPage;
|
|
}
|
|
|
|
bool GuiTab::SetSelectedPage(GuiTabPage* value)
|
|
{
|
|
if (!value)
|
|
{
|
|
if (tabPages.Count() == 0)
|
|
{
|
|
selectedPage = nullptr;
|
|
}
|
|
}
|
|
else if (value->GetOwnerTab() == this)
|
|
{
|
|
if (selectedPage == value)
|
|
{
|
|
return true;
|
|
}
|
|
|
|
selectedPage = value;
|
|
FOREACH(GuiTabPage*, tabPage, tabPages)
|
|
{
|
|
tabPage->SetVisible(tabPage == selectedPage);
|
|
}
|
|
}
|
|
if (auto ct = GetControlTemplateObject(false))
|
|
{
|
|
ct->SetSelectedTabPage(selectedPage);
|
|
}
|
|
SelectedPageChanged.Execute(GetNotifyEventArguments());
|
|
return selectedPage == value;
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiScrollView
|
|
***********************************************************************/
|
|
|
|
void GuiScrollView::BeforeControlTemplateUninstalled_()
|
|
{
|
|
auto ct = GetControlTemplateObject(false);
|
|
if (!ct) return;
|
|
|
|
if (auto scroll = ct->GetHorizontalScroll())
|
|
{
|
|
scroll->PositionChanged.Detach(hScrollHandler);
|
|
}
|
|
if (auto scroll = ct->GetVerticalScroll())
|
|
{
|
|
scroll->PositionChanged.Detach(vScrollHandler);
|
|
}
|
|
ct->GetEventReceiver()->horizontalWheel.Detach(hWheelHandler);
|
|
ct->GetEventReceiver()->verticalWheel.Detach(vWheelHandler);
|
|
ct->BoundsChanged.Detach(containerBoundsChangedHandler);
|
|
|
|
hScrollHandler = nullptr;
|
|
vScrollHandler = nullptr;
|
|
hWheelHandler = nullptr;
|
|
vWheelHandler = nullptr;
|
|
containerBoundsChangedHandler = nullptr;
|
|
supressScrolling = false;
|
|
}
|
|
|
|
void GuiScrollView::AfterControlTemplateInstalled_(bool initialize)
|
|
{
|
|
auto ct = GetControlTemplateObject(true);
|
|
if (auto scroll = ct->GetHorizontalScroll())
|
|
{
|
|
hScrollHandler = scroll->PositionChanged.AttachMethod(this, &GuiScrollView::OnHorizontalScroll);
|
|
}
|
|
if (auto scroll = ct->GetVerticalScroll())
|
|
{
|
|
vScrollHandler = scroll->PositionChanged.AttachMethod(this, &GuiScrollView::OnVerticalScroll);
|
|
}
|
|
hWheelHandler = ct->GetEventReceiver()->horizontalWheel.AttachMethod(this, &GuiScrollView::OnHorizontalWheel);
|
|
vWheelHandler = ct->GetEventReceiver()->verticalWheel.AttachMethod(this, &GuiScrollView::OnVerticalWheel);
|
|
containerBoundsChangedHandler = ct->BoundsChanged.AttachMethod(this, &GuiScrollView::OnContainerBoundsChanged);
|
|
CalculateView();
|
|
}
|
|
|
|
void GuiScrollView::UpdateDisplayFont()
|
|
{
|
|
GuiControl::UpdateDisplayFont();
|
|
CalculateView();
|
|
}
|
|
|
|
void GuiScrollView::OnContainerBoundsChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
InvokeOrDelayIfRendering([=]()
|
|
{
|
|
CalculateView();
|
|
});
|
|
}
|
|
|
|
void GuiScrollView::OnHorizontalScroll(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
if(!supressScrolling)
|
|
{
|
|
CallUpdateView();
|
|
}
|
|
}
|
|
|
|
void GuiScrollView::OnVerticalScroll(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
if(!supressScrolling)
|
|
{
|
|
CallUpdateView();
|
|
}
|
|
}
|
|
|
|
void GuiScrollView::OnHorizontalWheel(compositions::GuiGraphicsComposition* sender, compositions::GuiMouseEventArgs& arguments)
|
|
{
|
|
if(!supressScrolling)
|
|
{
|
|
if (auto scroll = GetControlTemplateObject(true)->GetHorizontalScroll())
|
|
{
|
|
if (scroll->GetEnabled())
|
|
{
|
|
vint position = scroll->GetPosition();
|
|
vint move = scroll->GetSmallMove();
|
|
position -= move * arguments.wheel / 60;
|
|
scroll->SetPosition(position);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiScrollView::OnVerticalWheel(compositions::GuiGraphicsComposition* sender, compositions::GuiMouseEventArgs& arguments)
|
|
{
|
|
if(!supressScrolling && GetVisuallyEnabled())
|
|
{
|
|
if (auto scroll = GetControlTemplateObject(true)->GetVerticalScroll())
|
|
{
|
|
if (scroll->GetEnabled())
|
|
{
|
|
vint position = scroll->GetPosition();
|
|
vint move = scroll->GetSmallMove();
|
|
position -= move * arguments.wheel / 60;
|
|
scroll->SetPosition(position);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiScrollView::CallUpdateView()
|
|
{
|
|
Rect viewBounds=GetViewBounds();
|
|
UpdateView(viewBounds);
|
|
}
|
|
|
|
bool GuiScrollView::AdjustView(Size fullSize)
|
|
{
|
|
auto ct = GetControlTemplateObject(true);
|
|
auto hScroll = ct->GetHorizontalScroll();
|
|
auto vScroll = ct->GetVerticalScroll();
|
|
Size viewSize = ct->GetContainerComposition()->GetBounds().GetSize();
|
|
|
|
auto hVisible = hScroll ? hScroll->GetVisible() : false;
|
|
auto vVisible = vScroll ? vScroll->GetVisible() : false;
|
|
|
|
if (hScroll)
|
|
{
|
|
if (fullSize.x <= viewSize.x)
|
|
{
|
|
hScroll->SetVisible(horizontalAlwaysVisible);
|
|
hScroll->SetEnabled(false);
|
|
hScroll->SetPosition(0);
|
|
}
|
|
else
|
|
{
|
|
hScroll->SetVisible(true);
|
|
hScroll->SetEnabled(true);
|
|
hScroll->SetTotalSize(fullSize.x);
|
|
hScroll->SetPageSize(viewSize.x);
|
|
}
|
|
}
|
|
|
|
if (vScroll)
|
|
{
|
|
if (fullSize.y <= viewSize.y)
|
|
{
|
|
vScroll->SetVisible(verticalAlwaysVisible);
|
|
vScroll->SetEnabled(false);
|
|
vScroll->SetPosition(0);
|
|
}
|
|
else
|
|
{
|
|
vScroll->SetVisible(true);
|
|
vScroll->SetEnabled(true);
|
|
vScroll->SetTotalSize(fullSize.y);
|
|
vScroll->SetPageSize(viewSize.y);
|
|
}
|
|
}
|
|
|
|
auto hVisible2 = hScroll ? hScroll->GetVisible() : false;
|
|
auto vVisible2 = vScroll ? vScroll->GetVisible() : false;
|
|
return hVisible != hVisible2 || vVisible != vVisible2;
|
|
}
|
|
|
|
GuiScrollView::GuiScrollView(theme::ThemeName themeName)
|
|
:GuiControl(themeName)
|
|
{
|
|
containerComposition->BoundsChanged.AttachMethod(this, &GuiScrollView::OnContainerBoundsChanged);
|
|
}
|
|
|
|
vint GuiScrollView::GetSmallMove()
|
|
{
|
|
return GetDisplayFont().size * 2;
|
|
}
|
|
|
|
Size GuiScrollView::GetBigMove()
|
|
{
|
|
return GetViewSize();
|
|
}
|
|
|
|
GuiScrollView::~GuiScrollView()
|
|
{
|
|
}
|
|
|
|
void GuiScrollView::CalculateView()
|
|
{
|
|
auto ct = GetControlTemplateObject(true);
|
|
if (!supressScrolling)
|
|
{
|
|
Size fullSize = QueryFullSize();
|
|
while (true)
|
|
{
|
|
bool flagA = AdjustView(fullSize);
|
|
bool flagB = AdjustView(fullSize);
|
|
supressScrolling = true;
|
|
CallUpdateView();
|
|
supressScrolling = false;
|
|
|
|
Size newSize = QueryFullSize();
|
|
if (fullSize == newSize)
|
|
{
|
|
vint smallMove = GetSmallMove();
|
|
Size bigMove = GetBigMove();
|
|
if (auto scroll = ct->GetHorizontalScroll())
|
|
{
|
|
scroll->SetSmallMove(smallMove);
|
|
scroll->SetBigMove(bigMove.x);
|
|
}
|
|
if (auto scroll = ct->GetVerticalScroll())
|
|
{
|
|
scroll->SetSmallMove(smallMove);
|
|
scroll->SetBigMove(bigMove.y);
|
|
}
|
|
|
|
if (!flagA && !flagB)
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
fullSize = newSize;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
Size GuiScrollView::GetViewSize()
|
|
{
|
|
Size viewSize = GetControlTemplateObject(true)->GetContainerComposition()->GetBounds().GetSize();
|
|
return viewSize;
|
|
}
|
|
|
|
Rect GuiScrollView::GetViewBounds()
|
|
{
|
|
return Rect(GetViewPosition(), GetViewSize());
|
|
}
|
|
|
|
Point GuiScrollView::GetViewPosition()
|
|
{
|
|
auto ct = GetControlTemplateObject(true);
|
|
auto hScroll = ct->GetHorizontalScroll();
|
|
auto vScroll = ct->GetVerticalScroll();
|
|
return Point(hScroll ? hScroll->GetPosition() : 0, vScroll ? vScroll->GetPosition() : 0);
|
|
}
|
|
|
|
void GuiScrollView::SetViewPosition(Point value)
|
|
{
|
|
auto ct = GetControlTemplateObject(true);
|
|
if (auto hScroll = ct->GetHorizontalScroll())
|
|
{
|
|
hScroll->SetPosition(value.x);
|
|
}
|
|
if (auto vScroll = ct->GetVerticalScroll())
|
|
{
|
|
vScroll->SetPosition(value.y);
|
|
}
|
|
}
|
|
|
|
GuiScroll* GuiScrollView::GetHorizontalScroll()
|
|
{
|
|
return GetControlTemplateObject(true)->GetHorizontalScroll();
|
|
}
|
|
|
|
GuiScroll* GuiScrollView::GetVerticalScroll()
|
|
{
|
|
return GetControlTemplateObject(true)->GetVerticalScroll();
|
|
}
|
|
|
|
bool GuiScrollView::GetHorizontalAlwaysVisible()
|
|
{
|
|
return horizontalAlwaysVisible;
|
|
}
|
|
|
|
void GuiScrollView::SetHorizontalAlwaysVisible(bool value)
|
|
{
|
|
if (horizontalAlwaysVisible != value)
|
|
{
|
|
horizontalAlwaysVisible = value;
|
|
CalculateView();
|
|
}
|
|
}
|
|
|
|
bool GuiScrollView::GetVerticalAlwaysVisible()
|
|
{
|
|
return verticalAlwaysVisible;
|
|
}
|
|
|
|
void GuiScrollView::SetVerticalAlwaysVisible(bool value)
|
|
{
|
|
if (verticalAlwaysVisible != value)
|
|
{
|
|
verticalAlwaysVisible = value;
|
|
CalculateView();
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiScrollContainer
|
|
***********************************************************************/
|
|
|
|
Size GuiScrollContainer::QueryFullSize()
|
|
{
|
|
return containerComposition->GetBounds().GetSize();
|
|
}
|
|
|
|
void GuiScrollContainer::UpdateView(Rect viewBounds)
|
|
{
|
|
auto leftTop = Point(-viewBounds.x1, -viewBounds.y1);
|
|
containerComposition->SetBounds(Rect(leftTop, Size(0, 0)));
|
|
}
|
|
|
|
GuiScrollContainer::GuiScrollContainer(theme::ThemeName themeName)
|
|
:GuiScrollView(themeName)
|
|
{
|
|
containerComposition->SetAlignmentToParent(Margin(-1, -1, -1, -1));
|
|
UpdateView(Rect(0, 0, 0, 0));
|
|
}
|
|
|
|
GuiScrollContainer::~GuiScrollContainer()
|
|
{
|
|
}
|
|
|
|
bool GuiScrollContainer::GetExtendToFullWidth()
|
|
{
|
|
return extendToFullWidth;
|
|
}
|
|
|
|
void GuiScrollContainer::SetExtendToFullWidth(bool value)
|
|
{
|
|
if (extendToFullWidth != value)
|
|
{
|
|
extendToFullWidth = value;
|
|
auto margin = containerComposition->GetAlignmentToParent();
|
|
if (value)
|
|
{
|
|
containerComposition->SetAlignmentToParent(Margin(0, margin.top, 0, margin.bottom));
|
|
}
|
|
else
|
|
{
|
|
containerComposition->SetAlignmentToParent(Margin(-1, margin.top, -1, margin.bottom));
|
|
}
|
|
}
|
|
}
|
|
|
|
bool GuiScrollContainer::GetExtendToFullHeight()
|
|
{
|
|
return extendToFullHeight;
|
|
}
|
|
|
|
void GuiScrollContainer::SetExtendToFullHeight(bool value)
|
|
{
|
|
if (extendToFullHeight != value)
|
|
{
|
|
extendToFullHeight = value;
|
|
auto margin = containerComposition->GetAlignmentToParent();
|
|
if (value)
|
|
{
|
|
containerComposition->SetAlignmentToParent(Margin(margin.left, 0, margin.right, 0));
|
|
}
|
|
else
|
|
{
|
|
containerComposition->SetAlignmentToParent(Margin(margin.left, -1, margin.right, -1));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\CONTROLS\GUIDATETIMECONTROLS.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace controls
|
|
{
|
|
using namespace collections;
|
|
using namespace compositions;
|
|
using namespace elements;
|
|
|
|
/***********************************************************************
|
|
GuiDatePicker::CommandExecutor
|
|
***********************************************************************/
|
|
|
|
GuiDatePicker::CommandExecutor::CommandExecutor(GuiDatePicker* _datePicker)
|
|
:datePicker(_datePicker)
|
|
{
|
|
}
|
|
|
|
GuiDatePicker::CommandExecutor::~CommandExecutor()
|
|
{
|
|
}
|
|
|
|
void GuiDatePicker::CommandExecutor::NotifyDateChanged()
|
|
{
|
|
datePicker->date = datePicker->GetControlTemplateObject(true)->GetDate();
|
|
datePicker->UpdateText();
|
|
datePicker->DateChanged.Execute(datePicker->GetNotifyEventArguments());
|
|
}
|
|
|
|
void GuiDatePicker::CommandExecutor::NotifyDateNavigated()
|
|
{
|
|
datePicker->DateNavigated.Execute(datePicker->GetNotifyEventArguments());
|
|
}
|
|
|
|
void GuiDatePicker::CommandExecutor::NotifyDateSelected()
|
|
{
|
|
datePicker->DateSelected.Execute(datePicker->GetNotifyEventArguments());
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiDatePicker
|
|
***********************************************************************/
|
|
|
|
void GuiDatePicker::BeforeControlTemplateUninstalled_()
|
|
{
|
|
auto ct = GetControlTemplateObject(false);
|
|
if (!ct) return;
|
|
|
|
ct->SetCommands(nullptr);
|
|
}
|
|
|
|
void GuiDatePicker::AfterControlTemplateInstalled_(bool initialize)
|
|
{
|
|
auto ct = GetControlTemplateObject(true);
|
|
ct->SetCommands(commandExecutor.Obj());
|
|
ct->SetDate(date);
|
|
ct->SetDateLocale(dateLocale);
|
|
UpdateText();
|
|
}
|
|
|
|
void GuiDatePicker::UpdateText()
|
|
{
|
|
GuiControl::SetText(dateLocale.FormatDate(dateFormat, date));
|
|
}
|
|
|
|
bool GuiDatePicker::IsAltAvailable()
|
|
{
|
|
if (nestedAlt)
|
|
{
|
|
return alt != L"";
|
|
}
|
|
else
|
|
{
|
|
return GuiControl::IsAltAvailable();
|
|
}
|
|
}
|
|
|
|
compositions::IGuiAltActionHost* GuiDatePicker::GetActivatingAltHost()
|
|
{
|
|
if (nestedAlt)
|
|
{
|
|
return this;
|
|
}
|
|
else
|
|
{
|
|
return GuiControl::GetActivatingAltHost();
|
|
}
|
|
}
|
|
|
|
GuiDatePicker::GuiDatePicker(theme::ThemeName themeName, bool _nestedAlt)
|
|
:GuiControl(themeName)
|
|
, nestedAlt(_nestedAlt)
|
|
{
|
|
commandExecutor = new CommandExecutor(this);
|
|
SetDateLocale(Locale::UserDefault());
|
|
SetDate(DateTime::LocalTime());
|
|
SetAltComposition(boundsComposition);
|
|
SetAltControl(this, false);
|
|
|
|
DateChanged.SetAssociatedComposition(boundsComposition);
|
|
DateNavigated.SetAssociatedComposition(boundsComposition);
|
|
DateSelected.SetAssociatedComposition(boundsComposition);
|
|
DateFormatChanged.SetAssociatedComposition(boundsComposition);
|
|
DateLocaleChanged.SetAssociatedComposition(boundsComposition);
|
|
|
|
commandExecutor->NotifyDateChanged();
|
|
}
|
|
|
|
GuiDatePicker::~GuiDatePicker()
|
|
{
|
|
}
|
|
|
|
const DateTime& GuiDatePicker::GetDate()
|
|
{
|
|
return date;
|
|
}
|
|
|
|
void GuiDatePicker::SetDate(const DateTime& value)
|
|
{
|
|
if (date != value)
|
|
{
|
|
date = value;
|
|
GetControlTemplateObject(true)->SetDate(value);
|
|
}
|
|
}
|
|
|
|
const WString& GuiDatePicker::GetDateFormat()
|
|
{
|
|
return dateFormat;
|
|
}
|
|
|
|
void GuiDatePicker::SetDateFormat(const WString& value)
|
|
{
|
|
dateFormat=value;
|
|
UpdateText();
|
|
DateFormatChanged.Execute(GetNotifyEventArguments());
|
|
}
|
|
|
|
const Locale& GuiDatePicker::GetDateLocale()
|
|
{
|
|
return dateLocale;
|
|
}
|
|
|
|
void GuiDatePicker::SetDateLocale(const Locale& value)
|
|
{
|
|
dateLocale=value;
|
|
List<WString> formats;
|
|
dateLocale.GetLongDateFormats(formats);
|
|
if(formats.Count()>0)
|
|
{
|
|
dateFormat=formats[0];
|
|
}
|
|
GetControlTemplateObject(true)->SetDateLocale(dateLocale);
|
|
|
|
UpdateText();
|
|
DateFormatChanged.Execute(GetNotifyEventArguments());
|
|
DateLocaleChanged.Execute(GetNotifyEventArguments());
|
|
}
|
|
|
|
void GuiDatePicker::SetText(const WString& value)
|
|
{
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiDateComboBox
|
|
***********************************************************************/
|
|
|
|
void GuiDateComboBox::BeforeControlTemplateUninstalled_()
|
|
{
|
|
}
|
|
|
|
void GuiDateComboBox::AfterControlTemplateInstalled_(bool initialize)
|
|
{
|
|
auto ct = GetControlTemplateObject(true);
|
|
datePicker->SetControlTemplate(ct->GetDatePickerTemplate());
|
|
}
|
|
|
|
void GuiDateComboBox::UpdateText()
|
|
{
|
|
SetText(datePicker->GetDateLocale().FormatDate(datePicker->GetDateFormat(), selectedDate));
|
|
}
|
|
|
|
void GuiDateComboBox::NotifyUpdateSelectedDate()
|
|
{
|
|
UpdateText();
|
|
SelectedDateChanged.Execute(GetNotifyEventArguments());
|
|
}
|
|
|
|
void GuiDateComboBox::OnSubMenuOpeningChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
datePicker->SetDate(selectedDate);
|
|
}
|
|
|
|
void GuiDateComboBox::datePicker_DateLocaleChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
UpdateText();
|
|
}
|
|
|
|
void GuiDateComboBox::datePicker_DateFormatChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
UpdateText();
|
|
}
|
|
|
|
void GuiDateComboBox::datePicker_DateSelected(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
selectedDate=datePicker->GetDate();
|
|
GetSubMenu()->Hide();
|
|
NotifyUpdateSelectedDate();
|
|
}
|
|
|
|
GuiDateComboBox::GuiDateComboBox(theme::ThemeName themeName)
|
|
:GuiComboBoxBase(themeName)
|
|
{
|
|
SelectedDateChanged.SetAssociatedComposition(GetBoundsComposition());
|
|
|
|
datePicker = new GuiDatePicker(theme::ThemeName::DatePicker, false);
|
|
datePicker->DateSelected.AttachMethod(this, &GuiDateComboBox::datePicker_DateSelected);
|
|
datePicker->DateLocaleChanged.AttachMethod(this, &GuiDateComboBox::datePicker_DateLocaleChanged);
|
|
datePicker->DateFormatChanged.AttachMethod(this, &GuiDateComboBox::datePicker_DateFormatChanged);
|
|
datePicker->GetBoundsComposition()->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
|
|
GetSubMenu()->GetContainerComposition()->AddChild(datePicker->GetBoundsComposition());
|
|
GetSubMenu()->SetHideOnDeactivateAltHost(false);
|
|
|
|
selectedDate=datePicker->GetDate();
|
|
SubMenuOpeningChanged.AttachMethod(this, &GuiDateComboBox::OnSubMenuOpeningChanged);
|
|
SetFont(GetFont());
|
|
SetText(datePicker->GetText());
|
|
}
|
|
|
|
GuiDateComboBox::~GuiDateComboBox()
|
|
{
|
|
}
|
|
|
|
void GuiDateComboBox::SetFont(const Nullable<FontProperties>& value)
|
|
{
|
|
GuiComboBoxBase::SetFont(value);
|
|
datePicker->SetFont(value);
|
|
}
|
|
|
|
const DateTime& GuiDateComboBox::GetSelectedDate()
|
|
{
|
|
return selectedDate;
|
|
}
|
|
|
|
void GuiDateComboBox::SetSelectedDate(const DateTime& value)
|
|
{
|
|
selectedDate=value;
|
|
NotifyUpdateSelectedDate();
|
|
}
|
|
|
|
GuiDatePicker* GuiDateComboBox::GetDatePicker()
|
|
{
|
|
return datePicker;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\CONTROLS\GUIDIALOGS.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace controls
|
|
{
|
|
using namespace elements;
|
|
using namespace compositions;
|
|
using namespace collections;
|
|
using namespace reflection::description;
|
|
|
|
/***********************************************************************
|
|
GuiDialogBase
|
|
***********************************************************************/
|
|
|
|
GuiWindow* GuiDialogBase::GetHostWindow()
|
|
{
|
|
if (rootObject)
|
|
{
|
|
if (auto control = dynamic_cast<GuiControl*>(rootObject))
|
|
{
|
|
if (auto host = control->GetRelatedControlHost())
|
|
{
|
|
return dynamic_cast<GuiWindow*>(host);
|
|
}
|
|
}
|
|
else if (auto composition = dynamic_cast<GuiGraphicsComposition*>(rootObject))
|
|
{
|
|
if (auto host = composition->GetRelatedControlHost())
|
|
{
|
|
return dynamic_cast<GuiWindow*>(host);
|
|
}
|
|
}
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
GuiDialogBase::GuiDialogBase()
|
|
{
|
|
}
|
|
|
|
GuiDialogBase::~GuiDialogBase()
|
|
{
|
|
}
|
|
|
|
void GuiDialogBase::Attach(GuiInstanceRootObject* _rootObject)
|
|
{
|
|
rootObject = _rootObject;
|
|
}
|
|
|
|
void GuiDialogBase::Detach(GuiInstanceRootObject* _rootObject)
|
|
{
|
|
rootObject = nullptr;
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiMessageDialog
|
|
***********************************************************************/
|
|
|
|
GuiMessageDialog::GuiMessageDialog()
|
|
{
|
|
}
|
|
|
|
GuiMessageDialog::~GuiMessageDialog()
|
|
{
|
|
}
|
|
|
|
INativeDialogService::MessageBoxButtonsInput GuiMessageDialog::GetInput()
|
|
{
|
|
return input;
|
|
}
|
|
|
|
void GuiMessageDialog::SetInput(INativeDialogService::MessageBoxButtonsInput value)
|
|
{
|
|
input = value;
|
|
}
|
|
|
|
INativeDialogService::MessageBoxDefaultButton GuiMessageDialog::GetDefaultButton()
|
|
{
|
|
return defaultButton;
|
|
}
|
|
|
|
void GuiMessageDialog::SetDefaultButton(INativeDialogService::MessageBoxDefaultButton value)
|
|
{
|
|
defaultButton = value;
|
|
}
|
|
|
|
INativeDialogService::MessageBoxIcons GuiMessageDialog::GetIcon()
|
|
{
|
|
return icon;
|
|
}
|
|
|
|
void GuiMessageDialog::SetIcon(INativeDialogService::MessageBoxIcons value)
|
|
{
|
|
icon = value;
|
|
}
|
|
|
|
INativeDialogService::MessageBoxModalOptions GuiMessageDialog::GetModalOption()
|
|
{
|
|
return modalOption;
|
|
}
|
|
|
|
void GuiMessageDialog::SetModalOption(INativeDialogService::MessageBoxModalOptions value)
|
|
{
|
|
modalOption = value;
|
|
}
|
|
|
|
const WString& GuiMessageDialog::GetText()
|
|
{
|
|
return text;
|
|
}
|
|
|
|
void GuiMessageDialog::SetText(const WString& value)
|
|
{
|
|
text = value;
|
|
}
|
|
|
|
const WString& GuiMessageDialog::GetTitle()
|
|
{
|
|
return title;
|
|
}
|
|
|
|
void GuiMessageDialog::SetTitle(const WString& value)
|
|
{
|
|
title = value;
|
|
}
|
|
|
|
INativeDialogService::MessageBoxButtonsOutput GuiMessageDialog::ShowDialog()
|
|
{
|
|
auto service = GetCurrentController()->DialogService();
|
|
return service->ShowMessageBox(GetHostWindow()->GetNativeWindow(), text, title, input, defaultButton, icon, modalOption);
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiColorDialog
|
|
***********************************************************************/
|
|
|
|
GuiColorDialog::GuiColorDialog()
|
|
{
|
|
for (vint i = 0; i < 16; i++)
|
|
{
|
|
customColors.Add(Color());
|
|
}
|
|
}
|
|
|
|
GuiColorDialog::~GuiColorDialog()
|
|
{
|
|
}
|
|
|
|
bool GuiColorDialog::GetEnabledCustomColor()
|
|
{
|
|
return enabledCustomColor;
|
|
}
|
|
|
|
void GuiColorDialog::SetEnabledCustomColor(bool value)
|
|
{
|
|
enabledCustomColor = value;
|
|
}
|
|
|
|
bool GuiColorDialog::GetOpenedCustomColor()
|
|
{
|
|
return openedCustomColor;
|
|
}
|
|
|
|
void GuiColorDialog::SetOpenedCustomColor(bool value)
|
|
{
|
|
openedCustomColor = value;
|
|
}
|
|
|
|
Color GuiColorDialog::GetSelectedColor()
|
|
{
|
|
return selectedColor;
|
|
}
|
|
|
|
void GuiColorDialog::SetSelectedColor(Color value)
|
|
{
|
|
if (selectedColor != value)
|
|
{
|
|
selectedColor = value;
|
|
SelectedColorChanged.Execute(GuiEventArgs());
|
|
}
|
|
}
|
|
|
|
collections::List<Color>& GuiColorDialog::GetCustomColors()
|
|
{
|
|
return customColors;
|
|
}
|
|
|
|
bool GuiColorDialog::ShowDialog()
|
|
{
|
|
Array<Color> colors;
|
|
CopyFrom(colors, customColors);
|
|
colors.Resize(16);
|
|
|
|
INativeDialogService::ColorDialogCustomColorOptions options =
|
|
!enabledCustomColor ? INativeDialogService::CustomColorDisabled :
|
|
!openedCustomColor ? INativeDialogService::CustomColorEnabled :
|
|
INativeDialogService::CustomColorOpened;
|
|
|
|
auto service = GetCurrentController()->DialogService();
|
|
if (!service->ShowColorDialog(GetHostWindow()->GetNativeWindow(), selectedColor, showSelection, options, &colors[0]))
|
|
{
|
|
return false;
|
|
}
|
|
|
|
CopyFrom(customColors, colors);
|
|
SelectedColorChanged.Execute(GuiEventArgs());
|
|
return true;
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiFontDialog
|
|
***********************************************************************/
|
|
|
|
GuiFontDialog::GuiFontDialog()
|
|
{
|
|
}
|
|
|
|
GuiFontDialog::~GuiFontDialog()
|
|
{
|
|
}
|
|
|
|
const FontProperties& GuiFontDialog::GetSelectedFont()
|
|
{
|
|
return selectedFont;
|
|
}
|
|
|
|
void GuiFontDialog::SetSelectedFont(const FontProperties& value)
|
|
{
|
|
if (selectedFont != value)
|
|
{
|
|
selectedFont = value;
|
|
SelectedFontChanged.Execute(GuiEventArgs());
|
|
}
|
|
}
|
|
|
|
Color GuiFontDialog::GetSelectedColor()
|
|
{
|
|
return selectedColor;
|
|
}
|
|
|
|
void GuiFontDialog::SetSelectedColor(Color value)
|
|
{
|
|
if (selectedColor != value)
|
|
{
|
|
selectedColor = value;
|
|
SelectedColorChanged.Execute(GuiEventArgs());
|
|
}
|
|
}
|
|
|
|
bool GuiFontDialog::GetShowSelection()
|
|
{
|
|
return showSelection;
|
|
}
|
|
|
|
void GuiFontDialog::SetShowSelection(bool value)
|
|
{
|
|
showSelection = value;
|
|
}
|
|
|
|
bool GuiFontDialog::GetShowEffect()
|
|
{
|
|
return showEffect;
|
|
}
|
|
|
|
void GuiFontDialog::SetShowEffect(bool value)
|
|
{
|
|
showEffect = value;
|
|
}
|
|
|
|
bool GuiFontDialog::GetForceFontExist()
|
|
{
|
|
return forceFontExist;
|
|
}
|
|
|
|
void GuiFontDialog::SetForceFontExist(bool value)
|
|
{
|
|
forceFontExist = value;
|
|
}
|
|
|
|
bool GuiFontDialog::ShowDialog()
|
|
{
|
|
auto service = GetCurrentController()->DialogService();
|
|
if (!service->ShowFontDialog(GetHostWindow()->GetNativeWindow(), selectedFont, selectedColor, showSelection, showEffect, forceFontExist))
|
|
{
|
|
return false;
|
|
}
|
|
|
|
SelectedColorChanged.Execute(GuiEventArgs());
|
|
SelectedFontChanged.Execute(GuiEventArgs());
|
|
return true;
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiFileDialogBase
|
|
***********************************************************************/
|
|
|
|
GuiFileDialogBase::GuiFileDialogBase()
|
|
{
|
|
}
|
|
|
|
GuiFileDialogBase::~GuiFileDialogBase()
|
|
{
|
|
}
|
|
|
|
const WString& GuiFileDialogBase::GetFilter()
|
|
{
|
|
return filter;
|
|
}
|
|
|
|
void GuiFileDialogBase::SetFilter(const WString& value)
|
|
{
|
|
filter = value;
|
|
}
|
|
|
|
vint GuiFileDialogBase::GetFilterIndex()
|
|
{
|
|
return filterIndex;
|
|
}
|
|
|
|
void GuiFileDialogBase::SetFilterIndex(vint value)
|
|
{
|
|
if (filterIndex != value)
|
|
{
|
|
filterIndex = value;
|
|
FilterIndexChanged.Execute(GuiEventArgs());
|
|
}
|
|
}
|
|
|
|
bool GuiFileDialogBase::GetEnabledPreview()
|
|
{
|
|
return enabledPreview;
|
|
}
|
|
|
|
void GuiFileDialogBase::SetEnabledPreview(bool value)
|
|
{
|
|
enabledPreview = value;
|
|
}
|
|
|
|
WString GuiFileDialogBase::GetTitle()
|
|
{
|
|
return title;
|
|
}
|
|
|
|
void GuiFileDialogBase::SetTitle(const WString& value)
|
|
{
|
|
title = value;
|
|
}
|
|
|
|
WString GuiFileDialogBase::GetFileName()
|
|
{
|
|
return fileName;
|
|
}
|
|
|
|
void GuiFileDialogBase::SetFileName(const WString& value)
|
|
{
|
|
if (fileName != value)
|
|
{
|
|
FileNameChanged.Execute(GuiEventArgs());
|
|
}
|
|
}
|
|
|
|
WString GuiFileDialogBase::GetDirectory()
|
|
{
|
|
return directory;
|
|
}
|
|
|
|
void GuiFileDialogBase::SetDirectory(const WString& value)
|
|
{
|
|
directory = value;
|
|
}
|
|
|
|
WString GuiFileDialogBase::GetDefaultExtension()
|
|
{
|
|
return defaultExtension;
|
|
}
|
|
|
|
void GuiFileDialogBase::SetDefaultExtension(const WString& value)
|
|
{
|
|
defaultExtension = value;
|
|
}
|
|
|
|
INativeDialogService::FileDialogOptions GuiFileDialogBase::GetOptions()
|
|
{
|
|
return options;
|
|
}
|
|
|
|
void GuiFileDialogBase::SetOptions(INativeDialogService::FileDialogOptions value)
|
|
{
|
|
options = value;
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiOpenFileDialog
|
|
***********************************************************************/
|
|
|
|
GuiOpenFileDialog::GuiOpenFileDialog()
|
|
{
|
|
}
|
|
|
|
GuiOpenFileDialog::~GuiOpenFileDialog()
|
|
{
|
|
}
|
|
|
|
collections::List<WString>& GuiOpenFileDialog::GetFileNames()
|
|
{
|
|
return fileNames;
|
|
}
|
|
|
|
bool GuiOpenFileDialog::ShowDialog()
|
|
{
|
|
fileNames.Clear();
|
|
auto service = GetCurrentController()->DialogService();
|
|
if (!service->ShowFileDialog(
|
|
GetHostWindow()->GetNativeWindow(),
|
|
fileNames,
|
|
filterIndex,
|
|
(enabledPreview ? INativeDialogService::FileDialogOpenPreview : INativeDialogService::FileDialogOpen),
|
|
title,
|
|
fileName,
|
|
directory,
|
|
defaultExtension,
|
|
filter,
|
|
options))
|
|
{
|
|
return false;
|
|
}
|
|
|
|
if (fileNames.Count() > 0)
|
|
{
|
|
fileName = fileNames[0];
|
|
FileNameChanged.Execute(GuiEventArgs());
|
|
FilterIndexChanged.Execute(GuiEventArgs());
|
|
}
|
|
return true;
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiSaveFileDialog
|
|
***********************************************************************/
|
|
|
|
GuiSaveFileDialog::GuiSaveFileDialog()
|
|
{
|
|
}
|
|
|
|
GuiSaveFileDialog::~GuiSaveFileDialog()
|
|
{
|
|
}
|
|
|
|
bool GuiSaveFileDialog::ShowDialog()
|
|
{
|
|
List<WString> fileNames;
|
|
auto service = GetCurrentController()->DialogService();
|
|
if (!service->ShowFileDialog(
|
|
GetHostWindow()->GetNativeWindow(),
|
|
fileNames,
|
|
filterIndex,
|
|
(enabledPreview ? INativeDialogService::FileDialogSavePreview : INativeDialogService::FileDialogSave),
|
|
title,
|
|
fileName,
|
|
directory,
|
|
defaultExtension,
|
|
filter,
|
|
options))
|
|
{
|
|
return false;
|
|
}
|
|
|
|
if (fileNames.Count() > 0)
|
|
{
|
|
fileName = fileNames[0];
|
|
FileNameChanged.Execute(GuiEventArgs());
|
|
FilterIndexChanged.Execute(GuiEventArgs());
|
|
}
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\CONTROLS\GUILABELCONTROLS.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace controls
|
|
{
|
|
using namespace elements;
|
|
using namespace compositions;
|
|
using namespace collections;
|
|
using namespace reflection::description;
|
|
|
|
/***********************************************************************
|
|
GuiLabel
|
|
***********************************************************************/
|
|
|
|
void GuiLabel::BeforeControlTemplateUninstalled_()
|
|
{
|
|
auto ct = GetControlTemplateObject(false);
|
|
if (!ct) return;
|
|
|
|
textColorConsisted = (textColor == ct->GetDefaultTextColor());
|
|
}
|
|
|
|
void GuiLabel::AfterControlTemplateInstalled_(bool initialize)
|
|
{
|
|
auto ct = GetControlTemplateObject(true);
|
|
if (initialize || textColorConsisted)
|
|
{
|
|
SetTextColor(ct->GetDefaultTextColor());
|
|
}
|
|
else
|
|
{
|
|
ct->SetTextColor(textColor);
|
|
}
|
|
}
|
|
|
|
GuiLabel::GuiLabel(theme::ThemeName themeName)
|
|
:GuiControl(themeName)
|
|
{
|
|
}
|
|
|
|
GuiLabel::~GuiLabel()
|
|
{
|
|
}
|
|
|
|
Color GuiLabel::GetTextColor()
|
|
{
|
|
return textColor;
|
|
}
|
|
|
|
void GuiLabel::SetTextColor(Color value)
|
|
{
|
|
if (textColor != value)
|
|
{
|
|
textColor = value;
|
|
GetControlTemplateObject(true)->SetTextColor(textColor);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\CONTROLS\GUISCROLLCONTROLS.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace controls
|
|
{
|
|
using namespace elements;
|
|
using namespace compositions;
|
|
using namespace collections;
|
|
using namespace reflection::description;
|
|
|
|
/***********************************************************************
|
|
GuiScroll::CommandExecutor
|
|
***********************************************************************/
|
|
|
|
GuiScroll::CommandExecutor::CommandExecutor(GuiScroll* _scroll)
|
|
:scroll(_scroll)
|
|
{
|
|
}
|
|
|
|
GuiScroll::CommandExecutor::~CommandExecutor()
|
|
{
|
|
}
|
|
|
|
void GuiScroll::CommandExecutor::SmallDecrease()
|
|
{
|
|
scroll->SetPosition(scroll->GetPosition()-scroll->GetSmallMove());
|
|
}
|
|
|
|
void GuiScroll::CommandExecutor::SmallIncrease()
|
|
{
|
|
scroll->SetPosition(scroll->GetPosition()+scroll->GetSmallMove());
|
|
}
|
|
|
|
void GuiScroll::CommandExecutor::BigDecrease()
|
|
{
|
|
scroll->SetPosition(scroll->GetPosition()-scroll->GetBigMove());
|
|
}
|
|
|
|
void GuiScroll::CommandExecutor::BigIncrease()
|
|
{
|
|
scroll->SetPosition(scroll->GetPosition()+scroll->GetBigMove());
|
|
}
|
|
|
|
void GuiScroll::CommandExecutor::SetTotalSize(vint value)
|
|
{
|
|
scroll->SetTotalSize(value);
|
|
}
|
|
|
|
void GuiScroll::CommandExecutor::SetPageSize(vint value)
|
|
{
|
|
scroll->SetPageSize(value);
|
|
}
|
|
|
|
void GuiScroll::CommandExecutor::SetPosition(vint value)
|
|
{
|
|
scroll->SetPosition(value);
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiScroll
|
|
***********************************************************************/
|
|
|
|
void GuiScroll::OnActiveAlt()
|
|
{
|
|
if (autoFocus)
|
|
{
|
|
GuiControl::OnActiveAlt();
|
|
}
|
|
}
|
|
|
|
bool GuiScroll::IsTabAvailable()
|
|
{
|
|
return autoFocus && GuiControl::IsTabAvailable();
|
|
}
|
|
|
|
void GuiScroll::OnKeyDown(compositions::GuiGraphicsComposition* sender, compositions::GuiKeyEventArgs& arguments)
|
|
{
|
|
if (arguments.eventSource == focusableComposition)
|
|
{
|
|
switch (arguments.code)
|
|
{
|
|
case VKEY::_HOME:
|
|
SetPosition(GetMinPosition());
|
|
arguments.handled = true;
|
|
break;
|
|
case VKEY::_END:
|
|
SetPosition(GetMaxPosition());
|
|
arguments.handled = true;
|
|
break;
|
|
case VKEY::_PRIOR:
|
|
commandExecutor->BigDecrease();
|
|
arguments.handled = true;
|
|
break;
|
|
case VKEY::_NEXT:
|
|
commandExecutor->BigIncrease();
|
|
arguments.handled = true;
|
|
break;
|
|
case VKEY::_LEFT:
|
|
case VKEY::_UP:
|
|
commandExecutor->SmallDecrease();
|
|
arguments.handled = true;
|
|
break;
|
|
case VKEY::_RIGHT:
|
|
case VKEY::_DOWN:
|
|
commandExecutor->SmallIncrease();
|
|
arguments.handled = true;
|
|
break;
|
|
default:;
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiScroll::OnMouseDown(compositions::GuiGraphicsComposition* sender, compositions::GuiMouseEventArgs& arguments)
|
|
{
|
|
if (autoFocus)
|
|
{
|
|
SetFocus();
|
|
}
|
|
}
|
|
|
|
void GuiScroll::BeforeControlTemplateUninstalled_()
|
|
{
|
|
auto ct = GetControlTemplateObject(false);
|
|
if (!ct) return;
|
|
|
|
ct->SetCommands(nullptr);
|
|
}
|
|
|
|
void GuiScroll::AfterControlTemplateInstalled_(bool initialize)
|
|
{
|
|
auto ct = GetControlTemplateObject(true);
|
|
ct->SetCommands(commandExecutor.Obj());
|
|
ct->SetPageSize(pageSize);
|
|
ct->SetTotalSize(totalSize);
|
|
ct->SetPosition(position);
|
|
}
|
|
|
|
GuiScroll::GuiScroll(theme::ThemeName themeName)
|
|
:GuiControl(themeName)
|
|
{
|
|
SetFocusableComposition(boundsComposition);
|
|
|
|
TotalSizeChanged.SetAssociatedComposition(boundsComposition);
|
|
PageSizeChanged.SetAssociatedComposition(boundsComposition);
|
|
PositionChanged.SetAssociatedComposition(boundsComposition);
|
|
SmallMoveChanged.SetAssociatedComposition(boundsComposition);
|
|
BigMoveChanged.SetAssociatedComposition(boundsComposition);
|
|
|
|
commandExecutor = new CommandExecutor(this);
|
|
boundsComposition->GetEventReceiver()->keyDown.AttachMethod(this, &GuiScroll::OnKeyDown);
|
|
boundsComposition->GetEventReceiver()->leftButtonDown.AttachMethod(this, &GuiScroll::OnMouseDown);
|
|
boundsComposition->GetEventReceiver()->rightButtonDown.AttachMethod(this, &GuiScroll::OnMouseDown);
|
|
}
|
|
|
|
GuiScroll::~GuiScroll()
|
|
{
|
|
}
|
|
|
|
vint GuiScroll::GetTotalSize()
|
|
{
|
|
return totalSize;
|
|
}
|
|
|
|
void GuiScroll::SetTotalSize(vint value)
|
|
{
|
|
if(totalSize!=value && 0<value)
|
|
{
|
|
totalSize=value;
|
|
if(pageSize>totalSize)
|
|
{
|
|
SetPageSize(totalSize);
|
|
}
|
|
if(position>GetMaxPosition())
|
|
{
|
|
SetPosition(GetMaxPosition());
|
|
}
|
|
GetControlTemplateObject(true)->SetTotalSize(totalSize);
|
|
TotalSizeChanged.Execute(GetNotifyEventArguments());
|
|
}
|
|
}
|
|
|
|
vint GuiScroll::GetPageSize()
|
|
{
|
|
return pageSize;
|
|
}
|
|
|
|
void GuiScroll::SetPageSize(vint value)
|
|
{
|
|
if(pageSize!=value && 0<=value && value<=totalSize)
|
|
{
|
|
pageSize=value;
|
|
if(position>GetMaxPosition())
|
|
{
|
|
SetPosition(GetMaxPosition());
|
|
}
|
|
GetControlTemplateObject(true)->SetPageSize(pageSize);
|
|
PageSizeChanged.Execute(GetNotifyEventArguments());
|
|
}
|
|
}
|
|
|
|
vint GuiScroll::GetPosition()
|
|
{
|
|
return position;
|
|
}
|
|
|
|
void GuiScroll::SetPosition(vint value)
|
|
{
|
|
vint min=GetMinPosition();
|
|
vint max=GetMaxPosition();
|
|
vint newPosition=
|
|
value<min?min:
|
|
value>max?max:
|
|
value;
|
|
if(position!=newPosition)
|
|
{
|
|
position=newPosition;
|
|
GetControlTemplateObject(true)->SetPosition(position);
|
|
PositionChanged.Execute(GetNotifyEventArguments());
|
|
}
|
|
}
|
|
|
|
vint GuiScroll::GetSmallMove()
|
|
{
|
|
return smallMove;
|
|
}
|
|
|
|
void GuiScroll::SetSmallMove(vint value)
|
|
{
|
|
if(value>0 && smallMove!=value)
|
|
{
|
|
smallMove=value;
|
|
SmallMoveChanged.Execute(GetNotifyEventArguments());
|
|
}
|
|
}
|
|
|
|
vint GuiScroll::GetBigMove()
|
|
{
|
|
return bigMove;
|
|
}
|
|
|
|
void GuiScroll::SetBigMove(vint value)
|
|
{
|
|
if(value>0 && bigMove!=value)
|
|
{
|
|
bigMove=value;
|
|
BigMoveChanged.Execute(GetNotifyEventArguments());
|
|
}
|
|
}
|
|
|
|
vint GuiScroll::GetMinPosition()
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
vint GuiScroll::GetMaxPosition()
|
|
{
|
|
return totalSize-pageSize;
|
|
}
|
|
|
|
bool GuiScroll::GetAutoFocus()
|
|
{
|
|
return autoFocus;
|
|
}
|
|
|
|
void GuiScroll::SetAutoFocus(bool value)
|
|
{
|
|
autoFocus = value;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\CONTROLS\GUIWINDOWCONTROLS.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace controls
|
|
{
|
|
using namespace elements;
|
|
using namespace compositions;
|
|
using namespace collections;
|
|
using namespace reflection::description;
|
|
|
|
/***********************************************************************
|
|
GuiControlHost
|
|
***********************************************************************/
|
|
|
|
void GuiControlHost::OnNativeWindowChanged()
|
|
{
|
|
}
|
|
|
|
void GuiControlHost::OnVisualStatusChanged()
|
|
{
|
|
}
|
|
|
|
controls::GuiControlHost* GuiControlHost::GetControlHostForInstance()
|
|
{
|
|
return this;
|
|
}
|
|
|
|
GuiControl* GuiControlHost::GetTooltipOwner(Point location)
|
|
{
|
|
GuiGraphicsComposition* composition=this->GetBoundsComposition()->FindComposition(location, false);
|
|
if(composition)
|
|
{
|
|
GuiControl* control=composition->GetRelatedControl();
|
|
while(control)
|
|
{
|
|
if(control->GetTooltipControl())
|
|
{
|
|
return control;
|
|
}
|
|
control=control->GetParent();
|
|
}
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
void GuiControlHost::MoveIntoTooltipControl(GuiControl* tooltipControl, Point location)
|
|
{
|
|
if (tooltipLocation != location)
|
|
{
|
|
tooltipLocation = location;
|
|
{
|
|
GuiControl* currentOwner = GetApplication()->GetTooltipOwner();
|
|
if (currentOwner && currentOwner != tooltipControl)
|
|
{
|
|
if (tooltipCloseDelay)
|
|
{
|
|
tooltipCloseDelay->Cancel();
|
|
tooltipCloseDelay = 0;
|
|
}
|
|
GetApplication()->DelayExecuteInMainThread([=]()
|
|
{
|
|
currentOwner->CloseTooltip();
|
|
}, TooltipDelayCloseTime);
|
|
}
|
|
}
|
|
if (!tooltipControl)
|
|
{
|
|
if (tooltipOpenDelay)
|
|
{
|
|
tooltipOpenDelay->Cancel();
|
|
tooltipOpenDelay = 0;
|
|
}
|
|
}
|
|
else if (tooltipOpenDelay)
|
|
{
|
|
tooltipOpenDelay->Delay(TooltipDelayOpenTime);
|
|
}
|
|
else if (GetApplication()->GetTooltipOwner() != tooltipControl)
|
|
{
|
|
tooltipOpenDelay = GetApplication()->DelayExecuteInMainThread([this]()
|
|
{
|
|
GuiControl* owner = GetTooltipOwner(tooltipLocation);
|
|
if (owner)
|
|
{
|
|
Point offset = owner->GetBoundsComposition()->GetGlobalBounds().LeftTop();
|
|
Point p(tooltipLocation.x - offset.x, tooltipLocation.y - offset.y + 24);
|
|
owner->DisplayTooltip(p);
|
|
tooltipOpenDelay = 0;
|
|
|
|
tooltipCloseDelay = GetApplication()->DelayExecuteInMainThread([this, owner]()
|
|
{
|
|
owner->CloseTooltip();
|
|
}, TooltipDelayLifeTime);
|
|
}
|
|
}, TooltipDelayOpenTime);
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiControlHost::MouseMoving(const NativeWindowMouseInfo& info)
|
|
{
|
|
if (!info.left && !info.middle && !info.right)
|
|
{
|
|
GuiControl* tooltipControl = GetTooltipOwner(tooltipLocation);
|
|
MoveIntoTooltipControl(tooltipControl, Point(host->GetNativeWindow()->Convert(NativePoint(info.x, info.y))));
|
|
}
|
|
}
|
|
|
|
void GuiControlHost::MouseLeaved()
|
|
{
|
|
MoveIntoTooltipControl(0, Point(-1, -1));
|
|
}
|
|
|
|
void GuiControlHost::Moved()
|
|
{
|
|
OnVisualStatusChanged();
|
|
}
|
|
|
|
void GuiControlHost::Enabled()
|
|
{
|
|
GuiControl::SetEnabled(true);
|
|
OnVisualStatusChanged();
|
|
}
|
|
|
|
void GuiControlHost::Disabled()
|
|
{
|
|
GuiControl::SetEnabled(false);
|
|
OnVisualStatusChanged();
|
|
}
|
|
|
|
void GuiControlHost::GotFocus()
|
|
{
|
|
WindowGotFocus.Execute(GetNotifyEventArguments());
|
|
OnVisualStatusChanged();
|
|
}
|
|
|
|
void GuiControlHost::LostFocus()
|
|
{
|
|
WindowLostFocus.Execute(GetNotifyEventArguments());
|
|
OnVisualStatusChanged();
|
|
}
|
|
|
|
void GuiControlHost::Activated()
|
|
{
|
|
WindowActivated.Execute(GetNotifyEventArguments());
|
|
OnVisualStatusChanged();
|
|
}
|
|
|
|
void GuiControlHost::Deactivated()
|
|
{
|
|
WindowDeactivated.Execute(GetNotifyEventArguments());
|
|
OnVisualStatusChanged();
|
|
}
|
|
|
|
void GuiControlHost::Opened()
|
|
{
|
|
WindowOpened.Execute(GetNotifyEventArguments());
|
|
}
|
|
|
|
void GuiControlHost::Closing(bool& cancel)
|
|
{
|
|
GuiRequestEventArgs arguments(boundsComposition);
|
|
arguments.cancel=cancel;
|
|
WindowClosing.Execute(arguments);
|
|
if(!arguments.handled)
|
|
{
|
|
cancel=arguments.cancel;
|
|
}
|
|
}
|
|
|
|
void GuiControlHost::Closed()
|
|
{
|
|
WindowClosed.Execute(GetNotifyEventArguments());
|
|
}
|
|
|
|
void GuiControlHost::Destroying()
|
|
{
|
|
WindowDestroying.Execute(GetNotifyEventArguments());
|
|
calledDestroyed = true;
|
|
if (deleteWhenDestroyed)
|
|
{
|
|
GetApplication()->InvokeInMainThread(this, [=]()
|
|
{
|
|
delete this;
|
|
});
|
|
}
|
|
SetNativeWindow(nullptr);
|
|
}
|
|
|
|
void GuiControlHost::UpdateClientSizeAfterRendering(Size clientSize)
|
|
{
|
|
SetClientSize(clientSize);
|
|
}
|
|
|
|
GuiControlHost::GuiControlHost(theme::ThemeName themeName)
|
|
:GuiControl(themeName)
|
|
{
|
|
boundsComposition->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
|
|
WindowGotFocus.SetAssociatedComposition(boundsComposition);
|
|
WindowLostFocus.SetAssociatedComposition(boundsComposition);
|
|
WindowActivated.SetAssociatedComposition(boundsComposition);
|
|
WindowDeactivated.SetAssociatedComposition(boundsComposition);
|
|
WindowOpened.SetAssociatedComposition(boundsComposition);
|
|
WindowClosing.SetAssociatedComposition(boundsComposition);
|
|
WindowClosed.SetAssociatedComposition(boundsComposition);
|
|
WindowDestroying.SetAssociatedComposition(boundsComposition);
|
|
|
|
host=new GuiGraphicsHost(this, boundsComposition);
|
|
sharedPtrDestructorProc = 0;
|
|
}
|
|
|
|
GuiControlHost::~GuiControlHost()
|
|
{
|
|
FinalizeInstanceRecursively(this);
|
|
OnBeforeReleaseGraphicsHost();
|
|
delete host;
|
|
}
|
|
|
|
void GuiControlHost::DeleteAfterProcessingAllEvents()
|
|
{
|
|
auto window = host->GetNativeWindow();
|
|
if (calledDestroyed || !window)
|
|
{
|
|
delete this;
|
|
}
|
|
else
|
|
{
|
|
deleteWhenDestroyed = true;
|
|
GetCurrentController()->WindowService()->DestroyNativeWindow(window);
|
|
}
|
|
}
|
|
|
|
compositions::GuiGraphicsHost* GuiControlHost::GetGraphicsHost()
|
|
{
|
|
return host;
|
|
}
|
|
|
|
compositions::GuiGraphicsComposition* GuiControlHost::GetMainComposition()
|
|
{
|
|
return host->GetMainComposition();
|
|
}
|
|
|
|
INativeWindow* GuiControlHost::GetNativeWindow()
|
|
{
|
|
return host->GetNativeWindow();
|
|
}
|
|
|
|
void GuiControlHost::SetNativeWindow(INativeWindow* window)
|
|
{
|
|
if(host->GetNativeWindow())
|
|
{
|
|
host->GetNativeWindow()->UninstallListener(this);
|
|
}
|
|
host->SetNativeWindow(window);
|
|
if(host->GetNativeWindow())
|
|
{
|
|
host->GetNativeWindow()->InstallListener(this);
|
|
}
|
|
OnNativeWindowChanged();
|
|
}
|
|
|
|
void GuiControlHost::ForceCalculateSizeImmediately()
|
|
{
|
|
auto size = GetClientSize();
|
|
boundsComposition->ForceCalculateSizeImmediately();
|
|
SetClientSize(size);
|
|
}
|
|
|
|
bool GuiControlHost::GetEnabled()
|
|
{
|
|
if(host->GetNativeWindow())
|
|
{
|
|
return host->GetNativeWindow()->IsEnabled();
|
|
}
|
|
else
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
void GuiControlHost::SetEnabled(bool value)
|
|
{
|
|
if(host->GetNativeWindow())
|
|
{
|
|
if(value)
|
|
{
|
|
host->GetNativeWindow()->Enable();
|
|
}
|
|
else
|
|
{
|
|
host->GetNativeWindow()->Disable();
|
|
}
|
|
}
|
|
}
|
|
|
|
bool GuiControlHost::GetFocused()
|
|
{
|
|
if(host->GetNativeWindow())
|
|
{
|
|
return host->GetNativeWindow()->IsFocused();
|
|
}
|
|
else
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
void GuiControlHost::SetFocused()
|
|
{
|
|
if(host->GetNativeWindow())
|
|
{
|
|
host->GetNativeWindow()->SetFocus();
|
|
}
|
|
}
|
|
|
|
bool GuiControlHost::GetActivated()
|
|
{
|
|
if(host->GetNativeWindow())
|
|
{
|
|
return host->GetNativeWindow()->IsActivated();
|
|
}
|
|
else
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
void GuiControlHost::SetActivated()
|
|
{
|
|
if(host->GetNativeWindow())
|
|
{
|
|
host->GetNativeWindow()->SetActivate();
|
|
}
|
|
}
|
|
|
|
bool GuiControlHost::GetShowInTaskBar()
|
|
{
|
|
if(host->GetNativeWindow())
|
|
{
|
|
return host->GetNativeWindow()->IsAppearedInTaskBar();
|
|
}
|
|
else
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
void GuiControlHost::SetShowInTaskBar(bool value)
|
|
{
|
|
if(host->GetNativeWindow())
|
|
{
|
|
if(value)
|
|
{
|
|
host->GetNativeWindow()->ShowInTaskBar();
|
|
}
|
|
else
|
|
{
|
|
host->GetNativeWindow()->HideInTaskBar();
|
|
}
|
|
}
|
|
}
|
|
|
|
bool GuiControlHost::GetEnabledActivate()
|
|
{
|
|
if(host->GetNativeWindow())
|
|
{
|
|
return host->GetNativeWindow()->IsEnabledActivate();
|
|
}
|
|
else
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
void GuiControlHost::SetEnabledActivate(bool value)
|
|
{
|
|
if(host->GetNativeWindow())
|
|
{
|
|
if(value)
|
|
{
|
|
host->GetNativeWindow()->EnableActivate();
|
|
}
|
|
else
|
|
{
|
|
host->GetNativeWindow()->DisableActivate();
|
|
}
|
|
}
|
|
}
|
|
|
|
bool GuiControlHost::GetTopMost()
|
|
{
|
|
if(host->GetNativeWindow())
|
|
{
|
|
return host->GetNativeWindow()->GetTopMost();
|
|
}
|
|
else
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
void GuiControlHost::SetTopMost(bool topmost)
|
|
{
|
|
if(host->GetNativeWindow())
|
|
{
|
|
host->GetNativeWindow()->SetTopMost(topmost);
|
|
}
|
|
}
|
|
|
|
compositions::IGuiShortcutKeyManager* GuiControlHost::GetShortcutKeyManager()
|
|
{
|
|
return host->GetShortcutKeyManager();
|
|
}
|
|
|
|
void GuiControlHost::SetShortcutKeyManager(compositions::IGuiShortcutKeyManager* value)
|
|
{
|
|
host->SetShortcutKeyManager(value);
|
|
}
|
|
|
|
compositions::GuiGraphicsTimerManager* GuiControlHost::GetTimerManager()
|
|
{
|
|
return host->GetTimerManager();
|
|
}
|
|
|
|
Size GuiControlHost::GetClientSize()
|
|
{
|
|
if (auto window = host->GetNativeWindow())
|
|
{
|
|
return window->Convert(window->GetClientSize());
|
|
}
|
|
else
|
|
{
|
|
return Size(0, 0);
|
|
}
|
|
}
|
|
|
|
void GuiControlHost::SetClientSize(Size value)
|
|
{
|
|
if (auto window = host->GetNativeWindow())
|
|
{
|
|
host->GetNativeWindow()->SetClientSize(window->Convert(value));
|
|
}
|
|
}
|
|
|
|
NativePoint GuiControlHost::GetLocation()
|
|
{
|
|
if(auto window = host->GetNativeWindow())
|
|
{
|
|
return window->GetBounds().LeftTop();
|
|
}
|
|
else
|
|
{
|
|
return NativePoint();
|
|
}
|
|
}
|
|
|
|
void GuiControlHost::SetLocation(NativePoint value)
|
|
{
|
|
if (auto window = host->GetNativeWindow())
|
|
{
|
|
auto bounds = window->GetBounds();
|
|
window->SetBounds(NativeRect(value, bounds.GetSize()));
|
|
}
|
|
}
|
|
|
|
void GuiControlHost::SetBounds(NativePoint location, Size size)
|
|
{
|
|
if (auto window = host->GetNativeWindow())
|
|
{
|
|
window->SetBounds(NativeRect(location, window->Convert(size)));
|
|
}
|
|
}
|
|
|
|
GuiControlHost* GuiControlHost::GetRelatedControlHost()
|
|
{
|
|
return this;
|
|
}
|
|
|
|
const WString& GuiControlHost::GetText()
|
|
{
|
|
WString result;
|
|
if(host->GetNativeWindow())
|
|
{
|
|
result=host->GetNativeWindow()->GetTitle();
|
|
}
|
|
if(result!=GuiControl::GetText())
|
|
{
|
|
GuiControl::SetText(result);
|
|
}
|
|
return GuiControl::GetText();
|
|
}
|
|
|
|
void GuiControlHost::SetText(const WString& value)
|
|
{
|
|
if(host->GetNativeWindow())
|
|
{
|
|
host->GetNativeWindow()->SetTitle(value);
|
|
GuiControl::SetText(value);
|
|
}
|
|
}
|
|
|
|
INativeScreen* GuiControlHost::GetRelatedScreen()
|
|
{
|
|
if(host->GetNativeWindow())
|
|
{
|
|
return GetCurrentController()->ScreenService()->GetScreen(host->GetNativeWindow());
|
|
}
|
|
else
|
|
{
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
void GuiControlHost::Show()
|
|
{
|
|
if(host->GetNativeWindow())
|
|
{
|
|
host->GetNativeWindow()->Show();
|
|
}
|
|
}
|
|
|
|
void GuiControlHost::ShowDeactivated()
|
|
{
|
|
if(host->GetNativeWindow())
|
|
{
|
|
host->GetNativeWindow()->ShowDeactivated();
|
|
}
|
|
}
|
|
|
|
void GuiControlHost::ShowRestored()
|
|
{
|
|
if(host->GetNativeWindow())
|
|
{
|
|
host->GetNativeWindow()->ShowRestored();
|
|
}
|
|
}
|
|
|
|
void GuiControlHost::ShowMaximized()
|
|
{
|
|
if(host->GetNativeWindow())
|
|
{
|
|
host->GetNativeWindow()->ShowMaximized();
|
|
}
|
|
}
|
|
|
|
void GuiControlHost::ShowMinimized()
|
|
{
|
|
if(host->GetNativeWindow())
|
|
{
|
|
host->GetNativeWindow()->ShowMinimized();
|
|
}
|
|
}
|
|
|
|
void GuiControlHost::Hide()
|
|
{
|
|
if(host->GetNativeWindow())
|
|
{
|
|
host->GetNativeWindow()->Hide(false);
|
|
}
|
|
}
|
|
|
|
void GuiControlHost::Close()
|
|
{
|
|
if (auto window = host->GetNativeWindow())
|
|
{
|
|
auto mainWindow = GetCurrentController()->WindowService()->GetMainWindow();
|
|
if (mainWindow == window)
|
|
{
|
|
SetNativeWindow(nullptr);
|
|
GetCurrentController()->WindowService()->DestroyNativeWindow(window);
|
|
}
|
|
else
|
|
{
|
|
window->Hide(false);
|
|
}
|
|
}
|
|
}
|
|
|
|
bool GuiControlHost::GetOpening()
|
|
{
|
|
INativeWindow* window=host->GetNativeWindow();
|
|
if(window)
|
|
{
|
|
return window->IsVisible();
|
|
}
|
|
return false;
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiWindow
|
|
***********************************************************************/
|
|
|
|
void GuiWindow::BeforeControlTemplateUninstalled_()
|
|
{
|
|
}
|
|
|
|
void GuiWindow::AfterControlTemplateInstalled_(bool initialize)
|
|
{
|
|
auto ct = GetControlTemplateObject(true);
|
|
#define FIX_WINDOW_PROPERTY(VARIABLE, NAME) \
|
|
switch (ct->Get ## NAME ## Option()) \
|
|
{ \
|
|
case templates::BoolOption::AlwaysTrue: \
|
|
VARIABLE = true; \
|
|
break; \
|
|
case templates::BoolOption::AlwaysFalse: \
|
|
VARIABLE = false; \
|
|
break; \
|
|
default:; \
|
|
} \
|
|
|
|
FIX_WINDOW_PROPERTY(hasMaximizedBox, MaximizedBox)
|
|
FIX_WINDOW_PROPERTY(hasMinimizedBox, MinimizedBox)
|
|
FIX_WINDOW_PROPERTY(hasBorder, Border)
|
|
FIX_WINDOW_PROPERTY(hasSizeBox, SizeBox)
|
|
FIX_WINDOW_PROPERTY(isIconVisible, IconVisible)
|
|
FIX_WINDOW_PROPERTY(hasTitleBar, TitleBar)
|
|
|
|
#undef FIX_WINDOW_PROPERTY
|
|
ct->SetMaximizedBox(hasMaximizedBox);
|
|
ct->SetMinimizedBox(hasMinimizedBox);
|
|
ct->SetBorder(hasBorder);
|
|
ct->SetSizeBox(hasSizeBox);
|
|
ct->SetIconVisible(isIconVisible);
|
|
ct->SetTitleBar(hasTitleBar);
|
|
ct->SetMaximized(GetNativeWindow()->GetSizeState() != INativeWindow::Maximized);
|
|
ct->SetActivated(GetActivated());
|
|
|
|
auto window = GetNativeWindow();
|
|
if (window)
|
|
{
|
|
window->SetIcon(icon);
|
|
}
|
|
UpdateCustomFramePadding(window, ct);
|
|
|
|
ct->SetIcon(icon ? icon : window ? window->GetIcon() : nullptr);
|
|
SyncNativeWindowProperties();
|
|
}
|
|
|
|
void GuiWindow::UpdateCustomFramePadding(INativeWindow* window, templates::GuiWindowTemplate* ct)
|
|
{
|
|
if (auto window = GetNativeWindow())
|
|
{
|
|
ct->SetCustomFramePadding(window->Convert(window->GetCustomFramePadding()));
|
|
}
|
|
else
|
|
{
|
|
ct->SetCustomFramePadding({8, 8, 8, 8});
|
|
}
|
|
}
|
|
|
|
void GuiWindow::SyncNativeWindowProperties()
|
|
{
|
|
if (auto window = GetNativeWindow())
|
|
{
|
|
if (GetControlTemplateObject(true)->GetCustomFrameEnabled())
|
|
{
|
|
window->EnableCustomFrameMode();
|
|
window->SetBorder(false);
|
|
}
|
|
else
|
|
{
|
|
window->DisableCustomFrameMode();
|
|
window->SetBorder(hasBorder);
|
|
}
|
|
|
|
window->SetMaximizedBox(hasMaximizedBox);
|
|
window->SetMinimizedBox(hasMinimizedBox);
|
|
window->SetSizeBox(hasSizeBox);
|
|
window->SetIconVisible(isIconVisible);
|
|
window->SetTitleBar(hasTitleBar);
|
|
}
|
|
}
|
|
|
|
void GuiWindow::Moved()
|
|
{
|
|
GuiControlHost::Moved();
|
|
GetControlTemplateObject(true)->SetMaximized(GetNativeWindow()->GetSizeState() != INativeWindow::Maximized);
|
|
}
|
|
|
|
void GuiWindow::DpiChanged()
|
|
{
|
|
if (auto ct = GetControlTemplateObject(false))
|
|
{
|
|
UpdateCustomFramePadding(GetNativeWindow(), ct);
|
|
}
|
|
}
|
|
|
|
void GuiWindow::OnNativeWindowChanged()
|
|
{
|
|
SyncNativeWindowProperties();
|
|
GuiControlHost::OnNativeWindowChanged();
|
|
}
|
|
|
|
void GuiWindow::OnVisualStatusChanged()
|
|
{
|
|
GuiControlHost::OnVisualStatusChanged();
|
|
}
|
|
|
|
void GuiWindow::MouseClickedOnOtherWindow(GuiWindow* window)
|
|
{
|
|
}
|
|
|
|
void GuiWindow::OnWindowActivated(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
if (auto ct = GetControlTemplateObject(false))
|
|
{
|
|
ct->SetActivated(true);
|
|
}
|
|
}
|
|
|
|
void GuiWindow::OnWindowDeactivated(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
if (auto ct = GetControlTemplateObject(false))
|
|
{
|
|
ct->SetActivated(false);
|
|
}
|
|
}
|
|
|
|
GuiWindow::GuiWindow(theme::ThemeName themeName)
|
|
:GuiControlHost(themeName)
|
|
{
|
|
SetAltComposition(boundsComposition);
|
|
SetAltControl(this, true);
|
|
|
|
INativeWindow* window = GetCurrentController()->WindowService()->CreateNativeWindow();
|
|
SetNativeWindow(window);
|
|
GetApplication()->RegisterWindow(this);
|
|
ClipboardUpdated.SetAssociatedComposition(boundsComposition);
|
|
|
|
WindowActivated.AttachMethod(this, &GuiWindow::OnWindowActivated);
|
|
WindowDeactivated.AttachMethod(this, &GuiWindow::OnWindowDeactivated);
|
|
}
|
|
|
|
GuiWindow::~GuiWindow()
|
|
{
|
|
FinalizeAggregation();
|
|
GetApplication()->UnregisterWindow(this);
|
|
INativeWindow* window=host->GetNativeWindow();
|
|
if(window)
|
|
{
|
|
SetNativeWindow(nullptr);
|
|
GetCurrentController()->WindowService()->DestroyNativeWindow(window);
|
|
}
|
|
}
|
|
|
|
IDescriptable* GuiWindow::QueryService(const WString& identifier)
|
|
{
|
|
if (identifier == IGuiAltActionHost::Identifier)
|
|
{
|
|
return (IGuiAltActionHost*)this;
|
|
}
|
|
else
|
|
{
|
|
return GuiControlHost::QueryService(identifier);
|
|
}
|
|
}
|
|
|
|
void GuiWindow::MoveToScreenCenter()
|
|
{
|
|
MoveToScreenCenter(GetRelatedScreen());
|
|
}
|
|
|
|
void GuiWindow::MoveToScreenCenter(INativeScreen* screen)
|
|
{
|
|
if (screen)
|
|
{
|
|
if (auto window = host->GetNativeWindow())
|
|
{
|
|
NativeRect screenBounds = screen->GetClientBounds();
|
|
NativeSize windowSize = window->GetBounds().GetSize();
|
|
SetLocation(
|
|
NativePoint(
|
|
screenBounds.Left() + (screenBounds.Width() - windowSize.x) / 2,
|
|
screenBounds.Top() + (screenBounds.Height() - windowSize.y) / 2
|
|
)
|
|
);
|
|
}
|
|
}
|
|
}
|
|
|
|
#define IMPL_WINDOW_PROPERTY(VARIABLE, NAME, CONDITION_BREAK) \
|
|
bool GuiWindow::Get ## NAME() \
|
|
{ \
|
|
return VARIABLE; \
|
|
} \
|
|
void GuiWindow::Set ## NAME(bool visible) \
|
|
{ \
|
|
auto ct = GetControlTemplateObject(true); \
|
|
if (ct->Get ## NAME ## Option() == templates::BoolOption::Customizable) \
|
|
{ \
|
|
VARIABLE = visible; \
|
|
ct->Set ## NAME(visible); \
|
|
auto window = GetNativeWindow(); \
|
|
if (window) \
|
|
{ \
|
|
CONDITION_BREAK \
|
|
window->Set ## NAME(visible); \
|
|
} \
|
|
UpdateCustomFramePadding(window, ct); \
|
|
} \
|
|
} \
|
|
|
|
#define IMPL_WINDOW_PROPERTY_EMPTY_CONDITION
|
|
#define IMPL_WINDOW_PROPERTY_BORDER_CONDITION if (!ct->GetCustomFrameEnabled())
|
|
|
|
IMPL_WINDOW_PROPERTY(hasMaximizedBox, MaximizedBox, IMPL_WINDOW_PROPERTY_EMPTY_CONDITION)
|
|
IMPL_WINDOW_PROPERTY(hasMinimizedBox, MinimizedBox, IMPL_WINDOW_PROPERTY_EMPTY_CONDITION)
|
|
IMPL_WINDOW_PROPERTY(hasBorder, Border, IMPL_WINDOW_PROPERTY_BORDER_CONDITION)
|
|
IMPL_WINDOW_PROPERTY(hasSizeBox, SizeBox, IMPL_WINDOW_PROPERTY_EMPTY_CONDITION)
|
|
IMPL_WINDOW_PROPERTY(isIconVisible, IconVisible, IMPL_WINDOW_PROPERTY_EMPTY_CONDITION)
|
|
IMPL_WINDOW_PROPERTY(hasTitleBar, TitleBar, IMPL_WINDOW_PROPERTY_EMPTY_CONDITION)
|
|
|
|
Ptr<GuiImageData> GuiWindow::GetIcon()
|
|
{
|
|
return icon;
|
|
}
|
|
|
|
void GuiWindow::SetIcon(Ptr<GuiImageData> value)
|
|
{
|
|
if (icon != value)
|
|
{
|
|
icon = value;
|
|
|
|
auto window = GetNativeWindow();
|
|
if (window)
|
|
{
|
|
window->SetIcon(icon);
|
|
}
|
|
|
|
if (auto ct = GetControlTemplateObject(false))
|
|
{
|
|
ct->SetIcon(icon ? icon : window ? window->GetIcon() : nullptr);
|
|
}
|
|
}
|
|
}
|
|
|
|
#undef IMPL_WINDOW_PROPERTY_BORDER_CONDITION
|
|
#undef IMPL_WINDOW_PROPERTY_EMPTY_CONDITION
|
|
#undef IMPL_WINDOW_PROPERTY
|
|
|
|
void GuiWindow::ShowModal(GuiWindow* owner, const Func<void()>& callback)
|
|
{
|
|
owner->SetEnabled(false);
|
|
GetNativeWindow()->SetParent(owner->GetNativeWindow());
|
|
auto container = MakePtr<IGuiGraphicsEventHandler::Container>();
|
|
container->handler = WindowClosed.AttachLambda([=](GuiGraphicsComposition* sender, GuiEventArgs& arguments)
|
|
{
|
|
GetApplication()->InvokeInMainThread(this, [=]()
|
|
{
|
|
WindowClosed.Detach(container->handler);
|
|
container->handler = nullptr;
|
|
GetNativeWindow()->SetParent(nullptr);
|
|
callback();
|
|
owner->SetEnabled(true);
|
|
owner->SetActivated();
|
|
});
|
|
});
|
|
Show();
|
|
}
|
|
|
|
void GuiWindow::ShowModalAndDelete(GuiWindow* owner, const Func<void()>& callback)
|
|
{
|
|
ShowModal(owner, [=]()
|
|
{
|
|
callback();
|
|
DeleteAfterProcessingAllEvents();
|
|
});
|
|
}
|
|
|
|
Ptr<reflection::description::IAsync> GuiWindow::ShowModalAsync(GuiWindow* owner)
|
|
{
|
|
auto future = IFuture::Create();
|
|
ShowModal(owner, [promise = future->GetPromise()]()
|
|
{
|
|
promise->SendResult({});
|
|
});
|
|
return future;
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiPopup
|
|
***********************************************************************/
|
|
|
|
void GuiPopup::UpdateClientSizeAfterRendering(Size clientSize)
|
|
{
|
|
if (popupType == -1)
|
|
{
|
|
GuiWindow::UpdateClientSizeAfterRendering(clientSize);
|
|
}
|
|
else
|
|
{
|
|
auto window = host->GetNativeWindow();
|
|
auto currentClientSize = window->GetClientSize();
|
|
auto currentWindowSize = window->GetBounds().GetSize();
|
|
auto offsetX = currentWindowSize.x - currentClientSize.x;
|
|
auto offsetY = currentWindowSize.y - currentClientSize.y;
|
|
auto nativeClientSize = window->Convert(clientSize);
|
|
auto position = CalculatePopupPosition(NativeSize(nativeClientSize.x + offsetX, nativeClientSize.y + offsetY), popupType, popupInfo);
|
|
SetBounds(position, clientSize);
|
|
}
|
|
}
|
|
|
|
void GuiPopup::MouseClickedOnOtherWindow(GuiWindow* window)
|
|
{
|
|
Hide();
|
|
}
|
|
|
|
void GuiPopup::PopupOpened(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
GetApplication()->RegisterPopupOpened(this);
|
|
}
|
|
|
|
void GuiPopup::PopupClosed(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
popupType = -1;
|
|
GetApplication()->RegisterPopupClosed(this);
|
|
if(auto window = GetNativeWindow())
|
|
{
|
|
window->SetParent(nullptr);
|
|
}
|
|
}
|
|
|
|
void GuiPopup::OnKeyDown(compositions::GuiGraphicsComposition* sender, compositions::GuiKeyEventArgs& arguments)
|
|
{
|
|
if (!arguments.handled)
|
|
{
|
|
Hide();
|
|
}
|
|
}
|
|
|
|
bool GuiPopup::IsClippedByScreen(NativeSize size, NativePoint location, INativeScreen* screen)
|
|
{
|
|
NativeRect screenBounds = screen->GetClientBounds();
|
|
NativeRect windowBounds(location, size);
|
|
return !screenBounds.Contains(windowBounds.LeftTop()) || !screenBounds.Contains(windowBounds.RightBottom());
|
|
}
|
|
|
|
NativePoint GuiPopup::CalculatePopupPosition(NativeSize windowSize, NativePoint location, INativeScreen* screen)
|
|
{
|
|
NativeRect screenBounds = screen->GetClientBounds();
|
|
|
|
if (location.x < screenBounds.x1)
|
|
{
|
|
location.x = screenBounds.x1;
|
|
}
|
|
else if (location.x + windowSize.x > screenBounds.x2)
|
|
{
|
|
location.x = screenBounds.x2 - windowSize.x;
|
|
}
|
|
|
|
if (location.y < screenBounds.y1)
|
|
{
|
|
location.y = screenBounds.y1;
|
|
}
|
|
else if (location.y + windowSize.y > screenBounds.y2)
|
|
{
|
|
location.y = screenBounds.y2 - windowSize.y;
|
|
}
|
|
|
|
return location;
|
|
}
|
|
|
|
NativePoint GuiPopup::CalculatePopupPosition(NativeSize windowSize, GuiControl* control, INativeWindow* controlWindow, Rect bounds, bool preferredTopBottomSide)
|
|
{
|
|
NativePoint controlClientOffset = controlWindow->Convert(control->GetBoundsComposition()->GetGlobalBounds().LeftTop());
|
|
NativePoint controlWindowOffset = controlWindow->GetClientBoundsInScreen().LeftTop();
|
|
NativeRect targetBounds(controlWindow->Convert(bounds.LeftTop()), controlWindow->Convert(bounds.GetSize()));
|
|
targetBounds.x1 += controlClientOffset.x + controlWindowOffset.x;
|
|
targetBounds.x2 += controlClientOffset.x + controlWindowOffset.x;
|
|
targetBounds.y1 += controlClientOffset.y + controlWindowOffset.y;
|
|
targetBounds.y2 += controlClientOffset.y + controlWindowOffset.y;
|
|
|
|
NativePoint locations[4];
|
|
if (preferredTopBottomSide)
|
|
{
|
|
locations[0] = NativePoint(targetBounds.x1, targetBounds.y2);
|
|
locations[1] = NativePoint(targetBounds.x2 - windowSize.x, targetBounds.y2);
|
|
locations[2] = NativePoint(targetBounds.x1, targetBounds.y1 - windowSize.y);
|
|
locations[3] = NativePoint(targetBounds.x2 - windowSize.x, targetBounds.y1 - windowSize.y);
|
|
}
|
|
else
|
|
{
|
|
locations[0] = NativePoint(targetBounds.x2, targetBounds.y1);
|
|
locations[1] = NativePoint(targetBounds.x2, targetBounds.y2 - windowSize.y);
|
|
locations[2] = NativePoint(targetBounds.x1 - windowSize.x, targetBounds.y1);
|
|
locations[3] = NativePoint(targetBounds.x1 - windowSize.x, targetBounds.y2 - windowSize.y);
|
|
}
|
|
|
|
auto screen = GetCurrentController()->ScreenService()->GetScreen(controlWindow);
|
|
for (vint i = 0; i < 4; i++)
|
|
{
|
|
if (!IsClippedByScreen(windowSize, locations[i], screen))
|
|
{
|
|
return CalculatePopupPosition(windowSize, locations[i], screen);
|
|
}
|
|
}
|
|
return CalculatePopupPosition(windowSize, locations[0], screen);
|
|
}
|
|
|
|
NativePoint GuiPopup::CalculatePopupPosition(NativeSize windowSize, GuiControl* control, INativeWindow* controlWindow, Point location)
|
|
{
|
|
NativePoint controlClientOffset = controlWindow->Convert(control->GetBoundsComposition()->GetGlobalBounds().LeftTop());
|
|
NativePoint controlWindowOffset = controlWindow->GetClientBoundsInScreen().LeftTop();
|
|
NativePoint targetLocation = controlWindow->Convert(location);
|
|
NativeCoordinate x = controlClientOffset.x + controlWindowOffset.x + targetLocation.x;
|
|
NativeCoordinate y = controlClientOffset.y + controlWindowOffset.y + targetLocation.y;
|
|
return CalculatePopupPosition(windowSize, NativePoint(x, y), GetCurrentController()->ScreenService()->GetScreen(controlWindow));
|
|
}
|
|
|
|
NativePoint GuiPopup::CalculatePopupPosition(NativeSize windowSize, GuiControl* control, INativeWindow* controlWindow, bool preferredTopBottomSide)
|
|
{
|
|
Rect bounds(Point(0, 0), control->GetBoundsComposition()->GetBounds().GetSize());
|
|
return CalculatePopupPosition(windowSize, control, controlWindow, bounds, preferredTopBottomSide);
|
|
}
|
|
|
|
NativePoint GuiPopup::CalculatePopupPosition(NativeSize windowSize, vint popupType, const PopupInfo& popupInfo)
|
|
{
|
|
switch (popupType)
|
|
{
|
|
case 1:
|
|
return CalculatePopupPosition(windowSize, popupInfo._1.location, popupInfo._1.screen);
|
|
case 2:
|
|
return CalculatePopupPosition(windowSize, popupInfo._2.control, popupInfo._2.controlWindow, popupInfo._2.bounds, popupInfo._2.preferredTopBottomSide);
|
|
case 3:
|
|
return CalculatePopupPosition(windowSize, popupInfo._3.control, popupInfo._3.controlWindow, popupInfo._3.location);
|
|
case 4:
|
|
return CalculatePopupPosition(windowSize, popupInfo._4.control, popupInfo._4.controlWindow, popupInfo._4.preferredTopBottomSide);
|
|
default:
|
|
CHECK_FAIL(L"vl::presentation::controls::GuiPopup::CalculatePopupPosition(Size, const PopupInfo&)#Internal error.");
|
|
}
|
|
}
|
|
|
|
void GuiPopup::ShowPopupInternal()
|
|
{
|
|
auto window = GetNativeWindow();
|
|
UpdateClientSizeAfterRendering(window->Convert(window->GetClientSize()));
|
|
|
|
INativeWindow* controlWindow = nullptr;
|
|
switch (popupType)
|
|
{
|
|
case 2: controlWindow = popupInfo._2.controlWindow; break;
|
|
case 3: controlWindow = popupInfo._3.controlWindow; break;
|
|
case 4: controlWindow = popupInfo._4.controlWindow; break;
|
|
}
|
|
|
|
if (controlWindow)
|
|
{
|
|
window->SetParent(controlWindow);
|
|
window->SetTopMost(controlWindow->GetTopMost());
|
|
}
|
|
else
|
|
{
|
|
window->SetTopMost(true);
|
|
}
|
|
ShowDeactivated();
|
|
}
|
|
|
|
GuiPopup::GuiPopup(theme::ThemeName themeName)
|
|
:GuiWindow(themeName)
|
|
{
|
|
SetMinimizedBox(false);
|
|
SetMaximizedBox(false);
|
|
SetSizeBox(false);
|
|
SetTitleBar(false);
|
|
SetShowInTaskBar(false);
|
|
|
|
WindowOpened.AttachMethod(this, &GuiPopup::PopupOpened);
|
|
WindowClosed.AttachMethod(this, &GuiPopup::PopupClosed);
|
|
boundsComposition->GetEventReceiver()->keyDown.AttachMethod(this, &GuiPopup::OnKeyDown);
|
|
}
|
|
|
|
GuiPopup::~GuiPopup()
|
|
{
|
|
GetApplication()->RegisterPopupClosed(this);
|
|
}
|
|
|
|
void GuiPopup::ShowPopup(NativePoint location, INativeScreen* screen)
|
|
{
|
|
if (auto window = GetNativeWindow())
|
|
{
|
|
if (!screen)
|
|
{
|
|
SetBounds(location, GetClientSize());
|
|
screen = GetCurrentController()->ScreenService()->GetScreen(window);
|
|
}
|
|
|
|
popupType = 1;
|
|
popupInfo._1.location = location;
|
|
popupInfo._1.screen = screen;
|
|
ShowPopupInternal();
|
|
}
|
|
}
|
|
|
|
void GuiPopup::ShowPopup(GuiControl* control, Rect bounds, bool preferredTopBottomSide)
|
|
{
|
|
if (auto window = GetNativeWindow())
|
|
{
|
|
if (auto controlHost = control->GetBoundsComposition()->GetRelatedControlHost())
|
|
{
|
|
if (auto controlWindow = controlHost->GetNativeWindow())
|
|
{
|
|
popupType = 2;
|
|
popupInfo._2.control = control;
|
|
popupInfo._2.controlWindow = controlWindow;
|
|
popupInfo._2.bounds = bounds;
|
|
popupInfo._2.preferredTopBottomSide = preferredTopBottomSide;
|
|
ShowPopupInternal();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiPopup::ShowPopup(GuiControl* control, Point location)
|
|
{
|
|
if (auto window = GetNativeWindow())
|
|
{
|
|
if (auto controlHost = control->GetBoundsComposition()->GetRelatedControlHost())
|
|
{
|
|
if (auto controlWindow = controlHost->GetNativeWindow())
|
|
{
|
|
popupType = 3;
|
|
popupInfo._3.control = control;
|
|
popupInfo._3.controlWindow = controlWindow;
|
|
popupInfo._3.location = location;
|
|
ShowPopupInternal();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiPopup::ShowPopup(GuiControl* control, bool preferredTopBottomSide)
|
|
{
|
|
if (auto window = GetNativeWindow())
|
|
{
|
|
if (auto controlHost = control->GetBoundsComposition()->GetRelatedControlHost())
|
|
{
|
|
if (auto controlWindow = controlHost->GetNativeWindow())
|
|
{
|
|
popupType = 4;
|
|
popupInfo._4.control = control;
|
|
popupInfo._4.controlWindow = controlWindow;
|
|
popupInfo._4.preferredTopBottomSide = preferredTopBottomSide;
|
|
ShowPopupInternal();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiPopup
|
|
***********************************************************************/
|
|
|
|
void GuiTooltip::GlobalTimer()
|
|
{
|
|
SetClientSize(GetClientSize());
|
|
}
|
|
|
|
void GuiTooltip::TooltipOpened(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
}
|
|
|
|
void GuiTooltip::TooltipClosed(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
SetTemporaryContentControl(0);
|
|
}
|
|
|
|
GuiTooltip::GuiTooltip(theme::ThemeName themeName)
|
|
:GuiPopup(themeName)
|
|
,temporaryContentControl(0)
|
|
{
|
|
containerComposition->SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
containerComposition->SetPreferredMinSize(Size(20, 10));
|
|
GetCurrentController()->CallbackService()->InstallListener(this);
|
|
|
|
WindowOpened.AttachMethod(this, &GuiTooltip::TooltipOpened);
|
|
WindowClosed.AttachMethod(this, &GuiTooltip::TooltipClosed);
|
|
}
|
|
|
|
GuiTooltip::~GuiTooltip()
|
|
{
|
|
GetCurrentController()->CallbackService()->UninstallListener(this);
|
|
}
|
|
|
|
vint GuiTooltip::GetPreferredContentWidth()
|
|
{
|
|
return containerComposition->GetPreferredMinSize().x;
|
|
}
|
|
|
|
void GuiTooltip::SetPreferredContentWidth(vint value)
|
|
{
|
|
containerComposition->SetPreferredMinSize(Size(value, 10));
|
|
}
|
|
|
|
GuiControl* GuiTooltip::GetTemporaryContentControl()
|
|
{
|
|
return temporaryContentControl;
|
|
}
|
|
|
|
void GuiTooltip::SetTemporaryContentControl(GuiControl* control)
|
|
{
|
|
if(temporaryContentControl && HasChild(temporaryContentControl))
|
|
{
|
|
containerComposition->RemoveChild(temporaryContentControl->GetBoundsComposition());
|
|
temporaryContentControl=0;
|
|
}
|
|
temporaryContentControl=control;
|
|
if(control)
|
|
{
|
|
control->GetBoundsComposition()->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
AddChild(control);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
.\CONTROLS\LISTCONTROLPACKAGE\GUIBINDABLEDATAGRID.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace controls
|
|
{
|
|
using namespace collections;
|
|
using namespace description;
|
|
using namespace templates;
|
|
|
|
namespace list
|
|
{
|
|
|
|
/***********************************************************************
|
|
DataFilterBase
|
|
***********************************************************************/
|
|
|
|
void DataFilterBase::InvokeOnProcessorChanged()
|
|
{
|
|
if (callback)
|
|
{
|
|
callback->OnProcessorChanged();
|
|
}
|
|
}
|
|
|
|
DataFilterBase::DataFilterBase()
|
|
{
|
|
}
|
|
|
|
void DataFilterBase::SetCallback(IDataProcessorCallback* value)
|
|
{
|
|
callback = value;
|
|
}
|
|
|
|
/***********************************************************************
|
|
DataMultipleFilter
|
|
***********************************************************************/
|
|
|
|
DataMultipleFilter::DataMultipleFilter()
|
|
{
|
|
}
|
|
|
|
bool DataMultipleFilter::AddSubFilter(Ptr<IDataFilter> value)
|
|
{
|
|
if (!value) return false;
|
|
if (filters.Contains(value.Obj())) return false;
|
|
filters.Add(value);
|
|
value->SetCallback(callback);
|
|
InvokeOnProcessorChanged();
|
|
return true;
|
|
}
|
|
|
|
bool DataMultipleFilter::RemoveSubFilter(Ptr<IDataFilter> value)
|
|
{
|
|
if (!value) return false;
|
|
if (!filters.Contains(value.Obj())) return false;
|
|
value->SetCallback(nullptr);
|
|
filters.Remove(value.Obj());
|
|
InvokeOnProcessorChanged();
|
|
return true;
|
|
}
|
|
|
|
void DataMultipleFilter::SetCallback(IDataProcessorCallback* value)
|
|
{
|
|
DataFilterBase::SetCallback(value);
|
|
for (vint i = 0; i < filters.Count(); i++)
|
|
{
|
|
filters[i]->SetCallback(value);
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
DataAndFilter
|
|
***********************************************************************/
|
|
|
|
DataAndFilter::DataAndFilter()
|
|
{
|
|
}
|
|
|
|
bool DataAndFilter::Filter(const description::Value& row)
|
|
{
|
|
return From(filters)
|
|
.All([row](Ptr<IDataFilter> filter)
|
|
{
|
|
return filter->Filter(row);
|
|
});
|
|
}
|
|
|
|
/***********************************************************************
|
|
DataOrFilter
|
|
***********************************************************************/
|
|
|
|
DataOrFilter::DataOrFilter()
|
|
{
|
|
}
|
|
|
|
bool DataOrFilter::Filter(const description::Value& row)
|
|
{
|
|
return From(filters)
|
|
.Any([row](Ptr<IDataFilter> filter)
|
|
{
|
|
return filter->Filter(row);
|
|
});
|
|
}
|
|
|
|
/***********************************************************************
|
|
DataNotFilter
|
|
***********************************************************************/
|
|
|
|
DataNotFilter::DataNotFilter()
|
|
{
|
|
}
|
|
|
|
bool DataNotFilter::SetSubFilter(Ptr<IDataFilter> value)
|
|
{
|
|
if (filter == value) return false;
|
|
if (filter) filter->SetCallback(nullptr);
|
|
filter = value;
|
|
if (filter) filter->SetCallback(callback);
|
|
InvokeOnProcessorChanged();
|
|
return true;
|
|
}
|
|
|
|
void DataNotFilter::SetCallback(IDataProcessorCallback* value)
|
|
{
|
|
DataFilterBase::SetCallback(value);
|
|
if (filter) filter->SetCallback(value);
|
|
}
|
|
|
|
bool DataNotFilter::Filter(const description::Value& row)
|
|
{
|
|
return filter ? true : !filter->Filter(row);
|
|
}
|
|
|
|
/***********************************************************************
|
|
DataSorterBase
|
|
***********************************************************************/
|
|
|
|
void DataSorterBase::InvokeOnProcessorChanged()
|
|
{
|
|
if (callback)
|
|
{
|
|
callback->OnProcessorChanged();
|
|
}
|
|
}
|
|
|
|
DataSorterBase::DataSorterBase()
|
|
{
|
|
}
|
|
|
|
void DataSorterBase::SetCallback(IDataProcessorCallback* value)
|
|
{
|
|
callback = value;
|
|
}
|
|
|
|
/***********************************************************************
|
|
DataMultipleSorter
|
|
***********************************************************************/
|
|
|
|
DataMultipleSorter::DataMultipleSorter()
|
|
{
|
|
}
|
|
|
|
bool DataMultipleSorter::SetLeftSorter(Ptr<IDataSorter> value)
|
|
{
|
|
if (leftSorter == value) return false;
|
|
if (leftSorter) leftSorter->SetCallback(nullptr);
|
|
leftSorter = value;
|
|
if (leftSorter) leftSorter->SetCallback(callback);
|
|
return true;
|
|
}
|
|
|
|
bool DataMultipleSorter::SetRightSorter(Ptr<IDataSorter> value)
|
|
{
|
|
if (rightSorter == value) return false;
|
|
if (rightSorter) rightSorter->SetCallback(nullptr);
|
|
rightSorter = value;
|
|
if (rightSorter) rightSorter->SetCallback(callback);
|
|
return true;
|
|
}
|
|
|
|
void DataMultipleSorter::SetCallback(IDataProcessorCallback* value)
|
|
{
|
|
DataSorterBase::SetCallback(value);
|
|
if (leftSorter) leftSorter->SetCallback(value);
|
|
if (rightSorter) rightSorter->SetCallback(value);
|
|
}
|
|
|
|
vint DataMultipleSorter::Compare(const description::Value& row1, const description::Value& row2)
|
|
{
|
|
if (leftSorter)
|
|
{
|
|
vint result = leftSorter->Compare(row1, row2);
|
|
if (result != 0) return result;
|
|
}
|
|
if (rightSorter)
|
|
{
|
|
vint result = rightSorter->Compare(row1, row2);
|
|
if (result != 0) return result;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/***********************************************************************
|
|
DataReverseSorter
|
|
***********************************************************************/
|
|
|
|
DataReverseSorter::DataReverseSorter()
|
|
{
|
|
}
|
|
|
|
bool DataReverseSorter::SetSubSorter(Ptr<IDataSorter> value)
|
|
{
|
|
if (sorter == value) return false;
|
|
if (sorter) sorter->SetCallback(nullptr);
|
|
sorter = value;
|
|
if (sorter) sorter->SetCallback(callback);
|
|
return true;
|
|
}
|
|
|
|
void DataReverseSorter::SetCallback(IDataProcessorCallback* value)
|
|
{
|
|
DataSorterBase::SetCallback(value);
|
|
if (sorter) sorter->SetCallback(value);
|
|
}
|
|
|
|
vint DataReverseSorter::Compare(const description::Value& row1, const description::Value& row2)
|
|
{
|
|
return sorter ? -sorter->Compare(row1, row2) : 0;
|
|
}
|
|
|
|
/***********************************************************************
|
|
DataColumn
|
|
***********************************************************************/
|
|
|
|
void DataColumn::NotifyAllColumnsUpdate(bool affectItem)
|
|
{
|
|
if (dataProvider)
|
|
{
|
|
vint index = dataProvider->columns.IndexOf(this);
|
|
if (index != -1)
|
|
{
|
|
dataProvider->columns.NotifyColumnUpdated(index, affectItem);
|
|
}
|
|
}
|
|
}
|
|
|
|
DataColumn::DataColumn()
|
|
{
|
|
}
|
|
|
|
DataColumn::~DataColumn()
|
|
{
|
|
if (popup && ownPopup)
|
|
{
|
|
SafeDeleteControl(popup);
|
|
}
|
|
}
|
|
|
|
WString DataColumn::GetText()
|
|
{
|
|
return text;
|
|
}
|
|
|
|
void DataColumn::SetText(const WString& value)
|
|
{
|
|
if (text != value)
|
|
{
|
|
text = value;
|
|
NotifyAllColumnsUpdate(false);
|
|
}
|
|
}
|
|
|
|
vint DataColumn::GetSize()
|
|
{
|
|
return size;
|
|
}
|
|
|
|
void DataColumn::SetSize(vint value)
|
|
{
|
|
if (size != value)
|
|
{
|
|
size = value;
|
|
NotifyAllColumnsUpdate(false);
|
|
}
|
|
}
|
|
|
|
bool DataColumn::GetOwnPopup()
|
|
{
|
|
return ownPopup;
|
|
}
|
|
|
|
void DataColumn::SetOwnPopup(bool value)
|
|
{
|
|
ownPopup = value;
|
|
}
|
|
|
|
GuiMenu* DataColumn::GetPopup()
|
|
{
|
|
return popup;
|
|
}
|
|
|
|
void DataColumn::SetPopup(GuiMenu* value)
|
|
{
|
|
if (popup != value)
|
|
{
|
|
popup = value;
|
|
NotifyAllColumnsUpdate(false);
|
|
}
|
|
}
|
|
|
|
Ptr<IDataFilter> DataColumn::GetFilter()
|
|
{
|
|
return associatedFilter;
|
|
}
|
|
|
|
void DataColumn::SetFilter(Ptr<IDataFilter> value)
|
|
{
|
|
if (associatedFilter) associatedFilter->SetCallback(nullptr);
|
|
associatedFilter = value;
|
|
if (associatedFilter) associatedFilter->SetCallback(dataProvider);
|
|
NotifyAllColumnsUpdate(false);
|
|
}
|
|
|
|
Ptr<IDataSorter> DataColumn::GetSorter()
|
|
{
|
|
return associatedSorter;
|
|
}
|
|
|
|
void DataColumn::SetSorter(Ptr<IDataSorter> value)
|
|
{
|
|
if (associatedSorter) associatedSorter->SetCallback(nullptr);
|
|
associatedSorter = value;
|
|
if (associatedSorter) associatedSorter->SetCallback(dataProvider);
|
|
NotifyAllColumnsUpdate(false);
|
|
}
|
|
|
|
Ptr<IDataVisualizerFactory> DataColumn::GetVisualizerFactory()
|
|
{
|
|
return visualizerFactory;
|
|
}
|
|
|
|
void DataColumn::SetVisualizerFactory(Ptr<IDataVisualizerFactory> value)
|
|
{
|
|
visualizerFactory = value;
|
|
NotifyAllColumnsUpdate(true);
|
|
}
|
|
|
|
Ptr<IDataEditorFactory> DataColumn::GetEditorFactory()
|
|
{
|
|
return editorFactory;
|
|
}
|
|
|
|
void DataColumn::SetEditorFactory(Ptr<IDataEditorFactory> value)
|
|
{
|
|
editorFactory = value;
|
|
NotifyAllColumnsUpdate(true);
|
|
}
|
|
|
|
WString DataColumn::GetCellText(vint row)
|
|
{
|
|
if (0 <= row && row < dataProvider->Count())
|
|
{
|
|
return ReadProperty(dataProvider->GetBindingValue(row), textProperty);
|
|
}
|
|
return L"";
|
|
}
|
|
|
|
description::Value DataColumn::GetCellValue(vint row)
|
|
{
|
|
if (0 <= row && row < dataProvider->Count())
|
|
{
|
|
return ReadProperty(dataProvider->GetBindingValue(row), valueProperty);
|
|
}
|
|
return Value();
|
|
}
|
|
|
|
void DataColumn::SetCellValue(vint row, description::Value value)
|
|
{
|
|
if (0 <= row && row < dataProvider->Count())
|
|
{
|
|
auto rowValue = dataProvider->GetBindingValue(row);
|
|
WriteProperty(rowValue, valueProperty, value);
|
|
dataProvider->InvokeOnItemModified(row, 1, 1);
|
|
}
|
|
}
|
|
|
|
ItemProperty<WString> DataColumn::GetTextProperty()
|
|
{
|
|
return textProperty;
|
|
}
|
|
|
|
void DataColumn::SetTextProperty(const ItemProperty<WString>& value)
|
|
{
|
|
if (textProperty != value)
|
|
{
|
|
textProperty = value;
|
|
NotifyAllColumnsUpdate(true);
|
|
compositions::GuiEventArgs arguments;
|
|
TextPropertyChanged.Execute(arguments);
|
|
}
|
|
}
|
|
|
|
WritableItemProperty<description::Value> DataColumn::GetValueProperty()
|
|
{
|
|
return valueProperty;
|
|
}
|
|
|
|
void DataColumn::SetValueProperty(const WritableItemProperty<description::Value>& value)
|
|
{
|
|
if (valueProperty != value)
|
|
{
|
|
valueProperty = value;
|
|
NotifyAllColumnsUpdate(true);
|
|
compositions::GuiEventArgs arguments;
|
|
ValuePropertyChanged.Execute(arguments);
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
DataColumns
|
|
***********************************************************************/
|
|
|
|
void DataColumns::NotifyColumnUpdated(vint index, bool affectItem)
|
|
{
|
|
affectItemFlag = affectItem;
|
|
NotifyUpdateInternal(index, 1, 1);
|
|
affectItemFlag = true;
|
|
}
|
|
|
|
void DataColumns::NotifyUpdateInternal(vint start, vint count, vint newCount)
|
|
{
|
|
dataProvider->NotifyAllColumnsUpdate();
|
|
if (affectItemFlag)
|
|
{
|
|
dataProvider->NotifyAllItemsUpdate();
|
|
}
|
|
}
|
|
|
|
bool DataColumns::QueryInsert(vint index, const Ptr<DataColumn>& value)
|
|
{
|
|
return !items.Contains(value.Obj());
|
|
}
|
|
|
|
void DataColumns::AfterInsert(vint index, const Ptr<DataColumn>& value)
|
|
{
|
|
value->dataProvider = dataProvider;
|
|
}
|
|
|
|
void DataColumns::BeforeRemove(vint index, const Ptr<DataColumn>& value)
|
|
{
|
|
value->dataProvider = nullptr;
|
|
}
|
|
|
|
DataColumns::DataColumns(DataProvider* _dataProvider)
|
|
:dataProvider(_dataProvider)
|
|
{
|
|
}
|
|
|
|
DataColumns::~DataColumns()
|
|
{
|
|
}
|
|
|
|
/***********************************************************************
|
|
DataProvider
|
|
***********************************************************************/
|
|
|
|
void DataProvider::NotifyAllItemsUpdate()
|
|
{
|
|
InvokeOnItemModified(0, Count(), Count());
|
|
}
|
|
|
|
void DataProvider::NotifyAllColumnsUpdate()
|
|
{
|
|
if (columnItemViewCallback)
|
|
{
|
|
columnItemViewCallback->OnColumnChanged();
|
|
}
|
|
}
|
|
|
|
GuiListControl::IItemProvider* DataProvider::GetItemProvider()
|
|
{
|
|
return this;
|
|
}
|
|
|
|
void DataProvider::OnProcessorChanged()
|
|
{
|
|
RebuildFilter();
|
|
ReorderRows(true);
|
|
}
|
|
|
|
void DataProvider::OnItemSourceModified(vint start, vint count, vint newCount)
|
|
{
|
|
if (!currentSorter && !currentFilter && count == newCount)
|
|
{
|
|
InvokeOnItemModified(start, count, newCount);
|
|
}
|
|
else
|
|
{
|
|
ReorderRows(true);
|
|
}
|
|
}
|
|
|
|
ListViewDataColumns& DataProvider::GetDataColumns()
|
|
{
|
|
return dataColumns;
|
|
}
|
|
|
|
DataColumns& DataProvider::GetColumns()
|
|
{
|
|
return columns;
|
|
}
|
|
|
|
Ptr<description::IValueEnumerable> DataProvider::GetItemSource()
|
|
{
|
|
return itemSource;
|
|
}
|
|
|
|
void DataProvider::SetItemSource(Ptr<description::IValueEnumerable> _itemSource)
|
|
{
|
|
vint oldCount = 0;
|
|
if (itemSource)
|
|
{
|
|
oldCount = itemSource->GetCount();
|
|
}
|
|
if (itemChangedEventHandler)
|
|
{
|
|
auto ol = itemSource.Cast<IValueObservableList>();
|
|
ol->ItemChanged.Remove(itemChangedEventHandler);
|
|
}
|
|
|
|
itemSource = nullptr;
|
|
itemChangedEventHandler = nullptr;
|
|
|
|
if (_itemSource)
|
|
{
|
|
if (auto ol = _itemSource.Cast<IValueObservableList>())
|
|
{
|
|
itemSource = ol;
|
|
itemChangedEventHandler = ol->ItemChanged.Add([this](vint start, vint oldCount, vint newCount)
|
|
{
|
|
OnItemSourceModified(start, oldCount, newCount);
|
|
});
|
|
}
|
|
else if (auto rl = _itemSource.Cast<IValueReadonlyList>())
|
|
{
|
|
itemSource = rl;
|
|
}
|
|
else
|
|
{
|
|
itemSource = IValueList::Create(GetLazyList<Value>(_itemSource));
|
|
}
|
|
}
|
|
|
|
OnItemSourceModified(0, oldCount, itemSource ? itemSource->GetCount() : 0);
|
|
}
|
|
|
|
void DataProvider::RebuildFilter()
|
|
{
|
|
if (currentFilter)
|
|
{
|
|
currentFilter->SetCallback(nullptr);
|
|
currentFilter = nullptr;
|
|
}
|
|
|
|
List<Ptr<IDataFilter>> selectedFilters;
|
|
CopyFrom(
|
|
selectedFilters,
|
|
From(columns)
|
|
.Select([](Ptr<DataColumn> column) {return column->GetFilter(); })
|
|
.Where([](Ptr<IDataFilter> filter) {return filter != nullptr; })
|
|
);
|
|
if (additionalFilter)
|
|
{
|
|
selectedFilters.Add(additionalFilter);
|
|
}
|
|
if (selectedFilters.Count() > 0)
|
|
{
|
|
auto andFilter = MakePtr<DataAndFilter>();
|
|
FOREACH(Ptr<IDataFilter>, filter, selectedFilters)
|
|
{
|
|
andFilter->AddSubFilter(filter);
|
|
}
|
|
currentFilter = andFilter;
|
|
}
|
|
|
|
if (currentFilter)
|
|
{
|
|
currentFilter->SetCallback(this);
|
|
}
|
|
}
|
|
|
|
void DataProvider::ReorderRows(bool invokeCallback)
|
|
{
|
|
vint oldRowCount = virtualRowToSourceRow.Count();
|
|
virtualRowToSourceRow.Clear();
|
|
vint rowCount = itemSource ? itemSource->GetCount() : 0;
|
|
|
|
if (currentFilter)
|
|
{
|
|
for (vint i = 0; i < rowCount; i++)
|
|
{
|
|
if (currentFilter->Filter(itemSource->Get(i)))
|
|
{
|
|
virtualRowToSourceRow.Add(i);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
for (vint i = 0; i < rowCount; i++)
|
|
{
|
|
virtualRowToSourceRow.Add(i);
|
|
}
|
|
}
|
|
|
|
if (currentSorter && virtualRowToSourceRow.Count() > 0)
|
|
{
|
|
IDataSorter* sorter = currentSorter.Obj();
|
|
SortLambda(
|
|
&virtualRowToSourceRow[0],
|
|
virtualRowToSourceRow.Count(),
|
|
[=](vint a, vint b)
|
|
{
|
|
return sorter->Compare(itemSource->Get(a), itemSource->Get(b));
|
|
});
|
|
}
|
|
|
|
if (invokeCallback)
|
|
{
|
|
NotifyAllItemsUpdate();
|
|
}
|
|
}
|
|
|
|
DataProvider::DataProvider()
|
|
:dataColumns(this)
|
|
, columns(this)
|
|
{
|
|
RebuildFilter();
|
|
ReorderRows(false);
|
|
}
|
|
|
|
DataProvider::~DataProvider()
|
|
{
|
|
}
|
|
|
|
Ptr<IDataFilter> DataProvider::GetAdditionalFilter()
|
|
{
|
|
return additionalFilter;
|
|
}
|
|
|
|
void DataProvider::SetAdditionalFilter(Ptr<IDataFilter> value)
|
|
{
|
|
additionalFilter = value;
|
|
RebuildFilter();
|
|
ReorderRows(true);
|
|
}
|
|
|
|
// ===================== GuiListControl::IItemProvider =====================
|
|
|
|
vint DataProvider::Count()
|
|
{
|
|
return virtualRowToSourceRow.Count();
|
|
}
|
|
|
|
WString DataProvider::GetTextValue(vint itemIndex)
|
|
{
|
|
return GetText(itemIndex);
|
|
}
|
|
|
|
description::Value DataProvider::GetBindingValue(vint itemIndex)
|
|
{
|
|
return itemSource ? itemSource->Get(virtualRowToSourceRow[itemIndex]) : Value();
|
|
}
|
|
|
|
IDescriptable* DataProvider::RequestView(const WString& identifier)
|
|
{
|
|
if (identifier == IListViewItemView::Identifier)
|
|
{
|
|
return (IListViewItemView*)this;
|
|
}
|
|
else if (identifier == ListViewColumnItemArranger::IColumnItemView::Identifier)
|
|
{
|
|
return (ListViewColumnItemArranger::IColumnItemView*)this;
|
|
}
|
|
else if (identifier == IDataGridView::Identifier)
|
|
{
|
|
return (IDataGridView*)this;
|
|
}
|
|
else
|
|
{
|
|
return nullptr;
|
|
}
|
|
}
|
|
|
|
// ===================== list::IListViewItemProvider =====================
|
|
|
|
Ptr<GuiImageData> DataProvider::GetSmallImage(vint itemIndex)
|
|
{
|
|
if (0 <= itemIndex && itemIndex < Count())
|
|
{
|
|
return ReadProperty(GetBindingValue(itemIndex), smallImageProperty);
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
Ptr<GuiImageData> DataProvider::GetLargeImage(vint itemIndex)
|
|
{
|
|
if (0 <= itemIndex && itemIndex < Count())
|
|
{
|
|
return ReadProperty(GetBindingValue(itemIndex), largeImageProperty);
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
WString DataProvider::GetText(vint itemIndex)
|
|
{
|
|
if (columns.Count() == 0)return L"";
|
|
return columns[0]->GetCellText(itemIndex);
|
|
}
|
|
|
|
WString DataProvider::GetSubItem(vint itemIndex, vint index)
|
|
{
|
|
return columns[index + 1]->GetCellText(itemIndex);
|
|
}
|
|
|
|
vint DataProvider::GetDataColumnCount()
|
|
{
|
|
return dataColumns.Count();
|
|
}
|
|
|
|
vint DataProvider::GetDataColumn(vint index)
|
|
{
|
|
return dataColumns[index];
|
|
}
|
|
|
|
vint DataProvider::GetColumnCount()
|
|
{
|
|
return columns.Count();
|
|
}
|
|
|
|
WString DataProvider::GetColumnText(vint index)
|
|
{
|
|
return columns[index]->GetText();
|
|
}
|
|
|
|
// ===================== list::ListViewColumnItemArranger::IColumnItemView =====================
|
|
|
|
bool DataProvider::AttachCallback(ListViewColumnItemArranger::IColumnItemViewCallback* value)
|
|
{
|
|
if (columnItemViewCallback)return false;
|
|
columnItemViewCallback = value;
|
|
return true;
|
|
}
|
|
|
|
bool DataProvider::DetachCallback(ListViewColumnItemArranger::IColumnItemViewCallback* value)
|
|
{
|
|
if (!columnItemViewCallback) return false;
|
|
columnItemViewCallback = nullptr;
|
|
return true;
|
|
}
|
|
|
|
vint DataProvider::GetColumnSize(vint index)
|
|
{
|
|
return columns[index]->GetSize();
|
|
}
|
|
|
|
void DataProvider::SetColumnSize(vint index, vint value)
|
|
{
|
|
columns[index]->SetSize(value);
|
|
}
|
|
|
|
GuiMenu* DataProvider::GetDropdownPopup(vint index)
|
|
{
|
|
return columns[index]->GetPopup();
|
|
}
|
|
|
|
ColumnSortingState DataProvider::GetSortingState(vint index)
|
|
{
|
|
return columns[index]->sortingState;
|
|
}
|
|
|
|
// ===================== list::IDataGridView =====================
|
|
|
|
bool DataProvider::IsColumnSortable(vint column)
|
|
{
|
|
return columns[column]->GetSorter();
|
|
}
|
|
|
|
void DataProvider::SortByColumn(vint column, bool ascending)
|
|
{
|
|
if (0 <= column && column < columns.Count())
|
|
{
|
|
auto sorter = columns[column]->GetSorter();
|
|
if (!sorter)
|
|
{
|
|
currentSorter = nullptr;
|
|
}
|
|
else if (ascending)
|
|
{
|
|
currentSorter = sorter;
|
|
}
|
|
else
|
|
{
|
|
Ptr<DataReverseSorter> reverseSorter = new DataReverseSorter();
|
|
reverseSorter->SetSubSorter(sorter);
|
|
currentSorter = reverseSorter;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
currentSorter = nullptr;
|
|
}
|
|
|
|
for (vint i = 0; i < columns.Count(); i++)
|
|
{
|
|
columns[i]->sortingState =
|
|
i != column ? ColumnSortingState::NotSorted :
|
|
ascending ? ColumnSortingState::Ascending :
|
|
ColumnSortingState::Descending
|
|
;
|
|
}
|
|
NotifyAllColumnsUpdate();
|
|
ReorderRows(true);
|
|
}
|
|
|
|
vint DataProvider::GetSortedColumn()
|
|
{
|
|
for (vint i = 0; i < columns.Count(); i++)
|
|
{
|
|
auto state = columns[i]->sortingState;
|
|
if (state != ColumnSortingState::NotSorted)
|
|
{
|
|
return i;
|
|
}
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
bool DataProvider::IsSortOrderAscending()
|
|
{
|
|
for (vint i = 0; i < columns.Count(); i++)
|
|
{
|
|
auto state = columns[i]->sortingState;
|
|
if (state != ColumnSortingState::NotSorted)
|
|
{
|
|
return state == ColumnSortingState::Ascending;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
vint DataProvider::GetCellSpan(vint row, vint column)
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
IDataVisualizerFactory* DataProvider::GetCellDataVisualizerFactory(vint row, vint column)
|
|
{
|
|
return columns[column]->GetVisualizerFactory().Obj();
|
|
}
|
|
|
|
IDataEditorFactory* DataProvider::GetCellDataEditorFactory(vint row, vint column)
|
|
{
|
|
return columns[column]->GetEditorFactory().Obj();
|
|
}
|
|
|
|
description::Value DataProvider::GetBindingCellValue(vint row, vint column)
|
|
{
|
|
return columns[column]->GetCellValue(row);
|
|
}
|
|
|
|
void DataProvider::SetBindingCellValue(vint row, vint column, const description::Value& value)
|
|
{
|
|
columns[column]->SetCellValue(row, value);
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiBindableDataGrid
|
|
***********************************************************************/
|
|
|
|
GuiBindableDataGrid::GuiBindableDataGrid(theme::ThemeName themeName)
|
|
:GuiVirtualDataGrid(themeName, new list::DataProvider)
|
|
{
|
|
dataProvider = dynamic_cast<list::DataProvider*>(GetItemProvider());
|
|
}
|
|
|
|
GuiBindableDataGrid::~GuiBindableDataGrid()
|
|
{
|
|
}
|
|
|
|
list::ListViewDataColumns& GuiBindableDataGrid::GetDataColumns()
|
|
{
|
|
return dataProvider->GetDataColumns();
|
|
}
|
|
|
|
list::DataColumns& GuiBindableDataGrid::GetColumns()
|
|
{
|
|
return dataProvider->GetColumns();
|
|
}
|
|
|
|
Ptr<description::IValueEnumerable> GuiBindableDataGrid::GetItemSource()
|
|
{
|
|
return dataProvider->GetItemSource();
|
|
}
|
|
|
|
void GuiBindableDataGrid::SetItemSource(Ptr<description::IValueEnumerable> _itemSource)
|
|
{
|
|
dataProvider->SetItemSource(_itemSource);
|
|
}
|
|
|
|
Ptr<list::IDataFilter> GuiBindableDataGrid::GetAdditionalFilter()
|
|
{
|
|
return dataProvider->GetAdditionalFilter();
|
|
}
|
|
|
|
void GuiBindableDataGrid::SetAdditionalFilter(Ptr<list::IDataFilter> value)
|
|
{
|
|
dataProvider->SetAdditionalFilter(value);
|
|
}
|
|
|
|
ItemProperty<Ptr<GuiImageData>> GuiBindableDataGrid::GetLargeImageProperty()
|
|
{
|
|
return dataProvider->largeImageProperty;
|
|
}
|
|
|
|
void GuiBindableDataGrid::SetLargeImageProperty(const ItemProperty<Ptr<GuiImageData>>& value)
|
|
{
|
|
if (dataProvider->largeImageProperty != value)
|
|
{
|
|
dataProvider->largeImageProperty = value;
|
|
dataProvider->NotifyAllItemsUpdate();
|
|
LargeImagePropertyChanged.Execute(GetNotifyEventArguments());
|
|
}
|
|
}
|
|
|
|
ItemProperty<Ptr<GuiImageData>> GuiBindableDataGrid::GetSmallImageProperty()
|
|
{
|
|
return dataProvider->smallImageProperty;
|
|
}
|
|
|
|
void GuiBindableDataGrid::SetSmallImageProperty(const ItemProperty<Ptr<GuiImageData>>& value)
|
|
{
|
|
if (dataProvider->smallImageProperty != value)
|
|
{
|
|
dataProvider->smallImageProperty = value;
|
|
dataProvider->NotifyAllItemsUpdate();
|
|
SmallImagePropertyChanged.Execute(GetNotifyEventArguments());
|
|
}
|
|
}
|
|
|
|
description::Value GuiBindableDataGrid::GetSelectedRowValue()
|
|
{
|
|
auto pos = GetSelectedCell();
|
|
if (pos.row == -1 || pos.column == -1)
|
|
{
|
|
return Value();
|
|
}
|
|
return dataProvider->GetBindingValue(GetSelectedCell().row);
|
|
}
|
|
|
|
description::Value GuiBindableDataGrid::GetSelectedCellValue()
|
|
{
|
|
auto pos = GetSelectedCell();
|
|
if (pos.row == -1 || pos.column == -1)
|
|
{
|
|
return Value();
|
|
}
|
|
return dataProvider->GetColumns()[pos.column]->GetCellValue(pos.row);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\CONTROLS\LISTCONTROLPACKAGE\GUIBINDABLELISTCONTROLS.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace controls
|
|
{
|
|
using namespace collections;
|
|
using namespace list;
|
|
using namespace tree;
|
|
using namespace reflection::description;
|
|
using namespace templates;
|
|
|
|
/***********************************************************************
|
|
GuiBindableTextList::ItemSource
|
|
***********************************************************************/
|
|
|
|
GuiBindableTextList::ItemSource::ItemSource()
|
|
{
|
|
}
|
|
|
|
GuiBindableTextList::ItemSource::~ItemSource()
|
|
{
|
|
SetItemSource(nullptr);
|
|
}
|
|
|
|
Ptr<description::IValueEnumerable> GuiBindableTextList::ItemSource::GetItemSource()
|
|
{
|
|
return itemSource;
|
|
}
|
|
|
|
void GuiBindableTextList::ItemSource::SetItemSource(Ptr<description::IValueEnumerable> _itemSource)
|
|
{
|
|
vint oldCount = 0;
|
|
if (itemSource)
|
|
{
|
|
oldCount = itemSource->GetCount();
|
|
}
|
|
if (itemChangedEventHandler)
|
|
{
|
|
auto ol = itemSource.Cast<IValueObservableList>();
|
|
ol->ItemChanged.Remove(itemChangedEventHandler);
|
|
}
|
|
|
|
itemSource = nullptr;
|
|
itemChangedEventHandler = nullptr;
|
|
|
|
if (_itemSource)
|
|
{
|
|
if (auto ol = _itemSource.Cast<IValueObservableList>())
|
|
{
|
|
itemSource = ol;
|
|
itemChangedEventHandler = ol->ItemChanged.Add([this](vint start, vint oldCount, vint newCount)
|
|
{
|
|
InvokeOnItemModified(start, oldCount, newCount);
|
|
});
|
|
}
|
|
else if (auto rl = _itemSource.Cast<IValueReadonlyList>())
|
|
{
|
|
itemSource = rl;
|
|
}
|
|
else
|
|
{
|
|
itemSource = IValueList::Create(GetLazyList<Value>(_itemSource));
|
|
}
|
|
}
|
|
|
|
InvokeOnItemModified(0, oldCount, itemSource ? itemSource->GetCount() : 0);
|
|
}
|
|
|
|
description::Value GuiBindableTextList::ItemSource::Get(vint index)
|
|
{
|
|
if (!itemSource) return Value();
|
|
return itemSource->Get(index);
|
|
}
|
|
|
|
void GuiBindableTextList::ItemSource::UpdateBindingProperties()
|
|
{
|
|
InvokeOnItemModified(0, Count(), Count());
|
|
}
|
|
|
|
// ===================== GuiListControl::IItemProvider =====================
|
|
|
|
vint GuiBindableTextList::ItemSource::Count()
|
|
{
|
|
if (!itemSource) return 0;
|
|
return itemSource->GetCount();
|
|
}
|
|
|
|
WString GuiBindableTextList::ItemSource::GetTextValue(vint itemIndex)
|
|
{
|
|
if (itemSource)
|
|
{
|
|
if (0 <= itemIndex && itemIndex < itemSource->GetCount())
|
|
{
|
|
return ReadProperty(itemSource->Get(itemIndex), textProperty);
|
|
}
|
|
}
|
|
return L"";
|
|
}
|
|
|
|
IDescriptable* GuiBindableTextList::ItemSource::RequestView(const WString& identifier)
|
|
{
|
|
if (identifier == ITextItemView::Identifier)
|
|
{
|
|
return (ITextItemView*)this;
|
|
}
|
|
else
|
|
{
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
// ===================== GuiListControl::IItemBindingView =====================
|
|
|
|
description::Value GuiBindableTextList::ItemSource::GetBindingValue(vint itemIndex)
|
|
{
|
|
if (itemSource)
|
|
{
|
|
if (0 <= itemIndex && itemIndex < itemSource->GetCount())
|
|
{
|
|
return itemSource->Get(itemIndex);
|
|
}
|
|
}
|
|
return Value();
|
|
}
|
|
|
|
// ===================== list::TextItemStyleProvider::ITextItemView =====================
|
|
|
|
bool GuiBindableTextList::ItemSource::GetChecked(vint itemIndex)
|
|
{
|
|
if (itemSource)
|
|
{
|
|
if (0 <= itemIndex && itemIndex < itemSource->GetCount())
|
|
{
|
|
return ReadProperty(itemSource->Get(itemIndex), checkedProperty);
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void GuiBindableTextList::ItemSource::SetChecked(vint itemIndex, bool value)
|
|
{
|
|
if (itemSource)
|
|
{
|
|
if (0 <= itemIndex && itemIndex < itemSource->GetCount())
|
|
{
|
|
auto thisValue = itemSource->Get(itemIndex);
|
|
WriteProperty(thisValue, checkedProperty, value);
|
|
InvokeOnItemModified(itemIndex, 1, 1);
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiBindableTextList
|
|
***********************************************************************/
|
|
|
|
GuiBindableTextList::GuiBindableTextList(theme::ThemeName themeName)
|
|
:GuiVirtualTextList(themeName, new ItemSource)
|
|
{
|
|
itemSource = dynamic_cast<ItemSource*>(GetItemProvider());
|
|
|
|
TextPropertyChanged.SetAssociatedComposition(boundsComposition);
|
|
TextPropertyChanged.SetAssociatedComposition(boundsComposition);
|
|
}
|
|
|
|
GuiBindableTextList::~GuiBindableTextList()
|
|
{
|
|
}
|
|
|
|
Ptr<description::IValueEnumerable> GuiBindableTextList::GetItemSource()
|
|
{
|
|
return itemSource->GetItemSource();
|
|
}
|
|
|
|
void GuiBindableTextList::SetItemSource(Ptr<description::IValueEnumerable> _itemSource)
|
|
{
|
|
itemSource->SetItemSource(_itemSource);
|
|
}
|
|
|
|
ItemProperty<WString> GuiBindableTextList::GetTextProperty()
|
|
{
|
|
return itemSource->textProperty;
|
|
}
|
|
|
|
void GuiBindableTextList::SetTextProperty(const ItemProperty<WString>& value)
|
|
{
|
|
if (itemSource->textProperty != value)
|
|
{
|
|
itemSource->textProperty = value;
|
|
itemSource->UpdateBindingProperties();
|
|
TextPropertyChanged.Execute(GetNotifyEventArguments());
|
|
}
|
|
}
|
|
|
|
WritableItemProperty<bool> GuiBindableTextList::GetCheckedProperty()
|
|
{
|
|
return itemSource->checkedProperty;
|
|
}
|
|
|
|
void GuiBindableTextList::SetCheckedProperty(const WritableItemProperty<bool>& value)
|
|
{
|
|
if (itemSource->checkedProperty != value)
|
|
{
|
|
itemSource->checkedProperty = value;
|
|
itemSource->UpdateBindingProperties();
|
|
CheckedPropertyChanged.Execute(GetNotifyEventArguments());
|
|
}
|
|
}
|
|
|
|
description::Value GuiBindableTextList::GetSelectedItem()
|
|
{
|
|
vint index = GetSelectedItemIndex();
|
|
if (index == -1) return Value();
|
|
return itemSource->Get(index);
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiBindableListView::ItemSource
|
|
***********************************************************************/
|
|
|
|
GuiBindableListView::ItemSource::ItemSource()
|
|
:columns(this)
|
|
, dataColumns(this)
|
|
{
|
|
}
|
|
|
|
GuiBindableListView::ItemSource::~ItemSource()
|
|
{
|
|
SetItemSource(nullptr);
|
|
}
|
|
|
|
Ptr<description::IValueEnumerable> GuiBindableListView::ItemSource::GetItemSource()
|
|
{
|
|
return itemSource;
|
|
}
|
|
|
|
void GuiBindableListView::ItemSource::SetItemSource(Ptr<description::IValueEnumerable> _itemSource)
|
|
{
|
|
vint oldCount = 0;
|
|
if (itemSource)
|
|
{
|
|
oldCount = itemSource->GetCount();
|
|
}
|
|
if (itemChangedEventHandler)
|
|
{
|
|
auto ol = itemSource.Cast<IValueObservableList>();
|
|
ol->ItemChanged.Remove(itemChangedEventHandler);
|
|
}
|
|
|
|
itemSource = nullptr;
|
|
itemChangedEventHandler = nullptr;
|
|
|
|
if (_itemSource)
|
|
{
|
|
if (auto ol = _itemSource.Cast<IValueObservableList>())
|
|
{
|
|
itemSource = ol;
|
|
itemChangedEventHandler = ol->ItemChanged.Add([this](vint start, vint oldCount, vint newCount)
|
|
{
|
|
InvokeOnItemModified(start, oldCount, newCount);
|
|
});
|
|
}
|
|
else if (auto rl = _itemSource.Cast<IValueReadonlyList>())
|
|
{
|
|
itemSource = rl;
|
|
}
|
|
else
|
|
{
|
|
itemSource = IValueList::Create(GetLazyList<Value>(_itemSource));
|
|
}
|
|
}
|
|
|
|
InvokeOnItemModified(0, oldCount, itemSource ? itemSource->GetCount() : 0);
|
|
}
|
|
|
|
description::Value GuiBindableListView::ItemSource::Get(vint index)
|
|
{
|
|
if (!itemSource) return Value();
|
|
return itemSource->Get(index);
|
|
}
|
|
|
|
void GuiBindableListView::ItemSource::UpdateBindingProperties()
|
|
{
|
|
InvokeOnItemModified(0, Count(), Count());
|
|
}
|
|
|
|
bool GuiBindableListView::ItemSource::NotifyUpdate(vint start, vint count)
|
|
{
|
|
if (!itemSource) return false;
|
|
if (start<0 || start >= itemSource->GetCount() || count <= 0 || start + count > itemSource->GetCount())
|
|
{
|
|
return false;
|
|
}
|
|
else
|
|
{
|
|
InvokeOnItemModified(start, count, count);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
list::ListViewDataColumns& GuiBindableListView::ItemSource::GetDataColumns()
|
|
{
|
|
return dataColumns;
|
|
}
|
|
|
|
list::ListViewColumns& GuiBindableListView::ItemSource::GetColumns()
|
|
{
|
|
return columns;
|
|
}
|
|
|
|
// ===================== list::IListViewItemProvider =====================
|
|
|
|
void GuiBindableListView::ItemSource::NotifyAllItemsUpdate()
|
|
{
|
|
NotifyUpdate(0, Count());
|
|
}
|
|
|
|
void GuiBindableListView::ItemSource::NotifyAllColumnsUpdate()
|
|
{
|
|
for (vint i = 0; i < columnItemViewCallbacks.Count(); i++)
|
|
{
|
|
columnItemViewCallbacks[i]->OnColumnChanged();
|
|
}
|
|
}
|
|
|
|
// ===================== GuiListControl::IItemProvider =====================
|
|
|
|
vint GuiBindableListView::ItemSource::Count()
|
|
{
|
|
if (!itemSource) return 0;
|
|
return itemSource->GetCount();
|
|
}
|
|
|
|
WString GuiBindableListView::ItemSource::GetTextValue(vint itemIndex)
|
|
{
|
|
return GetText(itemIndex);
|
|
}
|
|
|
|
description::Value GuiBindableListView::ItemSource::GetBindingValue(vint itemIndex)
|
|
{
|
|
if (itemSource)
|
|
{
|
|
if (0 <= itemIndex && itemIndex < itemSource->GetCount())
|
|
{
|
|
return itemSource->Get(itemIndex);
|
|
}
|
|
}
|
|
return Value();
|
|
}
|
|
|
|
IDescriptable* GuiBindableListView::ItemSource::RequestView(const WString& identifier)
|
|
{
|
|
if (identifier == IListViewItemView::Identifier)
|
|
{
|
|
return (IListViewItemView*)this;
|
|
}
|
|
else if (identifier == ListViewColumnItemArranger::IColumnItemView::Identifier)
|
|
{
|
|
return (ListViewColumnItemArranger::IColumnItemView*)this;
|
|
}
|
|
else
|
|
{
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
// ===================== list::ListViewItemStyleProvider::IListViewItemView =====================
|
|
|
|
Ptr<GuiImageData> GuiBindableListView::ItemSource::GetSmallImage(vint itemIndex)
|
|
{
|
|
if (itemSource)
|
|
{
|
|
if (0 <= itemIndex && itemIndex < itemSource->GetCount())
|
|
{
|
|
return ReadProperty(itemSource->Get(itemIndex), smallImageProperty);
|
|
}
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
Ptr<GuiImageData> GuiBindableListView::ItemSource::GetLargeImage(vint itemIndex)
|
|
{
|
|
if (itemSource)
|
|
{
|
|
if (0 <= itemIndex && itemIndex < itemSource->GetCount())
|
|
{
|
|
return ReadProperty(itemSource->Get(itemIndex), largeImageProperty);
|
|
}
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
WString GuiBindableListView::ItemSource::GetText(vint itemIndex)
|
|
{
|
|
if (itemSource)
|
|
{
|
|
if (0 <= itemIndex && itemIndex < itemSource->GetCount() && columns.Count()>0)
|
|
{
|
|
return ReadProperty(itemSource->Get(itemIndex), columns[0]->GetTextProperty());
|
|
}
|
|
}
|
|
return L"";
|
|
}
|
|
|
|
WString GuiBindableListView::ItemSource::GetSubItem(vint itemIndex, vint index)
|
|
{
|
|
if (itemSource)
|
|
{
|
|
if (0 <= itemIndex && itemIndex < itemSource->GetCount() && 0 <= index && index < columns.Count() - 1)
|
|
{
|
|
return ReadProperty(itemSource->Get(itemIndex), columns[index + 1]->GetTextProperty());
|
|
}
|
|
}
|
|
return L"";
|
|
}
|
|
|
|
vint GuiBindableListView::ItemSource::GetDataColumnCount()
|
|
{
|
|
return dataColumns.Count();
|
|
}
|
|
|
|
vint GuiBindableListView::ItemSource::GetDataColumn(vint index)
|
|
{
|
|
return dataColumns[index];
|
|
}
|
|
|
|
vint GuiBindableListView::ItemSource::GetColumnCount()
|
|
{
|
|
return columns.Count();
|
|
}
|
|
|
|
WString GuiBindableListView::ItemSource::GetColumnText(vint index)
|
|
{
|
|
if (index < 0 || index >= columns.Count())
|
|
{
|
|
return L"";
|
|
}
|
|
else
|
|
{
|
|
return columns[index]->GetText();
|
|
}
|
|
}
|
|
|
|
// ===================== list::ListViewColumnItemArranger::IColumnItemView =====================
|
|
|
|
bool GuiBindableListView::ItemSource::AttachCallback(ListViewColumnItemArranger::IColumnItemViewCallback* value)
|
|
{
|
|
if(columnItemViewCallbacks.Contains(value))
|
|
{
|
|
return false;
|
|
}
|
|
else
|
|
{
|
|
columnItemViewCallbacks.Add(value);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
bool GuiBindableListView::ItemSource::DetachCallback(ListViewColumnItemArranger::IColumnItemViewCallback* value)
|
|
{
|
|
vint index = columnItemViewCallbacks.IndexOf(value);
|
|
if (index == -1)
|
|
{
|
|
return false;
|
|
}
|
|
else
|
|
{
|
|
columnItemViewCallbacks.Remove(value);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
vint GuiBindableListView::ItemSource::GetColumnSize(vint index)
|
|
{
|
|
if (index < 0 || index >= columns.Count())
|
|
{
|
|
return 0;
|
|
}
|
|
else
|
|
{
|
|
return columns[index]->GetSize();
|
|
}
|
|
}
|
|
|
|
void GuiBindableListView::ItemSource::SetColumnSize(vint index, vint value)
|
|
{
|
|
if (index >= 0 && index < columns.Count())
|
|
{
|
|
columns[index]->SetSize(value);
|
|
}
|
|
}
|
|
|
|
GuiMenu* GuiBindableListView::ItemSource::GetDropdownPopup(vint index)
|
|
{
|
|
if (index < 0 || index >= columns.Count())
|
|
{
|
|
return 0;
|
|
}
|
|
else
|
|
{
|
|
return columns[index]->GetDropdownPopup();
|
|
}
|
|
}
|
|
|
|
ColumnSortingState GuiBindableListView::ItemSource::GetSortingState(vint index)
|
|
{
|
|
if (index < 0 || index >= columns.Count())
|
|
{
|
|
return ColumnSortingState::NotSorted;
|
|
}
|
|
else
|
|
{
|
|
return columns[index]->GetSortingState();
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiBindableListView
|
|
***********************************************************************/
|
|
|
|
GuiBindableListView::GuiBindableListView(theme::ThemeName themeName)
|
|
:GuiVirtualListView(themeName, new ItemSource)
|
|
{
|
|
itemSource = dynamic_cast<ItemSource*>(GetItemProvider());
|
|
|
|
LargeImagePropertyChanged.SetAssociatedComposition(boundsComposition);
|
|
SmallImagePropertyChanged.SetAssociatedComposition(boundsComposition);
|
|
}
|
|
|
|
GuiBindableListView::~GuiBindableListView()
|
|
{
|
|
}
|
|
|
|
list::ListViewDataColumns& GuiBindableListView::GetDataColumns()
|
|
{
|
|
return itemSource->GetDataColumns();
|
|
}
|
|
|
|
list::ListViewColumns& GuiBindableListView::GetColumns()
|
|
{
|
|
return itemSource->GetColumns();
|
|
}
|
|
|
|
Ptr<description::IValueEnumerable> GuiBindableListView::GetItemSource()
|
|
{
|
|
return itemSource->GetItemSource();
|
|
}
|
|
|
|
void GuiBindableListView::SetItemSource(Ptr<description::IValueEnumerable> _itemSource)
|
|
{
|
|
itemSource->SetItemSource(_itemSource);
|
|
}
|
|
|
|
ItemProperty<Ptr<GuiImageData>> GuiBindableListView::GetLargeImageProperty()
|
|
{
|
|
return itemSource->largeImageProperty;
|
|
}
|
|
|
|
void GuiBindableListView::SetLargeImageProperty(const ItemProperty<Ptr<GuiImageData>>& value)
|
|
{
|
|
if (itemSource->largeImageProperty != value)
|
|
{
|
|
itemSource->largeImageProperty = value;
|
|
itemSource->UpdateBindingProperties();
|
|
LargeImagePropertyChanged.Execute(GetNotifyEventArguments());
|
|
}
|
|
}
|
|
|
|
ItemProperty<Ptr<GuiImageData>> GuiBindableListView::GetSmallImageProperty()
|
|
{
|
|
return itemSource->smallImageProperty;
|
|
}
|
|
|
|
void GuiBindableListView::SetSmallImageProperty(const ItemProperty<Ptr<GuiImageData>>& value)
|
|
{
|
|
if (itemSource->smallImageProperty != value)
|
|
{
|
|
itemSource->smallImageProperty = value;
|
|
itemSource->UpdateBindingProperties();
|
|
SmallImagePropertyChanged.Execute(GetNotifyEventArguments());
|
|
}
|
|
}
|
|
|
|
description::Value GuiBindableListView::GetSelectedItem()
|
|
{
|
|
vint index = GetSelectedItemIndex();
|
|
if (index == -1) return Value();
|
|
return itemSource->Get(index);
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiBindableTreeView::ItemSourceNode
|
|
***********************************************************************/
|
|
|
|
Ptr<description::IValueReadonlyList> GuiBindableTreeView::ItemSourceNode::PrepareValueList(const description::Value& inputItemSource)
|
|
{
|
|
if (auto value = ReadProperty(inputItemSource, rootProvider->childrenProperty))
|
|
{
|
|
if (auto ol = value.Cast<IValueObservableList>())
|
|
{
|
|
return ol;
|
|
}
|
|
else if (auto rl = value.Cast<IValueReadonlyList>())
|
|
{
|
|
return rl;
|
|
}
|
|
else
|
|
{
|
|
return IValueList::Create(GetLazyList<Value>(value));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
return IValueList::Create();
|
|
}
|
|
}
|
|
|
|
void GuiBindableTreeView::ItemSourceNode::PrepareChildren(Ptr<description::IValueReadonlyList> newValueList)
|
|
{
|
|
if (!childrenVirtualList)
|
|
{
|
|
childrenVirtualList = newValueList;
|
|
if (auto ol = childrenVirtualList.Cast<IValueObservableList>())
|
|
{
|
|
itemChangedEventHandler = ol->ItemChanged.Add([this](vint start, vint oldCount, vint newCount)
|
|
{
|
|
callback->OnBeforeItemModified(this, start, oldCount, newCount);
|
|
children.RemoveRange(start, oldCount);
|
|
for (vint i = 0; i < newCount; i++)
|
|
{
|
|
Value value = childrenVirtualList->Get(start + i);
|
|
auto node = new ItemSourceNode(value, this);
|
|
children.Insert(start + i, node);
|
|
}
|
|
callback->OnAfterItemModified(this, start, oldCount, newCount);
|
|
});
|
|
}
|
|
|
|
vint count = childrenVirtualList->GetCount();
|
|
for (vint i = 0; i < count; i++)
|
|
{
|
|
Value value = childrenVirtualList->Get(i);
|
|
auto node = new ItemSourceNode(value, this);
|
|
children.Add(node);
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiBindableTreeView::ItemSourceNode::UnprepareChildren()
|
|
{
|
|
if (itemChangedEventHandler)
|
|
{
|
|
auto ol = childrenVirtualList.Cast<IValueObservableList>();
|
|
ol->ItemChanged.Remove(itemChangedEventHandler);
|
|
itemChangedEventHandler = nullptr;
|
|
}
|
|
childrenVirtualList = nullptr;
|
|
FOREACH(Ptr<ItemSourceNode>, node, children)
|
|
{
|
|
node->UnprepareChildren();
|
|
}
|
|
children.Clear();
|
|
}
|
|
|
|
GuiBindableTreeView::ItemSourceNode::ItemSourceNode(const description::Value& _itemSource, ItemSourceNode* _parent)
|
|
:itemSource(_itemSource)
|
|
, rootProvider(_parent->rootProvider)
|
|
, parent(_parent)
|
|
, callback(_parent->callback)
|
|
{
|
|
}
|
|
|
|
GuiBindableTreeView::ItemSourceNode::ItemSourceNode(ItemSource* _rootProvider)
|
|
:rootProvider(_rootProvider)
|
|
, parent(nullptr)
|
|
, callback(_rootProvider)
|
|
{
|
|
}
|
|
|
|
GuiBindableTreeView::ItemSourceNode::~ItemSourceNode()
|
|
{
|
|
}
|
|
|
|
description::Value GuiBindableTreeView::ItemSourceNode::GetItemSource()
|
|
{
|
|
return itemSource;
|
|
}
|
|
|
|
void GuiBindableTreeView::ItemSourceNode::SetItemSource(const description::Value& _itemSource)
|
|
{
|
|
auto newVirtualList = PrepareValueList(_itemSource);
|
|
vint oldCount = childrenVirtualList ? childrenVirtualList->GetCount() : 0;
|
|
vint newCount = newVirtualList->GetCount();
|
|
|
|
callback->OnBeforeItemModified(this, 0, oldCount, newCount);
|
|
UnprepareChildren();
|
|
itemSource = _itemSource;
|
|
PrepareChildren(newVirtualList);
|
|
callback->OnAfterItemModified(this, 0, oldCount, newCount);
|
|
}
|
|
|
|
bool GuiBindableTreeView::ItemSourceNode::GetExpanding()
|
|
{
|
|
return this == rootProvider->rootNode.Obj() ? true : expanding;
|
|
}
|
|
|
|
void GuiBindableTreeView::ItemSourceNode::SetExpanding(bool value)
|
|
{
|
|
if (this != rootProvider->rootNode.Obj() && expanding != value)
|
|
{
|
|
expanding = value;
|
|
if (expanding)
|
|
{
|
|
callback->OnItemExpanded(this);
|
|
}
|
|
else
|
|
{
|
|
callback->OnItemCollapsed(this);
|
|
}
|
|
}
|
|
}
|
|
|
|
vint GuiBindableTreeView::ItemSourceNode::CalculateTotalVisibleNodes()
|
|
{
|
|
if (!GetExpanding())
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
if (!childrenVirtualList)
|
|
{
|
|
PrepareChildren(PrepareValueList(itemSource));
|
|
}
|
|
vint count = 1;
|
|
FOREACH(Ptr<ItemSourceNode>, child, children)
|
|
{
|
|
count += child->CalculateTotalVisibleNodes();
|
|
}
|
|
return count;
|
|
}
|
|
|
|
vint GuiBindableTreeView::ItemSourceNode::GetChildCount()
|
|
{
|
|
if (!childrenVirtualList)
|
|
{
|
|
PrepareChildren(PrepareValueList(itemSource));
|
|
}
|
|
return children.Count();
|
|
}
|
|
|
|
Ptr<tree::INodeProvider> GuiBindableTreeView::ItemSourceNode::GetParent()
|
|
{
|
|
return parent;
|
|
}
|
|
|
|
Ptr<tree::INodeProvider> GuiBindableTreeView::ItemSourceNode::GetChild(vint index)
|
|
{
|
|
if (!childrenVirtualList)
|
|
{
|
|
PrepareChildren(PrepareValueList(itemSource));
|
|
}
|
|
if (0 <= index && index < children.Count())
|
|
{
|
|
return children[index];
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiBindableTreeView::ItemSource
|
|
***********************************************************************/
|
|
|
|
GuiBindableTreeView::ItemSource::ItemSource()
|
|
{
|
|
rootNode = new ItemSourceNode(this);
|
|
}
|
|
|
|
GuiBindableTreeView::ItemSource::~ItemSource()
|
|
{
|
|
}
|
|
|
|
description::Value GuiBindableTreeView::ItemSource::GetItemSource()
|
|
{
|
|
return rootNode->GetItemSource();
|
|
}
|
|
|
|
void GuiBindableTreeView::ItemSource::SetItemSource(const description::Value& _itemSource)
|
|
{
|
|
rootNode->SetItemSource(_itemSource);
|
|
}
|
|
|
|
void GuiBindableTreeView::ItemSource::UpdateBindingProperties(bool updateChildrenProperty)
|
|
{
|
|
vint oldCount = rootNode->GetChildCount();
|
|
if (updateChildrenProperty)
|
|
{
|
|
rootNode->UnprepareChildren();
|
|
}
|
|
vint newCount = rootNode->GetChildCount();
|
|
OnBeforeItemModified(rootNode.Obj(), 0, oldCount, newCount);
|
|
OnAfterItemModified(rootNode.Obj(), 0, oldCount, newCount);
|
|
}
|
|
|
|
// ===================== tree::INodeRootProvider =====================
|
|
|
|
Ptr<tree::INodeProvider> GuiBindableTreeView::ItemSource::GetRootNode()
|
|
{
|
|
return rootNode;
|
|
}
|
|
|
|
WString GuiBindableTreeView::ItemSource::GetTextValue(tree::INodeProvider* node)
|
|
{
|
|
return ReadProperty(GetBindingValue(node), textProperty);
|
|
}
|
|
|
|
description::Value GuiBindableTreeView::ItemSource::GetBindingValue(tree::INodeProvider* node)
|
|
{
|
|
if (auto itemSourceNode = dynamic_cast<ItemSourceNode*>(node))
|
|
{
|
|
return itemSourceNode->GetItemSource();
|
|
}
|
|
return Value();
|
|
}
|
|
|
|
IDescriptable* GuiBindableTreeView::ItemSource::RequestView(const WString& identifier)
|
|
{
|
|
if(identifier==ITreeViewItemView::Identifier)
|
|
{
|
|
return (ITreeViewItemView*)this;
|
|
}
|
|
else
|
|
{
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
// ===================== tree::ITreeViewItemView =====================
|
|
|
|
Ptr<GuiImageData> GuiBindableTreeView::ItemSource::GetNodeImage(tree::INodeProvider* node)
|
|
{
|
|
if (auto itemSourceNode = dynamic_cast<ItemSourceNode*>(node))
|
|
{
|
|
return ReadProperty(itemSourceNode->GetItemSource(), imageProperty);
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiBindableTreeView
|
|
***********************************************************************/
|
|
|
|
GuiBindableTreeView::GuiBindableTreeView(theme::ThemeName themeName)
|
|
:GuiVirtualTreeView(themeName, new ItemSource)
|
|
{
|
|
itemSource = dynamic_cast<ItemSource*>(GetNodeRootProvider());
|
|
|
|
TextPropertyChanged.SetAssociatedComposition(boundsComposition);
|
|
ImagePropertyChanged.SetAssociatedComposition(boundsComposition);
|
|
ChildrenPropertyChanged.SetAssociatedComposition(boundsComposition);
|
|
}
|
|
|
|
GuiBindableTreeView::~GuiBindableTreeView()
|
|
{
|
|
}
|
|
|
|
description::Value GuiBindableTreeView::GetItemSource()
|
|
{
|
|
return itemSource->GetItemSource();
|
|
}
|
|
|
|
void GuiBindableTreeView::SetItemSource(description::Value _itemSource)
|
|
{
|
|
itemSource->SetItemSource(_itemSource);
|
|
}
|
|
|
|
ItemProperty<WString> GuiBindableTreeView::GetTextProperty()
|
|
{
|
|
return itemSource->textProperty;
|
|
}
|
|
|
|
void GuiBindableTreeView::SetTextProperty(const ItemProperty<WString>& value)
|
|
{
|
|
if (itemSource->textProperty != value)
|
|
{
|
|
itemSource->textProperty = value;
|
|
itemSource->UpdateBindingProperties(false);
|
|
TextPropertyChanged.Execute(GetNotifyEventArguments());
|
|
}
|
|
}
|
|
|
|
ItemProperty<Ptr<GuiImageData>> GuiBindableTreeView::GetImageProperty()
|
|
{
|
|
return itemSource->imageProperty;
|
|
}
|
|
|
|
void GuiBindableTreeView::SetImageProperty(const ItemProperty<Ptr<GuiImageData>>& value)
|
|
{
|
|
if (itemSource->imageProperty != value)
|
|
{
|
|
itemSource->imageProperty = value;
|
|
itemSource->UpdateBindingProperties(false);
|
|
ImagePropertyChanged.Execute(GetNotifyEventArguments());
|
|
}
|
|
}
|
|
|
|
ItemProperty<Ptr<IValueEnumerable>> GuiBindableTreeView::GetChildrenProperty()
|
|
{
|
|
return itemSource->childrenProperty;
|
|
}
|
|
|
|
void GuiBindableTreeView::SetChildrenProperty(const ItemProperty<Ptr<IValueEnumerable>>& value)
|
|
{
|
|
if (itemSource->childrenProperty != value)
|
|
{
|
|
itemSource->childrenProperty = value;
|
|
itemSource->UpdateBindingProperties(true);
|
|
ChildrenPropertyChanged.Execute(GetNotifyEventArguments());
|
|
}
|
|
}
|
|
|
|
description::Value GuiBindableTreeView::GetSelectedItem()
|
|
{
|
|
vint index = GetSelectedItemIndex();
|
|
if (index == -1) return Value();
|
|
|
|
Value result;
|
|
if (auto node = nodeItemView->RequestNode(index))
|
|
{
|
|
if (auto itemSourceNode = node.Cast<ItemSourceNode>())
|
|
{
|
|
result = itemSourceNode->GetItemSource();
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
.\CONTROLS\LISTCONTROLPACKAGE\GUICOMBOCONTROLS.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace controls
|
|
{
|
|
|
|
/***********************************************************************
|
|
GuiComboBoxBase
|
|
***********************************************************************/
|
|
|
|
void GuiComboBoxBase::BeforeControlTemplateUninstalled_()
|
|
{
|
|
}
|
|
|
|
void GuiComboBoxBase::AfterControlTemplateInstalled_(bool initialize)
|
|
{
|
|
}
|
|
|
|
IGuiMenuService::Direction GuiComboBoxBase::GetSubMenuDirection()
|
|
{
|
|
return IGuiMenuService::Horizontal;
|
|
}
|
|
|
|
void GuiComboBoxBase::OnBoundsChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
Size size=GetPreferredMenuClientSize();
|
|
size.x=boundsComposition->GetBounds().Width();
|
|
SetPreferredMenuClientSize(size);
|
|
}
|
|
|
|
GuiComboBoxBase::GuiComboBoxBase(theme::ThemeName themeName)
|
|
:GuiMenuButton(themeName)
|
|
{
|
|
CreateSubMenu();
|
|
SetCascadeAction(false);
|
|
|
|
boundsComposition->BoundsChanged.AttachMethod(this, &GuiComboBoxBase::OnBoundsChanged);
|
|
}
|
|
|
|
GuiComboBoxBase::~GuiComboBoxBase()
|
|
{
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiComboBoxListControl
|
|
***********************************************************************/
|
|
|
|
void GuiComboBoxListControl::UpdateDisplayFont()
|
|
{
|
|
GuiControl::UpdateDisplayFont();
|
|
if (itemStyleController)
|
|
{
|
|
itemStyleController->SetFont(GetDisplayFont());
|
|
}
|
|
AdoptSubMenuSize();
|
|
}
|
|
|
|
void GuiComboBoxListControl::BeforeControlTemplateUninstalled()
|
|
{
|
|
GuiComboBoxBase::BeforeControlTemplateUninstalled();
|
|
}
|
|
|
|
void GuiComboBoxListControl::AfterControlTemplateInstalled(bool initialize)
|
|
{
|
|
GuiComboBoxBase::AfterControlTemplateInstalled(initialize);
|
|
GetControlTemplateObject(true)->SetTextVisible(!itemStyleProperty);
|
|
}
|
|
|
|
void GuiComboBoxListControl::RemoveStyleController()
|
|
{
|
|
if (itemStyleController)
|
|
{
|
|
SafeDeleteComposition(itemStyleController);
|
|
itemStyleController = nullptr;
|
|
}
|
|
}
|
|
|
|
void GuiComboBoxListControl::InstallStyleController(vint itemIndex)
|
|
{
|
|
if (itemStyleProperty)
|
|
{
|
|
if (itemIndex != -1)
|
|
{
|
|
auto item = containedListControl->GetItemProvider()->GetBindingValue(itemIndex);
|
|
if (!item.IsNull())
|
|
{
|
|
if (auto style = itemStyleProperty(item))
|
|
{
|
|
itemStyleController = style;
|
|
itemStyleController->SetText(GetText());
|
|
itemStyleController->SetFont(GetDisplayFont());
|
|
itemStyleController->SetContext(GetContext());
|
|
itemStyleController->SetVisuallyEnabled(GetVisuallyEnabled());
|
|
itemStyleController->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
containerComposition->AddChild(itemStyleController);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiComboBoxListControl::DisplaySelectedContent(vint itemIndex)
|
|
{
|
|
if (itemIndex == -1)
|
|
{
|
|
SetText(L"");
|
|
}
|
|
else
|
|
{
|
|
WString text = containedListControl->GetItemProvider()->GetTextValue(itemIndex);
|
|
SetText(text);
|
|
}
|
|
|
|
RemoveStyleController();
|
|
InstallStyleController(itemIndex);
|
|
if (selectedIndex != itemIndex)
|
|
{
|
|
selectedIndex = itemIndex;
|
|
SelectedIndexChanged.Execute(GetNotifyEventArguments());
|
|
}
|
|
}
|
|
|
|
void GuiComboBoxListControl::AdoptSubMenuSize()
|
|
{
|
|
Size expectedSize(0, GetDisplayFont().size * 20);
|
|
Size adoptedSize = containedListControl->GetAdoptedSize(expectedSize);
|
|
|
|
Size clientSize = GetPreferredMenuClientSize();
|
|
clientSize.y = adoptedSize.y + GetSubMenu()->GetClientSize().y - containedListControl->GetBoundsComposition()->GetBounds().Height();
|
|
SetPreferredMenuClientSize(clientSize);
|
|
|
|
if (GetSubMenuOpening())
|
|
{
|
|
GetSubMenu()->SetClientSize(clientSize);
|
|
}
|
|
}
|
|
|
|
void GuiComboBoxListControl::OnTextChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
if (itemStyleController)
|
|
{
|
|
itemStyleController->SetText(GetText());
|
|
}
|
|
}
|
|
|
|
void GuiComboBoxListControl::OnContextChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
if (itemStyleController)
|
|
{
|
|
itemStyleController->SetContext(GetContext());
|
|
}
|
|
AdoptSubMenuSize();
|
|
}
|
|
|
|
void GuiComboBoxListControl::OnVisuallyEnabledChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
if (itemStyleController)
|
|
{
|
|
itemStyleController->SetVisuallyEnabled(GetVisuallyEnabled());
|
|
}
|
|
}
|
|
|
|
void GuiComboBoxListControl::OnAfterSubMenuOpening(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
containedListControl->SelectItemsByClick(selectedIndex, false, false, true);
|
|
GetSubMenu()->GetNativeWindow()->SetFocus();
|
|
containedListControl->SetFocus();
|
|
}
|
|
|
|
void GuiComboBoxListControl::OnListControlAdoptedSizeInvalidated(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
AdoptSubMenuSize();
|
|
}
|
|
|
|
void GuiComboBoxListControl::OnListControlBoundsChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
auto flag = GetDisposedFlag();
|
|
GetApplication()->InvokeLambdaInMainThread(GetRelatedControlHost(), [=]()
|
|
{
|
|
if (!flag->IsDisposed())
|
|
{
|
|
AdoptSubMenuSize();
|
|
}
|
|
});
|
|
}
|
|
|
|
void GuiComboBoxListControl::OnListControlItemMouseDown(compositions::GuiGraphicsComposition* sender, compositions::GuiItemMouseEventArgs& arguments)
|
|
{
|
|
DisplaySelectedContent(containedListControl->GetSelectedItemIndex());
|
|
GetSubMenu()->Hide();
|
|
}
|
|
|
|
void GuiComboBoxListControl::OnListControlKeyDown(compositions::GuiGraphicsComposition* sender, compositions::GuiKeyEventArgs& arguments)
|
|
{
|
|
if (!arguments.autoRepeatKeyDown)
|
|
{
|
|
switch (arguments.code)
|
|
{
|
|
case VKEY::_RETURN:
|
|
DisplaySelectedContent(containedListControl->GetSelectedItemIndex());
|
|
arguments.handled = true;
|
|
case VKEY::_ESCAPE:
|
|
GetSubMenu()->Hide();
|
|
arguments.handled = true;
|
|
break;
|
|
default:;
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiComboBoxListControl::OnAttached(GuiListControl::IItemProvider* provider)
|
|
{
|
|
}
|
|
|
|
void GuiComboBoxListControl::OnItemModified(vint start, vint count, vint newCount)
|
|
{
|
|
if (count == newCount)
|
|
{
|
|
if (start <= selectedIndex && selectedIndex < start + count)
|
|
{
|
|
DisplaySelectedContent(selectedIndex);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
DisplaySelectedContent(-1);
|
|
}
|
|
}
|
|
|
|
GuiComboBoxListControl::GuiComboBoxListControl(theme::ThemeName themeName, GuiSelectableListControl* _containedListControl)
|
|
:GuiComboBoxBase(themeName)
|
|
, containedListControl(_containedListControl)
|
|
{
|
|
TextChanged.AttachMethod(this, &GuiComboBoxListControl::OnTextChanged);
|
|
ContextChanged.AttachMethod(this, &GuiComboBoxListControl::OnContextChanged);
|
|
VisuallyEnabledChanged.AttachMethod(this, &GuiComboBoxListControl::OnVisuallyEnabledChanged);
|
|
AfterSubMenuOpening.AttachMethod(this, &GuiComboBoxListControl::OnAfterSubMenuOpening);
|
|
|
|
containedListControl->GetItemProvider()->AttachCallback(this);
|
|
containedListControl->SetMultiSelect(false);
|
|
containedListControl->AdoptedSizeInvalidated.AttachMethod(this, &GuiComboBoxListControl::OnListControlAdoptedSizeInvalidated);
|
|
containedListControl->ItemLeftButtonDown.AttachMethod(this, &GuiComboBoxListControl::OnListControlItemMouseDown);
|
|
containedListControl->ItemRightButtonDown.AttachMethod(this, &GuiComboBoxListControl::OnListControlItemMouseDown);
|
|
containedListControl->GetFocusableComposition()->GetEventReceiver()->keyDown.AttachMethod(this, &GuiComboBoxListControl::OnListControlKeyDown);
|
|
boundsChangedHandler = containedListControl->GetBoundsComposition()->BoundsChanged.AttachMethod(this, &GuiComboBoxListControl::OnListControlBoundsChanged);
|
|
|
|
auto itemProvider = containedListControl->GetItemProvider();
|
|
|
|
SelectedIndexChanged.SetAssociatedComposition(boundsComposition);
|
|
|
|
containedListControl->GetBoundsComposition()->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
GetSubMenu()->GetContainerComposition()->AddChild(containedListControl->GetBoundsComposition());
|
|
SetFont(GetFont());
|
|
}
|
|
|
|
GuiComboBoxListControl::~GuiComboBoxListControl()
|
|
{
|
|
containedListControl->GetItemProvider()->DetachCallback(this);
|
|
containedListControl->GetBoundsComposition()->BoundsChanged.Detach(boundsChangedHandler);
|
|
boundsChangedHandler = nullptr;
|
|
}
|
|
|
|
GuiSelectableListControl* GuiComboBoxListControl::GetContainedListControl()
|
|
{
|
|
return containedListControl;
|
|
}
|
|
|
|
GuiComboBoxListControl::ItemStyleProperty GuiComboBoxListControl::GetItemTemplate()
|
|
{
|
|
return itemStyleProperty;
|
|
}
|
|
|
|
void GuiComboBoxListControl::SetItemTemplate(ItemStyleProperty value)
|
|
{
|
|
RemoveStyleController();
|
|
itemStyleProperty = value;
|
|
GetControlTemplateObject(true)->SetTextVisible(!itemStyleProperty);
|
|
InstallStyleController(selectedIndex);
|
|
ItemTemplateChanged.Execute(GetNotifyEventArguments());
|
|
}
|
|
|
|
vint GuiComboBoxListControl::GetSelectedIndex()
|
|
{
|
|
return selectedIndex;
|
|
}
|
|
|
|
void GuiComboBoxListControl::SetSelectedIndex(vint value)
|
|
{
|
|
if (selectedIndex != value)
|
|
{
|
|
if (0 <= value && value < containedListControl->GetItemProvider()->Count())
|
|
{
|
|
DisplaySelectedContent(value);
|
|
}
|
|
}
|
|
GetSubMenu()->Hide();
|
|
}
|
|
|
|
description::Value GuiComboBoxListControl::GetSelectedItem()
|
|
{
|
|
if (selectedIndex != -1)
|
|
{
|
|
return containedListControl->GetItemProvider()->GetBindingValue(selectedIndex);
|
|
}
|
|
return description::Value();
|
|
}
|
|
|
|
GuiListControl::IItemProvider* GuiComboBoxListControl::GetItemProvider()
|
|
{
|
|
return containedListControl->GetItemProvider();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
.\CONTROLS\LISTCONTROLPACKAGE\GUIDATAGRIDCONTROLS.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace controls
|
|
{
|
|
namespace list
|
|
{
|
|
using namespace compositions;
|
|
using namespace collections;
|
|
using namespace description;
|
|
using namespace templates;
|
|
|
|
const wchar_t* const IDataGridView::Identifier = L"vl::presentation::controls::list::IDataGridView";
|
|
|
|
/***********************************************************************
|
|
DefaultDataGridItemTemplate
|
|
***********************************************************************/
|
|
|
|
IDataVisualizerFactory* DefaultDataGridItemTemplate::GetDataVisualizerFactory(vint row, vint column)
|
|
{
|
|
if (auto dataGrid = dynamic_cast<GuiVirtualDataGrid*>(listControl))
|
|
{
|
|
if (auto factory = dataGrid->dataGridView->GetCellDataVisualizerFactory(row, column))
|
|
{
|
|
return factory;
|
|
}
|
|
|
|
if (column == 0)
|
|
{
|
|
return dataGrid->defaultMainColumnVisualizerFactory.Obj();
|
|
}
|
|
else
|
|
{
|
|
return dataGrid->defaultSubColumnVisualizerFactory.Obj();
|
|
}
|
|
|
|
}
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
IDataEditorFactory* DefaultDataGridItemTemplate::GetDataEditorFactory(vint row, vint column)
|
|
{
|
|
if (auto dataGrid = dynamic_cast<GuiVirtualDataGrid*>(listControl))
|
|
{
|
|
return dataGrid->dataGridView->GetCellDataEditorFactory(row, column);
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
vint DefaultDataGridItemTemplate::GetCellColumnIndex(compositions::GuiGraphicsComposition* composition)
|
|
{
|
|
for (vint i = 0; i < textTable->GetColumns(); i++)
|
|
{
|
|
auto cell = textTable->GetSitedCell(0, i);
|
|
if (composition == cell)
|
|
{
|
|
return i;
|
|
}
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
bool DefaultDataGridItemTemplate::IsInEditor(GuiVirtualDataGrid* dataGrid, compositions::GuiMouseEventArgs& arguments)
|
|
{
|
|
if (!dataGrid->currentEditor) return false;
|
|
auto editorComposition = dataGrid->currentEditor->GetTemplate();
|
|
auto currentComposition = arguments.eventSource;
|
|
|
|
while (currentComposition)
|
|
{
|
|
if (currentComposition == editorComposition)
|
|
{
|
|
return true;
|
|
}
|
|
else if (currentComposition == this)
|
|
{
|
|
break;
|
|
}
|
|
else
|
|
{
|
|
currentComposition = currentComposition->GetParent();
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
void DefaultDataGridItemTemplate::OnCellButtonDown(compositions::GuiGraphicsComposition* sender, compositions::GuiMouseEventArgs& arguments)
|
|
{
|
|
if (auto dataGrid = dynamic_cast<GuiVirtualDataGrid*>(listControl))
|
|
{
|
|
if (IsInEditor(dataGrid, arguments))
|
|
{
|
|
arguments.handled = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
void DefaultDataGridItemTemplate::OnCellLeftButtonUp(compositions::GuiGraphicsComposition* sender, compositions::GuiMouseEventArgs& arguments)
|
|
{
|
|
if (auto dataGrid = dynamic_cast<GuiVirtualDataGrid*>(listControl))
|
|
{
|
|
if (IsInEditor(dataGrid, arguments))
|
|
{
|
|
arguments.handled = true;
|
|
}
|
|
else if (dataGrid->GetVisuallyEnabled())
|
|
{
|
|
vint index = GetCellColumnIndex(sender);
|
|
if (index != -1)
|
|
{
|
|
vint currentRow = GetIndex();
|
|
dataGrid->SelectCell({ currentRow,index }, true);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void DefaultDataGridItemTemplate::OnCellRightButtonUp(compositions::GuiGraphicsComposition* sender, compositions::GuiMouseEventArgs& arguments)
|
|
{
|
|
if (auto dataGrid = dynamic_cast<GuiVirtualDataGrid*>(listControl))
|
|
{
|
|
if (IsInEditor(dataGrid, arguments))
|
|
{
|
|
arguments.handled = true;
|
|
}
|
|
else if (dataGrid->GetVisuallyEnabled())
|
|
{
|
|
vint index = GetCellColumnIndex(sender);
|
|
if (index != -1)
|
|
{
|
|
vint currentRow = GetIndex();
|
|
dataGrid->SelectCell({ currentRow,index }, false);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void DefaultDataGridItemTemplate::OnColumnChanged()
|
|
{
|
|
UpdateSubItemSize();
|
|
}
|
|
|
|
void DefaultDataGridItemTemplate::OnInitialize()
|
|
{
|
|
DefaultListViewItemTemplate::OnInitialize();
|
|
{
|
|
textTable = new GuiTableComposition;
|
|
textTable->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
textTable->SetRowsAndColumns(1, 1);
|
|
textTable->SetRowOption(0, GuiCellOption::MinSizeOption());
|
|
textTable->SetColumnOption(0, GuiCellOption::AbsoluteOption(0));
|
|
AddChild(textTable);
|
|
}
|
|
|
|
if (auto dataGrid = dynamic_cast<GuiVirtualDataGrid*>(listControl))
|
|
{
|
|
vint columnCount = dataGrid->listViewItemView->GetColumnCount();
|
|
vint itemIndex = GetIndex();
|
|
|
|
dataVisualizers.Resize(columnCount);
|
|
for (vint i = 0; i < dataVisualizers.Count(); i++)
|
|
{
|
|
auto factory = GetDataVisualizerFactory(itemIndex, i);
|
|
dataVisualizers[i] = factory->CreateVisualizer(dataGrid);
|
|
}
|
|
|
|
textTable->SetRowsAndColumns(1, columnCount);
|
|
for (vint i = 0; i < columnCount; i++)
|
|
{
|
|
auto cell = new GuiCellComposition;
|
|
textTable->AddChild(cell);
|
|
cell->SetSite(0, i, 1, 1);
|
|
cell->GetEventReceiver()->leftButtonDown.AttachMethod(this, &DefaultDataGridItemTemplate::OnCellButtonDown);
|
|
cell->GetEventReceiver()->rightButtonDown.AttachMethod(this, &DefaultDataGridItemTemplate::OnCellButtonDown);
|
|
cell->GetEventReceiver()->leftButtonUp.AttachMethod(this, &DefaultDataGridItemTemplate::OnCellLeftButtonUp);
|
|
cell->GetEventReceiver()->rightButtonUp.AttachMethod(this, &DefaultDataGridItemTemplate::OnCellRightButtonUp);
|
|
|
|
auto composition = dataVisualizers[i]->GetTemplate();
|
|
composition->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
cell->AddChild(composition);
|
|
}
|
|
|
|
for (vint i = 0; i < dataVisualizers.Count(); i++)
|
|
{
|
|
dataVisualizers[i]->BeforeVisualizeCell(dataGrid->GetItemProvider(), itemIndex, i);
|
|
}
|
|
|
|
GridPos selectedCell = dataGrid->GetSelectedCell();
|
|
if (selectedCell.row == itemIndex)
|
|
{
|
|
NotifySelectCell(selectedCell.column);
|
|
}
|
|
else
|
|
{
|
|
NotifySelectCell(-1);
|
|
}
|
|
UpdateSubItemSize();
|
|
}
|
|
|
|
SelectedChanged.AttachMethod(this, &DefaultDataGridItemTemplate::OnSelectedChanged);
|
|
FontChanged.AttachMethod(this, &DefaultDataGridItemTemplate::OnFontChanged);
|
|
ContextChanged.AttachMethod(this, &DefaultDataGridItemTemplate::OnContextChanged);
|
|
|
|
SelectedChanged.Execute(compositions::GuiEventArgs(this));
|
|
FontChanged.Execute(compositions::GuiEventArgs(this));
|
|
ContextChanged.Execute(compositions::GuiEventArgs(this));
|
|
}
|
|
|
|
void DefaultDataGridItemTemplate::OnSelectedChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
if (!GetSelected())
|
|
{
|
|
NotifySelectCell(-1);
|
|
}
|
|
}
|
|
|
|
void DefaultDataGridItemTemplate::OnFontChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
FOREACH(Ptr<IDataVisualizer>, visualizer, dataVisualizers)
|
|
{
|
|
visualizer->GetTemplate()->SetFont(GetFont());
|
|
}
|
|
if (currentEditor)
|
|
{
|
|
currentEditor->GetTemplate()->SetFont(GetFont());
|
|
}
|
|
}
|
|
|
|
void DefaultDataGridItemTemplate::OnContextChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
FOREACH(Ptr<IDataVisualizer>, visualizer, dataVisualizers)
|
|
{
|
|
visualizer->GetTemplate()->SetContext(GetContext());
|
|
}
|
|
if (currentEditor)
|
|
{
|
|
currentEditor->GetTemplate()->SetContext(GetContext());
|
|
}
|
|
}
|
|
|
|
DefaultDataGridItemTemplate::DefaultDataGridItemTemplate()
|
|
{
|
|
}
|
|
|
|
DefaultDataGridItemTemplate::~DefaultDataGridItemTemplate()
|
|
{
|
|
FOREACH(Ptr<IDataVisualizer>, visualizer, dataVisualizers)
|
|
{
|
|
visualizer->NotifyDeletedTemplate();
|
|
}
|
|
if (currentEditor)
|
|
{
|
|
currentEditor->NotifyDeletedTemplate();
|
|
}
|
|
}
|
|
|
|
void DefaultDataGridItemTemplate::UpdateSubItemSize()
|
|
{
|
|
if (auto dataGrid = dynamic_cast<GuiVirtualDataGrid*>(listControl))
|
|
{
|
|
vint columnCount = dataGrid->listViewItemView->GetColumnCount();
|
|
if (columnCount > textTable->GetColumns())
|
|
{
|
|
columnCount = textTable->GetColumns();
|
|
}
|
|
for (vint i = 0; i < columnCount; i++)
|
|
{
|
|
textTable->SetColumnOption(i, GuiCellOption::AbsoluteOption(dataGrid->columnItemView->GetColumnSize(i)));
|
|
}
|
|
textTable->UpdateCellBounds();
|
|
}
|
|
}
|
|
|
|
bool DefaultDataGridItemTemplate::IsEditorOpened()
|
|
{
|
|
return currentEditor != nullptr;
|
|
}
|
|
|
|
void DefaultDataGridItemTemplate::NotifyOpenEditor(vint column, IDataEditor* editor)
|
|
{
|
|
currentEditor = editor;
|
|
if (currentEditor)
|
|
{
|
|
auto cell = textTable->GetSitedCell(0, column);
|
|
auto* editorBounds = currentEditor->GetTemplate();
|
|
editorBounds->SetFont(GetFont());
|
|
editorBounds->SetContext(GetContext());
|
|
if (editorBounds->GetParent() && editorBounds->GetParent() != cell)
|
|
{
|
|
editorBounds->GetParent()->RemoveChild(editorBounds);
|
|
}
|
|
editorBounds->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
cell->AddChild(editorBounds);
|
|
if (auto focusControl = currentEditor->GetTemplate()->GetFocusControl())
|
|
{
|
|
focusControl->SetFocus();
|
|
}
|
|
}
|
|
}
|
|
|
|
void DefaultDataGridItemTemplate::NotifyCloseEditor()
|
|
{
|
|
if (currentEditor)
|
|
{
|
|
auto composition = currentEditor->GetTemplate();
|
|
if (composition->GetParent())
|
|
{
|
|
composition->GetParent()->RemoveChild(composition);
|
|
}
|
|
currentEditor = nullptr;
|
|
}
|
|
}
|
|
|
|
void DefaultDataGridItemTemplate::NotifySelectCell(vint column)
|
|
{
|
|
for (vint i = 0; i < dataVisualizers.Count(); i++)
|
|
{
|
|
dataVisualizers[i]->SetSelected(i == column);
|
|
}
|
|
}
|
|
|
|
void DefaultDataGridItemTemplate::NotifyCellEdited()
|
|
{
|
|
for (vint i = 0; i < dataVisualizers.Count(); i++)
|
|
{
|
|
dataVisualizers[i]->BeforeVisualizeCell(listControl->GetItemProvider(), GetIndex(), i);
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiVirtualDataGrid (Editor)
|
|
***********************************************************************/
|
|
|
|
using namespace list;
|
|
|
|
compositions::IGuiAltActionHost* GuiVirtualDataGrid::GetActivatingAltHost()
|
|
{
|
|
if (currentEditor)
|
|
{
|
|
if (auto focusControl = currentEditor->GetTemplate()->GetFocusControl())
|
|
{
|
|
if (auto action = focusControl->QueryTypedService<IGuiAltAction>())
|
|
{
|
|
if (action->IsAltAvailable() && action->IsAltEnabled())
|
|
{
|
|
SetAltComposition(currentEditor->GetTemplate());
|
|
SetAltControl(focusControl, true);
|
|
return this;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
SetAltComposition(nullptr);
|
|
SetAltControl(nullptr, false);
|
|
return GuiVirtualListView::GetActivatingAltHost();
|
|
}
|
|
|
|
void GuiVirtualDataGrid::OnItemModified(vint start, vint count, vint newCount)
|
|
{
|
|
GuiVirtualListView::OnItemModified(start, count, newCount);
|
|
if(!GetItemProvider()->IsEditing())
|
|
{
|
|
StopEdit();
|
|
}
|
|
}
|
|
|
|
void GuiVirtualDataGrid::OnStyleUninstalled(ItemStyle* style)
|
|
{
|
|
GuiVirtualListView::OnStyleUninstalled(style);
|
|
if (auto itemStyle = dynamic_cast<DefaultDataGridItemTemplate*>(style))
|
|
{
|
|
if (itemStyle->IsEditorOpened())
|
|
{
|
|
itemStyle->NotifyCloseEditor();
|
|
currentEditor = nullptr;
|
|
currentEditorPos = { -1,-1 };
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiVirtualDataGrid::NotifyCloseEditor()
|
|
{
|
|
if (currentEditorPos.row != -1 && GetArranger())
|
|
{
|
|
auto style = GetArranger()->GetVisibleStyle(currentEditorPos.row);
|
|
if (auto itemStyle = dynamic_cast<DefaultDataGridItemTemplate*>(style))
|
|
{
|
|
itemStyle->NotifyCloseEditor();
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiVirtualDataGrid::NotifySelectCell(vint row, vint column)
|
|
{
|
|
if (selectedCell.row != row || selectedCell.column != column)
|
|
{
|
|
selectedCell = { row, column };
|
|
SelectedCellChanged.Execute(GetNotifyEventArguments());
|
|
|
|
auto style = GetArranger()->GetVisibleStyle(row);
|
|
if (auto itemStyle = dynamic_cast<DefaultDataGridItemTemplate*>(style))
|
|
{
|
|
itemStyle->NotifySelectCell(column);
|
|
}
|
|
}
|
|
}
|
|
|
|
bool GuiVirtualDataGrid::StartEdit(vint row, vint column)
|
|
{
|
|
StopEdit();
|
|
NotifySelectCell(row, column);
|
|
|
|
auto style = GetArranger()->GetVisibleStyle(row);
|
|
if (auto itemStyle = dynamic_cast<DefaultDataGridItemTemplate*>(style))
|
|
{
|
|
if (auto factory = dataGridView->GetCellDataEditorFactory(row, column))
|
|
{
|
|
currentEditorOpeningEditor = true;
|
|
currentEditorPos = { row,column };
|
|
currentEditor = factory->CreateEditor(this);
|
|
if (auto focusControl = currentEditor->GetTemplate()->GetFocusControl())
|
|
{
|
|
focusControl->SetAlt(L"E");
|
|
}
|
|
currentEditor->BeforeEditCell(GetItemProvider(), row, column);
|
|
itemStyle->NotifyOpenEditor(column, currentEditor.Obj());
|
|
currentEditorOpeningEditor = false;
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void GuiVirtualDataGrid::StopEdit()
|
|
{
|
|
if (GetItemProvider()->IsEditing())
|
|
{
|
|
NotifyCloseEditor();
|
|
}
|
|
else
|
|
{
|
|
if (currentEditorPos != GridPos{-1, -1})
|
|
{
|
|
if (currentEditor)
|
|
{
|
|
NotifyCloseEditor();
|
|
}
|
|
}
|
|
}
|
|
SetAltComposition(nullptr);
|
|
SetAltControl(nullptr, false);
|
|
currentEditor = nullptr;
|
|
currentEditorPos = { -1,-1 };
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiVirtualDataGrid (IDataGridContext)
|
|
***********************************************************************/
|
|
|
|
templates::GuiListViewTemplate* GuiVirtualDataGrid::GetListViewControlTemplate()
|
|
{
|
|
return GetControlTemplateObject(true);
|
|
}
|
|
|
|
void GuiVirtualDataGrid::RequestSaveData()
|
|
{
|
|
if (currentEditor && !currentEditorOpeningEditor)
|
|
{
|
|
GuiControl* focusedControl = nullptr;
|
|
if (auto controlHost = GetRelatedControlHost())
|
|
{
|
|
if (auto graphicsHost = controlHost->GetGraphicsHost())
|
|
{
|
|
if (auto focusComposition = graphicsHost->GetFocusedComposition())
|
|
{
|
|
focusedControl = focusComposition->GetRelatedControl();
|
|
}
|
|
}
|
|
}
|
|
|
|
GetItemProvider()->PushEditing();
|
|
dataGridView->SetBindingCellValue(currentEditorPos.row, currentEditorPos.column, currentEditor->GetTemplate()->GetCellValue());
|
|
GetItemProvider()->PopEditing();
|
|
|
|
auto style = GetArranger()->GetVisibleStyle(currentEditorPos.row);
|
|
if (auto itemStyle = dynamic_cast<DefaultDataGridItemTemplate*>(style))
|
|
{
|
|
itemStyle->NotifyCellEdited();
|
|
}
|
|
|
|
if (currentEditor && focusedControl)
|
|
{
|
|
focusedControl->SetFocus();
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiVirtualDataGrid
|
|
***********************************************************************/
|
|
|
|
void GuiVirtualDataGrid::OnColumnClicked(compositions::GuiGraphicsComposition* sender, compositions::GuiItemEventArgs& arguments)
|
|
{
|
|
if(dataGridView->IsColumnSortable(arguments.itemIndex))
|
|
{
|
|
switch(columnItemView->GetSortingState(arguments.itemIndex))
|
|
{
|
|
case ColumnSortingState::NotSorted:
|
|
dataGridView->SortByColumn(arguments.itemIndex, true);
|
|
break;
|
|
case ColumnSortingState::Ascending:
|
|
dataGridView->SortByColumn(arguments.itemIndex, false);
|
|
break;
|
|
case ColumnSortingState::Descending:
|
|
dataGridView->SortByColumn(-1, false);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiVirtualDataGrid::OnSelectionChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
if (!skipOnSelectionChanged)
|
|
{
|
|
vint row = GetSelectedItemIndex();
|
|
if (row != -1)
|
|
{
|
|
if (selectedCell.row != row && selectedCell.column != -1)
|
|
{
|
|
SelectCell({ row,selectedCell.column }, false);
|
|
}
|
|
else
|
|
{
|
|
SelectCell({ row,0 }, false);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
StopEdit();
|
|
NotifySelectCell(-1, -1);
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiVirtualDataGrid::OnKeyDown(compositions::GuiGraphicsComposition* sender, compositions::GuiKeyEventArgs& arguments)
|
|
{
|
|
if (selectedCell.row != -1)
|
|
{
|
|
if (arguments.code == VKEY::_RETURN)
|
|
{
|
|
RequestSaveData();
|
|
SelectCell(selectedCell, !currentEditor);
|
|
arguments.handled = true;
|
|
if (!currentEditor)
|
|
{
|
|
SetFocus();
|
|
}
|
|
arguments.handled = true;
|
|
}
|
|
else if (arguments.code == VKEY::_ESCAPE)
|
|
{
|
|
if (currentEditor)
|
|
{
|
|
SelectCell(currentEditorPos, false);
|
|
SetFocus();
|
|
arguments.handled = true;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
vint columnOffset = 0;
|
|
switch (arguments.code)
|
|
{
|
|
case VKEY::_LEFT:
|
|
columnOffset = -1;
|
|
arguments.handled = true;
|
|
break;
|
|
case VKEY::_RIGHT:
|
|
columnOffset = 1;
|
|
arguments.handled = true;
|
|
break;
|
|
default:
|
|
return;
|
|
}
|
|
|
|
vint column = selectedCell.column + columnOffset;
|
|
if (column < 0)
|
|
{
|
|
column = 0;
|
|
}
|
|
else if (column >= listViewItemView->GetColumnCount())
|
|
{
|
|
column = listViewItemView->GetColumnCount();
|
|
}
|
|
SelectCell({ selectedCell.row, column }, false);
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiVirtualDataGrid::OnKeyUp(compositions::GuiGraphicsComposition* sender, compositions::GuiKeyEventArgs& arguments)
|
|
{
|
|
}
|
|
|
|
GuiVirtualDataGrid::GuiVirtualDataGrid(theme::ThemeName themeName, GuiListControl::IItemProvider* _itemProvider)
|
|
:GuiVirtualListView(themeName, _itemProvider)
|
|
{
|
|
listViewItemView = dynamic_cast<IListViewItemView*>(_itemProvider->RequestView(IListViewItemView::Identifier));
|
|
columnItemView = dynamic_cast<ListViewColumnItemArranger::IColumnItemView*>(_itemProvider->RequestView(ListViewColumnItemArranger::IColumnItemView::Identifier));
|
|
dataGridView = dynamic_cast<IDataGridView*>(_itemProvider->RequestView(IDataGridView::Identifier));
|
|
|
|
{
|
|
auto mainProperty = [](const Value&) { return new MainColumnVisualizerTemplate; };
|
|
auto subProperty = [](const Value&) { return new SubColumnVisualizerTemplate; };
|
|
auto focusRectangleProperty = [](const Value&) { return new FocusRectangleVisualizerTemplate; };
|
|
auto cellBorderProperty = [](const Value&) { return new CellBorderVisualizerTemplate; };
|
|
|
|
defaultMainColumnVisualizerFactory =
|
|
MakePtr<DataVisualizerFactory>(cellBorderProperty,
|
|
MakePtr<DataVisualizerFactory>(focusRectangleProperty,
|
|
MakePtr<DataVisualizerFactory>(mainProperty)
|
|
));
|
|
defaultSubColumnVisualizerFactory =
|
|
MakePtr<DataVisualizerFactory>(cellBorderProperty,
|
|
MakePtr<DataVisualizerFactory>(focusRectangleProperty,
|
|
MakePtr<DataVisualizerFactory>(subProperty)
|
|
));
|
|
}
|
|
|
|
CHECK_ERROR(listViewItemView != nullptr, L"GuiVirtualDataGrid::GuiVirtualDataGrid(IStyleController*, GuiListControl::IItemProvider*)#Missing IListViewItemView from item provider.");
|
|
CHECK_ERROR(columnItemView != nullptr, L"GuiVirtualDataGrid::GuiVirtualDataGrid(IStyleController*, GuiListControl::IItemProvider*)#Missing ListViewColumnItemArranger::IColumnItemView from item provider.");
|
|
CHECK_ERROR(dataGridView != nullptr, L"GuiVirtualDataGrid::GuiVirtualDataGrid(IStyleController*, GuiListControl::IItemProvider*)#Missing IDataGridView from item provider.");
|
|
|
|
SetViewToDefault();
|
|
|
|
ColumnClicked.AttachMethod(this, &GuiVirtualDataGrid::OnColumnClicked);
|
|
SelectionChanged.AttachMethod(this, &GuiVirtualDataGrid::OnSelectionChanged);
|
|
focusableComposition->GetEventReceiver()->keyDown.AttachMethod(this, &GuiVirtualDataGrid::OnKeyDown);
|
|
focusableComposition->GetEventReceiver()->keyUp.AttachMethod(this, &GuiVirtualDataGrid::OnKeyUp);
|
|
SelectedCellChanged.SetAssociatedComposition(boundsComposition);
|
|
}
|
|
|
|
GuiVirtualDataGrid::~GuiVirtualDataGrid()
|
|
{
|
|
}
|
|
|
|
GuiListControl::IItemProvider* GuiVirtualDataGrid::GetItemProvider()
|
|
{
|
|
return GuiVirtualListView::GetItemProvider();
|
|
}
|
|
|
|
void GuiVirtualDataGrid::SetViewToDefault()
|
|
{
|
|
SetStyleAndArranger(
|
|
[](const Value&) { return new list::DefaultDataGridItemTemplate; },
|
|
new list::ListViewColumnItemArranger
|
|
);
|
|
}
|
|
|
|
GridPos GuiVirtualDataGrid::GetSelectedCell()
|
|
{
|
|
return selectedCell;
|
|
}
|
|
|
|
bool GuiVirtualDataGrid::SelectCell(const GridPos& value, bool openEditor)
|
|
{
|
|
bool validPos = 0 <= value.row && value.row < GetItemProvider()->Count() && 0 <= value.column && value.column < listViewItemView->GetColumnCount();
|
|
|
|
if (validPos && selectedCell == value)
|
|
{
|
|
if (currentEditor && !openEditor)
|
|
{
|
|
StopEdit();
|
|
}
|
|
else if (!currentEditor && openEditor)
|
|
{
|
|
StartEdit(value.row, value.column);
|
|
}
|
|
return currentEditor != nullptr;
|
|
}
|
|
|
|
StopEdit();
|
|
if (validPos)
|
|
{
|
|
NotifySelectCell(value.row, value.column);
|
|
if (openEditor)
|
|
{
|
|
EnsureItemVisible(value.row);
|
|
if (GetMultiSelect())
|
|
{
|
|
ClearSelection();
|
|
}
|
|
|
|
skipOnSelectionChanged = true;
|
|
SetSelected(value.row, true);
|
|
skipOnSelectionChanged = false;
|
|
return StartEdit(value.row, value.column);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
NotifySelectCell(-1, -1);
|
|
ClearSelection();
|
|
}
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\CONTROLS\LISTCONTROLPACKAGE\GUIDATAGRIDEXTENSIONS.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace controls
|
|
{
|
|
namespace list
|
|
{
|
|
using namespace compositions;
|
|
using namespace elements;
|
|
using namespace theme;
|
|
using namespace templates;
|
|
|
|
/***********************************************************************
|
|
DataVisualizerBase
|
|
***********************************************************************/
|
|
|
|
DataVisualizerBase::DataVisualizerBase()
|
|
{
|
|
}
|
|
|
|
DataVisualizerBase::~DataVisualizerBase()
|
|
{
|
|
if (visualizerTemplate)
|
|
{
|
|
SafeDeleteComposition(visualizerTemplate);
|
|
}
|
|
}
|
|
|
|
IDataVisualizerFactory* DataVisualizerBase::GetFactory()
|
|
{
|
|
return factory;
|
|
}
|
|
|
|
templates::GuiGridVisualizerTemplate* DataVisualizerBase::GetTemplate()
|
|
{
|
|
return visualizerTemplate;
|
|
}
|
|
|
|
void DataVisualizerBase::NotifyDeletedTemplate()
|
|
{
|
|
visualizerTemplate = nullptr;
|
|
}
|
|
|
|
void DataVisualizerBase::BeforeVisualizeCell(GuiListControl::IItemProvider* itemProvider, vint row, vint column)
|
|
{
|
|
if (auto listViewItemView = dynamic_cast<IListViewItemView*>(dataGridContext->GetItemProvider()->RequestView(IListViewItemView::Identifier)))
|
|
{
|
|
auto style = dataGridContext->GetListViewControlTemplate();
|
|
visualizerTemplate->SetPrimaryTextColor(style->GetPrimaryTextColor());
|
|
visualizerTemplate->SetSecondaryTextColor(style->GetSecondaryTextColor());
|
|
visualizerTemplate->SetItemSeparatorColor(style->GetItemSeparatorColor());
|
|
|
|
visualizerTemplate->SetLargeImage(listViewItemView->GetLargeImage(row));
|
|
visualizerTemplate->SetSmallImage(listViewItemView->GetSmallImage(row));
|
|
visualizerTemplate->SetText(column == 0 ? listViewItemView->GetText(row) : listViewItemView->GetSubItem(row, column - 1));
|
|
}
|
|
if (auto dataGridView = dynamic_cast<IDataGridView*>(dataGridContext->GetItemProvider()->RequestView(IDataGridView::Identifier)))
|
|
{
|
|
visualizerTemplate->SetRowValue(itemProvider->GetBindingValue(row));
|
|
visualizerTemplate->SetCellValue(dataGridView->GetBindingCellValue(row, column));
|
|
}
|
|
}
|
|
|
|
void DataVisualizerBase::SetSelected(bool value)
|
|
{
|
|
if (visualizerTemplate)
|
|
{
|
|
visualizerTemplate->SetSelected(value);
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
DataVisualizerFactory
|
|
***********************************************************************/
|
|
|
|
DataVisualizerFactory::ItemTemplate* DataVisualizerFactory::CreateItemTemplate(controls::list::IDataGridContext* dataGridContext)
|
|
{
|
|
ItemTemplate* itemTemplate = templateFactory({});
|
|
CHECK_ERROR(itemTemplate, L"DataVisualizerFactory::CreateItemTemplate(IDataGridContext*)#An instance of GuiGridEditorTemplate is expected.");
|
|
if (decoratedFactory)
|
|
{
|
|
auto childTemplate = decoratedFactory->CreateItemTemplate(dataGridContext);
|
|
childTemplate->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
itemTemplate->GetContainerComposition()->AddChild(childTemplate);
|
|
|
|
#define FORWARD_EVENT(NAME)\
|
|
itemTemplate->NAME##Changed.AttachLambda([=](GuiGraphicsComposition* sender, GuiEventArgs& arguments)\
|
|
{\
|
|
childTemplate->Set##NAME(itemTemplate->Get##NAME());\
|
|
});\
|
|
|
|
#define FORWARD_EVENT_IMPL(CLASS, TYPE, NAME, VALUE) FORWARD_EVENT(NAME)
|
|
|
|
GuiTemplate_PROPERTIES(FORWARD_EVENT_IMPL)
|
|
GuiControlTemplate_PROPERTIES(FORWARD_EVENT_IMPL)
|
|
GuiGridCellTemplate_PROPERTIES(FORWARD_EVENT_IMPL)
|
|
GuiGridVisualizerTemplate_PROPERTIES(FORWARD_EVENT_IMPL)
|
|
|
|
#undef FORWARD_EVENT_IMPL
|
|
#undef FORWARD_EVENT
|
|
}
|
|
return itemTemplate;
|
|
}
|
|
|
|
DataVisualizerFactory::DataVisualizerFactory(TemplateProperty<ItemTemplate> _templateFactory, Ptr<DataVisualizerFactory> _decoratedFactory)
|
|
:templateFactory(_templateFactory)
|
|
, decoratedFactory(_decoratedFactory)
|
|
{
|
|
}
|
|
|
|
DataVisualizerFactory::~DataVisualizerFactory()
|
|
{
|
|
}
|
|
|
|
Ptr<controls::list::IDataVisualizer> DataVisualizerFactory::CreateVisualizer(controls::list::IDataGridContext* dataGridContext)
|
|
{
|
|
auto dataVisualizer = MakePtr<DataVisualizerBase>();
|
|
dataVisualizer->factory = this;
|
|
dataVisualizer->dataGridContext = dataGridContext;
|
|
dataVisualizer->visualizerTemplate = CreateItemTemplate(dataGridContext);
|
|
|
|
return dataVisualizer;
|
|
}
|
|
|
|
/***********************************************************************
|
|
DataEditorBase
|
|
***********************************************************************/
|
|
|
|
void DataEditorBase::OnCellValueChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
dataGridContext->RequestSaveData();
|
|
}
|
|
|
|
DataEditorBase::DataEditorBase()
|
|
{
|
|
}
|
|
|
|
DataEditorBase::~DataEditorBase()
|
|
{
|
|
if (editorTemplate)
|
|
{
|
|
SafeDeleteComposition(editorTemplate);
|
|
}
|
|
}
|
|
|
|
IDataEditorFactory* DataEditorBase::GetFactory()
|
|
{
|
|
return factory;
|
|
}
|
|
|
|
templates::GuiGridEditorTemplate* DataEditorBase::GetTemplate()
|
|
{
|
|
return editorTemplate;
|
|
}
|
|
|
|
void DataEditorBase::NotifyDeletedTemplate()
|
|
{
|
|
editorTemplate = nullptr;
|
|
}
|
|
|
|
void DataEditorBase::BeforeEditCell(GuiListControl::IItemProvider* itemProvider, vint row, vint column)
|
|
{
|
|
if (auto listViewItemView = dynamic_cast<IListViewItemView*>(dataGridContext->GetItemProvider()->RequestView(IListViewItemView::Identifier)))
|
|
{
|
|
auto style = dataGridContext->GetListViewControlTemplate();
|
|
editorTemplate->SetPrimaryTextColor(style->GetPrimaryTextColor());
|
|
editorTemplate->SetSecondaryTextColor(style->GetSecondaryTextColor());
|
|
editorTemplate->SetItemSeparatorColor(style->GetItemSeparatorColor());
|
|
|
|
editorTemplate->SetLargeImage(listViewItemView->GetLargeImage(row));
|
|
editorTemplate->SetSmallImage(listViewItemView->GetSmallImage(row));
|
|
editorTemplate->SetText(column == 0 ? listViewItemView->GetText(row) : listViewItemView->GetSubItem(row, column - 1));
|
|
}
|
|
if (auto dataGridView = dynamic_cast<IDataGridView*>(dataGridContext->GetItemProvider()->RequestView(IDataGridView::Identifier)))
|
|
{
|
|
editorTemplate->SetRowValue(itemProvider->GetBindingValue(row));
|
|
editorTemplate->SetCellValue(dataGridView->GetBindingCellValue(row, column));
|
|
}
|
|
editorTemplate->CellValueChanged.AttachMethod(this, &DataEditorBase::OnCellValueChanged);
|
|
}
|
|
|
|
bool DataEditorBase::GetCellValueSaved()
|
|
{
|
|
if (editorTemplate)
|
|
{
|
|
return editorTemplate->GetCellValueSaved();
|
|
}
|
|
return true;
|
|
}
|
|
|
|
/***********************************************************************
|
|
DataEditorFactory
|
|
***********************************************************************/
|
|
|
|
DataEditorFactory::DataEditorFactory(TemplateProperty<GuiGridEditorTemplate> _templateFactory)
|
|
:templateFactory(_templateFactory)
|
|
{
|
|
}
|
|
|
|
DataEditorFactory::~DataEditorFactory()
|
|
{
|
|
}
|
|
|
|
Ptr<IDataEditor> DataEditorFactory::CreateEditor(controls::list::IDataGridContext* dataGridContext)
|
|
{
|
|
auto editor = MakePtr<DataEditorBase>();
|
|
editor->factory = this;
|
|
editor->dataGridContext = dataGridContext;
|
|
|
|
ItemTemplate* itemTemplate = templateFactory({});
|
|
CHECK_ERROR(itemTemplate, L"DataEditorFactory::CreateEditor(IDataGridContext*)#An instance of GuiGridEditorTemplate is expected.");
|
|
editor->editorTemplate = itemTemplate;
|
|
return editor;
|
|
}
|
|
|
|
/***********************************************************************
|
|
MainColumnVisualizerTemplate
|
|
***********************************************************************/
|
|
|
|
void MainColumnVisualizerTemplate::OnTextChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
text->SetText(GetText());
|
|
}
|
|
|
|
void MainColumnVisualizerTemplate::OnFontChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
text->SetFont(GetFont());
|
|
}
|
|
|
|
void MainColumnVisualizerTemplate::OnTextColorChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
text->SetColor(GetPrimaryTextColor());
|
|
}
|
|
|
|
void MainColumnVisualizerTemplate::OnSmallImageChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
auto imageData = GetSmallImage();
|
|
if (imageData)
|
|
{
|
|
image->SetImage(imageData->GetImage(), imageData->GetFrameIndex());
|
|
}
|
|
else
|
|
{
|
|
image->SetImage(nullptr);
|
|
}
|
|
}
|
|
|
|
MainColumnVisualizerTemplate::MainColumnVisualizerTemplate()
|
|
{
|
|
GuiTableComposition* table = new GuiTableComposition;
|
|
table->SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
table->SetRowsAndColumns(3, 2);
|
|
table->SetRowOption(0, GuiCellOption::PercentageOption(0.5));
|
|
table->SetRowOption(1, GuiCellOption::MinSizeOption());
|
|
table->SetRowOption(2, GuiCellOption::PercentageOption(0.5));
|
|
table->SetColumnOption(0, GuiCellOption::MinSizeOption());
|
|
table->SetColumnOption(1, GuiCellOption::PercentageOption(1.0));
|
|
table->SetCellPadding(2);
|
|
{
|
|
GuiCellComposition* cell = new GuiCellComposition;
|
|
table->AddChild(cell);
|
|
cell->SetSite(1, 0, 1, 1);
|
|
cell->SetPreferredMinSize(Size(16, 16));
|
|
|
|
image = GuiImageFrameElement::Create();
|
|
image->SetStretch(true);
|
|
cell->SetOwnedElement(image);
|
|
}
|
|
{
|
|
GuiCellComposition* cell = new GuiCellComposition;
|
|
table->AddChild(cell);
|
|
cell->SetSite(0, 1, 3, 1);
|
|
cell->SetMargin(Margin(0, 0, 8, 0));
|
|
|
|
text = GuiSolidLabelElement::Create();
|
|
text->SetAlignments(Alignment::Left, Alignment::Center);
|
|
text->SetEllipse(true);
|
|
cell->SetOwnedElement(text);
|
|
}
|
|
table->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
|
|
SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
AddChild(table);
|
|
|
|
TextChanged.AttachMethod(this, &MainColumnVisualizerTemplate::OnTextChanged);
|
|
FontChanged.AttachMethod(this, &MainColumnVisualizerTemplate::OnFontChanged);
|
|
PrimaryTextColorChanged.AttachMethod(this, &MainColumnVisualizerTemplate::OnTextColorChanged);
|
|
SmallImageChanged.AttachMethod(this, &MainColumnVisualizerTemplate::OnSmallImageChanged);
|
|
|
|
TextChanged.Execute(compositions::GuiEventArgs(this));
|
|
FontChanged.Execute(compositions::GuiEventArgs(this));
|
|
PrimaryTextColorChanged.Execute(compositions::GuiEventArgs(this));
|
|
SmallImageChanged.Execute(compositions::GuiEventArgs(this));
|
|
}
|
|
|
|
MainColumnVisualizerTemplate::~MainColumnVisualizerTemplate()
|
|
{
|
|
}
|
|
|
|
/***********************************************************************
|
|
SubColumnVisualizerTemplate
|
|
***********************************************************************/
|
|
|
|
void SubColumnVisualizerTemplate::OnTextChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
text->SetText(GetText());
|
|
}
|
|
|
|
void SubColumnVisualizerTemplate::OnFontChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
text->SetFont(GetFont());
|
|
}
|
|
|
|
void SubColumnVisualizerTemplate::OnTextColorChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
text->SetColor(GetSecondaryTextColor());
|
|
}
|
|
|
|
void SubColumnVisualizerTemplate::Initialize(bool fixTextColor)
|
|
{
|
|
text = GuiSolidLabelElement::Create();
|
|
text->SetVerticalAlignment(Alignment::Center);
|
|
|
|
SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
SetMargin(Margin(8, 0, 8, 0));
|
|
SetOwnedElement(text);
|
|
|
|
TextChanged.AttachMethod(this, &SubColumnVisualizerTemplate::OnTextChanged);
|
|
FontChanged.AttachMethod(this, &SubColumnVisualizerTemplate::OnFontChanged);
|
|
if (!fixTextColor)
|
|
{
|
|
SecondaryTextColorChanged.AttachMethod(this, &SubColumnVisualizerTemplate::OnTextColorChanged);
|
|
}
|
|
|
|
TextChanged.Execute(compositions::GuiEventArgs(this));
|
|
FontChanged.Execute(compositions::GuiEventArgs(this));
|
|
if (!fixTextColor)
|
|
{
|
|
SecondaryTextColorChanged.Execute(compositions::GuiEventArgs(this));
|
|
}
|
|
}
|
|
|
|
SubColumnVisualizerTemplate::SubColumnVisualizerTemplate(bool fixTextColor)
|
|
{
|
|
Initialize(fixTextColor);
|
|
}
|
|
|
|
SubColumnVisualizerTemplate::SubColumnVisualizerTemplate()
|
|
{
|
|
Initialize(false);
|
|
}
|
|
|
|
SubColumnVisualizerTemplate::~SubColumnVisualizerTemplate()
|
|
{
|
|
}
|
|
|
|
/***********************************************************************
|
|
HyperlinkVisualizerTemplate
|
|
***********************************************************************/
|
|
|
|
void HyperlinkVisualizerTemplate::label_MouseEnter(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
FontProperties font = text->GetFont();
|
|
font.underline = true;
|
|
text->SetFont(font);
|
|
}
|
|
|
|
void HyperlinkVisualizerTemplate::label_MouseLeave(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
FontProperties font = text->GetFont();
|
|
font.underline = false;
|
|
text->SetFont(font);
|
|
}
|
|
|
|
HyperlinkVisualizerTemplate::HyperlinkVisualizerTemplate()
|
|
:SubColumnVisualizerTemplate(true)
|
|
{
|
|
text->SetColor(Color(0, 0, 255));
|
|
text->SetEllipse(true);
|
|
GetEventReceiver()->mouseEnter.AttachMethod(this, &HyperlinkVisualizerTemplate::label_MouseEnter);
|
|
GetEventReceiver()->mouseLeave.AttachMethod(this, &HyperlinkVisualizerTemplate::label_MouseLeave);
|
|
SetAssociatedCursor(GetCurrentController()->ResourceService()->GetSystemCursor(INativeCursor::Hand));
|
|
}
|
|
|
|
HyperlinkVisualizerTemplate::~HyperlinkVisualizerTemplate()
|
|
{
|
|
}
|
|
|
|
/***********************************************************************
|
|
CellBorderVisualizerTemplate
|
|
***********************************************************************/
|
|
|
|
void FocusRectangleVisualizerTemplate::OnSelectedChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
focusComposition->SetVisible(GetSelected());
|
|
}
|
|
|
|
FocusRectangleVisualizerTemplate::FocusRectangleVisualizerTemplate()
|
|
{
|
|
SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
|
|
focusComposition = new GuiBoundsComposition();
|
|
{
|
|
auto focus = GuiFocusRectangleElement::Create();
|
|
focusComposition->SetOwnedElement(focus);
|
|
focusComposition->SetAlignmentToParent(Margin(1, 1, 1, 1));
|
|
}
|
|
auto container = new GuiBoundsComposition();
|
|
{
|
|
container->SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
container->SetAlignmentToParent(Margin(2, 2, 2, 2));
|
|
}
|
|
|
|
AddChild(focusComposition);
|
|
AddChild(container);
|
|
SetContainerComposition(container);
|
|
|
|
SelectedChanged.AttachMethod(this, &FocusRectangleVisualizerTemplate::OnSelectedChanged);
|
|
SelectedChanged.Execute(compositions::GuiEventArgs(this));
|
|
}
|
|
|
|
FocusRectangleVisualizerTemplate::~FocusRectangleVisualizerTemplate()
|
|
{
|
|
}
|
|
|
|
/***********************************************************************
|
|
CellBorderVisualizerTemplate
|
|
***********************************************************************/
|
|
|
|
void CellBorderVisualizerTemplate::OnItemSeparatorColorChanged(GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
border1->SetColor(GetItemSeparatorColor());
|
|
border2->SetColor(GetItemSeparatorColor());
|
|
}
|
|
|
|
CellBorderVisualizerTemplate::CellBorderVisualizerTemplate()
|
|
{
|
|
SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
|
|
auto bounds1 = new GuiBoundsComposition;
|
|
{
|
|
border1 = GuiSolidBorderElement::Create();
|
|
bounds1->SetOwnedElement(border1);
|
|
bounds1->SetAlignmentToParent(Margin(-1, 0, 0, 0));
|
|
}
|
|
auto bounds2 = new GuiBoundsComposition;
|
|
{
|
|
border2 = GuiSolidBorderElement::Create();
|
|
bounds2->SetOwnedElement(border2);
|
|
bounds2->SetAlignmentToParent(Margin(0, -1, 0, 0));
|
|
}
|
|
auto container = new GuiBoundsComposition();
|
|
{
|
|
container->SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
container->SetAlignmentToParent(Margin(0, 0, 1, 1));
|
|
}
|
|
|
|
AddChild(bounds1);
|
|
AddChild(bounds2);
|
|
AddChild(container);
|
|
SetContainerComposition(container);
|
|
|
|
ItemSeparatorColorChanged.AttachMethod(this, &CellBorderVisualizerTemplate::OnItemSeparatorColorChanged);
|
|
ItemSeparatorColorChanged.Execute(compositions::GuiEventArgs(this));
|
|
}
|
|
|
|
CellBorderVisualizerTemplate::~CellBorderVisualizerTemplate()
|
|
{
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\CONTROLS\LISTCONTROLPACKAGE\GUILISTCONTROLITEMARRANGERS.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace controls
|
|
{
|
|
using namespace collections;
|
|
using namespace elements;
|
|
using namespace compositions;
|
|
|
|
namespace list
|
|
{
|
|
|
|
/***********************************************************************
|
|
RangedItemArrangerBase
|
|
***********************************************************************/
|
|
|
|
void RangedItemArrangerBase::InvalidateAdoptedSize()
|
|
{
|
|
if (listControl)
|
|
{
|
|
listControl->AdoptedSizeInvalidated.Execute(listControl->GetNotifyEventArguments());
|
|
}
|
|
}
|
|
|
|
vint RangedItemArrangerBase::CalculateAdoptedSize(vint expectedSize, vint count, vint itemSize)
|
|
{
|
|
vint visibleCount = expectedSize / itemSize;
|
|
if (count < visibleCount)
|
|
{
|
|
visibleCount = count;
|
|
}
|
|
else if (count > visibleCount)
|
|
{
|
|
vint deltaA = expectedSize - count * itemSize;
|
|
vint deltaB = itemSize - deltaA;
|
|
if (deltaB < deltaA)
|
|
{
|
|
visibleCount++;
|
|
}
|
|
}
|
|
return visibleCount * itemSize;
|
|
}
|
|
|
|
RangedItemArrangerBase::ItemStyleRecord RangedItemArrangerBase::CreateStyle(vint index)
|
|
{
|
|
GuiSelectableButton* backgroundButton = nullptr;
|
|
if (listControl->GetDisplayItemBackground())
|
|
{
|
|
backgroundButton = new GuiSelectableButton(theme::ThemeName::ListItemBackground);
|
|
if (auto style = listControl->GetControlTemplateObject(true)->GetBackgroundTemplate())
|
|
{
|
|
backgroundButton->SetControlTemplate(style);
|
|
}
|
|
backgroundButton->SetAutoFocus(false);
|
|
backgroundButton->SetAutoSelection(false);
|
|
}
|
|
|
|
auto itemStyle = callback->RequestItem(index, backgroundButton->GetBoundsComposition());
|
|
if (backgroundButton)
|
|
{
|
|
itemStyle->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
itemStyle->SelectedChanged.AttachLambda([=](GuiGraphicsComposition* sender, GuiEventArgs& arguments)
|
|
{
|
|
backgroundButton->SetSelected(itemStyle->GetSelected());
|
|
});
|
|
|
|
backgroundButton->SetSelected(itemStyle->GetSelected());
|
|
backgroundButton->GetContainerComposition()->AddChild(itemStyle);
|
|
}
|
|
return { itemStyle, backgroundButton };
|
|
}
|
|
|
|
void RangedItemArrangerBase::DeleteStyle(ItemStyleRecord style)
|
|
{
|
|
callback->ReleaseItem(style.key);
|
|
if (style.value)
|
|
{
|
|
SafeDeleteControl(style.value);
|
|
}
|
|
}
|
|
|
|
compositions::GuiBoundsComposition* RangedItemArrangerBase::GetStyleBounds(ItemStyleRecord style)
|
|
{
|
|
return style.value ? style.value->GetBoundsComposition() : style.key;
|
|
}
|
|
|
|
void RangedItemArrangerBase::ClearStyles()
|
|
{
|
|
startIndex = 0;
|
|
if (callback)
|
|
{
|
|
for (vint i = 0; i < visibleStyles.Count(); i++)
|
|
{
|
|
DeleteStyle(visibleStyles[i]);
|
|
}
|
|
}
|
|
visibleStyles.Clear();
|
|
viewBounds = Rect(0, 0, 0, 0);
|
|
InvalidateItemSizeCache();
|
|
InvalidateAdoptedSize();
|
|
}
|
|
|
|
void RangedItemArrangerBase::OnViewChangedInternal(Rect oldBounds, Rect newBounds)
|
|
{
|
|
vint endIndex = startIndex + visibleStyles.Count() - 1;
|
|
vint newStartIndex = 0;
|
|
vint itemCount = itemProvider->Count();
|
|
BeginPlaceItem(true, newBounds, newStartIndex);
|
|
if (newStartIndex < 0) newStartIndex = 0;
|
|
|
|
StyleList newVisibleStyles;
|
|
for (vint i = newStartIndex; i < itemCount; i++)
|
|
{
|
|
bool reuseOldStyle = startIndex <= i && i <= endIndex;
|
|
auto style = reuseOldStyle ? visibleStyles[i - startIndex] : CreateStyle(i);
|
|
newVisibleStyles.Add(style);
|
|
|
|
Rect bounds;
|
|
Margin alignmentToParent;
|
|
PlaceItem(true, !reuseOldStyle, i, style, newBounds, bounds, alignmentToParent);
|
|
if (IsItemOutOfViewBounds(i, style, bounds, newBounds))
|
|
{
|
|
break;
|
|
}
|
|
|
|
bounds.x1 -= newBounds.x1;
|
|
bounds.x2 -= newBounds.x1;
|
|
bounds.y1 -= newBounds.y1;
|
|
bounds.y2 -= newBounds.y1;
|
|
}
|
|
|
|
vint newEndIndex = newStartIndex + newVisibleStyles.Count() - 1;
|
|
for (vint i = 0; i < visibleStyles.Count(); i++)
|
|
{
|
|
vint index = startIndex + i;
|
|
if (index < newStartIndex || index > newEndIndex)
|
|
{
|
|
DeleteStyle(visibleStyles[i]);
|
|
}
|
|
}
|
|
CopyFrom(visibleStyles, newVisibleStyles);
|
|
|
|
if (EndPlaceItem(true, newBounds, newStartIndex))
|
|
{
|
|
callback->OnTotalSizeChanged();
|
|
InvalidateAdoptedSize();
|
|
}
|
|
startIndex = newStartIndex;
|
|
}
|
|
|
|
void RangedItemArrangerBase::RearrangeItemBounds()
|
|
{
|
|
vint newStartIndex = startIndex;
|
|
BeginPlaceItem(false, viewBounds, newStartIndex);
|
|
for (vint i = 0; i < visibleStyles.Count(); i++)
|
|
{
|
|
auto style = visibleStyles[i];
|
|
Rect bounds;
|
|
Margin alignmentToParent(-1, -1, -1, -1);
|
|
PlaceItem(false, false, startIndex + i, style, viewBounds, bounds, alignmentToParent);
|
|
|
|
bounds.x1 -= viewBounds.x1;
|
|
bounds.x2 -= viewBounds.x1;
|
|
bounds.y1 -= viewBounds.y1;
|
|
bounds.y2 -= viewBounds.y1;
|
|
|
|
callback->SetStyleAlignmentToParent(GetStyleBounds(style), alignmentToParent);
|
|
callback->SetStyleBounds(GetStyleBounds(style), bounds);
|
|
}
|
|
EndPlaceItem(false, viewBounds, startIndex);
|
|
}
|
|
|
|
RangedItemArrangerBase::RangedItemArrangerBase()
|
|
{
|
|
}
|
|
|
|
RangedItemArrangerBase::~RangedItemArrangerBase()
|
|
{
|
|
}
|
|
|
|
void RangedItemArrangerBase::OnAttached(GuiListControl::IItemProvider* provider)
|
|
{
|
|
itemProvider = provider;
|
|
if (provider)
|
|
{
|
|
OnItemModified(0, 0, provider->Count());
|
|
}
|
|
}
|
|
|
|
void RangedItemArrangerBase::OnItemModified(vint start, vint count, vint newCount)
|
|
{
|
|
if (callback && !itemProvider->IsEditing())
|
|
{
|
|
suppressOnViewChanged = true;
|
|
{
|
|
vint visibleCount = visibleStyles.Count();
|
|
vint itemCount = itemProvider->Count();
|
|
SortedList<ItemStyleRecord> reusedStyles;
|
|
for (vint i = 0; i < visibleCount; i++)
|
|
{
|
|
vint index = startIndex + i;
|
|
if (index >= itemCount)
|
|
{
|
|
break;
|
|
}
|
|
|
|
vint oldIndex = -1;
|
|
if (index < start)
|
|
{
|
|
oldIndex = index;
|
|
}
|
|
else if (index >= start + newCount)
|
|
{
|
|
oldIndex = index - newCount + count;
|
|
}
|
|
|
|
if (oldIndex != -1)
|
|
{
|
|
if (oldIndex >= startIndex && oldIndex < startIndex + visibleCount)
|
|
{
|
|
auto style = visibleStyles[oldIndex - startIndex];
|
|
reusedStyles.Add(style);
|
|
visibleStyles.Add(style);
|
|
}
|
|
else
|
|
{
|
|
oldIndex = -1;
|
|
}
|
|
}
|
|
if (oldIndex == -1)
|
|
{
|
|
visibleStyles.Add(CreateStyle(index));
|
|
}
|
|
}
|
|
|
|
for (vint i = 0; i < visibleCount; i++)
|
|
{
|
|
auto style = visibleStyles[i];
|
|
if (!reusedStyles.Contains(style))
|
|
{
|
|
DeleteStyle(style);
|
|
}
|
|
}
|
|
|
|
visibleStyles.RemoveRange(0, visibleCount);
|
|
for (vint i = 0; i < visibleStyles.Count(); i++)
|
|
{
|
|
visibleStyles[i].key->SetIndex(startIndex + i);
|
|
}
|
|
}
|
|
suppressOnViewChanged = false;
|
|
|
|
callback->OnTotalSizeChanged();
|
|
callback->SetViewLocation(viewBounds.LeftTop());
|
|
InvalidateAdoptedSize();
|
|
}
|
|
}
|
|
|
|
void RangedItemArrangerBase::AttachListControl(GuiListControl* value)
|
|
{
|
|
listControl = value;
|
|
InvalidateAdoptedSize();
|
|
}
|
|
|
|
void RangedItemArrangerBase::DetachListControl()
|
|
{
|
|
listControl = 0;
|
|
}
|
|
|
|
GuiListControl::IItemArrangerCallback* RangedItemArrangerBase::GetCallback()
|
|
{
|
|
return callback;
|
|
}
|
|
|
|
void RangedItemArrangerBase::SetCallback(GuiListControl::IItemArrangerCallback* value)
|
|
{
|
|
if (callback != value)
|
|
{
|
|
ClearStyles();
|
|
callback = value;
|
|
}
|
|
}
|
|
|
|
Size RangedItemArrangerBase::GetTotalSize()
|
|
{
|
|
if (callback)
|
|
{
|
|
return OnCalculateTotalSize();
|
|
}
|
|
else
|
|
{
|
|
return Size(0, 0);
|
|
}
|
|
}
|
|
|
|
GuiListControl::ItemStyle* RangedItemArrangerBase::GetVisibleStyle(vint itemIndex)
|
|
{
|
|
if (startIndex <= itemIndex && itemIndex < startIndex + visibleStyles.Count())
|
|
{
|
|
return visibleStyles[itemIndex - startIndex].key;
|
|
}
|
|
else
|
|
{
|
|
return nullptr;
|
|
}
|
|
}
|
|
|
|
vint RangedItemArrangerBase::GetVisibleIndex(GuiListControl::ItemStyle* style)
|
|
{
|
|
for (vint i = 0; i < visibleStyles.Count(); i++)
|
|
{
|
|
if (visibleStyles[i].key == style)
|
|
{
|
|
return i + startIndex;
|
|
}
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
void RangedItemArrangerBase::ReloadVisibleStyles()
|
|
{
|
|
ClearStyles();
|
|
}
|
|
|
|
void RangedItemArrangerBase::OnViewChanged(Rect bounds)
|
|
{
|
|
if (!suppressOnViewChanged)
|
|
{
|
|
suppressOnViewChanged = true;
|
|
Rect oldBounds = viewBounds;
|
|
viewBounds = bounds;
|
|
if (callback)
|
|
{
|
|
OnViewChangedInternal(oldBounds, viewBounds);
|
|
RearrangeItemBounds();
|
|
}
|
|
suppressOnViewChanged = false;
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
FreeHeightItemArranger
|
|
***********************************************************************/
|
|
|
|
void FreeHeightItemArranger::EnsureOffsetForItem(vint itemIndex)
|
|
{
|
|
if (heights.Count() == 0) return;
|
|
|
|
if (availableOffsetCount == 0)
|
|
{
|
|
availableOffsetCount = 1;
|
|
offsets[0] = 0;
|
|
}
|
|
|
|
for (vint i = availableOffsetCount; i < itemIndex && i < heights.Count(); i++)
|
|
{
|
|
offsets[i] = offsets[i - 1] + heights[i - 1];
|
|
}
|
|
}
|
|
|
|
void FreeHeightItemArranger::BeginPlaceItem(bool forMoving, Rect newBounds, vint& newStartIndex)
|
|
{
|
|
pim_heightUpdated = false;
|
|
EnsureOffsetForItem(heights.Count() - 1);
|
|
if (forMoving)
|
|
{
|
|
for (vint i = 0; i < heights.Count(); i++)
|
|
{
|
|
if (offsets[i] + heights[i] >= newBounds.Top())
|
|
{
|
|
newStartIndex = i;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void FreeHeightItemArranger::PlaceItem(bool forMoving, bool newCreatedStyle, vint index, ItemStyleRecord style, Rect viewBounds, Rect& bounds, Margin& alignmentToParent)
|
|
{
|
|
vint styleHeight = heights[index];
|
|
{
|
|
auto composition = GetStyleBounds(style);
|
|
auto currentBounds = callback->GetStyleBounds(composition);
|
|
callback->SetStyleBounds(composition, Rect(bounds.LeftTop(), Size(viewBounds.Width(), bounds.Height())));
|
|
vint newStyleHeight = callback->GetStylePreferredSize(composition).y;
|
|
callback->SetStyleBounds(composition, currentBounds);
|
|
|
|
if (!newCreatedStyle || styleHeight < newStyleHeight)
|
|
{
|
|
styleHeight = newStyleHeight;
|
|
}
|
|
}
|
|
|
|
if (heights[index] != styleHeight)
|
|
{
|
|
heights[index] = styleHeight;
|
|
pim_heightUpdated = true;
|
|
}
|
|
|
|
vint styleOffset = index == 0 ? 0 : offsets[index - 1] + heights[index - 1];
|
|
if (availableOffsetCount <= index || offsets[index] != styleOffset)
|
|
{
|
|
offsets[index] = styleOffset;
|
|
availableOffsetCount = index;
|
|
}
|
|
|
|
bounds = Rect(Point(0, offsets[index]), Size(viewBounds.Width(), heights[index]));
|
|
}
|
|
|
|
bool FreeHeightItemArranger::IsItemOutOfViewBounds(vint index, ItemStyleRecord style, Rect bounds, Rect viewBounds)
|
|
{
|
|
return bounds.Top() >= viewBounds.Bottom();
|
|
}
|
|
|
|
bool FreeHeightItemArranger::EndPlaceItem(bool forMoving, Rect newBounds, vint newStartIndex)
|
|
{
|
|
if (forMoving)
|
|
{
|
|
return pim_heightUpdated;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void FreeHeightItemArranger::InvalidateItemSizeCache()
|
|
{
|
|
availableOffsetCount = 0;
|
|
for (vint i = 0; i < heights.Count(); i++)
|
|
{
|
|
heights[i] = 1;
|
|
}
|
|
}
|
|
|
|
Size FreeHeightItemArranger::OnCalculateTotalSize()
|
|
{
|
|
if (heights.Count() == 0) return Size(0, 0);
|
|
EnsureOffsetForItem(heights.Count());
|
|
return Size(viewBounds.Width(), offsets[heights.Count() - 1] + heights[heights.Count() - 1]);
|
|
}
|
|
|
|
FreeHeightItemArranger::FreeHeightItemArranger()
|
|
{
|
|
}
|
|
|
|
FreeHeightItemArranger::~FreeHeightItemArranger()
|
|
{
|
|
}
|
|
|
|
void FreeHeightItemArranger::OnAttached(GuiListControl::IItemProvider* provider)
|
|
{
|
|
if (provider)
|
|
{
|
|
vint itemCount = provider->Count();
|
|
heights.Resize(itemCount);
|
|
offsets.Resize(itemCount);
|
|
for (vint i = 0; i < heights.Count(); i++)
|
|
{
|
|
heights[i] = 1;
|
|
}
|
|
availableOffsetCount = 0;
|
|
}
|
|
else
|
|
{
|
|
heights.Resize(0);
|
|
offsets.Resize(0);
|
|
availableOffsetCount = 0;
|
|
}
|
|
RangedItemArrangerBase::OnAttached(provider);
|
|
}
|
|
|
|
void FreeHeightItemArranger::OnItemModified(vint start, vint count, vint newCount)
|
|
{
|
|
availableOffsetCount = start;
|
|
vint itemCount = heights.Count() + newCount - count;
|
|
|
|
if (count < newCount)
|
|
{
|
|
heights.Resize(itemCount);
|
|
if (start + newCount < itemCount)
|
|
{
|
|
memmove(&heights[start + newCount], &heights[start + count], sizeof(vint) * (itemCount - start - newCount));
|
|
}
|
|
}
|
|
else if (count > newCount)
|
|
{
|
|
if (start + newCount < itemCount)
|
|
{
|
|
memmove(&heights[start + newCount], &heights[start + count], sizeof(vint) * (itemCount - start - newCount));
|
|
}
|
|
heights.Resize(itemCount);
|
|
}
|
|
|
|
for (vint i = 0; i < newCount; i++)
|
|
{
|
|
heights[start + i] = 1;
|
|
}
|
|
offsets.Resize(itemCount);
|
|
|
|
RangedItemArrangerBase::OnItemModified(start, count, newCount);
|
|
}
|
|
|
|
vint FreeHeightItemArranger::FindItem(vint itemIndex, compositions::KeyDirection key)
|
|
{
|
|
vint count = itemProvider->Count();
|
|
if (count == 0) return -1;
|
|
switch (key)
|
|
{
|
|
case KeyDirection::Up:
|
|
itemIndex--;
|
|
break;
|
|
case KeyDirection::Down:
|
|
itemIndex++;
|
|
break;
|
|
case KeyDirection::Home:
|
|
itemIndex = 0;
|
|
break;
|
|
case KeyDirection::End:
|
|
itemIndex = count;
|
|
break;
|
|
case KeyDirection::PageUp:
|
|
itemIndex -= visibleStyles.Count();
|
|
break;
|
|
case KeyDirection::PageDown:
|
|
itemIndex += visibleStyles.Count();
|
|
break;
|
|
default:
|
|
return -1;
|
|
}
|
|
|
|
if (itemIndex < 0) return 0;
|
|
else if (itemIndex >= count) return count - 1;
|
|
else return itemIndex;
|
|
}
|
|
|
|
GuiListControl::EnsureItemVisibleResult FreeHeightItemArranger::EnsureItemVisible(vint itemIndex)
|
|
{
|
|
if (callback)
|
|
{
|
|
bool moved = false;
|
|
while (true)
|
|
{
|
|
if (itemIndex < 0 || itemIndex >= itemProvider->Count())
|
|
{
|
|
return GuiListControl::EnsureItemVisibleResult::ItemNotExists;
|
|
}
|
|
|
|
EnsureOffsetForItem(itemIndex);
|
|
vint offset = viewBounds.y1;
|
|
vint top = offsets[itemIndex];
|
|
vint bottom = top + heights[itemIndex];
|
|
vint height = viewBounds.Height();
|
|
|
|
Point location = viewBounds.LeftTop();
|
|
if (offset > top)
|
|
{
|
|
location.y = top;
|
|
}
|
|
else if (offset < bottom - height)
|
|
{
|
|
location.y = bottom - height;
|
|
}
|
|
else
|
|
{
|
|
break;
|
|
}
|
|
|
|
auto oldLeftTop = viewBounds.LeftTop();
|
|
callback->SetViewLocation(location);
|
|
moved |= viewBounds.LeftTop() != oldLeftTop;
|
|
if (viewBounds.LeftTop() != location) break;
|
|
}
|
|
return moved ? GuiListControl::EnsureItemVisibleResult::Moved : GuiListControl::EnsureItemVisibleResult::NotMoved;
|
|
}
|
|
return GuiListControl::EnsureItemVisibleResult::NotMoved;
|
|
}
|
|
|
|
Size FreeHeightItemArranger::GetAdoptedSize(Size expectedSize)
|
|
{
|
|
vint h = expectedSize.x * 2;
|
|
if (expectedSize.y < h) expectedSize.y = h;
|
|
return expectedSize;
|
|
}
|
|
|
|
/***********************************************************************
|
|
FixedHeightItemArranger
|
|
***********************************************************************/
|
|
|
|
vint FixedHeightItemArranger::GetWidth()
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
vint FixedHeightItemArranger::GetYOffset()
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
void FixedHeightItemArranger::BeginPlaceItem(bool forMoving, Rect newBounds, vint& newStartIndex)
|
|
{
|
|
pi_width = GetWidth();
|
|
if (forMoving)
|
|
{
|
|
pim_rowHeight = rowHeight;
|
|
newStartIndex = (newBounds.Top() - GetYOffset()) / rowHeight;
|
|
}
|
|
}
|
|
|
|
void FixedHeightItemArranger::PlaceItem(bool forMoving, bool newCreatedStyle, vint index, ItemStyleRecord style, Rect viewBounds, Rect& bounds, Margin& alignmentToParent)
|
|
{
|
|
vint top = GetYOffset() + index * rowHeight;
|
|
if (pi_width == -1)
|
|
{
|
|
alignmentToParent = Margin(0, -1, 0, -1);
|
|
bounds = Rect(Point(0, top), Size(0, rowHeight));
|
|
}
|
|
else
|
|
{
|
|
alignmentToParent = Margin(-1, -1, -1, -1);
|
|
bounds = Rect(Point(0, top), Size(pi_width, rowHeight));
|
|
}
|
|
if (forMoving)
|
|
{
|
|
vint styleHeight = callback->GetStylePreferredSize(GetStyleBounds(style)).y;
|
|
if (pim_rowHeight < styleHeight)
|
|
{
|
|
pim_rowHeight = styleHeight;
|
|
}
|
|
}
|
|
}
|
|
|
|
bool FixedHeightItemArranger::IsItemOutOfViewBounds(vint index, ItemStyleRecord style, Rect bounds, Rect viewBounds)
|
|
{
|
|
return bounds.Top() >= viewBounds.Bottom();
|
|
}
|
|
|
|
bool FixedHeightItemArranger::EndPlaceItem(bool forMoving, Rect newBounds, vint newStartIndex)
|
|
{
|
|
if (forMoving)
|
|
{
|
|
if (pim_rowHeight != rowHeight)
|
|
{
|
|
vint offset = (pim_rowHeight - rowHeight) * newStartIndex;
|
|
rowHeight = pim_rowHeight;
|
|
callback->SetViewLocation(Point(0, newBounds.Top() + offset));
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void FixedHeightItemArranger::InvalidateItemSizeCache()
|
|
{
|
|
rowHeight = 1;
|
|
}
|
|
|
|
Size FixedHeightItemArranger::OnCalculateTotalSize()
|
|
{
|
|
vint width = GetWidth();
|
|
if (width < 0) width = 0;
|
|
return Size(width, rowHeight * itemProvider->Count() + GetYOffset());
|
|
}
|
|
|
|
FixedHeightItemArranger::FixedHeightItemArranger()
|
|
{
|
|
}
|
|
|
|
FixedHeightItemArranger::~FixedHeightItemArranger()
|
|
{
|
|
}
|
|
|
|
vint FixedHeightItemArranger::FindItem(vint itemIndex, compositions::KeyDirection key)
|
|
{
|
|
vint count = itemProvider->Count();
|
|
if (count == 0) return -1;
|
|
vint groupCount = viewBounds.Height() / rowHeight;
|
|
if (groupCount == 0) groupCount = 1;
|
|
switch (key)
|
|
{
|
|
case KeyDirection::Up:
|
|
itemIndex--;
|
|
break;
|
|
case KeyDirection::Down:
|
|
itemIndex++;
|
|
break;
|
|
case KeyDirection::Home:
|
|
itemIndex = 0;
|
|
break;
|
|
case KeyDirection::End:
|
|
itemIndex = count;
|
|
break;
|
|
case KeyDirection::PageUp:
|
|
itemIndex -= groupCount;
|
|
break;
|
|
case KeyDirection::PageDown:
|
|
itemIndex += groupCount;
|
|
break;
|
|
default:
|
|
return -1;
|
|
}
|
|
|
|
if (itemIndex < 0) return 0;
|
|
else if (itemIndex >= count) return count - 1;
|
|
else return itemIndex;
|
|
}
|
|
|
|
GuiListControl::EnsureItemVisibleResult FixedHeightItemArranger::EnsureItemVisible(vint itemIndex)
|
|
{
|
|
if (callback)
|
|
{
|
|
if (itemIndex < 0 || itemIndex >= itemProvider->Count())
|
|
{
|
|
return GuiListControl::EnsureItemVisibleResult::ItemNotExists;
|
|
}
|
|
bool moved = false;
|
|
while (true)
|
|
{
|
|
vint yOffset = GetYOffset();
|
|
vint top = itemIndex*rowHeight;
|
|
vint bottom = top + rowHeight + yOffset;
|
|
|
|
if (viewBounds.Height() < rowHeight)
|
|
{
|
|
if (viewBounds.Top() < bottom && top < viewBounds.Bottom())
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
Point location = viewBounds.LeftTop();
|
|
if (top < viewBounds.Top())
|
|
{
|
|
location.y = top;
|
|
}
|
|
else if (viewBounds.Bottom() < bottom)
|
|
{
|
|
location.y = bottom - viewBounds.Height();
|
|
}
|
|
else
|
|
{
|
|
break;
|
|
}
|
|
|
|
auto oldLeftTop = viewBounds.LeftTop();
|
|
callback->SetViewLocation(location);
|
|
moved |= viewBounds.LeftTop() != oldLeftTop;
|
|
if (viewBounds.LeftTop() != location) break;
|
|
}
|
|
return moved ? GuiListControl::EnsureItemVisibleResult::Moved : GuiListControl::EnsureItemVisibleResult::NotMoved;
|
|
}
|
|
return GuiListControl::EnsureItemVisibleResult::NotMoved;
|
|
}
|
|
|
|
Size FixedHeightItemArranger::GetAdoptedSize(Size expectedSize)
|
|
{
|
|
if (itemProvider)
|
|
{
|
|
vint yOffset = GetYOffset();
|
|
vint y = expectedSize.y - yOffset;
|
|
vint itemCount = itemProvider->Count();
|
|
return Size(expectedSize.x, yOffset + CalculateAdoptedSize(y, itemCount, rowHeight));
|
|
}
|
|
return expectedSize;
|
|
}
|
|
|
|
/***********************************************************************
|
|
FixedSizeMultiColumnItemArranger
|
|
***********************************************************************/
|
|
|
|
void FixedSizeMultiColumnItemArranger::BeginPlaceItem(bool forMoving, Rect newBounds, vint& newStartIndex)
|
|
{
|
|
if (forMoving)
|
|
{
|
|
pim_itemSize = itemSize;
|
|
vint rows = newBounds.Top() / itemSize.y;
|
|
if (rows < 0) rows = 0;
|
|
vint cols = newBounds.Width() / itemSize.x;
|
|
if (cols < 1) cols = 1;
|
|
newStartIndex = rows * cols;
|
|
}
|
|
}
|
|
|
|
void FixedSizeMultiColumnItemArranger::PlaceItem(bool forMoving, bool newCreatedStyle, vint index, ItemStyleRecord style, Rect viewBounds, Rect& bounds, Margin& alignmentToParent)
|
|
{
|
|
vint rowItems = viewBounds.Width() / itemSize.x;
|
|
if (rowItems < 1) rowItems = 1;
|
|
|
|
vint row = index / rowItems;
|
|
vint col = index % rowItems;
|
|
bounds = Rect(Point(col * itemSize.x, row * itemSize.y), itemSize);
|
|
if (forMoving)
|
|
{
|
|
Size styleSize = callback->GetStylePreferredSize(GetStyleBounds(style));
|
|
if (pim_itemSize.x < styleSize.x) pim_itemSize.x = styleSize.x;
|
|
if (pim_itemSize.y < styleSize.y) pim_itemSize.y = styleSize.y;
|
|
}
|
|
}
|
|
|
|
bool FixedSizeMultiColumnItemArranger::IsItemOutOfViewBounds(vint index, ItemStyleRecord style, Rect bounds, Rect viewBounds)
|
|
{
|
|
return bounds.Top() >= viewBounds.Bottom();
|
|
}
|
|
|
|
bool FixedSizeMultiColumnItemArranger::EndPlaceItem(bool forMoving, Rect newBounds, vint newStartIndex)
|
|
{
|
|
if (forMoving)
|
|
{
|
|
if (pim_itemSize != itemSize)
|
|
{
|
|
itemSize = pim_itemSize;
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void FixedSizeMultiColumnItemArranger::CalculateRange(Size itemSize, Rect bounds, vint count, vint& start, vint& end)
|
|
{
|
|
vint startRow = bounds.Top() / itemSize.y;
|
|
if (startRow < 0) startRow = 0;
|
|
vint endRow = (bounds.Bottom() - 1) / itemSize.y;
|
|
vint cols = bounds.Width() / itemSize.x;
|
|
if (cols < 1) cols = 1;
|
|
|
|
start = startRow*cols;
|
|
end = (endRow + 1)*cols - 1;
|
|
if (end >= count) end = count - 1;
|
|
}
|
|
|
|
void FixedSizeMultiColumnItemArranger::InvalidateItemSizeCache()
|
|
{
|
|
itemSize = Size(1, 1);
|
|
}
|
|
|
|
Size FixedSizeMultiColumnItemArranger::OnCalculateTotalSize()
|
|
{
|
|
vint rowItems = viewBounds.Width() / itemSize.x;
|
|
if (rowItems < 1) rowItems = 1;
|
|
vint rows = itemProvider->Count() / rowItems;
|
|
if (itemProvider->Count() % rowItems) rows++;
|
|
|
|
return Size(itemSize.x * rowItems, itemSize.y*rows);
|
|
}
|
|
|
|
FixedSizeMultiColumnItemArranger::FixedSizeMultiColumnItemArranger()
|
|
{
|
|
}
|
|
|
|
FixedSizeMultiColumnItemArranger::~FixedSizeMultiColumnItemArranger()
|
|
{
|
|
}
|
|
|
|
vint FixedSizeMultiColumnItemArranger::FindItem(vint itemIndex, compositions::KeyDirection key)
|
|
{
|
|
vint count = itemProvider->Count();
|
|
vint columnCount = viewBounds.Width() / itemSize.x;
|
|
if (columnCount == 0) columnCount = 1;
|
|
vint rowCount = viewBounds.Height() / itemSize.y;
|
|
if (rowCount == 0) rowCount = 1;
|
|
|
|
switch (key)
|
|
{
|
|
case KeyDirection::Up:
|
|
itemIndex -= columnCount;
|
|
break;
|
|
case KeyDirection::Down:
|
|
itemIndex += columnCount;
|
|
break;
|
|
case KeyDirection::Left:
|
|
itemIndex--;
|
|
break;
|
|
case KeyDirection::Right:
|
|
itemIndex++;
|
|
break;
|
|
case KeyDirection::Home:
|
|
itemIndex = 0;
|
|
break;
|
|
case KeyDirection::End:
|
|
itemIndex = count;
|
|
break;
|
|
case KeyDirection::PageUp:
|
|
itemIndex -= columnCount*rowCount;
|
|
break;
|
|
case KeyDirection::PageDown:
|
|
itemIndex += columnCount*rowCount;
|
|
break;
|
|
case KeyDirection::PageLeft:
|
|
itemIndex -= itemIndex%columnCount;
|
|
break;
|
|
case KeyDirection::PageRight:
|
|
itemIndex += columnCount - itemIndex%columnCount - 1;
|
|
break;
|
|
default:
|
|
return -1;
|
|
}
|
|
|
|
if (itemIndex < 0) return 0;
|
|
else if (itemIndex >= count) return count - 1;
|
|
else return itemIndex;
|
|
}
|
|
|
|
GuiListControl::EnsureItemVisibleResult FixedSizeMultiColumnItemArranger::EnsureItemVisible(vint itemIndex)
|
|
{
|
|
if (callback)
|
|
{
|
|
if (itemIndex < 0 || itemIndex >= itemProvider->Count())
|
|
{
|
|
return GuiListControl::EnsureItemVisibleResult::ItemNotExists;
|
|
}
|
|
bool moved = false;
|
|
while (true)
|
|
{
|
|
vint rowHeight = itemSize.y;
|
|
vint columnCount = viewBounds.Width() / itemSize.x;
|
|
if (columnCount == 0) columnCount = 1;
|
|
vint rowIndex = itemIndex / columnCount;
|
|
|
|
vint top = rowIndex*rowHeight;
|
|
vint bottom = top + rowHeight;
|
|
|
|
if (viewBounds.Height() < rowHeight)
|
|
{
|
|
if (viewBounds.Top() < bottom && top < viewBounds.Bottom())
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
Point location = viewBounds.LeftTop();
|
|
if (top < viewBounds.Top())
|
|
{
|
|
location.y = top;
|
|
}
|
|
else if (viewBounds.Bottom() < bottom)
|
|
{
|
|
location.y = bottom - viewBounds.Height();
|
|
}
|
|
else
|
|
{
|
|
break;
|
|
}
|
|
|
|
auto oldLeftTop = viewBounds.LeftTop();
|
|
callback->SetViewLocation(location);
|
|
moved |= viewBounds.LeftTop() != oldLeftTop;
|
|
if (viewBounds.LeftTop() != location) break;
|
|
}
|
|
return moved ? GuiListControl::EnsureItemVisibleResult::Moved : GuiListControl::EnsureItemVisibleResult::NotMoved;
|
|
}
|
|
return GuiListControl::EnsureItemVisibleResult::NotMoved;
|
|
}
|
|
|
|
Size FixedSizeMultiColumnItemArranger::GetAdoptedSize(Size expectedSize)
|
|
{
|
|
if (itemProvider)
|
|
{
|
|
vint count = itemProvider->Count();
|
|
vint columnCount = viewBounds.Width() / itemSize.x;
|
|
vint rowCount = viewBounds.Height() / itemSize.y;
|
|
return Size(
|
|
CalculateAdoptedSize(expectedSize.x, columnCount, itemSize.x),
|
|
CalculateAdoptedSize(expectedSize.y, rowCount, itemSize.y)
|
|
);
|
|
}
|
|
return expectedSize;
|
|
}
|
|
|
|
/***********************************************************************
|
|
FixedHeightMultiColumnItemArranger
|
|
***********************************************************************/
|
|
|
|
void FixedHeightMultiColumnItemArranger::CalculateRange(vint itemHeight, Rect bounds, vint& rows, vint& startColumn)
|
|
{
|
|
rows = bounds.Height() / itemHeight;
|
|
if (rows < 1) rows = 1;
|
|
startColumn = bounds.Left() / bounds.Width();
|
|
}
|
|
|
|
void FixedHeightMultiColumnItemArranger::BeginPlaceItem(bool forMoving, Rect newBounds, vint& newStartIndex)
|
|
{
|
|
pi_currentWidth = 0;
|
|
pi_totalWidth = 0;
|
|
if (forMoving)
|
|
{
|
|
pim_itemHeight = itemHeight;
|
|
vint rows = newBounds.Height() / itemHeight;
|
|
if (rows < 1) rows = 1;
|
|
vint columns = newBounds.Left() / newBounds.Width();
|
|
newStartIndex = rows * columns;
|
|
}
|
|
}
|
|
|
|
void FixedHeightMultiColumnItemArranger::PlaceItem(bool forMoving, bool newCreatedStyle, vint index, ItemStyleRecord style, Rect viewBounds, Rect& bounds, Margin& alignmentToParent)
|
|
{
|
|
vint rows = viewBounds.Height() / itemHeight;
|
|
if (rows < 1) rows = 1;
|
|
|
|
vint row = index % rows;
|
|
if (row == 0)
|
|
{
|
|
pi_totalWidth += pi_currentWidth;
|
|
pi_currentWidth = 0;
|
|
}
|
|
|
|
Size styleSize = callback->GetStylePreferredSize(GetStyleBounds(style));
|
|
if (pi_currentWidth < styleSize.x) pi_currentWidth = styleSize.x;
|
|
bounds = Rect(Point(pi_totalWidth + viewBounds.Left(), itemHeight * row), Size(0, 0));
|
|
if (forMoving)
|
|
{
|
|
if (pim_itemHeight < styleSize.y) pim_itemHeight = styleSize.y;
|
|
}
|
|
}
|
|
|
|
bool FixedHeightMultiColumnItemArranger::IsItemOutOfViewBounds(vint index, ItemStyleRecord style, Rect bounds, Rect viewBounds)
|
|
{
|
|
return bounds.Left() >= viewBounds.Right();
|
|
}
|
|
|
|
bool FixedHeightMultiColumnItemArranger::EndPlaceItem(bool forMoving, Rect newBounds, vint newStartIndex)
|
|
{
|
|
if (forMoving)
|
|
{
|
|
if (pim_itemHeight != itemHeight)
|
|
{
|
|
itemHeight = pim_itemHeight;
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void FixedHeightMultiColumnItemArranger::InvalidateItemSizeCache()
|
|
{
|
|
itemHeight = 1;
|
|
}
|
|
|
|
Size FixedHeightMultiColumnItemArranger::OnCalculateTotalSize()
|
|
{
|
|
vint rows = viewBounds.Height() / itemHeight;
|
|
if (rows < 1) rows = 1;
|
|
vint columns = itemProvider->Count() / rows;
|
|
if (itemProvider->Count() % rows) columns += 1;
|
|
return Size(viewBounds.Width() * columns, 0);
|
|
}
|
|
|
|
FixedHeightMultiColumnItemArranger::FixedHeightMultiColumnItemArranger()
|
|
:itemHeight(1)
|
|
{
|
|
}
|
|
|
|
FixedHeightMultiColumnItemArranger::~FixedHeightMultiColumnItemArranger()
|
|
{
|
|
}
|
|
|
|
vint FixedHeightMultiColumnItemArranger::FindItem(vint itemIndex, compositions::KeyDirection key)
|
|
{
|
|
vint count = itemProvider->Count();
|
|
vint groupCount = viewBounds.Height() / itemHeight;
|
|
if (groupCount == 0) groupCount = 1;
|
|
switch (key)
|
|
{
|
|
case KeyDirection::Up:
|
|
itemIndex--;
|
|
break;
|
|
case KeyDirection::Down:
|
|
itemIndex++;
|
|
break;
|
|
case KeyDirection::Left:
|
|
itemIndex -= groupCount;
|
|
break;
|
|
case KeyDirection::Right:
|
|
itemIndex += groupCount;
|
|
break;
|
|
case KeyDirection::Home:
|
|
itemIndex = 0;
|
|
break;
|
|
case KeyDirection::End:
|
|
itemIndex = count;
|
|
break;
|
|
case KeyDirection::PageUp:
|
|
itemIndex -= itemIndex%groupCount;
|
|
break;
|
|
case KeyDirection::PageDown:
|
|
itemIndex += groupCount - itemIndex%groupCount - 1;
|
|
break;
|
|
default:
|
|
return -1;
|
|
}
|
|
|
|
if (itemIndex < 0) return 0;
|
|
else if (itemIndex >= count) return count - 1;
|
|
else return itemIndex;
|
|
}
|
|
|
|
GuiListControl::EnsureItemVisibleResult FixedHeightMultiColumnItemArranger::EnsureItemVisible(vint itemIndex)
|
|
{
|
|
if (callback)
|
|
{
|
|
if (itemIndex < 0 || itemIndex >= itemProvider->Count())
|
|
{
|
|
return GuiListControl::EnsureItemVisibleResult::ItemNotExists;
|
|
}
|
|
bool moved = false;
|
|
while (true)
|
|
{
|
|
vint rowCount = viewBounds.Height() / itemHeight;
|
|
if (rowCount == 0) rowCount = 1;
|
|
vint columnIndex = itemIndex / rowCount;
|
|
vint minIndex = startIndex;
|
|
vint maxIndex = startIndex + visibleStyles.Count() - 1;
|
|
|
|
Point location = viewBounds.LeftTop();
|
|
if (minIndex <= itemIndex && itemIndex <= maxIndex)
|
|
{
|
|
Rect bounds = callback->GetStyleBounds(GetStyleBounds(visibleStyles[itemIndex - startIndex]));
|
|
if (0 < bounds.Bottom() && bounds.Top() < viewBounds.Width() && bounds.Width() > viewBounds.Width())
|
|
{
|
|
break;
|
|
}
|
|
else if (bounds.Left() < 0)
|
|
{
|
|
location.x -= viewBounds.Width();
|
|
}
|
|
else if (bounds.Right() > viewBounds.Width())
|
|
{
|
|
location.x += viewBounds.Width();
|
|
}
|
|
else
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
else if (columnIndex < minIndex / rowCount)
|
|
{
|
|
location.x -= viewBounds.Width();
|
|
}
|
|
else if (columnIndex >= maxIndex / rowCount)
|
|
{
|
|
location.x += viewBounds.Width();
|
|
}
|
|
else
|
|
{
|
|
break;
|
|
}
|
|
|
|
auto oldLeftTop = viewBounds.LeftTop();
|
|
callback->SetViewLocation(location);
|
|
moved |= viewBounds.LeftTop() != oldLeftTop;
|
|
if (viewBounds.LeftTop() != location) break;
|
|
}
|
|
return moved ? GuiListControl::EnsureItemVisibleResult::Moved : GuiListControl::EnsureItemVisibleResult::NotMoved;
|
|
}
|
|
return GuiListControl::EnsureItemVisibleResult::NotMoved;
|
|
}
|
|
|
|
Size FixedHeightMultiColumnItemArranger::GetAdoptedSize(Size expectedSize)
|
|
{
|
|
if (itemProvider)
|
|
{
|
|
vint count = itemProvider->Count();
|
|
return Size(expectedSize.x, CalculateAdoptedSize(expectedSize.y, count, itemHeight));
|
|
}
|
|
return expectedSize;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\CONTROLS\LISTCONTROLPACKAGE\GUILISTCONTROLS.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace controls
|
|
{
|
|
using namespace collections;
|
|
using namespace elements;
|
|
using namespace compositions;
|
|
|
|
/***********************************************************************
|
|
GuiListControl::ItemCallback
|
|
***********************************************************************/
|
|
|
|
Ptr<GuiListControl::ItemCallback::BoundsChangedHandler> GuiListControl::ItemCallback::InstallStyle(ItemStyle* style, vint itemIndex, compositions::GuiBoundsComposition* itemComposition)
|
|
{
|
|
auto handler = style->BoundsChanged.AttachMethod(this, &ItemCallback::OnStyleBoundsChanged);
|
|
listControl->GetContainerComposition()->AddChild(itemComposition ? itemComposition : style);
|
|
listControl->OnStyleInstalled(itemIndex, style);
|
|
return handler;
|
|
}
|
|
|
|
GuiListControl::ItemStyle* GuiListControl::ItemCallback::UninstallStyle(vint index)
|
|
{
|
|
auto style = installedStyles.Keys()[index];
|
|
auto handler = installedStyles.Values()[index];
|
|
listControl->OnStyleUninstalled(style);
|
|
listControl->GetContainerComposition()->RemoveChild(style);
|
|
style->BoundsChanged.Detach(handler);
|
|
return style;
|
|
}
|
|
|
|
void GuiListControl::ItemCallback::OnStyleBoundsChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
listControl->InvokeOrDelayIfRendering([=]()
|
|
{
|
|
listControl->CalculateView();
|
|
});
|
|
}
|
|
|
|
GuiListControl::ItemCallback::ItemCallback(GuiListControl* _listControl)
|
|
:listControl(_listControl)
|
|
{
|
|
}
|
|
|
|
GuiListControl::ItemCallback::~ItemCallback()
|
|
{
|
|
ClearCache();
|
|
}
|
|
|
|
void GuiListControl::ItemCallback::ClearCache()
|
|
{
|
|
for (vint i = 0; i < installedStyles.Count(); i++)
|
|
{
|
|
auto style = UninstallStyle(i);
|
|
SafeDeleteComposition(style);
|
|
}
|
|
installedStyles.Clear();
|
|
}
|
|
|
|
void GuiListControl::ItemCallback::OnAttached(IItemProvider* provider)
|
|
{
|
|
itemProvider = provider;
|
|
}
|
|
|
|
void GuiListControl::ItemCallback::OnItemModified(vint start, vint count, vint newCount)
|
|
{
|
|
listControl->OnItemModified(start, count, newCount);
|
|
}
|
|
|
|
GuiListControl::ItemStyle* GuiListControl::ItemCallback::RequestItem(vint itemIndex, compositions::GuiBoundsComposition* itemComposition)
|
|
{
|
|
CHECK_ERROR(0 <= itemIndex && itemIndex < itemProvider->Count(), L"GuiListControl::ItemCallback::RequestItem(vint)#Index out of range.");
|
|
CHECK_ERROR(listControl->itemStyleProperty, L"GuiListControl::ItemCallback::RequestItem(vint)#SetItemTemplate function should be called before adding items to the list control.");
|
|
|
|
auto style = listControl->itemStyleProperty(itemProvider->GetBindingValue(itemIndex));
|
|
auto handler = InstallStyle(style, itemIndex, itemComposition);
|
|
installedStyles.Add(style, handler);
|
|
return style;
|
|
}
|
|
|
|
void GuiListControl::ItemCallback::ReleaseItem(ItemStyle* style)
|
|
{
|
|
vint index = installedStyles.Keys().IndexOf(style);
|
|
if (index != -1)
|
|
{
|
|
auto style = UninstallStyle(index);
|
|
installedStyles.Remove(style);
|
|
SafeDeleteComposition(style);
|
|
}
|
|
}
|
|
|
|
void GuiListControl::ItemCallback::SetViewLocation(Point value)
|
|
{
|
|
Rect virtualRect(value, listControl->GetViewSize());
|
|
Rect realRect = listControl->axis->VirtualRectToRealRect(listControl->fullSize, virtualRect);
|
|
listControl->SetViewPosition(realRect.LeftTop());
|
|
}
|
|
|
|
Size GuiListControl::ItemCallback::GetStylePreferredSize(compositions::GuiBoundsComposition* style)
|
|
{
|
|
Size size = style->GetPreferredBounds().GetSize();
|
|
return listControl->axis->RealSizeToVirtualSize(size);
|
|
}
|
|
|
|
void GuiListControl::ItemCallback::SetStyleAlignmentToParent(compositions::GuiBoundsComposition* style, Margin margin)
|
|
{
|
|
Margin newMargin = listControl->axis->VirtualMarginToRealMargin(margin);
|
|
style->SetAlignmentToParent(newMargin);
|
|
}
|
|
|
|
Rect GuiListControl::ItemCallback::GetStyleBounds(compositions::GuiBoundsComposition* style)
|
|
{
|
|
Rect bounds = style->GetBounds();
|
|
return listControl->axis->RealRectToVirtualRect(listControl->GetViewSize(), bounds);
|
|
}
|
|
|
|
void GuiListControl::ItemCallback::SetStyleBounds(compositions::GuiBoundsComposition* style, Rect bounds)
|
|
{
|
|
Rect newBounds = listControl->axis->VirtualRectToRealRect(listControl->GetViewSize(), bounds);
|
|
return style->SetBounds(newBounds);
|
|
}
|
|
|
|
compositions::GuiGraphicsComposition* GuiListControl::ItemCallback::GetContainerComposition()
|
|
{
|
|
return listControl->GetContainerComposition();
|
|
}
|
|
|
|
void GuiListControl::ItemCallback::OnTotalSizeChanged()
|
|
{
|
|
listControl->CalculateView();
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiListControl
|
|
***********************************************************************/
|
|
|
|
void GuiListControl::BeforeControlTemplateUninstalled_()
|
|
{
|
|
}
|
|
|
|
void GuiListControl::AfterControlTemplateInstalled_(bool initialize)
|
|
{
|
|
if (itemArranger)
|
|
{
|
|
itemArranger->ReloadVisibleStyles();
|
|
CalculateView();
|
|
}
|
|
}
|
|
|
|
void GuiListControl::OnItemModified(vint start, vint count, vint newCount)
|
|
{
|
|
}
|
|
|
|
void GuiListControl::OnStyleInstalled(vint itemIndex, ItemStyle* style)
|
|
{
|
|
style->SetFont(GetDisplayFont());
|
|
style->SetContext(GetContext());
|
|
style->SetText(itemProvider->GetTextValue(itemIndex));
|
|
style->SetVisuallyEnabled(GetVisuallyEnabled());
|
|
style->SetSelected(false);
|
|
style->SetIndex(itemIndex);
|
|
style->Initialize(this);
|
|
AttachItemEvents(style);
|
|
}
|
|
|
|
void GuiListControl::OnStyleUninstalled(ItemStyle* style)
|
|
{
|
|
DetachItemEvents(style);
|
|
}
|
|
|
|
void GuiListControl::OnRenderTargetChanged(elements::IGuiGraphicsRenderTarget* renderTarget)
|
|
{
|
|
SetStyleAndArranger(itemStyleProperty, itemArranger);
|
|
GuiScrollView::OnRenderTargetChanged(renderTarget);
|
|
}
|
|
|
|
void GuiListControl::OnBeforeReleaseGraphicsHost()
|
|
{
|
|
GuiScrollView::OnBeforeReleaseGraphicsHost();
|
|
SetStyleAndArranger({}, nullptr);
|
|
}
|
|
|
|
Size GuiListControl::QueryFullSize()
|
|
{
|
|
Size virtualSize = itemArranger ? itemArranger->GetTotalSize() : Size(0, 0);
|
|
fullSize = axis->VirtualSizeToRealSize(virtualSize);
|
|
return fullSize;
|
|
}
|
|
|
|
void GuiListControl::UpdateView(Rect viewBounds)
|
|
{
|
|
if (itemArranger)
|
|
{
|
|
Rect newBounds = axis->RealRectToVirtualRect(fullSize, viewBounds);
|
|
itemArranger->OnViewChanged(newBounds);
|
|
}
|
|
}
|
|
|
|
void GuiListControl::OnBoundsMouseButtonDown(compositions::GuiGraphicsComposition* sender, compositions::GuiMouseEventArgs& arguments)
|
|
{
|
|
if(GetVisuallyEnabled())
|
|
{
|
|
SetFocus();
|
|
}
|
|
}
|
|
|
|
void GuiListControl::SetStyleAndArranger(ItemStyleProperty styleProperty, Ptr<IItemArranger> arranger)
|
|
{
|
|
if (itemArranger)
|
|
{
|
|
itemArranger->DetachListControl();
|
|
itemArranger->SetCallback(nullptr);
|
|
itemProvider->DetachCallback(itemArranger.Obj());
|
|
}
|
|
callback->ClearCache();
|
|
|
|
itemStyleProperty = styleProperty;
|
|
itemArranger = arranger;
|
|
|
|
if (auto scroll = GetVerticalScroll())
|
|
{
|
|
scroll->SetPosition(0);
|
|
}
|
|
if (auto scroll = GetHorizontalScroll())
|
|
{
|
|
scroll->SetPosition(0);
|
|
}
|
|
|
|
if (itemArranger)
|
|
{
|
|
itemProvider->AttachCallback(itemArranger.Obj());
|
|
itemArranger->SetCallback(callback.Obj());
|
|
itemArranger->AttachListControl(this);
|
|
}
|
|
CalculateView();
|
|
}
|
|
|
|
void GuiListControl::UpdateDisplayFont()
|
|
{
|
|
GuiControl::UpdateDisplayFont();
|
|
FOREACH(ItemStyle*, style, visibleStyles.Keys())
|
|
{
|
|
style->SetFont(GetDisplayFont());
|
|
}
|
|
}
|
|
|
|
void GuiListControl::OnClientBoundsChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
auto args = GetNotifyEventArguments();
|
|
AdoptedSizeInvalidated.Execute(args);
|
|
}
|
|
|
|
void GuiListControl::OnVisuallyEnabledChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
FOREACH(ItemStyle*, style, visibleStyles.Keys())
|
|
{
|
|
style->SetVisuallyEnabled(GetVisuallyEnabled());
|
|
}
|
|
}
|
|
|
|
void GuiListControl::OnContextChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
FOREACH(ItemStyle*, style, visibleStyles.Keys())
|
|
{
|
|
style->SetContext(GetContext());
|
|
}
|
|
}
|
|
|
|
void GuiListControl::OnItemMouseEvent(compositions::GuiItemMouseEvent& itemEvent, ItemStyle* style, compositions::GuiGraphicsComposition* sender, compositions::GuiMouseEventArgs& arguments)
|
|
{
|
|
if (itemArranger && GetVisuallyEnabled())
|
|
{
|
|
vint itemIndex = itemArranger->GetVisibleIndex(style);
|
|
if (itemIndex != -1)
|
|
{
|
|
GuiItemMouseEventArgs redirectArguments;
|
|
(GuiMouseEventArgs&)redirectArguments = arguments;
|
|
redirectArguments.compositionSource = boundsComposition;
|
|
redirectArguments.eventSource = boundsComposition;
|
|
redirectArguments.itemIndex = itemIndex;
|
|
itemEvent.Execute(redirectArguments);
|
|
arguments = redirectArguments;
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiListControl::OnItemNotifyEvent(compositions::GuiItemNotifyEvent& itemEvent, ItemStyle* style, compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
if (itemArranger && GetVisuallyEnabled())
|
|
{
|
|
vint itemIndex = itemArranger->GetVisibleIndex(style);
|
|
if (itemIndex != -1)
|
|
{
|
|
GuiItemEventArgs redirectArguments;
|
|
(GuiEventArgs&)redirectArguments = arguments;
|
|
redirectArguments.compositionSource = boundsComposition;
|
|
redirectArguments.eventSource = boundsComposition;
|
|
redirectArguments.itemIndex = itemIndex;
|
|
itemEvent.Execute(redirectArguments);
|
|
arguments = redirectArguments;
|
|
}
|
|
}
|
|
}
|
|
|
|
#define ATTACH_ITEM_MOUSE_EVENT(EVENTNAME, ITEMEVENTNAME)\
|
|
{\
|
|
Func<void(GuiItemMouseEvent&, ItemStyle*, GuiGraphicsComposition*, GuiMouseEventArgs&)> func(this, &GuiListControl::OnItemMouseEvent);\
|
|
helper->EVENTNAME##Handler = style->GetEventReceiver()->EVENTNAME.AttachFunction(\
|
|
Curry(Curry(func)(ITEMEVENTNAME))(style)\
|
|
);\
|
|
}\
|
|
|
|
#define ATTACH_ITEM_NOTIFY_EVENT(EVENTNAME, ITEMEVENTNAME)\
|
|
{\
|
|
Func<void(GuiItemNotifyEvent&, ItemStyle*, GuiGraphicsComposition*, GuiEventArgs&)> func(this, &GuiListControl::OnItemNotifyEvent);\
|
|
helper->EVENTNAME##Handler = style->GetEventReceiver()->EVENTNAME.AttachFunction(\
|
|
Curry(Curry(func)(ITEMEVENTNAME))(style)\
|
|
);\
|
|
}\
|
|
|
|
void GuiListControl::AttachItemEvents(ItemStyle* style)
|
|
{
|
|
vint index=visibleStyles.Keys().IndexOf(style);
|
|
if(index==-1)
|
|
{
|
|
Ptr<VisibleStyleHelper> helper=new VisibleStyleHelper;
|
|
visibleStyles.Add(style, helper);
|
|
|
|
ATTACH_ITEM_MOUSE_EVENT(leftButtonDown, ItemLeftButtonDown);
|
|
ATTACH_ITEM_MOUSE_EVENT(leftButtonUp, ItemLeftButtonUp);
|
|
ATTACH_ITEM_MOUSE_EVENT(leftButtonDoubleClick, ItemLeftButtonDoubleClick);
|
|
ATTACH_ITEM_MOUSE_EVENT(middleButtonDown, ItemMiddleButtonDown);
|
|
ATTACH_ITEM_MOUSE_EVENT(middleButtonUp, ItemMiddleButtonUp);
|
|
ATTACH_ITEM_MOUSE_EVENT(middleButtonDoubleClick, ItemMiddleButtonDoubleClick);
|
|
ATTACH_ITEM_MOUSE_EVENT(rightButtonDown, ItemRightButtonDown);
|
|
ATTACH_ITEM_MOUSE_EVENT(rightButtonUp, ItemRightButtonUp);
|
|
ATTACH_ITEM_MOUSE_EVENT(rightButtonDoubleClick, ItemRightButtonDoubleClick);
|
|
ATTACH_ITEM_MOUSE_EVENT(mouseMove, ItemMouseMove);
|
|
ATTACH_ITEM_NOTIFY_EVENT(mouseEnter, ItemMouseEnter);
|
|
ATTACH_ITEM_NOTIFY_EVENT(mouseLeave, ItemMouseLeave);
|
|
}
|
|
}
|
|
|
|
#undef ATTACH_ITEM_MOUSE_EVENT
|
|
#undef ATTACH_ITEM_NOTIFY_EVENT
|
|
|
|
#define DETACH_ITEM_EVENT(EVENTNAME) style->GetEventReceiver()->EVENTNAME.Detach(helper->EVENTNAME##Handler)
|
|
|
|
void GuiListControl::DetachItemEvents(ItemStyle* style)
|
|
{
|
|
vint index=visibleStyles.Keys().IndexOf(style);
|
|
if(index!=-1)
|
|
{
|
|
Ptr<VisibleStyleHelper> helper=visibleStyles.Values().Get(index);
|
|
visibleStyles.Remove(style);
|
|
|
|
DETACH_ITEM_EVENT(leftButtonDown);
|
|
DETACH_ITEM_EVENT(leftButtonUp);
|
|
DETACH_ITEM_EVENT(leftButtonDoubleClick);
|
|
DETACH_ITEM_EVENT(middleButtonDown);
|
|
DETACH_ITEM_EVENT(middleButtonUp);
|
|
DETACH_ITEM_EVENT(middleButtonDoubleClick);
|
|
DETACH_ITEM_EVENT(rightButtonDown);
|
|
DETACH_ITEM_EVENT(rightButtonUp);
|
|
DETACH_ITEM_EVENT(rightButtonDoubleClick);
|
|
DETACH_ITEM_EVENT(mouseMove);
|
|
DETACH_ITEM_EVENT(mouseEnter);
|
|
DETACH_ITEM_EVENT(mouseLeave);
|
|
}
|
|
}
|
|
|
|
#undef DETACH_ITEM_EVENT
|
|
|
|
GuiListControl::GuiListControl(theme::ThemeName themeName, IItemProvider* _itemProvider, bool acceptFocus)
|
|
:GuiScrollView(themeName)
|
|
, itemProvider(_itemProvider)
|
|
{
|
|
ContextChanged.AttachMethod(this, &GuiListControl::OnContextChanged);
|
|
VisuallyEnabledChanged.AttachMethod(this, &GuiListControl::OnVisuallyEnabledChanged);
|
|
containerComposition->BoundsChanged.AttachMethod(this, &GuiListControl::OnClientBoundsChanged);
|
|
|
|
ItemTemplateChanged.SetAssociatedComposition(boundsComposition);
|
|
ArrangerChanged.SetAssociatedComposition(boundsComposition);
|
|
AxisChanged.SetAssociatedComposition(boundsComposition);
|
|
AdoptedSizeInvalidated.SetAssociatedComposition(boundsComposition);
|
|
|
|
ItemLeftButtonDown.SetAssociatedComposition(boundsComposition);
|
|
ItemLeftButtonUp.SetAssociatedComposition(boundsComposition);
|
|
ItemLeftButtonDoubleClick.SetAssociatedComposition(boundsComposition);
|
|
ItemMiddleButtonDown.SetAssociatedComposition(boundsComposition);
|
|
ItemMiddleButtonUp.SetAssociatedComposition(boundsComposition);
|
|
ItemMiddleButtonDoubleClick.SetAssociatedComposition(boundsComposition);
|
|
ItemRightButtonDown.SetAssociatedComposition(boundsComposition);
|
|
ItemRightButtonUp.SetAssociatedComposition(boundsComposition);
|
|
ItemRightButtonDoubleClick.SetAssociatedComposition(boundsComposition);
|
|
ItemMouseMove.SetAssociatedComposition(boundsComposition);
|
|
ItemMouseEnter.SetAssociatedComposition(boundsComposition);
|
|
ItemMouseLeave.SetAssociatedComposition(boundsComposition);
|
|
|
|
callback = new ItemCallback(this);
|
|
itemProvider->AttachCallback(callback.Obj());
|
|
axis = new GuiDefaultAxis;
|
|
|
|
if (acceptFocus)
|
|
{
|
|
boundsComposition->GetEventReceiver()->leftButtonDown.AttachMethod(this, &GuiListControl::OnBoundsMouseButtonDown);
|
|
boundsComposition->GetEventReceiver()->middleButtonDown.AttachMethod(this, &GuiListControl::OnBoundsMouseButtonDown);
|
|
boundsComposition->GetEventReceiver()->rightButtonDown.AttachMethod(this, &GuiListControl::OnBoundsMouseButtonDown);
|
|
SetFocusableComposition(boundsComposition);
|
|
}
|
|
}
|
|
|
|
GuiListControl::~GuiListControl()
|
|
{
|
|
if(itemArranger)
|
|
{
|
|
itemProvider->DetachCallback(itemArranger.Obj());
|
|
}
|
|
callback->ClearCache();
|
|
itemStyleProperty = {};
|
|
itemArranger = nullptr;
|
|
}
|
|
|
|
GuiListControl::IItemProvider* GuiListControl::GetItemProvider()
|
|
{
|
|
return itemProvider.Obj();
|
|
}
|
|
|
|
GuiListControl::ItemStyleProperty GuiListControl::GetItemTemplate()
|
|
{
|
|
return itemStyleProperty;
|
|
}
|
|
|
|
void GuiListControl::SetItemTemplate(ItemStyleProperty value)
|
|
{
|
|
SetStyleAndArranger(value, itemArranger);
|
|
ItemTemplateChanged.Execute(GetNotifyEventArguments());
|
|
}
|
|
|
|
GuiListControl::IItemArranger* GuiListControl::GetArranger()
|
|
{
|
|
return itemArranger.Obj();
|
|
}
|
|
|
|
void GuiListControl::SetArranger(Ptr<IItemArranger> value)
|
|
{
|
|
SetStyleAndArranger(itemStyleProperty, value);
|
|
ArrangerChanged.Execute(GetNotifyEventArguments());
|
|
}
|
|
|
|
compositions::IGuiAxis* GuiListControl::GetAxis()
|
|
{
|
|
return axis.Obj();
|
|
}
|
|
|
|
void GuiListControl::SetAxis(Ptr<compositions::IGuiAxis> value)
|
|
{
|
|
Ptr<IGuiAxis> old = axis;
|
|
axis = value;
|
|
SetStyleAndArranger(itemStyleProperty, itemArranger);
|
|
AxisChanged.Execute(GetNotifyEventArguments());
|
|
}
|
|
|
|
bool GuiListControl::EnsureItemVisible(vint itemIndex)
|
|
{
|
|
if (itemIndex < 0 || itemIndex >= itemProvider->Count())
|
|
{
|
|
return false;
|
|
}
|
|
|
|
if (!itemArranger) return false;
|
|
auto result = itemArranger->EnsureItemVisible(itemIndex);
|
|
if (result == EnsureItemVisibleResult::Moved)
|
|
{
|
|
if (auto host = GetBoundsComposition()->GetRelatedGraphicsHost())
|
|
{
|
|
auto flag = GetDisposedFlag();
|
|
host->InvokeAfterRendering([=]()
|
|
{
|
|
if (!flag->IsDisposed())
|
|
{
|
|
EnsureItemVisible(itemIndex);
|
|
}
|
|
}, { this,0 });
|
|
}
|
|
}
|
|
return result != EnsureItemVisibleResult::ItemNotExists;
|
|
}
|
|
|
|
Size GuiListControl::GetAdoptedSize(Size expectedSize)
|
|
{
|
|
if (itemArranger)
|
|
{
|
|
Size controlSize = boundsComposition->GetBounds().GetSize();
|
|
Size viewSize = containerComposition->GetBounds().GetSize();
|
|
vint x = controlSize.x - viewSize.x;
|
|
vint y = controlSize.y - viewSize.y;
|
|
|
|
Size expectedViewSize(expectedSize.x - x, expectedSize.y - y);
|
|
if (axis)
|
|
{
|
|
expectedViewSize = axis->RealSizeToVirtualSize(expectedViewSize);
|
|
}
|
|
Size adoptedViewSize = itemArranger->GetAdoptedSize(expectedViewSize);
|
|
if (axis)
|
|
{
|
|
adoptedViewSize = axis->VirtualSizeToRealSize(adoptedViewSize);
|
|
}
|
|
return Size(adoptedViewSize.x + x, adoptedViewSize.y + y);
|
|
}
|
|
return expectedSize;
|
|
}
|
|
|
|
bool GuiListControl::GetDisplayItemBackground()
|
|
{
|
|
return displayItemBackground;
|
|
}
|
|
|
|
void GuiListControl::SetDisplayItemBackground(bool value)
|
|
{
|
|
if (displayItemBackground != value)
|
|
{
|
|
displayItemBackground = value;
|
|
SetStyleAndArranger(itemStyleProperty, itemArranger);
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiSelectableListControl
|
|
***********************************************************************/
|
|
|
|
void GuiSelectableListControl::NotifySelectionChanged()
|
|
{
|
|
SelectionChanged.Execute(GetNotifyEventArguments());
|
|
}
|
|
|
|
void GuiSelectableListControl::OnItemModified(vint start, vint count, vint newCount)
|
|
{
|
|
GuiListControl::OnItemModified(start, count, newCount);
|
|
if(count!=newCount)
|
|
{
|
|
ClearSelection();
|
|
}
|
|
}
|
|
|
|
void GuiSelectableListControl::OnStyleInstalled(vint itemIndex, ItemStyle* style)
|
|
{
|
|
GuiListControl::OnStyleInstalled(itemIndex, style);
|
|
style->SetSelected(selectedItems.Contains(itemIndex));
|
|
}
|
|
|
|
void GuiSelectableListControl::OnItemSelectionChanged(vint itemIndex, bool value)
|
|
{
|
|
if(auto style = itemArranger->GetVisibleStyle(itemIndex))
|
|
{
|
|
style->SetSelected(value);
|
|
}
|
|
}
|
|
|
|
void GuiSelectableListControl::OnItemSelectionCleared()
|
|
{
|
|
FOREACH(ItemStyle*, style, visibleStyles.Keys())
|
|
{
|
|
style->SetSelected(false);
|
|
}
|
|
}
|
|
|
|
void GuiSelectableListControl::OnItemLeftButtonDown(compositions::GuiGraphicsComposition* sender, compositions::GuiItemMouseEventArgs& arguments)
|
|
{
|
|
if(GetVisuallyEnabled())
|
|
{
|
|
SelectItemsByClick(arguments.itemIndex, arguments.ctrl, arguments.shift, true);
|
|
}
|
|
}
|
|
|
|
void GuiSelectableListControl::OnItemRightButtonDown(compositions::GuiGraphicsComposition* sender, compositions::GuiItemMouseEventArgs& arguments)
|
|
{
|
|
if(GetVisuallyEnabled())
|
|
{
|
|
SelectItemsByClick(arguments.itemIndex, arguments.ctrl, arguments.shift, false);
|
|
}
|
|
}
|
|
|
|
void GuiSelectableListControl::NormalizeSelectedItemIndexStartEnd()
|
|
{
|
|
if (selectedItemIndexStart < 0 || selectedItemIndexStart >= itemProvider->Count())
|
|
{
|
|
selectedItemIndexStart = 0;
|
|
}
|
|
if (selectedItemIndexEnd < 0 || selectedItemIndexEnd >= itemProvider->Count())
|
|
{
|
|
selectedItemIndexEnd = 0;
|
|
}
|
|
}
|
|
|
|
void GuiSelectableListControl::SetMultipleItemsSelectedSilently(vint start, vint end, bool selected)
|
|
{
|
|
if(start>end)
|
|
{
|
|
vint temp=start;
|
|
start=end;
|
|
end=temp;
|
|
}
|
|
vint count=itemProvider->Count();
|
|
if(start<0) start=0;
|
|
if(end>=count) end=count-1;
|
|
for(vint i=start;i<=end;i++)
|
|
{
|
|
if(selected)
|
|
{
|
|
if(!selectedItems.Contains(i))
|
|
{
|
|
selectedItems.Add(i);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
selectedItems.Remove(i);
|
|
}
|
|
OnItemSelectionChanged(i, selected);
|
|
}
|
|
}
|
|
|
|
void GuiSelectableListControl::OnKeyDown(compositions::GuiGraphicsComposition* sender, compositions::GuiKeyEventArgs& arguments)
|
|
{
|
|
if(GetVisuallyEnabled())
|
|
{
|
|
if(SelectItemsByKey(arguments.code, arguments.ctrl, arguments.shift))
|
|
{
|
|
arguments.handled=true;
|
|
}
|
|
}
|
|
}
|
|
|
|
GuiSelectableListControl::GuiSelectableListControl(theme::ThemeName themeName, IItemProvider* _itemProvider)
|
|
:GuiListControl(themeName, _itemProvider, true)
|
|
, multiSelect(false)
|
|
, selectedItemIndexStart(-1)
|
|
, selectedItemIndexEnd(-1)
|
|
{
|
|
SelectionChanged.SetAssociatedComposition(boundsComposition);
|
|
ItemLeftButtonDown.AttachMethod(this, &GuiSelectableListControl::OnItemLeftButtonDown);
|
|
ItemRightButtonDown.AttachMethod(this, &GuiSelectableListControl::OnItemRightButtonDown);
|
|
if (focusableComposition)
|
|
{
|
|
focusableComposition->GetEventReceiver()->keyDown.AttachMethod(this, &GuiSelectableListControl::OnKeyDown);
|
|
}
|
|
}
|
|
|
|
GuiSelectableListControl::~GuiSelectableListControl()
|
|
{
|
|
}
|
|
|
|
bool GuiSelectableListControl::GetMultiSelect()
|
|
{
|
|
return multiSelect;
|
|
}
|
|
|
|
void GuiSelectableListControl::SetMultiSelect(bool value)
|
|
{
|
|
if (multiSelect != value)
|
|
{
|
|
multiSelect = value;
|
|
ClearSelection();
|
|
}
|
|
}
|
|
|
|
const collections::SortedList<vint>& GuiSelectableListControl::GetSelectedItems()
|
|
{
|
|
return selectedItems;
|
|
}
|
|
|
|
vint GuiSelectableListControl::GetSelectedItemIndex()
|
|
{
|
|
return selectedItems.Count() == 1 ? selectedItems[0] : -1;
|
|
}
|
|
|
|
WString GuiSelectableListControl::GetSelectedItemText()
|
|
{
|
|
vint index = GetSelectedItemIndex();
|
|
if (index != -1)
|
|
{
|
|
return itemProvider->GetTextValue(index);
|
|
}
|
|
return L"";
|
|
}
|
|
|
|
bool GuiSelectableListControl::GetSelected(vint itemIndex)
|
|
{
|
|
return selectedItems.Contains(itemIndex);
|
|
}
|
|
|
|
void GuiSelectableListControl::SetSelected(vint itemIndex, bool value)
|
|
{
|
|
if(value)
|
|
{
|
|
if(!selectedItems.Contains(itemIndex))
|
|
{
|
|
if(!multiSelect)
|
|
{
|
|
selectedItems.Clear();
|
|
OnItemSelectionCleared();
|
|
}
|
|
selectedItems.Add(itemIndex);
|
|
OnItemSelectionChanged(itemIndex, value);
|
|
NotifySelectionChanged();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if(selectedItems.Remove(itemIndex))
|
|
{
|
|
OnItemSelectionChanged(itemIndex, value);
|
|
NotifySelectionChanged();
|
|
}
|
|
}
|
|
}
|
|
|
|
bool GuiSelectableListControl::SelectItemsByClick(vint itemIndex, bool ctrl, bool shift, bool leftButton)
|
|
{
|
|
NormalizeSelectedItemIndexStartEnd();
|
|
if (0 <= itemIndex && itemIndex < itemProvider->Count())
|
|
{
|
|
if (!leftButton)
|
|
{
|
|
if (selectedItems.Contains(itemIndex))
|
|
{
|
|
return true;
|
|
}
|
|
}
|
|
if (!multiSelect)
|
|
{
|
|
shift = false;
|
|
ctrl = false;
|
|
}
|
|
if (shift)
|
|
{
|
|
if (!ctrl)
|
|
{
|
|
SetMultipleItemsSelectedSilently(selectedItemIndexStart, selectedItemIndexEnd, false);
|
|
}
|
|
selectedItemIndexEnd = itemIndex;
|
|
SetMultipleItemsSelectedSilently(selectedItemIndexStart, selectedItemIndexEnd, true);
|
|
NotifySelectionChanged();
|
|
}
|
|
else
|
|
{
|
|
if (ctrl)
|
|
{
|
|
vint index = selectedItems.IndexOf(itemIndex);
|
|
if (index == -1)
|
|
{
|
|
selectedItems.Add(itemIndex);
|
|
}
|
|
else
|
|
{
|
|
selectedItems.RemoveAt(index);
|
|
}
|
|
OnItemSelectionChanged(itemIndex, index == -1);
|
|
NotifySelectionChanged();
|
|
}
|
|
else
|
|
{
|
|
selectedItems.Clear();
|
|
OnItemSelectionCleared();
|
|
selectedItems.Add(itemIndex);
|
|
OnItemSelectionChanged(itemIndex, true);
|
|
NotifySelectionChanged();
|
|
}
|
|
selectedItemIndexStart = itemIndex;
|
|
selectedItemIndexEnd = itemIndex;
|
|
}
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool GuiSelectableListControl::SelectItemsByKey(VKEY code, bool ctrl, bool shift)
|
|
{
|
|
if (!GetArranger()) return false;
|
|
|
|
NormalizeSelectedItemIndexStartEnd();
|
|
KeyDirection keyDirection = KeyDirection::Up;
|
|
switch (code)
|
|
{
|
|
case VKEY::_UP:
|
|
keyDirection = KeyDirection::Up;
|
|
break;
|
|
case VKEY::_DOWN:
|
|
keyDirection = KeyDirection::Down;
|
|
break;
|
|
case VKEY::_LEFT:
|
|
keyDirection = KeyDirection::Left;
|
|
break;
|
|
case VKEY::_RIGHT:
|
|
keyDirection = KeyDirection::Right;
|
|
break;
|
|
case VKEY::_HOME:
|
|
keyDirection = KeyDirection::Home;
|
|
break;
|
|
case VKEY::_END:
|
|
keyDirection = KeyDirection::End;
|
|
break;
|
|
case VKEY::_PRIOR:
|
|
keyDirection = KeyDirection::PageUp;
|
|
break;
|
|
case VKEY::_NEXT:
|
|
keyDirection = KeyDirection::PageDown;
|
|
break;
|
|
default:
|
|
return false;
|
|
}
|
|
|
|
if (GetAxis())
|
|
{
|
|
keyDirection = GetAxis()->RealKeyDirectionToVirtualKeyDirection(keyDirection);
|
|
}
|
|
vint itemIndex = GetArranger()->FindItem(selectedItemIndexEnd, keyDirection);
|
|
if (SelectItemsByClick(itemIndex, ctrl, shift, true))
|
|
{
|
|
return EnsureItemVisible(itemIndex);
|
|
}
|
|
else
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
void GuiSelectableListControl::ClearSelection()
|
|
{
|
|
if(selectedItems.Count()>0)
|
|
{
|
|
selectedItems.Clear();
|
|
OnItemSelectionCleared();
|
|
NotifySelectionChanged();
|
|
}
|
|
}
|
|
|
|
namespace list
|
|
{
|
|
|
|
/***********************************************************************
|
|
ItemProviderBase
|
|
***********************************************************************/
|
|
|
|
void ItemProviderBase::InvokeOnItemModified(vint start, vint count, vint newCount)
|
|
{
|
|
CHECK_ERROR(!callingOnItemModified, L"ItemProviderBase::InvokeOnItemModified(vint, vint, vint)#Canning modify the observable data source during its item modified event, which will cause this event to be executed recursively.");
|
|
callingOnItemModified = true;
|
|
for (vint i = 0; i < callbacks.Count(); i++)
|
|
{
|
|
callbacks[i]->OnItemModified(start, count, newCount);
|
|
}
|
|
callingOnItemModified = false;
|
|
}
|
|
|
|
ItemProviderBase::ItemProviderBase()
|
|
{
|
|
}
|
|
|
|
ItemProviderBase::~ItemProviderBase()
|
|
{
|
|
for(vint i=0;i<callbacks.Count();i++)
|
|
{
|
|
callbacks[i]->OnAttached(0);
|
|
}
|
|
}
|
|
|
|
bool ItemProviderBase::AttachCallback(GuiListControl::IItemProviderCallback* value)
|
|
{
|
|
if(callbacks.Contains(value))
|
|
{
|
|
return false;
|
|
}
|
|
else
|
|
{
|
|
callbacks.Add(value);
|
|
value->OnAttached(this);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
bool ItemProviderBase::DetachCallback(GuiListControl::IItemProviderCallback* value)
|
|
{
|
|
vint index=callbacks.IndexOf(value);
|
|
if(index==-1)
|
|
{
|
|
return false;
|
|
}
|
|
else
|
|
{
|
|
value->OnAttached(0);
|
|
callbacks.Remove(value);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
void ItemProviderBase::PushEditing()
|
|
{
|
|
editingCounter++;
|
|
}
|
|
|
|
bool ItemProviderBase::PopEditing()
|
|
{
|
|
if (editingCounter == 0)return false;
|
|
editingCounter--;
|
|
return true;
|
|
}
|
|
|
|
bool ItemProviderBase::IsEditing()
|
|
{
|
|
return editingCounter > 0;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\CONTROLS\LISTCONTROLPACKAGE\GUILISTVIEWCONTROLS.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace controls
|
|
{
|
|
using namespace elements;
|
|
using namespace compositions;
|
|
using namespace collections;
|
|
using namespace reflection::description;
|
|
|
|
/***********************************************************************
|
|
GuiListViewColumnHeader
|
|
***********************************************************************/
|
|
|
|
void GuiListViewColumnHeader::BeforeControlTemplateUninstalled_()
|
|
{
|
|
}
|
|
|
|
void GuiListViewColumnHeader::AfterControlTemplateInstalled_(bool initialize)
|
|
{
|
|
GetControlTemplateObject(true)->SetSortingState(columnSortingState);
|
|
}
|
|
|
|
GuiListViewColumnHeader::GuiListViewColumnHeader(theme::ThemeName themeName)
|
|
:GuiMenuButton(themeName)
|
|
{
|
|
}
|
|
|
|
GuiListViewColumnHeader::~GuiListViewColumnHeader()
|
|
{
|
|
}
|
|
|
|
bool GuiListViewColumnHeader::IsAltAvailable()
|
|
{
|
|
return false;
|
|
}
|
|
|
|
ColumnSortingState GuiListViewColumnHeader::GetColumnSortingState()
|
|
{
|
|
return columnSortingState;
|
|
}
|
|
|
|
void GuiListViewColumnHeader::SetColumnSortingState(ColumnSortingState value)
|
|
{
|
|
if (columnSortingState != value)
|
|
{
|
|
columnSortingState = value;
|
|
GetControlTemplateObject(true)->SetSortingState(columnSortingState);
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiListViewBase
|
|
***********************************************************************/
|
|
|
|
void GuiListViewBase::BeforeControlTemplateUninstalled_()
|
|
{
|
|
}
|
|
|
|
void GuiListViewBase::AfterControlTemplateInstalled_(bool initialize)
|
|
{
|
|
}
|
|
|
|
GuiListViewBase::GuiListViewBase(theme::ThemeName themeName, GuiListControl::IItemProvider* _itemProvider)
|
|
:GuiSelectableListControl(themeName, _itemProvider)
|
|
{
|
|
ColumnClicked.SetAssociatedComposition(boundsComposition);
|
|
}
|
|
|
|
GuiListViewBase::~GuiListViewBase()
|
|
{
|
|
}
|
|
|
|
namespace list
|
|
{
|
|
|
|
const wchar_t* const IListViewItemView::Identifier = L"vl::presentation::controls::list::IListViewItemView";
|
|
|
|
/***********************************************************************
|
|
ListViewColumnItemArranger::ColumnItemViewCallback
|
|
***********************************************************************/
|
|
|
|
ListViewColumnItemArranger::ColumnItemViewCallback::ColumnItemViewCallback(ListViewColumnItemArranger* _arranger)
|
|
:arranger(_arranger)
|
|
{
|
|
}
|
|
|
|
ListViewColumnItemArranger::ColumnItemViewCallback::~ColumnItemViewCallback()
|
|
{
|
|
}
|
|
|
|
void ListViewColumnItemArranger::ColumnItemViewCallback::OnColumnChanged()
|
|
{
|
|
arranger->RebuildColumns();
|
|
FOREACH(ItemStyleRecord, style, arranger->visibleStyles)
|
|
{
|
|
if (auto callback = dynamic_cast<IColumnItemViewCallback*>(style.key))
|
|
{
|
|
callback->OnColumnChanged();
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
ListViewColumnItemArranger
|
|
***********************************************************************/
|
|
|
|
const wchar_t* const ListViewColumnItemArranger::IColumnItemView::Identifier = L"vl::presentation::controls::list::ListViewColumnItemArranger::IColumnItemView";
|
|
|
|
void ListViewColumnItemArranger::ColumnClicked(vint index, compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
GuiItemEventArgs args(listView->ColumnClicked.GetAssociatedComposition());
|
|
args.itemIndex=index;
|
|
listView->ColumnClicked.Execute(args);
|
|
}
|
|
|
|
void ListViewColumnItemArranger::ColumnBoundsChanged(vint index, compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
GuiBoundsComposition* buttonBounds=columnHeaderButtons[index]->GetBoundsComposition();
|
|
vint size=buttonBounds->GetBounds().Width();
|
|
if(size>columnItemView->GetColumnSize(index))
|
|
{
|
|
columnItemView->SetColumnSize(index, size);
|
|
}
|
|
}
|
|
|
|
void ListViewColumnItemArranger::ColumnHeaderSplitterLeftButtonDown(compositions::GuiGraphicsComposition* sender, compositions::GuiMouseEventArgs& arguments)
|
|
{
|
|
if(listView->GetVisuallyEnabled())
|
|
{
|
|
arguments.handled=true;
|
|
splitterDragging=true;
|
|
splitterLatestX=arguments.x;
|
|
}
|
|
}
|
|
|
|
void ListViewColumnItemArranger::ColumnHeaderSplitterLeftButtonUp(compositions::GuiGraphicsComposition* sender, compositions::GuiMouseEventArgs& arguments)
|
|
{
|
|
if(listView->GetVisuallyEnabled())
|
|
{
|
|
arguments.handled=true;
|
|
splitterDragging=false;
|
|
splitterLatestX=0;
|
|
}
|
|
}
|
|
|
|
void ListViewColumnItemArranger::ColumnHeaderSplitterMouseMove(compositions::GuiGraphicsComposition* sender, compositions::GuiMouseEventArgs& arguments)
|
|
{
|
|
if(splitterDragging)
|
|
{
|
|
vint offset=arguments.x-splitterLatestX;
|
|
vint index=columnHeaderSplitters.IndexOf(dynamic_cast<GuiBoundsComposition*>(sender));
|
|
if(index!=-1)
|
|
{
|
|
GuiBoundsComposition* buttonBounds=columnHeaderButtons[index]->GetBoundsComposition();
|
|
Rect bounds=buttonBounds->GetBounds();
|
|
Rect newBounds(bounds.LeftTop(), Size(bounds.Width()+offset, bounds.Height()));
|
|
buttonBounds->SetBounds(newBounds);
|
|
|
|
vint finalSize=buttonBounds->GetBounds().Width();
|
|
columnItemView->SetColumnSize(index, finalSize);
|
|
}
|
|
}
|
|
}
|
|
|
|
void ListViewColumnItemArranger::RearrangeItemBounds()
|
|
{
|
|
FixedHeightItemArranger::RearrangeItemBounds();
|
|
vint count = columnHeaders->GetParent()->Children().Count();
|
|
columnHeaders->GetParent()->MoveChild(columnHeaders, count - 1);
|
|
columnHeaders->SetBounds(Rect(Point(-viewBounds.Left(), 0), Size(0, 0)));
|
|
}
|
|
|
|
vint ListViewColumnItemArranger::GetWidth()
|
|
{
|
|
vint width=columnHeaders->GetBounds().Width()-SplitterWidth;
|
|
if(width<SplitterWidth)
|
|
{
|
|
width=SplitterWidth;
|
|
}
|
|
return width;
|
|
}
|
|
|
|
vint ListViewColumnItemArranger::GetYOffset()
|
|
{
|
|
return columnHeaders->GetBounds().Height();
|
|
}
|
|
|
|
Size ListViewColumnItemArranger::OnCalculateTotalSize()
|
|
{
|
|
Size size=FixedHeightItemArranger::OnCalculateTotalSize();
|
|
size.x+=SplitterWidth;
|
|
return size;
|
|
}
|
|
|
|
void ListViewColumnItemArranger::DeleteColumnButtons()
|
|
{
|
|
for(vint i=columnHeaders->GetStackItems().Count()-1;i>=0;i--)
|
|
{
|
|
GuiStackItemComposition* item=columnHeaders->GetStackItems().Get(i);
|
|
columnHeaders->RemoveChild(item);
|
|
|
|
GuiControl* button=item->Children().Get(0)->GetAssociatedControl();
|
|
if(button)
|
|
{
|
|
item->RemoveChild(button->GetBoundsComposition());
|
|
delete button;
|
|
}
|
|
delete item;
|
|
}
|
|
columnHeaderButtons.Clear();
|
|
columnHeaderSplitters.Clear();
|
|
}
|
|
|
|
void ListViewColumnItemArranger::RebuildColumns()
|
|
{
|
|
if (columnItemView && columnHeaderButtons.Count() == listViewItemView->GetColumnCount())
|
|
{
|
|
for (vint i = 0; i < listViewItemView->GetColumnCount(); i++)
|
|
{
|
|
GuiListViewColumnHeader* button = columnHeaderButtons[i];
|
|
button->SetText(listViewItemView->GetColumnText(i));
|
|
button->SetSubMenu(columnItemView->GetDropdownPopup(i), false);
|
|
button->SetColumnSortingState(columnItemView->GetSortingState(i));
|
|
button->GetBoundsComposition()->SetBounds(Rect(Point(0, 0), Size(columnItemView->GetColumnSize(i), 0)));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
DeleteColumnButtons();
|
|
if (columnItemView && listViewItemView)
|
|
{
|
|
for (vint i = 0; i < listViewItemView->GetColumnCount(); i++)
|
|
{
|
|
GuiBoundsComposition* splitterComposition = new GuiBoundsComposition;
|
|
splitterComposition->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
splitterComposition->SetAssociatedCursor(GetCurrentController()->ResourceService()->GetSystemCursor(INativeCursor::SizeWE));
|
|
splitterComposition->SetAlignmentToParent(Margin(0, 0, -1, 0));
|
|
splitterComposition->SetPreferredMinSize(Size(SplitterWidth, 0));
|
|
columnHeaderSplitters.Add(splitterComposition);
|
|
|
|
splitterComposition->GetEventReceiver()->leftButtonDown.AttachMethod(this, &ListViewColumnItemArranger::ColumnHeaderSplitterLeftButtonDown);
|
|
splitterComposition->GetEventReceiver()->leftButtonUp.AttachMethod(this, &ListViewColumnItemArranger::ColumnHeaderSplitterLeftButtonUp);
|
|
splitterComposition->GetEventReceiver()->mouseMove.AttachMethod(this, &ListViewColumnItemArranger::ColumnHeaderSplitterMouseMove);
|
|
}
|
|
for (vint i = 0; i < listViewItemView->GetColumnCount(); i++)
|
|
{
|
|
GuiListViewColumnHeader* button = new GuiListViewColumnHeader(theme::ThemeName::Unknown);
|
|
button->SetAutoFocus(false);
|
|
button->SetControlTemplate(listView->GetControlTemplateObject(true)->GetColumnHeaderTemplate());
|
|
button->SetText(listViewItemView->GetColumnText(i));
|
|
button->SetSubMenu(columnItemView->GetDropdownPopup(i), false);
|
|
button->SetColumnSortingState(columnItemView->GetSortingState(i));
|
|
button->GetBoundsComposition()->SetBounds(Rect(Point(0, 0), Size(columnItemView->GetColumnSize(i), 0)));
|
|
button->Clicked.AttachLambda(Curry(Func<void(vint, GuiGraphicsComposition*, GuiEventArgs&)>(this, &ListViewColumnItemArranger::ColumnClicked))(i));
|
|
button->GetBoundsComposition()->BoundsChanged.AttachLambda(Curry(Func<void(vint, GuiGraphicsComposition*, GuiEventArgs&)>(this, &ListViewColumnItemArranger::ColumnBoundsChanged))(i));
|
|
columnHeaderButtons.Add(button);
|
|
if (i > 0)
|
|
{
|
|
button->GetContainerComposition()->AddChild(columnHeaderSplitters[i - 1]);
|
|
}
|
|
|
|
GuiStackItemComposition* item = new GuiStackItemComposition;
|
|
item->SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
item->AddChild(button->GetBoundsComposition());
|
|
columnHeaders->AddChild(item);
|
|
}
|
|
if (listViewItemView->GetColumnCount() > 0)
|
|
{
|
|
GuiBoundsComposition* splitterComposition = columnHeaderSplitters[listViewItemView->GetColumnCount() - 1];
|
|
|
|
GuiStackItemComposition* item = new GuiStackItemComposition;
|
|
item->SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
item->AddChild(splitterComposition);
|
|
columnHeaders->AddChild(item);
|
|
}
|
|
}
|
|
}
|
|
callback->OnTotalSizeChanged();
|
|
}
|
|
|
|
ListViewColumnItemArranger::ListViewColumnItemArranger()
|
|
{
|
|
columnHeaders = new GuiStackComposition;
|
|
columnHeaders->SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
columnItemViewCallback = new ColumnItemViewCallback(this);
|
|
}
|
|
|
|
ListViewColumnItemArranger::~ListViewColumnItemArranger()
|
|
{
|
|
if(!columnHeaders->GetParent())
|
|
{
|
|
DeleteColumnButtons();
|
|
delete columnHeaders;
|
|
}
|
|
}
|
|
|
|
void ListViewColumnItemArranger::AttachListControl(GuiListControl* value)
|
|
{
|
|
FixedHeightItemArranger::AttachListControl(value);
|
|
listView = dynamic_cast<GuiListViewBase*>(value);
|
|
if (listView)
|
|
{
|
|
listViewItemView = dynamic_cast<IListViewItemView*>(listView->GetItemProvider()->RequestView(IListViewItemView::Identifier));
|
|
columnItemView = dynamic_cast<IColumnItemView*>(listView->GetItemProvider()->RequestView(IColumnItemView::Identifier));
|
|
listView->GetContainerComposition()->AddChild(columnHeaders);
|
|
if (columnItemView)
|
|
{
|
|
columnItemView->AttachCallback(columnItemViewCallback.Obj());
|
|
RebuildColumns();
|
|
}
|
|
}
|
|
}
|
|
|
|
void ListViewColumnItemArranger::DetachListControl()
|
|
{
|
|
if (listView)
|
|
{
|
|
if (columnItemView)
|
|
{
|
|
columnItemView->DetachCallback(columnItemViewCallback.Obj());
|
|
columnItemView = nullptr;
|
|
}
|
|
listViewItemView = nullptr;
|
|
listView->GetContainerComposition()->RemoveChild(columnHeaders);
|
|
listView = nullptr;
|
|
}
|
|
FixedHeightItemArranger::DetachListControl();
|
|
}
|
|
|
|
/***********************************************************************
|
|
ListViewSubItems
|
|
***********************************************************************/
|
|
|
|
void ListViewSubItems::NotifyUpdateInternal(vint start, vint count, vint newCount)
|
|
{
|
|
owner->NotifyUpdate();
|
|
}
|
|
|
|
/***********************************************************************
|
|
ListViewItem
|
|
***********************************************************************/
|
|
|
|
void ListViewItem::NotifyUpdate()
|
|
{
|
|
if (owner)
|
|
{
|
|
vint index = owner->IndexOf(this);
|
|
owner->NotifyUpdateInternal(index, 1, 1);
|
|
}
|
|
}
|
|
|
|
ListViewItem::ListViewItem()
|
|
:owner(0)
|
|
{
|
|
subItems.owner = this;
|
|
}
|
|
|
|
ListViewSubItems& ListViewItem::GetSubItems()
|
|
{
|
|
return subItems;
|
|
}
|
|
|
|
Ptr<GuiImageData> ListViewItem::GetSmallImage()
|
|
{
|
|
return smallImage;
|
|
}
|
|
|
|
void ListViewItem::SetSmallImage(Ptr<GuiImageData> value)
|
|
{
|
|
smallImage = value;
|
|
NotifyUpdate();
|
|
}
|
|
|
|
Ptr<GuiImageData> ListViewItem::GetLargeImage()
|
|
{
|
|
return largeImage;
|
|
}
|
|
|
|
void ListViewItem::SetLargeImage(Ptr<GuiImageData> value)
|
|
{
|
|
largeImage = value;
|
|
NotifyUpdate();
|
|
}
|
|
|
|
const WString& ListViewItem::GetText()
|
|
{
|
|
return text;
|
|
}
|
|
|
|
void ListViewItem::SetText(const WString& value)
|
|
{
|
|
text = value;
|
|
NotifyUpdate();
|
|
}
|
|
|
|
description::Value ListViewItem::GetTag()
|
|
{
|
|
return tag;
|
|
}
|
|
|
|
void ListViewItem::SetTag(const description::Value& value)
|
|
{
|
|
tag = value;
|
|
NotifyUpdate();
|
|
}
|
|
|
|
/***********************************************************************
|
|
ListViewColumn
|
|
***********************************************************************/
|
|
|
|
void ListViewColumn::NotifyUpdate(bool affectItem)
|
|
{
|
|
if (owner)
|
|
{
|
|
vint index = owner->IndexOf(this);
|
|
owner->NotifyColumnUpdated(index, affectItem);
|
|
}
|
|
}
|
|
|
|
ListViewColumn::ListViewColumn(const WString& _text, vint _size)
|
|
:text(_text)
|
|
,size(_size)
|
|
{
|
|
}
|
|
|
|
ListViewColumn::~ListViewColumn()
|
|
{
|
|
if (dropdownPopup && ownPopup)
|
|
{
|
|
SafeDeleteControl(dropdownPopup);
|
|
}
|
|
}
|
|
|
|
const WString& ListViewColumn::GetText()
|
|
{
|
|
return text;
|
|
}
|
|
|
|
void ListViewColumn::SetText(const WString& value)
|
|
{
|
|
if (text != value)
|
|
{
|
|
text = value;
|
|
NotifyUpdate(false);
|
|
}
|
|
}
|
|
|
|
ItemProperty<WString> ListViewColumn::GetTextProperty()
|
|
{
|
|
return textProperty;
|
|
}
|
|
|
|
void ListViewColumn::SetTextProperty(const ItemProperty<WString>& value)
|
|
{
|
|
textProperty = value;
|
|
NotifyUpdate(true);
|
|
}
|
|
|
|
vint ListViewColumn::GetSize()
|
|
{
|
|
return size;
|
|
}
|
|
|
|
void ListViewColumn::SetSize(vint value)
|
|
{
|
|
if (size != value)
|
|
{
|
|
size = value;
|
|
NotifyUpdate(false);
|
|
}
|
|
}
|
|
|
|
bool ListViewColumn::GetOwnPopup()
|
|
{
|
|
return ownPopup;
|
|
}
|
|
|
|
void ListViewColumn::SetOwnPopup(bool value)
|
|
{
|
|
ownPopup = value;
|
|
}
|
|
|
|
GuiMenu* ListViewColumn::GetDropdownPopup()
|
|
{
|
|
return dropdownPopup;
|
|
}
|
|
|
|
void ListViewColumn::SetDropdownPopup(GuiMenu* value)
|
|
{
|
|
if (dropdownPopup != value)
|
|
{
|
|
dropdownPopup = value;
|
|
NotifyUpdate(false);
|
|
}
|
|
}
|
|
|
|
ColumnSortingState ListViewColumn::GetSortingState()
|
|
{
|
|
return sortingState;
|
|
}
|
|
|
|
void ListViewColumn::SetSortingState(ColumnSortingState value)
|
|
{
|
|
if (sortingState != value)
|
|
{
|
|
sortingState = value;
|
|
NotifyUpdate(false);
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
ListViewDataColumns
|
|
***********************************************************************/
|
|
|
|
void ListViewDataColumns::NotifyUpdateInternal(vint start, vint count, vint newCount)
|
|
{
|
|
itemProvider->NotifyAllItemsUpdate();
|
|
}
|
|
|
|
ListViewDataColumns::ListViewDataColumns(IListViewItemProvider* _itemProvider)
|
|
:itemProvider(_itemProvider)
|
|
{
|
|
}
|
|
|
|
ListViewDataColumns::~ListViewDataColumns()
|
|
{
|
|
}
|
|
|
|
/***********************************************************************
|
|
ListViewColumns
|
|
***********************************************************************/
|
|
|
|
void ListViewColumns::NotifyColumnUpdated(vint column, bool affectItem)
|
|
{
|
|
affectItemFlag = affectItem;
|
|
NotifyUpdate(column, 1);
|
|
affectItemFlag = true;
|
|
}
|
|
|
|
void ListViewColumns::AfterInsert(vint index, const Ptr<ListViewColumn>& value)
|
|
{
|
|
collections::ObservableListBase<Ptr<ListViewColumn>>::AfterInsert(index, value);
|
|
value->owner = this;
|
|
}
|
|
|
|
void ListViewColumns::BeforeRemove(vint index, const Ptr<ListViewColumn>& value)
|
|
{
|
|
value->owner = 0;
|
|
collections::ObservableListBase<Ptr<ListViewColumn>>::BeforeRemove(index, value);
|
|
}
|
|
|
|
void ListViewColumns::NotifyUpdateInternal(vint start, vint count, vint newCount)
|
|
{
|
|
itemProvider->NotifyAllColumnsUpdate();
|
|
if (affectItemFlag)
|
|
{
|
|
itemProvider->NotifyAllItemsUpdate();
|
|
}
|
|
}
|
|
|
|
ListViewColumns::ListViewColumns(IListViewItemProvider* _itemProvider)
|
|
:itemProvider(_itemProvider)
|
|
{
|
|
}
|
|
|
|
ListViewColumns::~ListViewColumns()
|
|
{
|
|
}
|
|
|
|
/***********************************************************************
|
|
ListViewItemProvider
|
|
***********************************************************************/
|
|
|
|
void ListViewItemProvider::AfterInsert(vint index, const Ptr<ListViewItem>& value)
|
|
{
|
|
ListProvider<Ptr<ListViewItem>>::AfterInsert(index, value);
|
|
value->owner = this;
|
|
}
|
|
|
|
void ListViewItemProvider::BeforeRemove(vint index, const Ptr<ListViewItem>& value)
|
|
{
|
|
value->owner = 0;
|
|
ListProvider<Ptr<ListViewItem>>::AfterInsert(index, value);
|
|
}
|
|
|
|
void ListViewItemProvider::NotifyAllItemsUpdate()
|
|
{
|
|
NotifyUpdate(0, Count());
|
|
}
|
|
|
|
void ListViewItemProvider::NotifyAllColumnsUpdate()
|
|
{
|
|
for (vint i = 0; i < columnItemViewCallbacks.Count(); i++)
|
|
{
|
|
columnItemViewCallbacks[i]->OnColumnChanged();
|
|
}
|
|
}
|
|
|
|
Ptr<GuiImageData> ListViewItemProvider::GetSmallImage(vint itemIndex)
|
|
{
|
|
return Get(itemIndex)->smallImage;
|
|
}
|
|
|
|
Ptr<GuiImageData> ListViewItemProvider::GetLargeImage(vint itemIndex)
|
|
{
|
|
return Get(itemIndex)->largeImage;
|
|
}
|
|
|
|
WString ListViewItemProvider::GetText(vint itemIndex)
|
|
{
|
|
return Get(itemIndex)->text;
|
|
}
|
|
|
|
WString ListViewItemProvider::GetSubItem(vint itemIndex, vint index)
|
|
{
|
|
Ptr<ListViewItem> item=Get(itemIndex);
|
|
if(index<0 || index>=item->GetSubItems().Count())
|
|
{
|
|
return L"";
|
|
}
|
|
else
|
|
{
|
|
return item->GetSubItems()[index];
|
|
}
|
|
}
|
|
|
|
vint ListViewItemProvider::GetDataColumnCount()
|
|
{
|
|
return dataColumns.Count();
|
|
}
|
|
|
|
vint ListViewItemProvider::GetDataColumn(vint index)
|
|
{
|
|
return dataColumns[index];
|
|
}
|
|
|
|
vint ListViewItemProvider::GetColumnCount()
|
|
{
|
|
return columns.Count();
|
|
}
|
|
|
|
WString ListViewItemProvider::GetColumnText(vint index)
|
|
{
|
|
if (index<0 || index >= columns.Count())
|
|
{
|
|
return L"";
|
|
}
|
|
else
|
|
{
|
|
return columns[index]->GetText();
|
|
}
|
|
}
|
|
|
|
bool ListViewItemProvider::AttachCallback(ListViewColumnItemArranger::IColumnItemViewCallback* value)
|
|
{
|
|
if(columnItemViewCallbacks.Contains(value))
|
|
{
|
|
return false;
|
|
}
|
|
else
|
|
{
|
|
columnItemViewCallbacks.Add(value);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
bool ListViewItemProvider::DetachCallback(ListViewColumnItemArranger::IColumnItemViewCallback* value)
|
|
{
|
|
vint index=columnItemViewCallbacks.IndexOf(value);
|
|
if(index==-1)
|
|
{
|
|
return false;
|
|
}
|
|
else
|
|
{
|
|
columnItemViewCallbacks.Remove(value);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
vint ListViewItemProvider::GetColumnSize(vint index)
|
|
{
|
|
if(index<0 || index>=columns.Count())
|
|
{
|
|
return 0;
|
|
}
|
|
else
|
|
{
|
|
return columns[index]->GetSize();
|
|
}
|
|
}
|
|
|
|
void ListViewItemProvider::SetColumnSize(vint index, vint value)
|
|
{
|
|
if(index>=0 && index<columns.Count())
|
|
{
|
|
columns[index]->SetSize(value);
|
|
}
|
|
}
|
|
|
|
GuiMenu* ListViewItemProvider::GetDropdownPopup(vint index)
|
|
{
|
|
if(index<0 || index>=columns.Count())
|
|
{
|
|
return 0;
|
|
}
|
|
else
|
|
{
|
|
return columns[index]->GetDropdownPopup();
|
|
}
|
|
}
|
|
|
|
ColumnSortingState ListViewItemProvider::GetSortingState(vint index)
|
|
{
|
|
if (index < 0 || index >= columns.Count())
|
|
{
|
|
return ColumnSortingState::NotSorted;
|
|
}
|
|
else
|
|
{
|
|
return columns[index]->GetSortingState();
|
|
}
|
|
}
|
|
|
|
WString ListViewItemProvider::GetTextValue(vint itemIndex)
|
|
{
|
|
return GetText(itemIndex);
|
|
}
|
|
|
|
description::Value ListViewItemProvider::GetBindingValue(vint itemIndex)
|
|
{
|
|
return Value::From(Get(itemIndex));
|
|
}
|
|
|
|
ListViewItemProvider::ListViewItemProvider()
|
|
:columns(this)
|
|
, dataColumns(this)
|
|
{
|
|
}
|
|
|
|
ListViewItemProvider::~ListViewItemProvider()
|
|
{
|
|
}
|
|
|
|
IDescriptable* ListViewItemProvider::RequestView(const WString& identifier)
|
|
{
|
|
if (identifier == IListViewItemView::Identifier)
|
|
{
|
|
return (IListViewItemView*)this;
|
|
}
|
|
else if (identifier == ListViewColumnItemArranger::IColumnItemView::Identifier)
|
|
{
|
|
return (ListViewColumnItemArranger::IColumnItemView*)this;
|
|
}
|
|
else
|
|
{
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
ListViewDataColumns& ListViewItemProvider::GetDataColumns()
|
|
{
|
|
return dataColumns;
|
|
}
|
|
|
|
ListViewColumns& ListViewItemProvider::GetColumns()
|
|
{
|
|
return columns;
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiListView
|
|
***********************************************************************/
|
|
|
|
void GuiVirtualListView::OnStyleInstalled(vint itemIndex, ItemStyle* style)
|
|
{
|
|
GuiListViewBase::OnStyleInstalled(itemIndex, style);
|
|
}
|
|
|
|
void GuiVirtualListView::OnItemTemplateChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
view = ListViewView::Unknown;
|
|
}
|
|
|
|
GuiVirtualListView::GuiVirtualListView(theme::ThemeName themeName, GuiListControl::IItemProvider* _itemProvider)
|
|
:GuiListViewBase(themeName, _itemProvider)
|
|
{
|
|
SetView(ListViewView::Detail);
|
|
}
|
|
|
|
GuiVirtualListView::~GuiVirtualListView()
|
|
{
|
|
}
|
|
|
|
ListViewView GuiVirtualListView::GetView()
|
|
{
|
|
return view;
|
|
}
|
|
|
|
void GuiVirtualListView::SetView(ListViewView _view)
|
|
{
|
|
switch (_view)
|
|
{
|
|
case ListViewView::BigIcon:
|
|
SetStyleAndArranger(
|
|
[](const Value&) { return new list::BigIconListViewItemTemplate; },
|
|
new list::FixedSizeMultiColumnItemArranger
|
|
);
|
|
break;
|
|
case ListViewView::SmallIcon:
|
|
SetStyleAndArranger(
|
|
[](const Value&) { return new list::SmallIconListViewItemTemplate; },
|
|
new list::FixedSizeMultiColumnItemArranger
|
|
);
|
|
break;
|
|
case ListViewView::List:
|
|
SetStyleAndArranger(
|
|
[](const Value&) { return new list::ListListViewItemTemplate; },
|
|
new list::FixedHeightMultiColumnItemArranger
|
|
);
|
|
break;
|
|
case ListViewView::Tile:
|
|
SetStyleAndArranger(
|
|
[](const Value&) { return new list::TileListViewItemTemplate; },
|
|
new list::FixedSizeMultiColumnItemArranger
|
|
);
|
|
break;
|
|
case ListViewView::Information:
|
|
SetStyleAndArranger(
|
|
[](const Value&) { return new list::InformationListViewItemTemplate; },
|
|
new list::FixedHeightItemArranger
|
|
);
|
|
break;
|
|
case ListViewView::Detail:
|
|
SetStyleAndArranger(
|
|
[](const Value&) { return new list::DetailListViewItemTemplate; },
|
|
new list::ListViewColumnItemArranger
|
|
);
|
|
break;
|
|
default:;
|
|
}
|
|
view = _view;
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiListView
|
|
***********************************************************************/
|
|
|
|
GuiListView::GuiListView(theme::ThemeName themeName)
|
|
:GuiVirtualListView(themeName, new list::ListViewItemProvider)
|
|
{
|
|
items=dynamic_cast<list::ListViewItemProvider*>(itemProvider.Obj());
|
|
}
|
|
|
|
GuiListView::~GuiListView()
|
|
{
|
|
}
|
|
|
|
list::ListViewItemProvider& GuiListView::GetItems()
|
|
{
|
|
return *items;
|
|
}
|
|
|
|
list::ListViewDataColumns& GuiListView::GetDataColumns()
|
|
{
|
|
return items->GetDataColumns();
|
|
}
|
|
|
|
list::ListViewColumns& GuiListView::GetColumns()
|
|
{
|
|
return items->GetColumns();
|
|
}
|
|
|
|
Ptr<list::ListViewItem> GuiListView::GetSelectedItem()
|
|
{
|
|
vint index = GetSelectedItemIndex();
|
|
if (index == -1) return 0;
|
|
return items->Get(index);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\CONTROLS\LISTCONTROLPACKAGE\GUILISTVIEWITEMTEMPLATES.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace controls
|
|
{
|
|
using namespace elements;
|
|
using namespace compositions;
|
|
using namespace collections;
|
|
using namespace reflection::description;
|
|
|
|
namespace list
|
|
{
|
|
|
|
/***********************************************************************
|
|
DefaultListViewItemTemplate
|
|
***********************************************************************/
|
|
|
|
DefaultListViewItemTemplate::DefaultListViewItemTemplate()
|
|
{
|
|
SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
}
|
|
|
|
DefaultListViewItemTemplate::~DefaultListViewItemTemplate()
|
|
{
|
|
}
|
|
|
|
/***********************************************************************
|
|
BigIconListViewItemTemplate
|
|
***********************************************************************/
|
|
|
|
void BigIconListViewItemTemplate::OnInitialize()
|
|
{
|
|
DefaultListViewItemTemplate::OnInitialize();
|
|
{
|
|
auto table = new GuiTableComposition;
|
|
AddChild(table);
|
|
table->SetRowsAndColumns(2, 3);
|
|
table->SetRowOption(0, GuiCellOption::MinSizeOption());
|
|
table->SetRowOption(1, GuiCellOption::MinSizeOption());
|
|
table->SetColumnOption(0, GuiCellOption::PercentageOption(0.5));
|
|
table->SetColumnOption(1, GuiCellOption::MinSizeOption());
|
|
table->SetColumnOption(2, GuiCellOption::PercentageOption(0.5));
|
|
table->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
table->SetCellPadding(5);
|
|
{
|
|
auto cell = new GuiCellComposition;
|
|
table->AddChild(cell);
|
|
cell->SetSite(0, 1, 1, 1);
|
|
cell->SetPreferredMinSize(Size(32, 32));
|
|
|
|
image = GuiImageFrameElement::Create();
|
|
image->SetStretch(true);
|
|
cell->SetOwnedElement(image);
|
|
}
|
|
{
|
|
auto cell = new GuiCellComposition;
|
|
table->AddChild(cell);
|
|
cell->SetMinSizeLimitation(GuiGraphicsComposition::NoLimit);
|
|
cell->SetSite(1, 0, 1, 3);
|
|
cell->SetPreferredMinSize(Size(64, 40));
|
|
|
|
text = GuiSolidLabelElement::Create();
|
|
text->SetAlignments(Alignment::Center, Alignment::Top);
|
|
text->SetWrapLine(true);
|
|
text->SetEllipse(true);
|
|
cell->SetOwnedElement(text);
|
|
}
|
|
}
|
|
|
|
if (auto listView = dynamic_cast<GuiVirtualListView*>(listControl))
|
|
{
|
|
auto itemIndex = GetIndex();
|
|
if (auto view = dynamic_cast<IListViewItemView*>(listView->GetItemProvider()->RequestView(IListViewItemView::Identifier)))
|
|
{
|
|
auto imageData = view->GetLargeImage(itemIndex);
|
|
if (imageData)
|
|
{
|
|
image->SetImage(imageData->GetImage(), imageData->GetFrameIndex());
|
|
}
|
|
else
|
|
{
|
|
image->SetImage(nullptr);
|
|
}
|
|
text->SetText(view->GetText(itemIndex));
|
|
text->SetColor(listView->GetControlTemplateObject(true)->GetPrimaryTextColor());
|
|
}
|
|
}
|
|
|
|
FontChanged.AttachMethod(this, &BigIconListViewItemTemplate::OnFontChanged);
|
|
|
|
FontChanged.Execute(compositions::GuiEventArgs(this));
|
|
}
|
|
|
|
void BigIconListViewItemTemplate::OnFontChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
text->SetFont(GetFont());
|
|
}
|
|
|
|
BigIconListViewItemTemplate::BigIconListViewItemTemplate()
|
|
{
|
|
}
|
|
|
|
BigIconListViewItemTemplate::~BigIconListViewItemTemplate()
|
|
{
|
|
}
|
|
|
|
/***********************************************************************
|
|
SmallIconListViewItemTemplate
|
|
***********************************************************************/
|
|
|
|
void SmallIconListViewItemTemplate::OnInitialize()
|
|
{
|
|
DefaultListViewItemTemplate::OnInitialize();
|
|
{
|
|
auto table = new GuiTableComposition;
|
|
AddChild(table);
|
|
table->SetRowsAndColumns(3, 2);
|
|
table->SetRowOption(0, GuiCellOption::PercentageOption(0.5));
|
|
table->SetRowOption(1, GuiCellOption::MinSizeOption());
|
|
table->SetRowOption(2, GuiCellOption::PercentageOption(0.5));
|
|
table->SetColumnOption(0, GuiCellOption::MinSizeOption());
|
|
table->SetColumnOption(1, GuiCellOption::MinSizeOption());
|
|
table->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
table->SetCellPadding(2);
|
|
{
|
|
GuiCellComposition* cell = new GuiCellComposition;
|
|
table->AddChild(cell);
|
|
cell->SetSite(1, 0, 1, 1);
|
|
cell->SetPreferredMinSize(Size(16, 16));
|
|
|
|
image = GuiImageFrameElement::Create();
|
|
image->SetStretch(true);
|
|
cell->SetOwnedElement(image);
|
|
}
|
|
{
|
|
GuiCellComposition* cell = new GuiCellComposition;
|
|
table->AddChild(cell);
|
|
cell->SetSite(0, 1, 3, 1);
|
|
cell->SetPreferredMinSize(Size(192, 0));
|
|
|
|
text = GuiSolidLabelElement::Create();
|
|
text->SetAlignments(Alignment::Left, Alignment::Center);
|
|
text->SetEllipse(true);
|
|
cell->SetOwnedElement(text);
|
|
}
|
|
}
|
|
|
|
if (auto listView = dynamic_cast<GuiVirtualListView*>(listControl))
|
|
{
|
|
auto itemIndex = GetIndex();
|
|
if (auto view = dynamic_cast<IListViewItemView*>(listView->GetItemProvider()->RequestView(IListViewItemView::Identifier)))
|
|
{
|
|
auto imageData = view->GetSmallImage(itemIndex);
|
|
if (imageData)
|
|
{
|
|
image->SetImage(imageData->GetImage(), imageData->GetFrameIndex());
|
|
}
|
|
else
|
|
{
|
|
image->SetImage(nullptr);
|
|
}
|
|
text->SetText(view->GetText(itemIndex));
|
|
text->SetColor(listView->GetControlTemplateObject(true)->GetPrimaryTextColor());
|
|
}
|
|
}
|
|
|
|
FontChanged.AttachMethod(this, &SmallIconListViewItemTemplate::OnFontChanged);
|
|
|
|
FontChanged.Execute(compositions::GuiEventArgs(this));
|
|
}
|
|
|
|
void SmallIconListViewItemTemplate::OnFontChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
text->SetFont(GetFont());
|
|
}
|
|
|
|
SmallIconListViewItemTemplate::SmallIconListViewItemTemplate()
|
|
{
|
|
}
|
|
|
|
SmallIconListViewItemTemplate::~SmallIconListViewItemTemplate()
|
|
{
|
|
}
|
|
|
|
/***********************************************************************
|
|
ListListViewItemTemplate
|
|
***********************************************************************/
|
|
|
|
void ListListViewItemTemplate::OnInitialize()
|
|
{
|
|
DefaultListViewItemTemplate::OnInitialize();
|
|
{
|
|
auto table = new GuiTableComposition;
|
|
AddChild(table);
|
|
table->SetRowsAndColumns(3, 2);
|
|
table->SetRowOption(0, GuiCellOption::PercentageOption(0.5));
|
|
table->SetRowOption(1, GuiCellOption::MinSizeOption());
|
|
table->SetRowOption(2, GuiCellOption::PercentageOption(0.5));
|
|
table->SetColumnOption(0, GuiCellOption::MinSizeOption());
|
|
table->SetColumnOption(1, GuiCellOption::MinSizeOption());
|
|
table->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
table->SetCellPadding(2);
|
|
{
|
|
auto cell = new GuiCellComposition;
|
|
table->AddChild(cell);
|
|
cell->SetSite(1, 0, 1, 1);
|
|
cell->SetPreferredMinSize(Size(16, 16));
|
|
|
|
image = GuiImageFrameElement::Create();
|
|
image->SetStretch(true);
|
|
cell->SetOwnedElement(image);
|
|
}
|
|
{
|
|
auto cell = new GuiCellComposition;
|
|
table->AddChild(cell);
|
|
cell->SetSite(0, 1, 3, 1);
|
|
cell->SetMargin(Margin(0, 0, 16, 0));
|
|
|
|
text = GuiSolidLabelElement::Create();
|
|
text->SetAlignments(Alignment::Left, Alignment::Center);
|
|
cell->SetOwnedElement(text);
|
|
}
|
|
}
|
|
|
|
if (auto listView = dynamic_cast<GuiVirtualListView*>(listControl))
|
|
{
|
|
auto itemIndex = GetIndex();
|
|
if (auto view = dynamic_cast<IListViewItemView*>(listView->GetItemProvider()->RequestView(IListViewItemView::Identifier)))
|
|
{
|
|
auto imageData = view->GetSmallImage(itemIndex);
|
|
if (imageData)
|
|
{
|
|
image->SetImage(imageData->GetImage(), imageData->GetFrameIndex());
|
|
}
|
|
else
|
|
{
|
|
image->SetImage(nullptr);
|
|
}
|
|
text->SetText(view->GetText(itemIndex));
|
|
text->SetColor(listView->GetControlTemplateObject(true)->GetPrimaryTextColor());
|
|
}
|
|
}
|
|
|
|
FontChanged.AttachMethod(this, &ListListViewItemTemplate::OnFontChanged);
|
|
|
|
FontChanged.Execute(compositions::GuiEventArgs(this));
|
|
}
|
|
|
|
void ListListViewItemTemplate::OnFontChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
text->SetFont(GetFont());
|
|
}
|
|
|
|
ListListViewItemTemplate::ListListViewItemTemplate()
|
|
{
|
|
}
|
|
|
|
ListListViewItemTemplate::~ListListViewItemTemplate()
|
|
{
|
|
}
|
|
|
|
/***********************************************************************
|
|
TileListViewItemTemplate
|
|
***********************************************************************/
|
|
|
|
elements::GuiSolidLabelElement* TileListViewItemTemplate::CreateTextElement(vint textRow)
|
|
{
|
|
auto cell = new GuiCellComposition;
|
|
textTable->AddChild(cell);
|
|
cell->SetSite(textRow + 1, 0, 1, 1);
|
|
|
|
auto textElement = GuiSolidLabelElement::Create();
|
|
textElement->SetAlignments(Alignment::Left, Alignment::Center);
|
|
textElement->SetEllipse(true);
|
|
cell->SetOwnedElement(textElement);
|
|
return textElement;
|
|
}
|
|
|
|
void TileListViewItemTemplate::ResetTextTable(vint textRows)
|
|
{
|
|
textTable->SetRowsAndColumns(textRows + 2, 1);
|
|
textTable->SetRowOption(0, GuiCellOption::PercentageOption(0.5));
|
|
for (vint i = 0; i<textRows; i++)
|
|
{
|
|
textTable->SetRowOption(i + 1, GuiCellOption::MinSizeOption());
|
|
}
|
|
textTable->SetRowOption(textRows + 1, GuiCellOption::PercentageOption(0.5));
|
|
textTable->SetColumnOption(0, GuiCellOption::PercentageOption(1.0));
|
|
}
|
|
|
|
void TileListViewItemTemplate::OnInitialize()
|
|
{
|
|
DefaultListViewItemTemplate::OnInitialize();
|
|
{
|
|
auto table = new GuiTableComposition;
|
|
AddChild(table);
|
|
table->SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
table->SetRowsAndColumns(3, 2);
|
|
table->SetRowOption(0, GuiCellOption::PercentageOption(0.5));
|
|
table->SetRowOption(1, GuiCellOption::MinSizeOption());
|
|
table->SetRowOption(2, GuiCellOption::PercentageOption(0.5));
|
|
table->SetColumnOption(0, GuiCellOption::MinSizeOption());
|
|
table->SetColumnOption(1, GuiCellOption::MinSizeOption());
|
|
table->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
table->SetCellPadding(4);
|
|
{
|
|
auto cell = new GuiCellComposition;
|
|
table->AddChild(cell);
|
|
cell->SetSite(1, 0, 1, 1);
|
|
cell->SetPreferredMinSize(Size(32, 32));
|
|
|
|
image = GuiImageFrameElement::Create();
|
|
image->SetStretch(true);
|
|
cell->SetOwnedElement(image);
|
|
}
|
|
{
|
|
auto cell = new GuiCellComposition;
|
|
table->AddChild(cell);
|
|
cell->SetSite(0, 1, 3, 1);
|
|
cell->SetPreferredMinSize(Size(224, 0));
|
|
|
|
textTable = new GuiTableComposition;
|
|
textTable->SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
textTable->SetCellPadding(1);
|
|
ResetTextTable(1);
|
|
textTable->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
cell->AddChild(textTable);
|
|
{
|
|
text = CreateTextElement(0);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (auto listView = dynamic_cast<GuiVirtualListView*>(listControl))
|
|
{
|
|
auto itemIndex = GetIndex();
|
|
if (auto view = dynamic_cast<IListViewItemView*>(listView->GetItemProvider()->RequestView(IListViewItemView::Identifier)))
|
|
{
|
|
auto imageData = view->GetLargeImage(itemIndex);
|
|
if (imageData)
|
|
{
|
|
image->SetImage(imageData->GetImage(), imageData->GetFrameIndex());
|
|
}
|
|
else
|
|
{
|
|
image->SetImage(nullptr);
|
|
}
|
|
text->SetText(view->GetText(itemIndex));
|
|
text->SetColor(listView->GetControlTemplateObject(true)->GetPrimaryTextColor());
|
|
|
|
vint dataColumnCount = view->GetDataColumnCount();
|
|
ResetTextTable(dataColumnCount + 1);
|
|
dataTexts.Resize(dataColumnCount);
|
|
for (vint i = 0; i < dataColumnCount; i++)
|
|
{
|
|
dataTexts[i] = CreateTextElement(i + 1);
|
|
dataTexts[i]->SetText(view->GetSubItem(itemIndex, view->GetDataColumn(i)));
|
|
dataTexts[i]->SetColor(listView->GetControlTemplateObject(true)->GetSecondaryTextColor());
|
|
}
|
|
}
|
|
}
|
|
|
|
FontChanged.AttachMethod(this, &TileListViewItemTemplate::OnFontChanged);
|
|
|
|
FontChanged.Execute(compositions::GuiEventArgs(this));
|
|
}
|
|
|
|
void TileListViewItemTemplate::OnFontChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
text->SetFont(GetFont());
|
|
if (auto view = dynamic_cast<IListViewItemView*>(listControl->GetItemProvider()->RequestView(IListViewItemView::Identifier)))
|
|
{
|
|
vint dataColumnCount = view->GetDataColumnCount();
|
|
for (vint i = 0; i < dataColumnCount; i++)
|
|
{
|
|
dataTexts[i]->SetFont(GetFont());
|
|
}
|
|
}
|
|
}
|
|
|
|
TileListViewItemTemplate::TileListViewItemTemplate()
|
|
{
|
|
}
|
|
|
|
TileListViewItemTemplate::~TileListViewItemTemplate()
|
|
{
|
|
}
|
|
|
|
/***********************************************************************
|
|
InformationListViewItemTemplate
|
|
***********************************************************************/
|
|
|
|
void InformationListViewItemTemplate::OnInitialize()
|
|
{
|
|
DefaultListViewItemTemplate::OnInitialize();
|
|
{
|
|
bottomLine = GuiSolidBackgroundElement::Create();
|
|
bottomLineComposition = new GuiBoundsComposition;
|
|
bottomLineComposition->SetOwnedElement(bottomLine);
|
|
bottomLineComposition->SetAlignmentToParent(Margin(8, -1, 8, 0));
|
|
bottomLineComposition->SetPreferredMinSize(Size(0, 1));
|
|
AddChild(bottomLineComposition);
|
|
|
|
auto table = new GuiTableComposition;
|
|
AddChild(table);
|
|
table->SetRowsAndColumns(3, 3);
|
|
table->SetRowOption(0, GuiCellOption::PercentageOption(0.5));
|
|
table->SetRowOption(1, GuiCellOption::MinSizeOption());
|
|
table->SetRowOption(2, GuiCellOption::PercentageOption(0.5));
|
|
table->SetColumnOption(0, GuiCellOption::MinSizeOption());
|
|
table->SetColumnOption(1, GuiCellOption::PercentageOption(1.0));
|
|
table->SetColumnOption(2, GuiCellOption::MinSizeOption());
|
|
table->SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
table->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
table->SetCellPadding(4);
|
|
{
|
|
auto cell = new GuiCellComposition;
|
|
table->AddChild(cell);
|
|
cell->SetSite(1, 0, 1, 1);
|
|
cell->SetPreferredMinSize(Size(32, 32));
|
|
|
|
image = GuiImageFrameElement::Create();
|
|
image->SetStretch(true);
|
|
cell->SetOwnedElement(image);
|
|
}
|
|
{
|
|
auto cell = new GuiCellComposition;
|
|
table->AddChild(cell);
|
|
cell->SetSite(0, 1, 3, 1);
|
|
|
|
text = GuiSolidLabelElement::Create();
|
|
text->SetEllipse(true);
|
|
cell->SetOwnedElement(text);
|
|
}
|
|
{
|
|
auto cell = new GuiCellComposition;
|
|
table->AddChild(cell);
|
|
cell->SetSite(0, 2, 3, 1);
|
|
cell->SetPreferredMinSize(Size(224, 0));
|
|
|
|
textTable = new GuiTableComposition;
|
|
textTable->SetCellPadding(4);
|
|
textTable->SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
textTable->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
cell->AddChild(textTable);
|
|
}
|
|
}
|
|
|
|
if (auto listView = dynamic_cast<GuiVirtualListView*>(listControl))
|
|
{
|
|
auto itemIndex = GetIndex();
|
|
if (auto view = dynamic_cast<IListViewItemView*>(listView->GetItemProvider()->RequestView(IListViewItemView::Identifier)))
|
|
{
|
|
auto imageData = view->GetLargeImage(itemIndex);
|
|
if (imageData)
|
|
{
|
|
image->SetImage(imageData->GetImage(), imageData->GetFrameIndex());
|
|
}
|
|
else
|
|
{
|
|
image->SetImage(nullptr);
|
|
}
|
|
text->SetText(view->GetText(itemIndex));
|
|
text->SetColor(listView->GetControlTemplateObject(true)->GetPrimaryTextColor());
|
|
bottomLine->SetColor(listView->GetControlTemplateObject(true)->GetItemSeparatorColor());
|
|
|
|
vint dataColumnCount = view->GetDataColumnCount();
|
|
columnTexts.Resize(dataColumnCount);
|
|
dataTexts.Resize(dataColumnCount);
|
|
|
|
textTable->SetRowsAndColumns(dataColumnCount + 2, 1);
|
|
textTable->SetRowOption(0, GuiCellOption::PercentageOption(0.5));
|
|
for (vint i = 0; i < dataColumnCount; i++)
|
|
{
|
|
textTable->SetRowOption(i + 1, GuiCellOption::MinSizeOption());
|
|
}
|
|
textTable->SetRowOption(dataColumnCount + 1, GuiCellOption::PercentageOption(0.5));
|
|
textTable->SetColumnOption(0, GuiCellOption::PercentageOption(1.0));
|
|
|
|
for (vint i = 0; i < dataColumnCount; i++)
|
|
{
|
|
auto cell = new GuiCellComposition;
|
|
textTable->AddChild(cell);
|
|
cell->SetSite(i + 1, 0, 1, 1);
|
|
|
|
auto dataTable = new GuiTableComposition;
|
|
dataTable->SetRowsAndColumns(1, 2);
|
|
dataTable->SetRowOption(0, GuiCellOption::MinSizeOption());
|
|
dataTable->SetColumnOption(0, GuiCellOption::MinSizeOption());
|
|
dataTable->SetColumnOption(1, GuiCellOption::PercentageOption(1.0));
|
|
dataTable->SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
dataTable->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
cell->AddChild(dataTable);
|
|
{
|
|
auto cell = new GuiCellComposition;
|
|
dataTable->AddChild(cell);
|
|
cell->SetSite(0, 0, 1, 1);
|
|
|
|
columnTexts[i] = GuiSolidLabelElement::Create();
|
|
columnTexts[i]->SetText(view->GetColumnText(view->GetDataColumn(i) + 1) + L": ");
|
|
columnTexts[i]->SetColor(listView->GetControlTemplateObject(true)->GetSecondaryTextColor());
|
|
cell->SetOwnedElement(columnTexts[i]);
|
|
}
|
|
{
|
|
auto cell = new GuiCellComposition;
|
|
dataTable->AddChild(cell);
|
|
cell->SetSite(0, 1, 1, 1);
|
|
|
|
dataTexts[i]= GuiSolidLabelElement::Create();
|
|
dataTexts[i]->SetEllipse(true);
|
|
dataTexts[i]->SetText(view->GetSubItem(itemIndex, view->GetDataColumn(i)));
|
|
dataTexts[i]->SetColor(listView->GetControlTemplateObject(true)->GetPrimaryTextColor());
|
|
cell->SetOwnedElement(dataTexts[i]);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
FontChanged.AttachMethod(this, &InformationListViewItemTemplate::OnFontChanged);
|
|
|
|
FontChanged.Execute(compositions::GuiEventArgs(this));
|
|
}
|
|
|
|
void InformationListViewItemTemplate::OnFontChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
{
|
|
auto font = GetFont();
|
|
font.size = (vint)(font.size * 1.2);
|
|
text->SetFont(font);
|
|
}
|
|
if (auto view = dynamic_cast<IListViewItemView*>(listControl->GetItemProvider()->RequestView(IListViewItemView::Identifier)))
|
|
{
|
|
vint dataColumnCount = view->GetDataColumnCount();
|
|
for (vint i = 0; i < dataColumnCount; i++)
|
|
{
|
|
columnTexts[i]->SetFont(GetFont());
|
|
dataTexts[i]->SetFont(GetFont());
|
|
}
|
|
}
|
|
}
|
|
|
|
InformationListViewItemTemplate::InformationListViewItemTemplate()
|
|
{
|
|
}
|
|
|
|
InformationListViewItemTemplate::~InformationListViewItemTemplate()
|
|
{
|
|
}
|
|
|
|
/***********************************************************************
|
|
DetailListViewItemTemplate
|
|
***********************************************************************/
|
|
|
|
void DetailListViewItemTemplate::OnInitialize()
|
|
{
|
|
DefaultListViewItemTemplate::OnInitialize();
|
|
columnItemView = dynamic_cast<ListViewColumnItemArranger::IColumnItemView*>(listControl->GetItemProvider()->RequestView(ListViewColumnItemArranger::IColumnItemView::Identifier));
|
|
|
|
{
|
|
textTable = new GuiTableComposition;
|
|
textTable->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
textTable->SetRowsAndColumns(1, 1);
|
|
textTable->SetRowOption(0, GuiCellOption::MinSizeOption());
|
|
textTable->SetColumnOption(0, GuiCellOption::AbsoluteOption(0));
|
|
AddChild(textTable);
|
|
{
|
|
auto cell = new GuiCellComposition;
|
|
textTable->AddChild(cell);
|
|
cell->SetSite(0, 0, 1, 1);
|
|
|
|
auto table = new GuiTableComposition;
|
|
cell->AddChild(table);
|
|
table->SetRowsAndColumns(3, 2);
|
|
table->SetRowOption(0, GuiCellOption::PercentageOption(0.5));
|
|
table->SetRowOption(1, GuiCellOption::MinSizeOption());
|
|
table->SetRowOption(2, GuiCellOption::PercentageOption(0.5));
|
|
table->SetColumnOption(0, GuiCellOption::MinSizeOption());
|
|
table->SetColumnOption(1, GuiCellOption::PercentageOption(1.0));
|
|
table->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
table->SetCellPadding(2);
|
|
{
|
|
auto cell = new GuiCellComposition;
|
|
table->AddChild(cell);
|
|
cell->SetSite(1, 0, 1, 1);
|
|
cell->SetPreferredMinSize(Size(16, 16));
|
|
|
|
image = GuiImageFrameElement::Create();
|
|
image->SetStretch(true);
|
|
cell->SetOwnedElement(image);
|
|
}
|
|
{
|
|
auto cell = new GuiCellComposition;
|
|
table->AddChild(cell);
|
|
cell->SetSite(0, 1, 3, 1);
|
|
cell->SetMargin(Margin(0, 0, 8, 0));
|
|
|
|
text = GuiSolidLabelElement::Create();
|
|
text->SetAlignments(Alignment::Left, Alignment::Center);
|
|
text->SetEllipse(true);
|
|
cell->SetOwnedElement(text);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (auto listView = dynamic_cast<GuiVirtualListView*>(listControl))
|
|
{
|
|
auto itemIndex = GetIndex();
|
|
if (auto view = dynamic_cast<IListViewItemView*>(listView->GetItemProvider()->RequestView(IListViewItemView::Identifier)))
|
|
{
|
|
auto imageData = view->GetSmallImage(itemIndex);
|
|
if (imageData)
|
|
{
|
|
image->SetImage(imageData->GetImage(), imageData->GetFrameIndex());
|
|
}
|
|
else
|
|
{
|
|
image->SetImage(0);
|
|
}
|
|
text->SetText(view->GetText(itemIndex));
|
|
text->SetColor(listView->GetControlTemplateObject(true)->GetPrimaryTextColor());
|
|
|
|
vint columnCount = view->GetColumnCount() - 1;
|
|
subItems.Resize(columnCount);
|
|
textTable->SetRowsAndColumns(1, columnCount + 1);
|
|
for (vint i = 0; i < columnCount; i++)
|
|
{
|
|
auto cell = new GuiCellComposition;
|
|
textTable->AddChild(cell);
|
|
cell->SetSite(0, i + 1, 1, 1);
|
|
cell->SetMargin(Margin(8, 0, 8, 0));
|
|
|
|
subItems[i] = GuiSolidLabelElement::Create();
|
|
subItems[i]->SetAlignments(Alignment::Left, Alignment::Center);
|
|
subItems[i]->SetFont(text->GetFont());
|
|
subItems[i]->SetEllipse(true);
|
|
subItems[i]->SetText(view->GetSubItem(itemIndex, i));
|
|
subItems[i]->SetColor(listView->GetControlTemplateObject(true)->GetSecondaryTextColor());
|
|
cell->SetOwnedElement(subItems[i]);
|
|
}
|
|
OnColumnChanged();
|
|
}
|
|
}
|
|
|
|
FontChanged.AttachMethod(this, &DetailListViewItemTemplate::OnFontChanged);
|
|
|
|
FontChanged.Execute(compositions::GuiEventArgs(this));
|
|
}
|
|
|
|
void DetailListViewItemTemplate::OnColumnChanged()
|
|
{
|
|
if (auto view = dynamic_cast<IListViewItemView*>(listControl->GetItemProvider()->RequestView(IListViewItemView::Identifier)))
|
|
{
|
|
if (columnItemView)
|
|
{
|
|
vint columnCount = view->GetColumnCount();
|
|
if (columnCount>textTable->GetColumns())
|
|
{
|
|
columnCount = textTable->GetColumns();
|
|
}
|
|
for (vint i = 0; i<columnCount; i++)
|
|
{
|
|
textTable->SetColumnOption(i, GuiCellOption::AbsoluteOption(columnItemView->GetColumnSize(i)));
|
|
}
|
|
textTable->UpdateCellBounds();
|
|
}
|
|
}
|
|
}
|
|
|
|
void DetailListViewItemTemplate::OnFontChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
text->SetFont(GetFont());
|
|
if (auto view = dynamic_cast<IListViewItemView*>(listControl->GetItemProvider()->RequestView(IListViewItemView::Identifier)))
|
|
{
|
|
vint columnCount = view->GetColumnCount() - 1;
|
|
for (vint i = 0; i < columnCount; i++)
|
|
{
|
|
subItems[i]->SetFont(GetFont());
|
|
}
|
|
}
|
|
}
|
|
|
|
DetailListViewItemTemplate::DetailListViewItemTemplate()
|
|
{
|
|
}
|
|
|
|
DetailListViewItemTemplate::~DetailListViewItemTemplate()
|
|
{
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\CONTROLS\LISTCONTROLPACKAGE\GUITEXTLISTCONTROLS.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace controls
|
|
{
|
|
using namespace collections;
|
|
using namespace elements;
|
|
using namespace compositions;
|
|
using namespace reflection::description;
|
|
|
|
namespace list
|
|
{
|
|
const wchar_t* const ITextItemView::Identifier = L"vl::presentation::controls::list::ITextItemView";
|
|
|
|
/***********************************************************************
|
|
DefaultTextListItemTemplate
|
|
***********************************************************************/
|
|
|
|
TemplateProperty<DefaultTextListItemTemplate::BulletStyle> DefaultTextListItemTemplate::CreateBulletStyle()
|
|
{
|
|
return {};
|
|
}
|
|
|
|
void DefaultTextListItemTemplate::OnInitialize()
|
|
{
|
|
SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
|
|
textElement = GuiSolidLabelElement::Create();
|
|
textElement->SetAlignments(Alignment::Left, Alignment::Center);
|
|
|
|
GuiBoundsComposition* textComposition = new GuiBoundsComposition;
|
|
textComposition->SetOwnedElement(textElement);
|
|
textComposition->SetMinSizeLimitation(GuiGraphicsComposition::LimitToElement);
|
|
|
|
if (auto bulletStyleController = CreateBulletStyle())
|
|
{
|
|
bulletButton = new GuiSelectableButton(theme::ThemeName::Unknown);
|
|
bulletButton->SetAutoFocus(false);
|
|
bulletButton->SetControlTemplate(bulletStyleController);
|
|
bulletButton->GetBoundsComposition()->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
bulletButton->SelectedChanged.AttachMethod(this, &DefaultTextListItemTemplate::OnBulletSelectedChanged);
|
|
|
|
GuiTableComposition* table = new GuiTableComposition;
|
|
AddChild(table);
|
|
table->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
table->SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
table->SetRowsAndColumns(1, 2);
|
|
table->SetRowOption(0, GuiCellOption::PercentageOption(1.0));
|
|
table->SetColumnOption(0, GuiCellOption::MinSizeOption());
|
|
table->SetColumnOption(1, GuiCellOption::PercentageOption(1.0));
|
|
{
|
|
GuiCellComposition* cell = new GuiCellComposition;
|
|
table->AddChild(cell);
|
|
cell->SetSite(0, 0, 1, 1);
|
|
cell->AddChild(bulletButton->GetBoundsComposition());
|
|
}
|
|
{
|
|
GuiCellComposition* cell = new GuiCellComposition;
|
|
table->AddChild(cell);
|
|
cell->SetSite(0, 1, 1, 1);
|
|
cell->AddChild(textComposition);
|
|
textComposition->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
AddChild(textComposition);
|
|
textComposition->SetAlignmentToParent(Margin(5, 2, 0, 2));
|
|
}
|
|
|
|
FontChanged.AttachMethod(this, &DefaultTextListItemTemplate::OnFontChanged);
|
|
TextChanged.AttachMethod(this, &DefaultTextListItemTemplate::OnTextChanged);
|
|
TextColorChanged.AttachMethod(this, &DefaultTextListItemTemplate::OnTextColorChanged);
|
|
CheckedChanged.AttachMethod(this, &DefaultTextListItemTemplate::OnCheckedChanged);
|
|
|
|
FontChanged.Execute(compositions::GuiEventArgs(this));
|
|
TextChanged.Execute(compositions::GuiEventArgs(this));
|
|
TextColorChanged.Execute(compositions::GuiEventArgs(this));
|
|
CheckedChanged.Execute(compositions::GuiEventArgs(this));
|
|
}
|
|
|
|
void DefaultTextListItemTemplate::OnFontChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
textElement->SetFont(GetFont());
|
|
}
|
|
|
|
void DefaultTextListItemTemplate::OnTextChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
textElement->SetText(GetText());
|
|
}
|
|
|
|
void DefaultTextListItemTemplate::OnTextColorChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
textElement->SetColor(GetTextColor());
|
|
}
|
|
|
|
void DefaultTextListItemTemplate::OnCheckedChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
if (bulletButton)
|
|
{
|
|
supressEdit = true;
|
|
bulletButton->SetSelected(GetChecked());
|
|
supressEdit = false;
|
|
}
|
|
}
|
|
|
|
void DefaultTextListItemTemplate::OnBulletSelectedChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
if (!supressEdit)
|
|
{
|
|
if (auto textItemView = dynamic_cast<ITextItemView*>(listControl->GetItemProvider()->RequestView(ITextItemView::Identifier)))
|
|
{
|
|
BeginEditListItem();
|
|
textItemView->SetChecked(GetIndex(), bulletButton->GetSelected());
|
|
EndEditListItem();
|
|
}
|
|
}
|
|
}
|
|
|
|
DefaultTextListItemTemplate::DefaultTextListItemTemplate()
|
|
{
|
|
}
|
|
|
|
DefaultTextListItemTemplate::~DefaultTextListItemTemplate()
|
|
{
|
|
}
|
|
|
|
/***********************************************************************
|
|
DefaultCheckTextListItemTemplate
|
|
***********************************************************************/
|
|
|
|
TemplateProperty<DefaultTextListItemTemplate::BulletStyle> DefaultCheckTextListItemTemplate::CreateBulletStyle()
|
|
{
|
|
if (auto textList = dynamic_cast<GuiVirtualTextList*>(listControl))
|
|
{
|
|
auto style = textList->GetControlTemplateObject(true)->GetCheckBulletTemplate();
|
|
if (style) return style;
|
|
}
|
|
return theme::GetCurrentTheme()->CreateStyle(theme::ThemeName::CheckTextListItem);
|
|
}
|
|
|
|
/***********************************************************************
|
|
DefaultRadioTextListItemTemplate
|
|
***********************************************************************/
|
|
|
|
TemplateProperty<DefaultTextListItemTemplate::BulletStyle> DefaultRadioTextListItemTemplate::CreateBulletStyle()
|
|
{
|
|
if (auto textList = dynamic_cast<GuiVirtualTextList*>(listControl))
|
|
{
|
|
auto style = textList->GetControlTemplateObject(true)->GetRadioBulletTemplate();
|
|
if (style) return style;
|
|
}
|
|
return theme::GetCurrentTheme()->CreateStyle(theme::ThemeName::RadioTextListItem);
|
|
}
|
|
|
|
/***********************************************************************
|
|
TextItem
|
|
***********************************************************************/
|
|
|
|
TextItem::TextItem()
|
|
:owner(0)
|
|
, checked(false)
|
|
{
|
|
}
|
|
|
|
TextItem::TextItem(const WString& _text, bool _checked)
|
|
:owner(0)
|
|
, text(_text)
|
|
, checked(_checked)
|
|
{
|
|
}
|
|
|
|
TextItem::~TextItem()
|
|
{
|
|
}
|
|
|
|
bool TextItem::operator==(const TextItem& value)const
|
|
{
|
|
return text==value.text;
|
|
}
|
|
|
|
bool TextItem::operator!=(const TextItem& value)const
|
|
{
|
|
return text!=value.text;
|
|
}
|
|
|
|
const WString& TextItem::GetText()
|
|
{
|
|
return text;
|
|
}
|
|
|
|
void TextItem::SetText(const WString& value)
|
|
{
|
|
if (text != value)
|
|
{
|
|
text = value;
|
|
if (owner)
|
|
{
|
|
vint index = owner->IndexOf(this);
|
|
owner->InvokeOnItemModified(index, 1, 1);
|
|
}
|
|
}
|
|
}
|
|
|
|
bool TextItem::GetChecked()
|
|
{
|
|
return checked;
|
|
}
|
|
|
|
void TextItem::SetChecked(bool value)
|
|
{
|
|
if (checked != value)
|
|
{
|
|
checked = value;
|
|
if (owner)
|
|
{
|
|
vint index = owner->IndexOf(this);
|
|
owner->InvokeOnItemModified(index, 1, 1);
|
|
|
|
GuiItemEventArgs arguments;
|
|
arguments.itemIndex = index;
|
|
owner->listControl->ItemChecked.Execute(arguments);
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
TextItemProvider
|
|
***********************************************************************/
|
|
|
|
void TextItemProvider::AfterInsert(vint item, const Ptr<TextItem>& value)
|
|
{
|
|
ListProvider<Ptr<TextItem>>::AfterInsert(item, value);
|
|
value->owner = this;
|
|
}
|
|
|
|
void TextItemProvider::BeforeRemove(vint item, const Ptr<TextItem>& value)
|
|
{
|
|
value->owner = 0;
|
|
ListProvider<Ptr<TextItem>>::BeforeRemove(item, value);
|
|
}
|
|
|
|
WString TextItemProvider::GetTextValue(vint itemIndex)
|
|
{
|
|
return Get(itemIndex)->GetText();
|
|
}
|
|
|
|
description::Value TextItemProvider::GetBindingValue(vint itemIndex)
|
|
{
|
|
return Value::From(Get(itemIndex));
|
|
}
|
|
|
|
bool TextItemProvider::GetChecked(vint itemIndex)
|
|
{
|
|
return Get(itemIndex)->GetChecked();
|
|
}
|
|
|
|
void TextItemProvider::SetChecked(vint itemIndex, bool value)
|
|
{
|
|
return Get(itemIndex)->SetChecked(value);
|
|
}
|
|
|
|
TextItemProvider::TextItemProvider()
|
|
:listControl(0)
|
|
{
|
|
}
|
|
|
|
TextItemProvider::~TextItemProvider()
|
|
{
|
|
}
|
|
|
|
IDescriptable* TextItemProvider::RequestView(const WString& identifier)
|
|
{
|
|
if (identifier == ITextItemView::Identifier)
|
|
{
|
|
return (ITextItemView*)this;
|
|
}
|
|
else
|
|
{
|
|
return nullptr;
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiTextList
|
|
***********************************************************************/
|
|
|
|
void GuiVirtualTextList::BeforeControlTemplateUninstalled_()
|
|
{
|
|
}
|
|
|
|
void GuiVirtualTextList::AfterControlTemplateInstalled_(bool initialize)
|
|
{
|
|
}
|
|
|
|
void GuiVirtualTextList::OnStyleInstalled(vint itemIndex, ItemStyle* style)
|
|
{
|
|
GuiSelectableListControl::OnStyleInstalled(itemIndex, style);
|
|
if (auto textItemStyle = dynamic_cast<templates::GuiTextListItemTemplate*>(style))
|
|
{
|
|
textItemStyle->SetTextColor(GetControlTemplateObject(true)->GetTextColor());
|
|
if (auto textItemView = dynamic_cast<list::ITextItemView*>(itemProvider->RequestView(list::ITextItemView::Identifier)))
|
|
{
|
|
textItemStyle->SetChecked(textItemView->GetChecked(itemIndex));
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiVirtualTextList::OnItemTemplateChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
view = TextListView::Unknown;
|
|
}
|
|
|
|
GuiVirtualTextList::GuiVirtualTextList(theme::ThemeName themeName, GuiListControl::IItemProvider* _itemProvider)
|
|
:GuiSelectableListControl(themeName, _itemProvider)
|
|
{
|
|
ItemTemplateChanged.AttachMethod(this, &GuiVirtualTextList::OnItemTemplateChanged);
|
|
ItemChecked.SetAssociatedComposition(boundsComposition);
|
|
|
|
SetView(TextListView::Text);
|
|
}
|
|
|
|
GuiVirtualTextList::~GuiVirtualTextList()
|
|
{
|
|
}
|
|
|
|
TextListView GuiVirtualTextList::GetView()
|
|
{
|
|
return view;
|
|
}
|
|
|
|
void GuiVirtualTextList::SetView(TextListView _view)
|
|
{
|
|
switch (_view)
|
|
{
|
|
case TextListView::Text:
|
|
SetStyleAndArranger(
|
|
[](const Value&) { return new list::DefaultTextListItemTemplate; },
|
|
new list::FixedHeightItemArranger
|
|
);
|
|
break;
|
|
case TextListView::Check:
|
|
SetStyleAndArranger(
|
|
[](const Value&) { return new list::DefaultCheckTextListItemTemplate; },
|
|
new list::FixedHeightItemArranger
|
|
);
|
|
break;
|
|
case TextListView::Radio:
|
|
SetStyleAndArranger(
|
|
[](const Value&) { return new list::DefaultRadioTextListItemTemplate; },
|
|
new list::FixedHeightItemArranger
|
|
);
|
|
break;
|
|
default:;
|
|
}
|
|
view = _view;
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiTextList
|
|
***********************************************************************/
|
|
|
|
GuiTextList::GuiTextList(theme::ThemeName themeName)
|
|
:GuiVirtualTextList(themeName, new list::TextItemProvider)
|
|
{
|
|
items=dynamic_cast<list::TextItemProvider*>(itemProvider.Obj());
|
|
items->listControl=this;
|
|
}
|
|
|
|
GuiTextList::~GuiTextList()
|
|
{
|
|
}
|
|
|
|
list::TextItemProvider& GuiTextList::GetItems()
|
|
{
|
|
return *items;
|
|
}
|
|
|
|
Ptr<list::TextItem> GuiTextList::GetSelectedItem()
|
|
{
|
|
vint index = GetSelectedItemIndex();
|
|
if (index == -1) return 0;
|
|
return items->Get(index);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\CONTROLS\LISTCONTROLPACKAGE\GUITREEVIEWCONTROLS.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace controls
|
|
{
|
|
using namespace elements;
|
|
using namespace compositions;
|
|
using namespace reflection::description;
|
|
|
|
namespace tree
|
|
{
|
|
const wchar_t* const INodeItemView::Identifier = L"vl::presentation::controls::tree::INodeItemView";
|
|
|
|
/***********************************************************************
|
|
NodeItemProvider
|
|
***********************************************************************/
|
|
|
|
Ptr<INodeProvider> NodeItemProvider::GetNodeByOffset(Ptr<INodeProvider> provider, vint offset)
|
|
{
|
|
if (offset == 0) return provider;
|
|
Ptr<INodeProvider> result;
|
|
if (provider->GetExpanding() && offset > 0)
|
|
{
|
|
offset -= 1;
|
|
vint count = provider->GetChildCount();
|
|
for (vint i = 0; (!result && i < count); i++)
|
|
{
|
|
auto child = provider->GetChild(i);
|
|
vint visibleCount = child->CalculateTotalVisibleNodes();
|
|
if (offset < visibleCount)
|
|
{
|
|
result = GetNodeByOffset(child, offset);
|
|
}
|
|
else
|
|
{
|
|
offset -= visibleCount;
|
|
}
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
void NodeItemProvider::OnAttached(INodeRootProvider* provider)
|
|
{
|
|
}
|
|
|
|
void NodeItemProvider::OnBeforeItemModified(INodeProvider* parentNode, vint start, vint count, vint newCount)
|
|
{
|
|
vint offset = 0;
|
|
vint base = CalculateNodeVisibilityIndexInternal(parentNode);
|
|
if (base != -2 && parentNode->GetExpanding())
|
|
{
|
|
for (vint i = 0; i < count; i++)
|
|
{
|
|
auto child = parentNode->GetChild(start + i);
|
|
offset += child->CalculateTotalVisibleNodes();
|
|
}
|
|
}
|
|
offsetBeforeChildModifieds.Set(parentNode, offset);
|
|
}
|
|
|
|
void NodeItemProvider::OnAfterItemModified(INodeProvider* parentNode, vint start, vint count, vint newCount)
|
|
{
|
|
vint offsetBeforeChildModified = 0;
|
|
{
|
|
vint index = offsetBeforeChildModifieds.Keys().IndexOf(parentNode);
|
|
if (index != -1)
|
|
{
|
|
offsetBeforeChildModified = offsetBeforeChildModifieds.Values().Get(index);
|
|
offsetBeforeChildModifieds.Remove(parentNode);
|
|
}
|
|
}
|
|
|
|
vint base = CalculateNodeVisibilityIndexInternal(parentNode);
|
|
if (base != -2 && parentNode->GetExpanding())
|
|
{
|
|
vint offset = 0;
|
|
vint firstChildStart = -1;
|
|
for (vint i = 0; i < newCount; i++)
|
|
{
|
|
auto child = parentNode->GetChild(start + i);
|
|
if (i == 0)
|
|
{
|
|
firstChildStart = CalculateNodeVisibilityIndexInternal(child.Obj());
|
|
}
|
|
offset += child->CalculateTotalVisibleNodes();
|
|
}
|
|
|
|
if (firstChildStart == -1)
|
|
{
|
|
vint childCount = parentNode->GetChildCount();
|
|
if (childCount == 0)
|
|
{
|
|
firstChildStart = base + 1;
|
|
}
|
|
else if (start < childCount)
|
|
{
|
|
auto child = parentNode->GetChild(start);
|
|
firstChildStart = CalculateNodeVisibilityIndexInternal(child.Obj());
|
|
}
|
|
else
|
|
{
|
|
auto child = parentNode->GetChild(start - 1);
|
|
firstChildStart = CalculateNodeVisibilityIndexInternal(child.Obj());
|
|
firstChildStart += child->CalculateTotalVisibleNodes();
|
|
}
|
|
}
|
|
InvokeOnItemModified(firstChildStart, offsetBeforeChildModified, offset);
|
|
}
|
|
}
|
|
|
|
void NodeItemProvider::OnItemExpanded(INodeProvider* node)
|
|
{
|
|
vint base = CalculateNodeVisibilityIndexInternal(node);
|
|
if (base != -2)
|
|
{
|
|
vint visibility = node->CalculateTotalVisibleNodes();
|
|
InvokeOnItemModified(base + 1, 0, visibility - 1);
|
|
}
|
|
}
|
|
|
|
void NodeItemProvider::OnItemCollapsed(INodeProvider* node)
|
|
{
|
|
vint base = CalculateNodeVisibilityIndexInternal(node);
|
|
if (base != -2)
|
|
{
|
|
vint visibility = 0;
|
|
vint count = node->GetChildCount();
|
|
for (vint i = 0; i < count; i++)
|
|
{
|
|
auto child = node->GetChild(i);
|
|
visibility += child->CalculateTotalVisibleNodes();
|
|
}
|
|
InvokeOnItemModified(base + 1, visibility, 0);
|
|
}
|
|
}
|
|
|
|
vint NodeItemProvider::CalculateNodeVisibilityIndexInternal(INodeProvider* node)
|
|
{
|
|
auto parent = node->GetParent();
|
|
if (!parent)
|
|
{
|
|
return -1;
|
|
}
|
|
if (!parent->GetExpanding())
|
|
{
|
|
return -2;
|
|
}
|
|
|
|
vint index = CalculateNodeVisibilityIndexInternal(parent.Obj());
|
|
if (index == -2)
|
|
{
|
|
return -2;
|
|
}
|
|
|
|
vint count = parent->GetChildCount();
|
|
for (vint i = 0; i < count; i++)
|
|
{
|
|
auto child = parent->GetChild(i);
|
|
bool findResult = child == node;
|
|
if (findResult)
|
|
{
|
|
index++;
|
|
}
|
|
else
|
|
{
|
|
index += child->CalculateTotalVisibleNodes();
|
|
}
|
|
if (findResult)
|
|
{
|
|
return index;
|
|
}
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
vint NodeItemProvider::CalculateNodeVisibilityIndex(INodeProvider* node)
|
|
{
|
|
vint result = CalculateNodeVisibilityIndexInternal(node);
|
|
return result < 0 ? -1 : result;
|
|
}
|
|
|
|
Ptr<INodeProvider> NodeItemProvider::RequestNode(vint index)
|
|
{
|
|
if(root->CanGetNodeByVisibleIndex())
|
|
{
|
|
return root->GetNodeByVisibleIndex(index+1);
|
|
}
|
|
else
|
|
{
|
|
return GetNodeByOffset(root->GetRootNode(), index+1);
|
|
}
|
|
}
|
|
|
|
NodeItemProvider::NodeItemProvider(Ptr<INodeRootProvider> _root)
|
|
:root(_root)
|
|
{
|
|
root->AttachCallback(this);
|
|
}
|
|
|
|
NodeItemProvider::~NodeItemProvider()
|
|
{
|
|
root->DetachCallback(this);
|
|
}
|
|
|
|
Ptr<INodeRootProvider> NodeItemProvider::GetRoot()
|
|
{
|
|
return root;
|
|
}
|
|
|
|
vint NodeItemProvider::Count()
|
|
{
|
|
return root->GetRootNode()->CalculateTotalVisibleNodes()-1;
|
|
}
|
|
|
|
WString NodeItemProvider::GetTextValue(vint itemIndex)
|
|
{
|
|
if (auto node = RequestNode(itemIndex))
|
|
{
|
|
return root->GetTextValue(node.Obj());
|
|
}
|
|
return L"";
|
|
}
|
|
|
|
description::Value NodeItemProvider::GetBindingValue(vint itemIndex)
|
|
{
|
|
if (auto node = RequestNode(itemIndex))
|
|
{
|
|
return root->GetBindingValue(node.Obj());
|
|
}
|
|
return Value();
|
|
}
|
|
|
|
IDescriptable* NodeItemProvider::RequestView(const WString& identifier)
|
|
{
|
|
if(identifier==INodeItemView::Identifier)
|
|
{
|
|
return (INodeItemView*)this;
|
|
}
|
|
else
|
|
{
|
|
return root->RequestView(identifier);
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
MemoryNodeProvider::NodeCollection
|
|
***********************************************************************/
|
|
|
|
void MemoryNodeProvider::NodeCollection::OnBeforeChildModified(vint start, vint count, vint newCount)
|
|
{
|
|
if (ownerProvider->expanding)
|
|
{
|
|
for (vint i = 0; i < count; i++)
|
|
{
|
|
offsetBeforeChildModified += items[start + i]->totalVisibleNodeCount;
|
|
}
|
|
}
|
|
INodeProviderCallback* proxy = ownerProvider->GetCallbackProxyInternal();
|
|
if (proxy)
|
|
{
|
|
proxy->OnBeforeItemModified(ownerProvider, start, count, newCount);
|
|
}
|
|
}
|
|
|
|
void MemoryNodeProvider::NodeCollection::OnAfterChildModified(vint start, vint count, vint newCount)
|
|
{
|
|
ownerProvider->childCount += (newCount - count);
|
|
if (ownerProvider->expanding)
|
|
{
|
|
vint offset = 0;
|
|
for (vint i = 0; i < newCount; i++)
|
|
{
|
|
offset += items[start + i]->totalVisibleNodeCount;
|
|
}
|
|
ownerProvider->OnChildTotalVisibleNodesChanged(offset - offsetBeforeChildModified);
|
|
}
|
|
offsetBeforeChildModified = 0;
|
|
INodeProviderCallback* proxy = ownerProvider->GetCallbackProxyInternal();
|
|
if (proxy)
|
|
{
|
|
proxy->OnAfterItemModified(ownerProvider, start, count, newCount);
|
|
}
|
|
}
|
|
|
|
bool MemoryNodeProvider::NodeCollection::QueryInsert(vint index, Ptr<MemoryNodeProvider> const& child)
|
|
{
|
|
return child->parent == 0;
|
|
}
|
|
|
|
bool MemoryNodeProvider::NodeCollection::QueryRemove(vint index, Ptr<MemoryNodeProvider> const& child)
|
|
{
|
|
return child->parent == ownerProvider;
|
|
}
|
|
|
|
void MemoryNodeProvider::NodeCollection::BeforeInsert(vint index, Ptr<MemoryNodeProvider> const& child)
|
|
{
|
|
OnBeforeChildModified(index, 0, 1);
|
|
child->parent = ownerProvider;
|
|
}
|
|
|
|
void MemoryNodeProvider::NodeCollection::BeforeRemove(vint index, Ptr<MemoryNodeProvider> const& child)
|
|
{
|
|
OnBeforeChildModified(index, 1, 0);
|
|
child->parent = 0;
|
|
}
|
|
|
|
void MemoryNodeProvider::NodeCollection::AfterInsert(vint index, Ptr<MemoryNodeProvider> const& child)
|
|
{
|
|
OnAfterChildModified(index, 0, 1);
|
|
}
|
|
|
|
void MemoryNodeProvider::NodeCollection::AfterRemove(vint index, vint count)
|
|
{
|
|
OnAfterChildModified(index, count, 0);
|
|
}
|
|
|
|
MemoryNodeProvider::NodeCollection::NodeCollection()
|
|
:ownerProvider(0)
|
|
{
|
|
}
|
|
|
|
/***********************************************************************
|
|
MemoryNodeProvider
|
|
***********************************************************************/
|
|
|
|
INodeProviderCallback* MemoryNodeProvider::GetCallbackProxyInternal()
|
|
{
|
|
if(parent)
|
|
{
|
|
return parent->GetCallbackProxyInternal();
|
|
}
|
|
else
|
|
{
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
void MemoryNodeProvider::OnChildTotalVisibleNodesChanged(vint offset)
|
|
{
|
|
totalVisibleNodeCount+=offset;
|
|
if(parent)
|
|
{
|
|
parent->OnChildTotalVisibleNodesChanged(offset);
|
|
}
|
|
}
|
|
|
|
MemoryNodeProvider::MemoryNodeProvider(Ptr<DescriptableObject> _data)
|
|
:data(_data)
|
|
{
|
|
children.ownerProvider=this;
|
|
}
|
|
|
|
MemoryNodeProvider::~MemoryNodeProvider()
|
|
{
|
|
}
|
|
|
|
Ptr<DescriptableObject> MemoryNodeProvider::GetData()
|
|
{
|
|
return data;
|
|
}
|
|
|
|
void MemoryNodeProvider::SetData(const Ptr<DescriptableObject>& value)
|
|
{
|
|
data=value;
|
|
NotifyDataModified();
|
|
}
|
|
|
|
void MemoryNodeProvider::NotifyDataModified()
|
|
{
|
|
if(parent)
|
|
{
|
|
vint index=parent->children.IndexOf(this);
|
|
INodeProviderCallback* proxy=GetCallbackProxyInternal();
|
|
if(proxy)
|
|
{
|
|
proxy->OnBeforeItemModified(parent, index, 1, 1);
|
|
proxy->OnAfterItemModified(parent, index, 1, 1);
|
|
}
|
|
}
|
|
}
|
|
|
|
MemoryNodeProvider::NodeCollection& MemoryNodeProvider::Children()
|
|
{
|
|
return children;
|
|
}
|
|
|
|
bool MemoryNodeProvider::GetExpanding()
|
|
{
|
|
return expanding;
|
|
}
|
|
|
|
void MemoryNodeProvider::SetExpanding(bool value)
|
|
{
|
|
if(expanding!=value)
|
|
{
|
|
expanding=value;
|
|
vint offset=0;
|
|
for(vint i=0;i<childCount;i++)
|
|
{
|
|
offset+=children[i]->totalVisibleNodeCount;
|
|
}
|
|
|
|
OnChildTotalVisibleNodesChanged(expanding?offset:-offset);
|
|
INodeProviderCallback* proxy=GetCallbackProxyInternal();
|
|
if(proxy)
|
|
{
|
|
if(expanding)
|
|
{
|
|
proxy->OnItemExpanded(this);
|
|
}
|
|
else
|
|
{
|
|
proxy->OnItemCollapsed(this);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
vint MemoryNodeProvider::CalculateTotalVisibleNodes()
|
|
{
|
|
return totalVisibleNodeCount;
|
|
}
|
|
|
|
vint MemoryNodeProvider::GetChildCount()
|
|
{
|
|
return childCount;
|
|
}
|
|
|
|
Ptr<INodeProvider> MemoryNodeProvider::GetParent()
|
|
{
|
|
return parent;
|
|
}
|
|
|
|
Ptr<INodeProvider> MemoryNodeProvider::GetChild(vint index)
|
|
{
|
|
if(0<=index && index<childCount)
|
|
{
|
|
return children[index].Obj();
|
|
}
|
|
else
|
|
{
|
|
return nullptr;
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
NodeRootProviderBase
|
|
***********************************************************************/
|
|
|
|
void NodeRootProviderBase::OnAttached(INodeRootProvider* provider)
|
|
{
|
|
}
|
|
|
|
void NodeRootProviderBase::OnBeforeItemModified(INodeProvider* parentNode, vint start, vint count, vint newCount)
|
|
{
|
|
for(vint i=0;i<callbacks.Count();i++)
|
|
{
|
|
callbacks[i]->OnBeforeItemModified(parentNode, start, count, newCount);
|
|
}
|
|
}
|
|
|
|
void NodeRootProviderBase::OnAfterItemModified(INodeProvider* parentNode, vint start, vint count, vint newCount)
|
|
{
|
|
for(vint i=0;i<callbacks.Count();i++)
|
|
{
|
|
callbacks[i]->OnAfterItemModified(parentNode, start, count, newCount);
|
|
}
|
|
}
|
|
|
|
void NodeRootProviderBase::OnItemExpanded(INodeProvider* node)
|
|
{
|
|
for(vint i=0;i<callbacks.Count();i++)
|
|
{
|
|
callbacks[i]->OnItemExpanded(node);
|
|
}
|
|
}
|
|
|
|
void NodeRootProviderBase::OnItemCollapsed(INodeProvider* node)
|
|
{
|
|
for(vint i=0;i<callbacks.Count();i++)
|
|
{
|
|
callbacks[i]->OnItemCollapsed(node);
|
|
}
|
|
}
|
|
|
|
NodeRootProviderBase::NodeRootProviderBase()
|
|
{
|
|
}
|
|
|
|
NodeRootProviderBase::~NodeRootProviderBase()
|
|
{
|
|
}
|
|
|
|
bool NodeRootProviderBase::CanGetNodeByVisibleIndex()
|
|
{
|
|
return false;
|
|
}
|
|
|
|
Ptr<INodeProvider> NodeRootProviderBase::GetNodeByVisibleIndex(vint index)
|
|
{
|
|
return nullptr;
|
|
}
|
|
|
|
bool NodeRootProviderBase::AttachCallback(INodeProviderCallback* value)
|
|
{
|
|
if(callbacks.Contains(value))
|
|
{
|
|
return false;
|
|
}
|
|
else
|
|
{
|
|
callbacks.Add(value);
|
|
value->OnAttached(this);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
bool NodeRootProviderBase::DetachCallback(INodeProviderCallback* value)
|
|
{
|
|
vint index=callbacks.IndexOf(value);
|
|
if(index==-1)
|
|
{
|
|
return false;
|
|
}
|
|
else
|
|
{
|
|
value->OnAttached(0);
|
|
callbacks.Remove(value);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
IDescriptable* NodeRootProviderBase::RequestView(const WString& identifier)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
/***********************************************************************
|
|
MemoryNodeRootProvider
|
|
***********************************************************************/
|
|
|
|
INodeProviderCallback* MemoryNodeRootProvider::GetCallbackProxyInternal()
|
|
{
|
|
return this;
|
|
}
|
|
|
|
MemoryNodeRootProvider::MemoryNodeRootProvider()
|
|
{
|
|
SetExpanding(true);
|
|
}
|
|
|
|
MemoryNodeRootProvider::~MemoryNodeRootProvider()
|
|
{
|
|
}
|
|
|
|
Ptr<INodeProvider> MemoryNodeRootProvider::GetRootNode()
|
|
{
|
|
return this;
|
|
}
|
|
|
|
MemoryNodeProvider* MemoryNodeRootProvider::GetMemoryNode(INodeProvider* node)
|
|
{
|
|
return dynamic_cast<MemoryNodeProvider*>(node);
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiVirtualTreeListControl
|
|
***********************************************************************/
|
|
|
|
void GuiVirtualTreeListControl::BeforeControlTemplateUninstalled_()
|
|
{
|
|
}
|
|
|
|
void GuiVirtualTreeListControl::AfterControlTemplateInstalled_(bool initialize)
|
|
{
|
|
}
|
|
|
|
void GuiVirtualTreeListControl::OnAttached(tree::INodeRootProvider* provider)
|
|
{
|
|
}
|
|
|
|
void GuiVirtualTreeListControl::OnBeforeItemModified(tree::INodeProvider* parentNode, vint start, vint count, vint newCount)
|
|
{
|
|
}
|
|
|
|
void GuiVirtualTreeListControl::OnAfterItemModified(tree::INodeProvider* parentNode, vint start, vint count, vint newCount)
|
|
{
|
|
}
|
|
|
|
void GuiVirtualTreeListControl::OnItemExpanded(tree::INodeProvider* node)
|
|
{
|
|
GuiNodeEventArgs arguments;
|
|
(GuiEventArgs&)arguments=GetNotifyEventArguments();
|
|
arguments.node=node;
|
|
NodeExpanded.Execute(arguments);
|
|
}
|
|
|
|
void GuiVirtualTreeListControl::OnItemCollapsed(tree::INodeProvider* node)
|
|
{
|
|
GuiNodeEventArgs arguments;
|
|
(GuiEventArgs&)arguments=GetNotifyEventArguments();
|
|
arguments.node=node;
|
|
NodeCollapsed.Execute(arguments);
|
|
}
|
|
|
|
void GuiVirtualTreeListControl::OnItemMouseEvent(compositions::GuiNodeMouseEvent& nodeEvent, compositions::GuiGraphicsComposition* sender, compositions::GuiItemMouseEventArgs& arguments)
|
|
{
|
|
auto node = GetNodeItemView()->RequestNode(arguments.itemIndex);
|
|
if (node)
|
|
{
|
|
GuiNodeMouseEventArgs redirectArguments;
|
|
(GuiMouseEventArgs&)redirectArguments = arguments;
|
|
redirectArguments.node = node.Obj();
|
|
nodeEvent.Execute(redirectArguments);
|
|
(GuiMouseEventArgs&)arguments = redirectArguments;
|
|
}
|
|
}
|
|
|
|
void GuiVirtualTreeListControl::OnItemNotifyEvent(compositions::GuiNodeNotifyEvent& nodeEvent, compositions::GuiGraphicsComposition* sender, compositions::GuiItemEventArgs& arguments)
|
|
{
|
|
if (auto node = GetNodeItemView()->RequestNode(arguments.itemIndex))
|
|
{
|
|
GuiNodeEventArgs redirectArguments;
|
|
(GuiEventArgs&)redirectArguments = arguments;
|
|
redirectArguments.node = node.Obj();
|
|
nodeEvent.Execute(redirectArguments);
|
|
(GuiEventArgs&)arguments = redirectArguments;
|
|
}
|
|
}
|
|
|
|
#define ATTACH_ITEM_MOUSE_EVENT(NODEEVENTNAME, ITEMEVENTNAME)\
|
|
{\
|
|
Func<void(GuiNodeMouseEvent&, GuiGraphicsComposition*, GuiItemMouseEventArgs&)> func(this, &GuiVirtualTreeListControl::OnItemMouseEvent);\
|
|
ITEMEVENTNAME.AttachFunction(Curry(func)(NODEEVENTNAME));\
|
|
}\
|
|
|
|
#define ATTACH_ITEM_NOTIFY_EVENT(NODEEVENTNAME, ITEMEVENTNAME)\
|
|
{\
|
|
Func<void(GuiNodeNotifyEvent&, GuiGraphicsComposition*, GuiItemEventArgs&)> func(this, &GuiVirtualTreeListControl::OnItemNotifyEvent);\
|
|
ITEMEVENTNAME.AttachFunction(Curry(func)(NODEEVENTNAME));\
|
|
}\
|
|
|
|
void GuiVirtualTreeListControl::OnNodeLeftButtonDoubleClick(compositions::GuiGraphicsComposition* sender, compositions::GuiNodeMouseEventArgs& arguments)
|
|
{
|
|
if (arguments.node->GetChildCount() > 0)
|
|
{
|
|
arguments.node->SetExpanding(!arguments.node->GetExpanding());
|
|
}
|
|
}
|
|
|
|
GuiVirtualTreeListControl::GuiVirtualTreeListControl(theme::ThemeName themeName, Ptr<tree::INodeRootProvider> _nodeRootProvider)
|
|
:GuiSelectableListControl(themeName, new tree::NodeItemProvider(_nodeRootProvider))
|
|
{
|
|
nodeItemProvider = dynamic_cast<tree::NodeItemProvider*>(GetItemProvider());
|
|
nodeItemView = dynamic_cast<tree::INodeItemView*>(GetItemProvider()->RequestView(tree::INodeItemView::Identifier));
|
|
|
|
NodeLeftButtonDown.SetAssociatedComposition(boundsComposition);
|
|
NodeLeftButtonUp.SetAssociatedComposition(boundsComposition);
|
|
NodeLeftButtonDoubleClick.SetAssociatedComposition(boundsComposition);
|
|
NodeMiddleButtonDown.SetAssociatedComposition(boundsComposition);
|
|
NodeMiddleButtonUp.SetAssociatedComposition(boundsComposition);
|
|
NodeMiddleButtonDoubleClick.SetAssociatedComposition(boundsComposition);
|
|
NodeRightButtonDown.SetAssociatedComposition(boundsComposition);
|
|
NodeRightButtonUp.SetAssociatedComposition(boundsComposition);
|
|
NodeRightButtonDoubleClick.SetAssociatedComposition(boundsComposition);
|
|
NodeMouseMove.SetAssociatedComposition(boundsComposition);
|
|
NodeMouseEnter.SetAssociatedComposition(boundsComposition);
|
|
NodeMouseLeave.SetAssociatedComposition(boundsComposition);
|
|
NodeExpanded.SetAssociatedComposition(boundsComposition);
|
|
NodeCollapsed.SetAssociatedComposition(boundsComposition);
|
|
|
|
ATTACH_ITEM_MOUSE_EVENT(NodeLeftButtonDown, ItemLeftButtonDown);
|
|
ATTACH_ITEM_MOUSE_EVENT(NodeLeftButtonUp, ItemLeftButtonUp);
|
|
ATTACH_ITEM_MOUSE_EVENT(NodeLeftButtonDoubleClick, ItemLeftButtonDoubleClick);
|
|
ATTACH_ITEM_MOUSE_EVENT(NodeMiddleButtonDown, ItemMiddleButtonDown);
|
|
ATTACH_ITEM_MOUSE_EVENT(NodeMiddleButtonUp, ItemMiddleButtonUp);
|
|
ATTACH_ITEM_MOUSE_EVENT(NodeMiddleButtonDoubleClick, ItemMiddleButtonDoubleClick);
|
|
ATTACH_ITEM_MOUSE_EVENT(NodeRightButtonDown, ItemRightButtonDown);
|
|
ATTACH_ITEM_MOUSE_EVENT(NodeRightButtonUp, ItemRightButtonUp);
|
|
ATTACH_ITEM_MOUSE_EVENT(NodeRightButtonDoubleClick, ItemRightButtonDoubleClick);
|
|
ATTACH_ITEM_MOUSE_EVENT(NodeMouseMove, ItemMouseMove);
|
|
ATTACH_ITEM_NOTIFY_EVENT(NodeMouseEnter, ItemMouseEnter);
|
|
ATTACH_ITEM_NOTIFY_EVENT(NodeMouseLeave, ItemMouseLeave);
|
|
|
|
nodeItemProvider->GetRoot()->AttachCallback(this);
|
|
NodeLeftButtonDoubleClick.AttachMethod(this, &GuiVirtualTreeListControl::OnNodeLeftButtonDoubleClick);
|
|
}
|
|
|
|
#undef ATTACH_ITEM_MOUSE_EVENT
|
|
#undef ATTACH_ITEM_NOTIFY_EVENT
|
|
|
|
GuiVirtualTreeListControl::~GuiVirtualTreeListControl()
|
|
{
|
|
}
|
|
|
|
tree::INodeItemView* GuiVirtualTreeListControl::GetNodeItemView()
|
|
{
|
|
return nodeItemView;
|
|
}
|
|
|
|
tree::INodeRootProvider* GuiVirtualTreeListControl::GetNodeRootProvider()
|
|
{
|
|
return nodeItemProvider->GetRoot().Obj();
|
|
}
|
|
|
|
namespace tree
|
|
{
|
|
|
|
/***********************************************************************
|
|
TreeViewItem
|
|
***********************************************************************/
|
|
|
|
const wchar_t* const ITreeViewItemView::Identifier = L"vl::presentation::controls::tree::ITreeViewItemView";
|
|
|
|
TreeViewItem::TreeViewItem()
|
|
{
|
|
}
|
|
|
|
TreeViewItem::TreeViewItem(const Ptr<GuiImageData>& _image, const WString& _text)
|
|
:image(_image)
|
|
,text(_text)
|
|
{
|
|
}
|
|
|
|
/***********************************************************************
|
|
TreeViewItemRootProvider
|
|
***********************************************************************/
|
|
|
|
Ptr<GuiImageData> TreeViewItemRootProvider::GetNodeImage(INodeProvider* node)
|
|
{
|
|
MemoryNodeProvider* memoryNode=dynamic_cast<MemoryNodeProvider*>(node);
|
|
if(memoryNode)
|
|
{
|
|
Ptr<TreeViewItem> data=memoryNode->GetData().Cast<TreeViewItem>();
|
|
if(data)
|
|
{
|
|
return data->image;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
WString TreeViewItemRootProvider::GetTextValue(INodeProvider* node)
|
|
{
|
|
MemoryNodeProvider* memoryNode = dynamic_cast<MemoryNodeProvider*>(node);
|
|
if (memoryNode)
|
|
{
|
|
Ptr<TreeViewItem> data = memoryNode->GetData().Cast<TreeViewItem>();
|
|
if (data)
|
|
{
|
|
return data->text;
|
|
}
|
|
}
|
|
return L"";
|
|
}
|
|
|
|
description::Value TreeViewItemRootProvider::GetBindingValue(INodeProvider* node)
|
|
{
|
|
return Value::From(GetTreeViewData(node));
|
|
}
|
|
|
|
TreeViewItemRootProvider::TreeViewItemRootProvider()
|
|
{
|
|
}
|
|
|
|
TreeViewItemRootProvider::~TreeViewItemRootProvider()
|
|
{
|
|
}
|
|
|
|
IDescriptable* TreeViewItemRootProvider::RequestView(const WString& identifier)
|
|
{
|
|
if(identifier==ITreeViewItemView::Identifier)
|
|
{
|
|
return (ITreeViewItemView*)this;
|
|
}
|
|
else
|
|
{
|
|
return MemoryNodeRootProvider::RequestView(identifier);
|
|
}
|
|
}
|
|
|
|
Ptr<TreeViewItem> TreeViewItemRootProvider::GetTreeViewData(INodeProvider* node)
|
|
{
|
|
MemoryNodeProvider* memoryNode=GetMemoryNode(node);
|
|
if(memoryNode)
|
|
{
|
|
return memoryNode->GetData().Cast<TreeViewItem>();
|
|
}
|
|
else
|
|
{
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
void TreeViewItemRootProvider::SetTreeViewData(INodeProvider* node, Ptr<TreeViewItem> value)
|
|
{
|
|
MemoryNodeProvider* memoryNode=GetMemoryNode(node);
|
|
if(memoryNode)
|
|
{
|
|
memoryNode->SetData(value);
|
|
}
|
|
}
|
|
|
|
void TreeViewItemRootProvider::UpdateTreeViewData(INodeProvider* node)
|
|
{
|
|
MemoryNodeProvider* memoryNode=GetMemoryNode(node);
|
|
if(memoryNode)
|
|
{
|
|
memoryNode->NotifyDataModified();
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiVirtualTreeView
|
|
***********************************************************************/
|
|
|
|
templates::GuiTreeItemTemplate* GuiVirtualTreeView::GetStyleFromNode(tree::INodeProvider* node)
|
|
{
|
|
if (itemArranger)
|
|
{
|
|
vint index = nodeItemView->CalculateNodeVisibilityIndex(node);
|
|
if (index != -1)
|
|
{
|
|
auto style = itemArranger->GetVisibleStyle(index);
|
|
return dynamic_cast<templates::GuiTreeItemTemplate*>(style);
|
|
}
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
void GuiVirtualTreeView::SetStyleExpanding(tree::INodeProvider* node, bool expanding)
|
|
{
|
|
if (auto treeItemStyle = GetStyleFromNode(node))
|
|
{
|
|
treeItemStyle->SetExpanding(expanding);
|
|
}
|
|
}
|
|
|
|
void GuiVirtualTreeView::SetStyleExpandable(tree::INodeProvider* node, bool expandable)
|
|
{
|
|
if (auto treeItemStyle = GetStyleFromNode(node))
|
|
{
|
|
treeItemStyle->SetExpandable(expandable);
|
|
}
|
|
}
|
|
|
|
void GuiVirtualTreeView::OnAfterItemModified(tree::INodeProvider* parentNode, vint start, vint count, vint newCount)
|
|
{
|
|
GuiVirtualTreeListControl::OnAfterItemModified(parentNode, start, count, newCount);
|
|
SetStyleExpandable(parentNode, parentNode->GetChildCount() > 0);
|
|
}
|
|
|
|
void GuiVirtualTreeView::OnItemExpanded(tree::INodeProvider* node)
|
|
{
|
|
GuiVirtualTreeListControl::OnItemExpanded(node);
|
|
SetStyleExpanding(node, true);
|
|
}
|
|
|
|
void GuiVirtualTreeView::OnItemCollapsed(tree::INodeProvider* node)
|
|
{
|
|
GuiVirtualTreeListControl::OnItemCollapsed(node);
|
|
SetStyleExpanding(node, false);
|
|
}
|
|
|
|
void GuiVirtualTreeView::OnStyleInstalled(vint itemIndex, ItemStyle* style)
|
|
{
|
|
GuiVirtualTreeListControl::OnStyleInstalled(itemIndex, style);
|
|
if (auto treeItemStyle = dynamic_cast<templates::GuiTreeItemTemplate*>(style))
|
|
{
|
|
treeItemStyle->SetTextColor(GetControlTemplateObject(true)->GetTextColor());
|
|
|
|
if (treeViewItemView)
|
|
{
|
|
if (auto node = nodeItemView->RequestNode(itemIndex))
|
|
{
|
|
treeItemStyle->SetImage(treeViewItemView->GetNodeImage(node.Obj()));
|
|
treeItemStyle->SetExpanding(node->GetExpanding());
|
|
treeItemStyle->SetExpandable(node->GetChildCount() > 0);
|
|
{
|
|
vint level = -1;
|
|
auto current = node;
|
|
while (current->GetParent())
|
|
{
|
|
level++;
|
|
current = current->GetParent();
|
|
}
|
|
treeItemStyle->SetLevel(level);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
GuiVirtualTreeView::GuiVirtualTreeView(theme::ThemeName themeName, Ptr<tree::INodeRootProvider> _nodeRootProvider)
|
|
:GuiVirtualTreeListControl(themeName, _nodeRootProvider)
|
|
{
|
|
treeViewItemView = dynamic_cast<tree::ITreeViewItemView*>(GetNodeRootProvider()->RequestView(tree::ITreeViewItemView::Identifier));
|
|
SetStyleAndArranger(
|
|
[](const Value&) { return new tree::DefaultTreeItemTemplate; },
|
|
new list::FixedHeightItemArranger
|
|
);
|
|
}
|
|
|
|
GuiVirtualTreeView::~GuiVirtualTreeView()
|
|
{
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiTreeView
|
|
***********************************************************************/
|
|
|
|
GuiTreeView::GuiTreeView(theme::ThemeName themeName)
|
|
:GuiVirtualTreeView(themeName, new tree::TreeViewItemRootProvider)
|
|
{
|
|
nodes = nodeItemProvider->GetRoot().Cast<tree::TreeViewItemRootProvider>();
|
|
}
|
|
|
|
GuiTreeView::~GuiTreeView()
|
|
{
|
|
}
|
|
|
|
Ptr<tree::TreeViewItemRootProvider> GuiTreeView::Nodes()
|
|
{
|
|
return nodes;
|
|
}
|
|
|
|
Ptr<tree::TreeViewItem> GuiTreeView::GetSelectedItem()
|
|
{
|
|
Ptr<tree::TreeViewItem> result;
|
|
vint index = GetSelectedItemIndex();
|
|
if (index != -1)
|
|
{
|
|
if (auto node = nodeItemView->RequestNode(index))
|
|
{
|
|
if (auto memoryNode = node.Cast<tree::MemoryNodeProvider>())
|
|
{
|
|
result = memoryNode->GetData().Cast<tree::TreeViewItem>();
|
|
}
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
namespace tree
|
|
{
|
|
|
|
/***********************************************************************
|
|
DefaultTreeItemTemplate
|
|
***********************************************************************/
|
|
|
|
void DefaultTreeItemTemplate::OnInitialize()
|
|
{
|
|
templates::GuiTreeItemTemplate::OnInitialize();
|
|
SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
|
|
table = new GuiTableComposition;
|
|
AddChild(table);
|
|
table->SetRowsAndColumns(3, 4);
|
|
table->SetRowOption(0, GuiCellOption::PercentageOption(0.5));
|
|
table->SetRowOption(1, GuiCellOption::MinSizeOption());
|
|
table->SetRowOption(2, GuiCellOption::PercentageOption(0.5));
|
|
table->SetColumnOption(0, GuiCellOption::AbsoluteOption(0));
|
|
table->SetColumnOption(1, GuiCellOption::MinSizeOption());
|
|
table->SetColumnOption(2, GuiCellOption::MinSizeOption());
|
|
table->SetColumnOption(3, GuiCellOption::MinSizeOption());
|
|
table->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
table->SetCellPadding(2);
|
|
{
|
|
GuiCellComposition* cell = new GuiCellComposition;
|
|
table->AddChild(cell);
|
|
cell->SetSite(0, 1, 3, 1);
|
|
cell->SetPreferredMinSize(Size(16, 16));
|
|
|
|
expandingButton = new GuiSelectableButton(theme::ThemeName::TreeItemExpander);
|
|
if (auto treeView = dynamic_cast<GuiVirtualTreeView*>(listControl))
|
|
{
|
|
if (auto expanderStyle = treeView->GetControlTemplateObject(true)->GetExpandingDecoratorTemplate())
|
|
{
|
|
expandingButton->SetControlTemplate(expanderStyle);
|
|
}
|
|
}
|
|
expandingButton->SetAutoFocus(false);
|
|
expandingButton->SetAutoSelection(false);
|
|
expandingButton->GetBoundsComposition()->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
expandingButton->GetBoundsComposition()->GetEventReceiver()->leftButtonDoubleClick.AttachMethod(this, &DefaultTreeItemTemplate::OnExpandingButtonDoubleClick);
|
|
expandingButton->Clicked.AttachMethod(this, &DefaultTreeItemTemplate::OnExpandingButtonClicked);
|
|
cell->AddChild(expandingButton->GetBoundsComposition());
|
|
}
|
|
{
|
|
GuiCellComposition* cell = new GuiCellComposition;
|
|
table->AddChild(cell);
|
|
cell->SetSite(1, 2, 1, 1);
|
|
cell->SetPreferredMinSize(Size(16, 16));
|
|
|
|
imageElement = GuiImageFrameElement::Create();
|
|
imageElement->SetStretch(true);
|
|
cell->SetOwnedElement(imageElement);
|
|
}
|
|
{
|
|
GuiCellComposition* cell = new GuiCellComposition;
|
|
table->AddChild(cell);
|
|
cell->SetSite(0, 3, 3, 1);
|
|
cell->SetPreferredMinSize(Size(192, 0));
|
|
|
|
textElement = GuiSolidLabelElement::Create();
|
|
textElement->SetAlignments(Alignment::Left, Alignment::Center);
|
|
textElement->SetEllipse(true);
|
|
cell->SetOwnedElement(textElement);
|
|
}
|
|
|
|
FontChanged.AttachMethod(this, &DefaultTreeItemTemplate::OnFontChanged);
|
|
TextChanged.AttachMethod(this, &DefaultTreeItemTemplate::OnTextChanged);
|
|
TextColorChanged.AttachMethod(this, &DefaultTreeItemTemplate::OnTextColorChanged);
|
|
ExpandingChanged.AttachMethod(this, &DefaultTreeItemTemplate::OnExpandingChanged);
|
|
ExpandableChanged.AttachMethod(this, &DefaultTreeItemTemplate::OnExpandableChanged);
|
|
LevelChanged.AttachMethod(this, &DefaultTreeItemTemplate::OnLevelChanged);
|
|
ImageChanged.AttachMethod(this, &DefaultTreeItemTemplate::OnImageChanged);
|
|
|
|
FontChanged.Execute(compositions::GuiEventArgs(this));
|
|
TextChanged.Execute(compositions::GuiEventArgs(this));
|
|
TextColorChanged.Execute(compositions::GuiEventArgs(this));
|
|
ExpandingChanged.Execute(compositions::GuiEventArgs(this));
|
|
ExpandableChanged.Execute(compositions::GuiEventArgs(this));
|
|
LevelChanged.Execute(compositions::GuiEventArgs(this));
|
|
ImageChanged.Execute(compositions::GuiEventArgs(this));
|
|
}
|
|
|
|
void DefaultTreeItemTemplate::OnFontChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
textElement->SetFont(GetFont());
|
|
}
|
|
|
|
void DefaultTreeItemTemplate::OnTextChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
textElement->SetText(GetText());
|
|
}
|
|
|
|
void DefaultTreeItemTemplate::OnTextColorChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
textElement->SetColor(GetTextColor());
|
|
}
|
|
|
|
void DefaultTreeItemTemplate::OnExpandingChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
expandingButton->SetSelected(GetExpanding());
|
|
}
|
|
|
|
void DefaultTreeItemTemplate::OnExpandableChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
expandingButton->SetVisible(GetExpandable());
|
|
}
|
|
|
|
void DefaultTreeItemTemplate::OnLevelChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
table->SetColumnOption(0, GuiCellOption::AbsoluteOption(GetLevel() * 12));
|
|
}
|
|
|
|
void DefaultTreeItemTemplate::OnImageChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
if (auto imageData = GetImage())
|
|
{
|
|
imageElement->SetImage(imageData->GetImage(), imageData->GetFrameIndex());
|
|
}
|
|
else
|
|
{
|
|
imageElement->SetImage(nullptr);
|
|
}
|
|
}
|
|
|
|
void DefaultTreeItemTemplate::OnExpandingButtonDoubleClick(compositions::GuiGraphicsComposition* sender, compositions::GuiMouseEventArgs& arguments)
|
|
{
|
|
arguments.handled = true;
|
|
}
|
|
|
|
void DefaultTreeItemTemplate::OnExpandingButtonClicked(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
if (expandingButton->GetVisuallyEnabled())
|
|
{
|
|
if (auto treeControl = dynamic_cast<GuiVirtualTreeListControl*>(listControl))
|
|
{
|
|
if (auto view = treeControl->GetNodeItemView())
|
|
{
|
|
vint index = treeControl->GetArranger()->GetVisibleIndex(this);
|
|
if (index != -1)
|
|
{
|
|
if (auto node = view->RequestNode(index))
|
|
{
|
|
bool expanding = node->GetExpanding();
|
|
node->SetExpanding(!expanding);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
DefaultTreeItemTemplate::DefaultTreeItemTemplate()
|
|
{
|
|
}
|
|
|
|
DefaultTreeItemTemplate::~DefaultTreeItemTemplate()
|
|
{
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\CONTROLS\TEMPLATES\GUIANIMATION.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace controls
|
|
{
|
|
using namespace collections;
|
|
using namespace reflection::description;
|
|
|
|
/***********************************************************************
|
|
GuiTimedAnimation
|
|
***********************************************************************/
|
|
|
|
class GuiTimedAnimation : public Object, public virtual IGuiAnimation
|
|
{
|
|
protected:
|
|
DateTime startTime;
|
|
vuint64_t time;
|
|
bool running = false;
|
|
|
|
public:
|
|
GuiTimedAnimation()
|
|
{
|
|
}
|
|
|
|
~GuiTimedAnimation()
|
|
{
|
|
}
|
|
|
|
void Start()override
|
|
{
|
|
startTime = DateTime::LocalTime();
|
|
time = 0;
|
|
running = true;
|
|
}
|
|
|
|
void Pause()override
|
|
{
|
|
time = GetTime();
|
|
running = false;
|
|
}
|
|
|
|
void Resume()override
|
|
{
|
|
startTime = DateTime::LocalTime();
|
|
running = true;
|
|
}
|
|
|
|
vuint64_t GetTime()
|
|
{
|
|
if (running)
|
|
{
|
|
return time + (DateTime::LocalTime().totalMilliseconds - startTime.totalMilliseconds);
|
|
}
|
|
else
|
|
{
|
|
return time;
|
|
}
|
|
}
|
|
};
|
|
|
|
/***********************************************************************
|
|
GuiFiniteAnimation
|
|
***********************************************************************/
|
|
|
|
class GuiFiniteAnimation : public GuiTimedAnimation
|
|
{
|
|
protected:
|
|
vuint64_t length = 0;
|
|
vuint64_t currentTime = 0;
|
|
Func<void(vuint64_t)> run;
|
|
|
|
public:
|
|
GuiFiniteAnimation(const Func<void(vuint64_t)>& _run, vuint64_t _length)
|
|
:run(_run)
|
|
, length(_length)
|
|
{
|
|
}
|
|
|
|
~GuiFiniteAnimation()
|
|
{
|
|
}
|
|
|
|
void Run()override
|
|
{
|
|
currentTime = GetTime();
|
|
if (currentTime < length && run)
|
|
{
|
|
run(currentTime);
|
|
}
|
|
}
|
|
|
|
bool GetStopped()override
|
|
{
|
|
return currentTime >= length;
|
|
}
|
|
};
|
|
|
|
/***********************************************************************
|
|
GuiInfiniteAnimation
|
|
***********************************************************************/
|
|
|
|
class GuiInfiniteAnimation : public GuiTimedAnimation
|
|
{
|
|
protected:
|
|
Func<void(vuint64_t)> run;
|
|
|
|
public:
|
|
GuiInfiniteAnimation(const Func<void(vuint64_t)>& _run)
|
|
:run(_run)
|
|
{
|
|
}
|
|
|
|
~GuiInfiniteAnimation()
|
|
{
|
|
}
|
|
|
|
void Run()override
|
|
{
|
|
if (run)
|
|
{
|
|
run(GetTime());
|
|
}
|
|
}
|
|
|
|
bool GetStopped()override
|
|
{
|
|
return false;
|
|
}
|
|
};
|
|
|
|
/***********************************************************************
|
|
IGuiAnimation
|
|
***********************************************************************/
|
|
|
|
Ptr<IGuiAnimation> IGuiAnimation::CreateAnimation(const Func<void(vuint64_t)>& run, vuint64_t milliseconds)
|
|
{
|
|
return new GuiFiniteAnimation(run, milliseconds);
|
|
}
|
|
|
|
Ptr<IGuiAnimation> IGuiAnimation::CreateAnimation(const Func<void(vuint64_t)>& run)
|
|
{
|
|
return new GuiInfiniteAnimation(run);
|
|
}
|
|
|
|
/***********************************************************************
|
|
IGuiAnimationCoroutine
|
|
***********************************************************************/
|
|
|
|
class GuiCoroutineAnimation : public Object, public virtual IGuiAnimationCoroutine::IImpl
|
|
{
|
|
protected:
|
|
IGuiAnimationCoroutine::Creator creator;
|
|
Ptr<ICoroutine> coroutine;
|
|
|
|
Ptr<IGuiAnimation> waitingAnimation;
|
|
vint waitingGroup = -1;
|
|
Group<vint, Ptr<IGuiAnimation>> groupAnimations;
|
|
|
|
public:
|
|
GuiCoroutineAnimation(const IGuiAnimationCoroutine::Creator& _creator)
|
|
:creator(_creator)
|
|
{
|
|
}
|
|
|
|
~GuiCoroutineAnimation()
|
|
{
|
|
}
|
|
|
|
void OnPlayAndWait(Ptr<IGuiAnimation> animation)override
|
|
{
|
|
CHECK_ERROR(!waitingAnimation && waitingGroup == -1, L"GuiCoroutineAnimation::OnPlayAndWait(Ptr<IGuiAnimation>)#Cannot be called when an animation or a group has already been waiting for.");
|
|
waitingAnimation = animation;
|
|
waitingAnimation->Start();
|
|
}
|
|
|
|
void OnPlayInGroup(Ptr<IGuiAnimation> animation, vint groupId)override
|
|
{
|
|
groupAnimations.Add(groupId, animation);
|
|
animation->Start();
|
|
}
|
|
|
|
void OnWaitForGroup(vint groupId)override
|
|
{
|
|
CHECK_ERROR(!waitingAnimation && waitingGroup == -1, L"GuiCoroutineAnimation::OnWaitForGroup(vint)#Cannot be called when an animation or a group has already been waiting for.");
|
|
if (groupAnimations.Keys().Contains(groupId))
|
|
{
|
|
waitingGroup = groupId;
|
|
}
|
|
}
|
|
|
|
void Start()override
|
|
{
|
|
CHECK_ERROR(!coroutine, L"GuiCoroutineAnimation::Start()#Cannot be called more than once.");
|
|
coroutine = creator(this);
|
|
}
|
|
|
|
void Pause()override
|
|
{
|
|
if (waitingAnimation)
|
|
{
|
|
waitingAnimation->Pause();
|
|
}
|
|
for (vint i = 0; i < groupAnimations.Count(); i++)
|
|
{
|
|
FOREACH(Ptr<IGuiAnimation>, animation, groupAnimations.GetByIndex(i))
|
|
{
|
|
animation->Pause();
|
|
}
|
|
}
|
|
}
|
|
|
|
void Resume()override
|
|
{
|
|
if (waitingAnimation)
|
|
{
|
|
waitingAnimation->Resume();
|
|
}
|
|
for (vint i = 0; i < groupAnimations.Count(); i++)
|
|
{
|
|
FOREACH(Ptr<IGuiAnimation>, animation, groupAnimations.GetByIndex(i))
|
|
{
|
|
animation->Resume();
|
|
}
|
|
}
|
|
}
|
|
|
|
void Run()override
|
|
{
|
|
CHECK_ERROR(coroutine, L"GuiCoroutineAnimation::Run()#Cannot be called before calling Start.");
|
|
|
|
if (waitingAnimation)
|
|
{
|
|
waitingAnimation->Run();
|
|
if (waitingAnimation->GetStopped())
|
|
{
|
|
waitingAnimation = nullptr;
|
|
}
|
|
}
|
|
|
|
for (vint i = groupAnimations.Count() - 1; i >= 0; i--)
|
|
{
|
|
auto& animations = groupAnimations.GetByIndex(i);
|
|
for (vint j = animations.Count() - 1; j >= 0; j--)
|
|
{
|
|
auto animation = animations[j];
|
|
animation->Run();
|
|
if (animation->GetStopped())
|
|
{
|
|
groupAnimations.Remove(i, animation.Obj());
|
|
}
|
|
}
|
|
}
|
|
|
|
if (waitingGroup != -1 && !groupAnimations.Keys().Contains(waitingGroup))
|
|
{
|
|
waitingGroup = -1;
|
|
}
|
|
|
|
if (coroutine->GetStatus() == CoroutineStatus::Waiting)
|
|
{
|
|
if (waitingAnimation || waitingGroup != -1)
|
|
{
|
|
return;
|
|
}
|
|
coroutine->Resume(true, nullptr);
|
|
}
|
|
}
|
|
|
|
bool GetStopped()override
|
|
{
|
|
if (!coroutine) return false;
|
|
if (coroutine->GetStatus() != CoroutineStatus::Stopped) return false;
|
|
if (waitingAnimation || groupAnimations.Count() > 0) return false;
|
|
return true;
|
|
}
|
|
};
|
|
|
|
void IGuiAnimationCoroutine::WaitAndPause(IImpl* impl, vuint64_t milliseconds)
|
|
{
|
|
return PlayAndWaitAndPause(impl, IGuiAnimation::CreateAnimation({}, milliseconds));
|
|
}
|
|
|
|
void IGuiAnimationCoroutine::PlayAndWaitAndPause(IImpl* impl, Ptr<IGuiAnimation> animation)
|
|
{
|
|
impl->OnPlayAndWait(animation);
|
|
}
|
|
|
|
void IGuiAnimationCoroutine::PlayInGroupAndPause(IImpl* impl, Ptr<IGuiAnimation> animation, vint groupId)
|
|
{
|
|
impl->OnPlayInGroup(animation, groupId);
|
|
}
|
|
|
|
void IGuiAnimationCoroutine::WaitForGroupAndPause(IImpl* impl, vint groupId)
|
|
{
|
|
impl->OnWaitForGroup(groupId);
|
|
}
|
|
|
|
void IGuiAnimationCoroutine::ReturnAndExit(IImpl* impl)
|
|
{
|
|
}
|
|
|
|
Ptr<IGuiAnimation> IGuiAnimationCoroutine::Create(const Creator& creator)
|
|
{
|
|
return new GuiCoroutineAnimation(creator);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\CONTROLS\TEMPLATES\GUICOMMONTEMPLATES.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace templates
|
|
{
|
|
using namespace elements;
|
|
using namespace compositions;
|
|
using namespace templates;
|
|
using namespace controls;
|
|
using namespace theme;
|
|
|
|
/***********************************************************************
|
|
GuiCommonDatePickerLook
|
|
***********************************************************************/
|
|
|
|
vint GetDayCountForMonth(vint year, vint month)
|
|
{
|
|
bool isLeapYear = (year % 100 == 0) ? (year % 400 == 0) : (year % 4 == 0);
|
|
switch (month)
|
|
{
|
|
case 1:case 3:case 5:case 7:case 8:case 10:case 12:
|
|
return 31;
|
|
case 4:case 6:case 9:case 11:
|
|
return 30;
|
|
default:
|
|
return isLeapYear ? 29 : 28;
|
|
}
|
|
}
|
|
|
|
void StepPreviousMonth(vint& year, vint& month)
|
|
{
|
|
if (month == 1)
|
|
{
|
|
year--;
|
|
month = 12;
|
|
}
|
|
else
|
|
{
|
|
month--;
|
|
}
|
|
}
|
|
|
|
void StepNextMonth(vint& year, vint& month)
|
|
{
|
|
if (month == 12)
|
|
{
|
|
year++;
|
|
month = 1;
|
|
}
|
|
else
|
|
{
|
|
month++;
|
|
}
|
|
}
|
|
|
|
void GuiCommonDatePickerLook::SetDay(const DateTime& day, vint& index, vint monthOffset)
|
|
{
|
|
dateDays[index] = day;
|
|
GuiSolidLabelElement* label = labelDays[index];
|
|
label->SetText(itow(day.day));
|
|
label->SetColor(monthOffset == 0 ? primaryTextColor : secondaryTextColor);
|
|
|
|
wchar_t alt[] = L"D00";
|
|
if (monthOffset == -1) alt[0] = L'C';
|
|
else if (monthOffset == 1) alt[0] = L'E';
|
|
alt[1] = (wchar_t)(L'0' + day.day / 10);
|
|
alt[2] = (wchar_t)(L'0' + day.day % 10);
|
|
buttonDays[index]->SetAlt(alt);
|
|
|
|
index++;
|
|
}
|
|
|
|
void GuiCommonDatePickerLook::comboYearMonth_SelectedIndexChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
if (!preventComboEvent)
|
|
{
|
|
if (comboYear->GetSelectedIndex() != -1 && comboMonth->GetSelectedIndex() != -1)
|
|
{
|
|
vint year = comboYear->GetSelectedIndex() + YearFirst;
|
|
vint month = comboMonth->GetSelectedIndex() + 1;
|
|
SetDate(DateTime::FromDateTime(year, month, 1));
|
|
|
|
GuiEventArgs arguments(this);
|
|
DateChanged.Execute(arguments);
|
|
commands->NotifyDateChanged();
|
|
commands->NotifyDateNavigated();
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiCommonDatePickerLook::buttonDay_SelectedChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
if (!preventButtonEvent)
|
|
{
|
|
GuiSelectableButton* button = dynamic_cast<GuiSelectableButton*>(sender->GetRelatedControl());
|
|
if (button->GetSelected())
|
|
{
|
|
vint index = buttonDays.IndexOf(button);
|
|
if (index != -1)
|
|
{
|
|
DateTime day = dateDays[index];
|
|
if (day.year != currentDate.year || day.month != currentDate.month)
|
|
{
|
|
SetDate(day);
|
|
}
|
|
else
|
|
{
|
|
currentDate = day;
|
|
}
|
|
|
|
GuiEventArgs arguments(this);
|
|
DateChanged.Execute(arguments);
|
|
commands->NotifyDateChanged();
|
|
commands->NotifyDateSelected();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiCommonDatePickerLook::DisplayMonth(vint year, vint month)
|
|
{
|
|
if (YearFirst <= year && year <= YearLast && 1 <= month && month <= 12)
|
|
{
|
|
preventComboEvent = true;
|
|
comboYear->SetSelectedIndex(year - YearFirst);
|
|
comboMonth->SetSelectedIndex(month - 1);
|
|
preventComboEvent = false;
|
|
}
|
|
|
|
vint yearPrev = year, yearNext = year, monthPrev = month, monthNext = month;
|
|
StepPreviousMonth(yearPrev, monthPrev);
|
|
StepNextMonth(yearNext, monthNext);
|
|
|
|
vint countPrev = GetDayCountForMonth(yearPrev, monthPrev);
|
|
vint count = GetDayCountForMonth(year, month);
|
|
vint countNext = GetDayCountForMonth(yearNext, monthNext);
|
|
|
|
DateTime firstDay = DateTime::FromDateTime(year, month, 1);
|
|
vint showPrev = firstDay.dayOfWeek;
|
|
if (showPrev == 0) showPrev = DaysOfWeek;
|
|
vint show = count;
|
|
vint showNext = DaysOfWeek*DayRows - showPrev - show;
|
|
|
|
vint index = 0;
|
|
for (vint i = 0; i < showPrev; i++)
|
|
{
|
|
DateTime day = DateTime::FromDateTime(yearPrev, monthPrev, countPrev - (showPrev - i - 1));
|
|
SetDay(day, index, -1);
|
|
}
|
|
for (vint i = 0; i < show; i++)
|
|
{
|
|
DateTime day = DateTime::FromDateTime(year, month, i + 1);
|
|
SetDay(day, index, 0);
|
|
}
|
|
for (vint i = 0; i < showNext; i++)
|
|
{
|
|
DateTime day = DateTime::FromDateTime(yearNext, monthNext, i + 1);
|
|
SetDay(day, index, 1);
|
|
}
|
|
}
|
|
|
|
void GuiCommonDatePickerLook::SelectDay(vint day)
|
|
{
|
|
for (vint i = 0; i < dateDays.Count(); i++)
|
|
{
|
|
const DateTime& dt = dateDays[i];
|
|
if (dt.year == currentDate.year && dt.month == currentDate.month && dt.day == day)
|
|
{
|
|
preventButtonEvent = true;
|
|
buttonDays[i]->SetSelected(true);
|
|
preventButtonEvent = false;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
GuiCommonDatePickerLook::GuiCommonDatePickerLook(Color _backgroundColor, Color _primaryTextColor, Color _secondaryTextColor)
|
|
:backgroundColor(_backgroundColor)
|
|
, primaryTextColor(_primaryTextColor)
|
|
, secondaryTextColor(_secondaryTextColor)
|
|
{
|
|
DateChanged.SetAssociatedComposition(this);
|
|
SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
|
|
GuiTableComposition* monthTable = 0;
|
|
GuiTableComposition* dayTable = 0;
|
|
{
|
|
listYears = new GuiTextList(theme::ThemeName::TextList);
|
|
listYears->SetHorizontalAlwaysVisible(false);
|
|
for (vint i = YearFirst; i <= YearLast; i++)
|
|
{
|
|
listYears->GetItems().Add(new list::TextItem(itow(i)));
|
|
}
|
|
comboYear = new GuiComboBoxListControl(theme::ThemeName::ComboBox, listYears);
|
|
comboYear->SetAlt(L"Y");
|
|
comboYear->GetBoundsComposition()->SetAlignmentToParent(Margin(0, 0, 2, 0));
|
|
comboYear->SelectedIndexChanged.AttachMethod(this, &GuiCommonDatePickerLook::comboYearMonth_SelectedIndexChanged);
|
|
}
|
|
{
|
|
listMonths = new GuiTextList(theme::ThemeName::TextList);
|
|
listMonths->SetHorizontalAlwaysVisible(false);
|
|
comboMonth = new GuiComboBoxListControl(theme::ThemeName::ComboBox, listMonths);
|
|
comboMonth->SetAlt(L"M");
|
|
comboMonth->GetBoundsComposition()->SetAlignmentToParent(Margin(2, 0, 0, 0));
|
|
comboMonth->SelectedIndexChanged.AttachMethod(this, &GuiCommonDatePickerLook::comboYearMonth_SelectedIndexChanged);
|
|
}
|
|
{
|
|
monthTable = new GuiTableComposition;
|
|
monthTable->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
monthTable->SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
monthTable->SetRowsAndColumns(1, 2);
|
|
monthTable->SetRowOption(0, GuiCellOption::MinSizeOption());
|
|
monthTable->SetColumnOption(0, GuiCellOption::PercentageOption(0.5));
|
|
monthTable->SetColumnOption(1, GuiCellOption::PercentageOption(0.5));
|
|
{
|
|
GuiCellComposition* cell = new GuiCellComposition;
|
|
monthTable->AddChild(cell);
|
|
cell->SetSite(0, 0, 1, 1);
|
|
cell->AddChild(comboYear->GetBoundsComposition());
|
|
}
|
|
{
|
|
GuiCellComposition* cell = new GuiCellComposition;
|
|
monthTable->AddChild(cell);
|
|
cell->SetSite(0, 1, 1, 1);
|
|
cell->AddChild(comboMonth->GetBoundsComposition());
|
|
}
|
|
}
|
|
{
|
|
dayTable = new GuiTableComposition;
|
|
dayTable->SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
dayTable->SetCellPadding(4);
|
|
dayTable->SetRowsAndColumns(DayRows + DayRowStart, DaysOfWeek);
|
|
|
|
for (vint i = 0; i < DayRowStart; i++)
|
|
{
|
|
dayTable->SetRowOption(i, GuiCellOption::MinSizeOption());
|
|
}
|
|
for (vint i = 0; i < DayRows; i++)
|
|
{
|
|
dayTable->SetRowOption(i + DayRowStart, GuiCellOption::PercentageOption(1.0));
|
|
}
|
|
for (vint i = 0; i < DaysOfWeek; i++)
|
|
{
|
|
dayTable->SetColumnOption(i, GuiCellOption::PercentageOption(1.0));
|
|
}
|
|
|
|
{
|
|
GuiCellComposition* cell = new GuiCellComposition;
|
|
dayTable->AddChild(cell);
|
|
cell->SetSite(0, 0, 1, DaysOfWeek);
|
|
cell->AddChild(monthTable);
|
|
}
|
|
|
|
labelDaysOfWeek.Resize(7);
|
|
for (vint i = 0; i < DaysOfWeek; i++)
|
|
{
|
|
GuiCellComposition* cell = new GuiCellComposition;
|
|
dayTable->AddChild(cell);
|
|
cell->SetSite(1, i, 1, 1);
|
|
|
|
GuiSolidLabelElement* element = GuiSolidLabelElement::Create();
|
|
element->SetAlignments(Alignment::Center, Alignment::Center);
|
|
element->SetColor(primaryTextColor);
|
|
labelDaysOfWeek[i] = element;
|
|
cell->SetOwnedElement(element);
|
|
}
|
|
|
|
buttonDays.Resize(DaysOfWeek*DayRows);
|
|
labelDays.Resize(DaysOfWeek*DayRows);
|
|
dateDays.Resize(DaysOfWeek*DayRows);
|
|
|
|
auto dayMutexController = new GuiSelectableButton::MutexGroupController;
|
|
AddComponent(dayMutexController);
|
|
|
|
for (vint i = 0; i < DaysOfWeek; i++)
|
|
{
|
|
for (vint j = 0; j < DayRows; j++)
|
|
{
|
|
GuiCellComposition* cell = new GuiCellComposition;
|
|
dayTable->AddChild(cell);
|
|
cell->SetSite(j + DayRowStart, i, 1, 1);
|
|
|
|
GuiSelectableButton* button = new GuiSelectableButton(theme::ThemeName::CheckBox);
|
|
button->GetBoundsComposition()->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
button->SetGroupController(dayMutexController);
|
|
button->SelectedChanged.AttachMethod(this, &GuiCommonDatePickerLook::buttonDay_SelectedChanged);
|
|
cell->AddChild(button->GetBoundsComposition());
|
|
buttonDays[j*DaysOfWeek + i] = button;
|
|
|
|
GuiSolidLabelElement* element = GuiSolidLabelElement::Create();
|
|
element->SetAlignments(Alignment::Center, Alignment::Center);
|
|
element->SetText(L"0");
|
|
labelDays[j*DaysOfWeek + i] = element;
|
|
|
|
GuiBoundsComposition* elementBounds = new GuiBoundsComposition;
|
|
elementBounds->SetOwnedElement(element);
|
|
elementBounds->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
elementBounds->SetMinSizeLimitation(GuiGraphicsComposition::LimitToElement);
|
|
button->GetContainerComposition()->AddChild(elementBounds);
|
|
}
|
|
}
|
|
}
|
|
{
|
|
GuiSolidBackgroundElement* element = GuiSolidBackgroundElement::Create();
|
|
element->SetColor(backgroundColor);
|
|
dayTable->SetOwnedElement(element);
|
|
}
|
|
|
|
dayTable->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
AddChild(dayTable);
|
|
|
|
SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
SetFont(font);
|
|
}
|
|
|
|
GuiCommonDatePickerLook::~GuiCommonDatePickerLook()
|
|
{
|
|
FinalizeInstanceRecursively(this);
|
|
}
|
|
|
|
controls::IDatePickerCommandExecutor* GuiCommonDatePickerLook::GetCommands()
|
|
{
|
|
return commands;
|
|
}
|
|
|
|
void GuiCommonDatePickerLook::SetCommands(controls::IDatePickerCommandExecutor* value)
|
|
{
|
|
commands = value;
|
|
}
|
|
|
|
TemplateProperty<GuiSelectableButtonTemplate> GuiCommonDatePickerLook::GetDateButtonTemplate()
|
|
{
|
|
return dateButtonTemplate;
|
|
}
|
|
|
|
void GuiCommonDatePickerLook::SetDateButtonTemplate(const TemplateProperty<GuiSelectableButtonTemplate>& value)
|
|
{
|
|
dateButtonTemplate = value;
|
|
FOREACH(GuiSelectableButton*, button, buttonDays)
|
|
{
|
|
button->SetControlTemplate(value);
|
|
}
|
|
}
|
|
|
|
TemplateProperty<GuiTextListTemplate> GuiCommonDatePickerLook::GetDateTextListTemplate()
|
|
{
|
|
return dateTextListTemplate;
|
|
}
|
|
|
|
void GuiCommonDatePickerLook::SetDateTextListTemplate(const TemplateProperty<GuiTextListTemplate>& value)
|
|
{
|
|
dateTextListTemplate = value;
|
|
listYears->SetControlTemplate(value);
|
|
listMonths->SetControlTemplate(value);
|
|
}
|
|
|
|
TemplateProperty<GuiComboBoxTemplate> GuiCommonDatePickerLook::GetDateComboBoxTemplate()
|
|
{
|
|
return dateComboBoxTemplate;
|
|
}
|
|
|
|
void GuiCommonDatePickerLook::SetDateComboBoxTemplate(const TemplateProperty<GuiComboBoxTemplate>& value)
|
|
{
|
|
dateComboBoxTemplate = value;
|
|
comboYear->SetControlTemplate(value);
|
|
comboMonth->SetControlTemplate(value);
|
|
}
|
|
|
|
const Locale& GuiCommonDatePickerLook::GetDateLocale()
|
|
{
|
|
return dateLocale;
|
|
}
|
|
|
|
void GuiCommonDatePickerLook::SetDateLocale(const Locale& value)
|
|
{
|
|
if (dateLocale != value)
|
|
{
|
|
dateLocale = value;
|
|
for (vint i = 0; i < DaysOfWeek; i++)
|
|
{
|
|
labelDaysOfWeek[i]->SetText(dateLocale.GetShortDayOfWeekName(i));
|
|
}
|
|
|
|
listMonths->GetItems().Clear();
|
|
for (vint i = 1; i <= 12; i++)
|
|
{
|
|
listMonths->GetItems().Add(new list::TextItem(dateLocale.GetLongMonthName(i)));
|
|
}
|
|
|
|
SetDate(currentDate);
|
|
}
|
|
}
|
|
|
|
const DateTime& GuiCommonDatePickerLook::GetDate()
|
|
{
|
|
return currentDate;
|
|
}
|
|
|
|
void GuiCommonDatePickerLook::SetDate(const DateTime& value)
|
|
{
|
|
currentDate = value;
|
|
DisplayMonth(value.year, value.month);
|
|
SelectDay(value.day);
|
|
}
|
|
|
|
const FontProperties& GuiCommonDatePickerLook::GetFont()
|
|
{
|
|
return font;
|
|
}
|
|
|
|
void GuiCommonDatePickerLook::SetFont(const FontProperties& value)
|
|
{
|
|
if (font != value)
|
|
{
|
|
font = value;
|
|
comboYear->SetFont(value);
|
|
listYears->SetFont(value);
|
|
comboMonth->SetFont(value);
|
|
listMonths->SetFont(value);
|
|
FOREACH(GuiSolidLabelElement*, label, From(labelDaysOfWeek).Concat(labelDays))
|
|
{
|
|
label->SetFont(value);
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiCommonScrollViewLook
|
|
***********************************************************************/
|
|
|
|
void GuiCommonScrollViewLook::UpdateTable()
|
|
{
|
|
if (horizontalScroll->GetVisible())
|
|
{
|
|
tableComposition->SetRowOption(1, GuiCellOption::AbsoluteOption(defaultScrollSize));
|
|
}
|
|
else
|
|
{
|
|
tableComposition->SetRowOption(1, GuiCellOption::AbsoluteOption(0));
|
|
}
|
|
|
|
if (verticalScroll->GetVisible())
|
|
{
|
|
tableComposition->SetColumnOption(1, GuiCellOption::AbsoluteOption(defaultScrollSize));
|
|
}
|
|
else
|
|
{
|
|
tableComposition->SetColumnOption(1, GuiCellOption::AbsoluteOption(0));
|
|
}
|
|
|
|
tableComposition->UpdateCellBounds();
|
|
}
|
|
|
|
void GuiCommonScrollViewLook::hScroll_OnVisibleChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
UpdateTable();
|
|
}
|
|
|
|
void GuiCommonScrollViewLook::vScroll_OnVisibleChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
UpdateTable();
|
|
}
|
|
|
|
GuiCommonScrollViewLook::GuiCommonScrollViewLook(vint _defaultScrollSize)
|
|
:defaultScrollSize(_defaultScrollSize)
|
|
{
|
|
SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
|
|
horizontalScroll = new GuiScroll(theme::ThemeName::HScroll);
|
|
horizontalScroll->GetBoundsComposition()->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
horizontalScroll->SetEnabled(false);
|
|
horizontalScroll->SetAutoFocus(false);
|
|
verticalScroll = new GuiScroll(theme::ThemeName::VScroll);
|
|
verticalScroll->GetBoundsComposition()->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
verticalScroll->SetEnabled(false);
|
|
verticalScroll->SetAutoFocus(false);
|
|
|
|
tableComposition = new GuiTableComposition;
|
|
AddChild(tableComposition);
|
|
tableComposition->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
tableComposition->SetRowsAndColumns(2, 2);
|
|
tableComposition->SetRowOption(0, GuiCellOption::PercentageOption(1.0));
|
|
tableComposition->SetRowOption(1, GuiCellOption::MinSizeOption());
|
|
tableComposition->SetColumnOption(0, GuiCellOption::PercentageOption(1.0));
|
|
tableComposition->SetColumnOption(1, GuiCellOption::MinSizeOption());
|
|
UpdateTable();
|
|
{
|
|
GuiCellComposition* cell = new GuiCellComposition;
|
|
tableComposition->AddChild(cell);
|
|
cell->SetSite(1, 0, 1, 1);
|
|
cell->AddChild(horizontalScroll->GetBoundsComposition());
|
|
}
|
|
{
|
|
GuiCellComposition* cell = new GuiCellComposition;
|
|
tableComposition->AddChild(cell);
|
|
cell->SetSite(0, 1, 1, 1);
|
|
cell->AddChild(verticalScroll->GetBoundsComposition());
|
|
}
|
|
|
|
containerCellComposition = new GuiCellComposition;
|
|
tableComposition->AddChild(containerCellComposition);
|
|
containerCellComposition->SetSite(0, 0, 1, 1);
|
|
|
|
containerComposition = new GuiBoundsComposition;
|
|
containerComposition->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
containerCellComposition->AddChild(containerComposition);
|
|
|
|
horizontalScroll->VisibleChanged.AttachMethod(this, &GuiCommonScrollViewLook::hScroll_OnVisibleChanged);
|
|
verticalScroll->VisibleChanged.AttachMethod(this, &GuiCommonScrollViewLook::vScroll_OnVisibleChanged);
|
|
UpdateTable();
|
|
}
|
|
|
|
GuiCommonScrollViewLook::~GuiCommonScrollViewLook()
|
|
{
|
|
}
|
|
|
|
controls::GuiScroll* GuiCommonScrollViewLook::GetHScroll()
|
|
{
|
|
return horizontalScroll;
|
|
}
|
|
|
|
controls::GuiScroll* GuiCommonScrollViewLook::GetVScroll()
|
|
{
|
|
return verticalScroll;
|
|
}
|
|
|
|
compositions::GuiGraphicsComposition* GuiCommonScrollViewLook::GetContainerComposition()
|
|
{
|
|
return containerComposition;
|
|
}
|
|
|
|
TemplateProperty<GuiScrollTemplate> GuiCommonScrollViewLook::GetHScrollTemplate()
|
|
{
|
|
return hScrollTemplate;
|
|
}
|
|
|
|
void GuiCommonScrollViewLook::SetHScrollTemplate(const TemplateProperty<GuiScrollTemplate>& value)
|
|
{
|
|
hScrollTemplate = value;
|
|
horizontalScroll->SetControlTemplate(value);
|
|
}
|
|
|
|
TemplateProperty<GuiScrollTemplate> GuiCommonScrollViewLook::GetVScrollTemplate()
|
|
{
|
|
return vScrollTemplate;
|
|
}
|
|
|
|
void GuiCommonScrollViewLook::SetVScrollTemplate(const TemplateProperty<GuiScrollTemplate>& value)
|
|
{
|
|
vScrollTemplate = value;
|
|
verticalScroll->SetControlTemplate(value);
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiCommonScrollBehavior
|
|
***********************************************************************/
|
|
|
|
void GuiCommonScrollBehavior::SetScroll(vint totalPixels, vint newOffset)
|
|
{
|
|
vint totalSize = scrollTemplate->GetTotalSize();
|
|
double ratio = (double)newOffset / totalPixels;
|
|
vint newPosition = (vint)round(ratio * totalSize);
|
|
|
|
vint offset1 = (vint)round(((double)newPosition / totalSize) * totalPixels);
|
|
vint offset2 = (vint)round(((double)(newPosition + 1)) / totalSize * totalPixels);
|
|
vint delta1 = offset1 - newOffset;
|
|
vint delta2 = offset2 - newOffset;
|
|
|
|
if (delta1 < 0) { delta1 = -delta1; }
|
|
if (delta2 < 0) { delta2 = -delta2; }
|
|
|
|
if (delta1 < delta2)
|
|
{
|
|
scrollTemplate->GetCommands()->SetPosition(newPosition);
|
|
}
|
|
else
|
|
{
|
|
scrollTemplate->GetCommands()->SetPosition(newPosition + 1);
|
|
}
|
|
}
|
|
|
|
void GuiCommonScrollBehavior::AttachHandle(compositions::GuiGraphicsComposition* handle)
|
|
{
|
|
handle->GetEventReceiver()->leftButtonDown.AttachLambda([=](GuiGraphicsComposition*, GuiMouseEventArgs& arguments)
|
|
{
|
|
if (scrollTemplate->GetVisuallyEnabled())
|
|
{
|
|
dragging = true;
|
|
location.x = arguments.x;
|
|
location.y = arguments.y;
|
|
}
|
|
});
|
|
|
|
handle->GetEventReceiver()->leftButtonUp.AttachLambda([=](GuiGraphicsComposition*, GuiMouseEventArgs&)
|
|
{
|
|
if (scrollTemplate->GetVisuallyEnabled())
|
|
{
|
|
dragging = false;
|
|
}
|
|
});
|
|
}
|
|
|
|
GuiCommonScrollBehavior::GuiCommonScrollBehavior()
|
|
{
|
|
}
|
|
|
|
GuiCommonScrollBehavior::~GuiCommonScrollBehavior()
|
|
{
|
|
}
|
|
|
|
void GuiCommonScrollBehavior::AttachScrollTemplate(GuiScrollTemplate* value)
|
|
{
|
|
scrollTemplate = value;
|
|
}
|
|
|
|
void GuiCommonScrollBehavior::AttachDecreaseButton(controls::GuiButton* button)
|
|
{
|
|
button->Clicked.AttachLambda([=](GuiGraphicsComposition*, GuiEventArgs&)
|
|
{
|
|
scrollTemplate->GetCommands()->SmallDecrease();
|
|
});
|
|
}
|
|
|
|
void GuiCommonScrollBehavior::AttachIncreaseButton(controls::GuiButton* button)
|
|
{
|
|
button->Clicked.AttachLambda([=](GuiGraphicsComposition*, GuiEventArgs&)
|
|
{
|
|
scrollTemplate->GetCommands()->SmallIncrease();
|
|
});
|
|
}
|
|
|
|
void GuiCommonScrollBehavior::AttachHorizontalScrollHandle(compositions::GuiPartialViewComposition* partialView)
|
|
{
|
|
partialView->GetParent()->GetEventReceiver()->leftButtonDown.AttachLambda([=](GuiGraphicsComposition*, GuiMouseEventArgs& arguments)
|
|
{
|
|
if (scrollTemplate->GetVisuallyEnabled())
|
|
{
|
|
if (arguments.x < partialView->GetBounds().x1)
|
|
{
|
|
scrollTemplate->GetCommands()->BigDecrease();
|
|
}
|
|
else if (arguments.x >= partialView->GetBounds().x2)
|
|
{
|
|
scrollTemplate->GetCommands()->BigIncrease();
|
|
}
|
|
}
|
|
});
|
|
|
|
AttachHorizontalTrackerHandle(partialView);
|
|
}
|
|
|
|
void GuiCommonScrollBehavior::AttachVerticalScrollHandle(compositions::GuiPartialViewComposition* partialView)
|
|
{
|
|
partialView->GetParent()->GetEventReceiver()->leftButtonDown.AttachLambda([=](GuiGraphicsComposition*, GuiMouseEventArgs& arguments)
|
|
{
|
|
if (scrollTemplate->GetVisuallyEnabled())
|
|
{
|
|
if (arguments.y < partialView->GetBounds().y1)
|
|
{
|
|
scrollTemplate->GetCommands()->BigDecrease();
|
|
}
|
|
else if (arguments.y >= partialView->GetBounds().y2)
|
|
{
|
|
scrollTemplate->GetCommands()->BigIncrease();
|
|
}
|
|
}
|
|
});
|
|
|
|
AttachVerticalTrackerHandle(partialView);
|
|
}
|
|
|
|
void GuiCommonScrollBehavior::AttachHorizontalTrackerHandle(compositions::GuiPartialViewComposition* partialView)
|
|
{
|
|
partialView->GetEventReceiver()->mouseMove.AttachLambda([=](GuiGraphicsComposition*, GuiMouseEventArgs& arguments)
|
|
{
|
|
if (dragging)
|
|
{
|
|
auto bounds = partialView->GetParent()->GetBounds();
|
|
vint totalPixels = bounds.x2 - bounds.x1;
|
|
vint currentOffset = partialView->GetBounds().x1;
|
|
vint newOffset = currentOffset + (arguments.x - location.x);
|
|
SetScroll(totalPixels, newOffset);
|
|
}
|
|
});
|
|
|
|
AttachHandle(partialView);
|
|
}
|
|
|
|
void GuiCommonScrollBehavior::AttachVerticalTrackerHandle(compositions::GuiPartialViewComposition* partialView)
|
|
{
|
|
partialView->GetEventReceiver()->mouseMove.AttachLambda([=](GuiGraphicsComposition*, GuiMouseEventArgs& arguments)
|
|
{
|
|
if (dragging)
|
|
{
|
|
auto bounds = partialView->GetParent()->GetBounds();
|
|
vint totalPixels = bounds.y2 - bounds.y1;
|
|
vint currentOffset = partialView->GetBounds().y1;
|
|
vint newOffset = currentOffset + (arguments.y - location.y);
|
|
SetScroll(totalPixels, newOffset);
|
|
}
|
|
});
|
|
|
|
AttachHandle(partialView);
|
|
}
|
|
|
|
vint GuiCommonScrollBehavior::GetHorizontalTrackerHandlerPosition(compositions::GuiBoundsComposition* handle, vint totalSize, vint pageSize, vint position)
|
|
{
|
|
vint width = handle->GetParent()->GetBounds().Width() - handle->GetBounds().Width();
|
|
vint max = totalSize - pageSize;
|
|
return max == 0 ? 0 : width * position / max;
|
|
}
|
|
|
|
vint GuiCommonScrollBehavior::GetVerticalTrackerHandlerPosition(compositions::GuiBoundsComposition* handle, vint totalSize, vint pageSize, vint position)
|
|
{
|
|
vint height = handle->GetParent()->GetBounds().Height() - handle->GetBounds().Height();
|
|
vint max = totalSize - pageSize;
|
|
return max == 0 ? 0 : height * position / max;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\CONTROLS\TEMPLATES\GUICONTROLSHARED.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace controls
|
|
{
|
|
using namespace reflection::description;
|
|
using namespace compositions;
|
|
|
|
/***********************************************************************
|
|
GuiComponent
|
|
***********************************************************************/
|
|
|
|
GuiComponent::GuiComponent()
|
|
{
|
|
}
|
|
|
|
GuiComponent::~GuiComponent()
|
|
{
|
|
}
|
|
|
|
void GuiComponent::Attach(GuiInstanceRootObject* rootObject)
|
|
{
|
|
}
|
|
|
|
void GuiComponent::Detach(GuiInstanceRootObject* rootObject)
|
|
{
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiInstanceRootObject
|
|
***********************************************************************/
|
|
|
|
class RootObjectTimerCallback : public Object, public IGuiGraphicsTimerCallback
|
|
{
|
|
public:
|
|
GuiControlHost* controlHost;
|
|
GuiInstanceRootObject* rootObject;
|
|
bool alive = true;
|
|
|
|
RootObjectTimerCallback(GuiInstanceRootObject* _rootObject, GuiControlHost* _controlHost)
|
|
:rootObject(_rootObject)
|
|
, controlHost(_controlHost)
|
|
{
|
|
}
|
|
|
|
bool Play()override
|
|
{
|
|
if (alive)
|
|
{
|
|
for (vint i = rootObject->runningAnimations.Count() - 1; i >= 0; i--)
|
|
{
|
|
auto animation = rootObject->runningAnimations[i];
|
|
animation->Run();
|
|
if (animation->GetStopped())
|
|
{
|
|
rootObject->runningAnimations.RemoveAt(i);
|
|
}
|
|
}
|
|
|
|
if (rootObject->runningAnimations.Count() == 0)
|
|
{
|
|
rootObject->UninstallTimerCallback(nullptr);
|
|
return false;
|
|
}
|
|
}
|
|
return alive;
|
|
}
|
|
};
|
|
|
|
void GuiInstanceRootObject::InstallTimerCallback(controls::GuiControlHost* controlHost)
|
|
{
|
|
if (!timerCallback)
|
|
{
|
|
timerCallback = new RootObjectTimerCallback(this, controlHost);
|
|
controlHost->GetTimerManager()->AddCallback(timerCallback);
|
|
}
|
|
}
|
|
|
|
bool GuiInstanceRootObject::UninstallTimerCallback(controls::GuiControlHost* controlHost)
|
|
{
|
|
if (timerCallback && timerCallback->controlHost != controlHost)
|
|
{
|
|
timerCallback->alive = false;
|
|
timerCallback = nullptr;
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void GuiInstanceRootObject::OnControlHostForInstanceChanged()
|
|
{
|
|
auto controlHost = GetControlHostForInstance();
|
|
if (UninstallTimerCallback(controlHost))
|
|
{
|
|
FOREACH(Ptr<IGuiAnimation>, animation, runningAnimations)
|
|
{
|
|
animation->Pause();
|
|
}
|
|
}
|
|
|
|
if (controlHost)
|
|
{
|
|
InstallTimerCallback(controlHost);
|
|
FOREACH(Ptr<IGuiAnimation>, animation, runningAnimations)
|
|
{
|
|
animation->Resume();
|
|
}
|
|
StartPendingAnimations();
|
|
}
|
|
}
|
|
|
|
void GuiInstanceRootObject::StartPendingAnimations()
|
|
{
|
|
FOREACH(Ptr<IGuiAnimation>, animation, pendingAnimations)
|
|
{
|
|
animation->Start();
|
|
}
|
|
|
|
CopyFrom(runningAnimations, pendingAnimations, true);
|
|
pendingAnimations.Clear();
|
|
}
|
|
|
|
GuiInstanceRootObject::GuiInstanceRootObject()
|
|
{
|
|
}
|
|
|
|
GuiInstanceRootObject::~GuiInstanceRootObject()
|
|
{
|
|
UninstallTimerCallback(nullptr);
|
|
}
|
|
|
|
void GuiInstanceRootObject::FinalizeInstance()
|
|
{
|
|
if (!finalized)
|
|
{
|
|
finalized = true;
|
|
|
|
FOREACH(Ptr<IValueSubscription>, subscription, subscriptions)
|
|
{
|
|
subscription->Close();
|
|
}
|
|
FOREACH(GuiComponent*, component, components)
|
|
{
|
|
component->Detach(this);
|
|
}
|
|
|
|
subscriptions.Clear();
|
|
for (vint i = 0; i<components.Count(); i++)
|
|
{
|
|
delete components[i];
|
|
}
|
|
components.Clear();
|
|
}
|
|
}
|
|
|
|
bool GuiInstanceRootObject::IsFinalized()
|
|
{
|
|
return finalized;
|
|
}
|
|
|
|
void GuiInstanceRootObject::FinalizeInstanceRecursively(templates::GuiTemplate* thisObject)
|
|
{
|
|
if (!finalized)
|
|
{
|
|
NotifyFinalizeInstance(thisObject);
|
|
}
|
|
}
|
|
|
|
void GuiInstanceRootObject::FinalizeInstanceRecursively(GuiCustomControl* thisObject)
|
|
{
|
|
if (!finalized)
|
|
{
|
|
NotifyFinalizeInstance(thisObject);
|
|
}
|
|
}
|
|
|
|
void GuiInstanceRootObject::FinalizeInstanceRecursively(GuiControlHost* thisObject)
|
|
{
|
|
if (!finalized)
|
|
{
|
|
NotifyFinalizeInstance(thisObject);
|
|
}
|
|
}
|
|
|
|
void GuiInstanceRootObject::FinalizeGeneralInstance(GuiInstanceRootObject* thisObject)
|
|
{
|
|
}
|
|
|
|
void GuiInstanceRootObject::SetResourceResolver(Ptr<GuiResourcePathResolver> resolver)
|
|
{
|
|
resourceResolver = resolver;
|
|
}
|
|
|
|
Ptr<DescriptableObject> GuiInstanceRootObject::ResolveResource(const WString& protocol, const WString& path, bool ensureExist)
|
|
{
|
|
Ptr<DescriptableObject> object;
|
|
if (resourceResolver)
|
|
{
|
|
object = resourceResolver->ResolveResource(protocol, path);
|
|
}
|
|
if (ensureExist && !object)
|
|
{
|
|
throw ArgumentException(L"Resource \"" + protocol + L"://" + path + L"\" does not exist.");
|
|
}
|
|
return object;
|
|
}
|
|
|
|
Ptr<description::IValueSubscription> GuiInstanceRootObject::AddSubscription(Ptr<description::IValueSubscription> subscription)
|
|
{
|
|
CHECK_ERROR(finalized == false, L"GuiInstanceRootObject::AddSubscription(Ptr<IValueSubscription>)#Cannot add subscription after finalizing.");
|
|
if (subscriptions.Contains(subscription.Obj()))
|
|
{
|
|
return nullptr;
|
|
}
|
|
else
|
|
{
|
|
subscriptions.Add(subscription);
|
|
subscription->Open();
|
|
subscription->Update();
|
|
return subscription;
|
|
}
|
|
}
|
|
|
|
void GuiInstanceRootObject::UpdateSubscriptions()
|
|
{
|
|
FOREACH(Ptr<IValueSubscription>, subscription, subscriptions)
|
|
{
|
|
subscription->Update();
|
|
}
|
|
}
|
|
|
|
bool GuiInstanceRootObject::AddComponent(GuiComponent* component)
|
|
{
|
|
CHECK_ERROR(finalized == false, L"GuiInstanceRootObject::AddComponent(GuiComponent*)#Cannot add component after finalizing.");
|
|
if(components.Contains(component))
|
|
{
|
|
return false;
|
|
}
|
|
else
|
|
{
|
|
components.Add(component);
|
|
component->Attach(this);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
bool GuiInstanceRootObject::AddControlHostComponent(GuiControlHost* controlHost)
|
|
{
|
|
return AddComponent(new GuiObjectComponent<GuiControlHost>(controlHost));
|
|
}
|
|
|
|
bool GuiInstanceRootObject::AddAnimation(Ptr<IGuiAnimation> animation)
|
|
{
|
|
CHECK_ERROR(finalized == false, L"GuiInstanceRootObject::AddAnimation(Ptr<IGuiAnimation>)#Cannot add animation after finalizing.");
|
|
if (runningAnimations.Contains(animation.Obj()) || pendingAnimations.Contains(animation.Obj()))
|
|
{
|
|
return false;
|
|
}
|
|
else
|
|
{
|
|
pendingAnimations.Add(animation);
|
|
|
|
if (auto controlHost = GetControlHostForInstance())
|
|
{
|
|
InstallTimerCallback(controlHost);
|
|
StartPendingAnimations();
|
|
}
|
|
return true;
|
|
}
|
|
}
|
|
|
|
bool GuiInstanceRootObject::KillAnimation(Ptr<IGuiAnimation> animation)
|
|
{
|
|
if (!animation) return false;
|
|
if (runningAnimations.Contains(animation.Obj()))
|
|
{
|
|
runningAnimations.Remove(animation.Obj());
|
|
return true;
|
|
}
|
|
if (pendingAnimations.Contains(animation.Obj()))
|
|
{
|
|
pendingAnimations.Remove(animation.Obj());
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\CONTROLS\TEMPLATES\GUICONTROLTEMPLATES.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace templates
|
|
{
|
|
using namespace collections;
|
|
using namespace controls;
|
|
using namespace compositions;
|
|
using namespace elements;
|
|
|
|
/***********************************************************************
|
|
GuiTemplate
|
|
***********************************************************************/
|
|
|
|
GuiTemplate_PROPERTIES(GUI_TEMPLATE_PROPERTY_IMPL)
|
|
|
|
controls::GuiControlHost* GuiTemplate::GetControlHostForInstance()
|
|
{
|
|
return GetRelatedControlHost();
|
|
}
|
|
|
|
void GuiTemplate::OnParentLineChanged()
|
|
{
|
|
GuiBoundsComposition::OnParentLineChanged();
|
|
OnControlHostForInstanceChanged();
|
|
}
|
|
|
|
GuiTemplate::GuiTemplate()
|
|
{
|
|
GuiTemplate_PROPERTIES(GUI_TEMPLATE_PROPERTY_EVENT_INIT)
|
|
}
|
|
|
|
GuiTemplate::~GuiTemplate()
|
|
{
|
|
FinalizeInstanceRecursively(this);
|
|
}
|
|
|
|
/***********************************************************************
|
|
Item GuiListItemTemplate
|
|
***********************************************************************/
|
|
|
|
void GuiListItemTemplate::OnInitialize()
|
|
{
|
|
}
|
|
|
|
GuiListItemTemplate_PROPERTIES(GUI_TEMPLATE_PROPERTY_IMPL)
|
|
|
|
GuiListItemTemplate::GuiListItemTemplate()
|
|
{
|
|
GuiListItemTemplate_PROPERTIES(GUI_TEMPLATE_PROPERTY_EVENT_INIT)
|
|
}
|
|
|
|
GuiListItemTemplate::~GuiListItemTemplate()
|
|
{
|
|
FinalizeAggregation();
|
|
}
|
|
|
|
void GuiListItemTemplate::BeginEditListItem()
|
|
{
|
|
listControl->GetItemProvider()->PushEditing();
|
|
}
|
|
|
|
void GuiListItemTemplate::EndEditListItem()
|
|
{
|
|
CHECK_ERROR(listControl->GetItemProvider()->PopEditing(), L"GuiListItemTemplate::EndEditListItem()#BeginEditListItem and EndEditListItem calls are not paired.");
|
|
}
|
|
|
|
void GuiListItemTemplate::Initialize(controls::GuiListControl* _listControl)
|
|
{
|
|
CHECK_ERROR(listControl == nullptr, L"GuiListItemTemplate::Initialize(GuiListControl*)#This function can only be called once.");
|
|
listControl = _listControl;
|
|
OnInitialize();
|
|
}
|
|
|
|
/***********************************************************************
|
|
Template Declarations
|
|
***********************************************************************/
|
|
|
|
GUI_CONTROL_TEMPLATE_DECL(GUI_TEMPLATE_CLASS_IMPL)
|
|
GUI_ITEM_TEMPLATE_DECL(GUI_TEMPLATE_CLASS_IMPL)
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\CONTROLS\TEMPLATES\GUITHEMESTYLEFACTORY.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace theme
|
|
{
|
|
using namespace collections;
|
|
using namespace controls;
|
|
using namespace templates;
|
|
|
|
class Theme : public Object, public virtual theme::ITheme
|
|
{
|
|
public:
|
|
Dictionary<WString, Ptr<ThemeTemplates>> templates;
|
|
ThemeTemplates* first = nullptr;
|
|
ThemeTemplates* last = nullptr;
|
|
|
|
bool RegisterTheme(const WString& name, Ptr<ThemeTemplates> theme)
|
|
{
|
|
CHECK_ERROR(theme->previous == nullptr, L"vl::presentation::theme::RegisterTheme(const WString&, Ptr<ThemeTemplates>)#Theme object has been registered");
|
|
CHECK_ERROR(theme->next == nullptr, L"vl::presentation::theme::RegisterTheme(const WString&, Ptr<ThemeTemplates>)#Theme object has been registered");
|
|
|
|
if (templates.Keys().Contains(name))
|
|
{
|
|
return false;
|
|
}
|
|
templates.Add(name, theme);
|
|
|
|
if (last)
|
|
{
|
|
last->next = theme.Obj();
|
|
}
|
|
theme->previous = last;
|
|
last = theme.Obj();
|
|
|
|
return true;
|
|
}
|
|
|
|
Ptr<ThemeTemplates> UnregisterTheme(const WString& name)
|
|
{
|
|
vint index = templates.Keys().IndexOf(name);
|
|
if (index == -1)
|
|
{
|
|
return nullptr;
|
|
}
|
|
|
|
auto themeTemplates = templates.Values().Get(index);
|
|
|
|
if (themeTemplates->previous)
|
|
{
|
|
themeTemplates->previous->next = themeTemplates->next;
|
|
}
|
|
else
|
|
{
|
|
first = themeTemplates->next;
|
|
}
|
|
|
|
if (themeTemplates->next)
|
|
{
|
|
themeTemplates->next->previous = themeTemplates->previous;
|
|
|
|
}
|
|
else
|
|
{
|
|
last = themeTemplates->previous;
|
|
}
|
|
|
|
templates.Remove(name);
|
|
return themeTemplates;
|
|
}
|
|
|
|
TemplateProperty<GuiControlTemplate> CreateStyle(ThemeName themeName)override
|
|
{
|
|
switch (themeName)
|
|
{
|
|
#define GUI_DEFINE_ITEM_PROPERTY(TEMPLATE, CONTROL) \
|
|
case ThemeName::CONTROL:\
|
|
{\
|
|
auto current = last;\
|
|
while (current) \
|
|
{\
|
|
if (current->CONTROL)\
|
|
{\
|
|
return current->CONTROL; \
|
|
}\
|
|
current = current->previous;\
|
|
}\
|
|
throw Exception(L"Control template for \"" L ## #CONTROL L"\" is not defined.");\
|
|
}\
|
|
|
|
GUI_CONTROL_TEMPLATE_TYPES(GUI_DEFINE_ITEM_PROPERTY)
|
|
#undef GUI_DEFINE_ITEM_PROPERTY
|
|
default:
|
|
CHECK_FAIL(L"vl::presentation::theme::ITheme::CreateStyle(ThemeName)#Unknown theme name.");
|
|
}
|
|
}
|
|
};
|
|
|
|
controls::GuiControlHost* ThemeTemplates::GetControlHostForInstance()
|
|
{
|
|
return nullptr;
|
|
}
|
|
|
|
ThemeTemplates::~ThemeTemplates()
|
|
{
|
|
FinalizeAggregation();
|
|
}
|
|
|
|
Theme* currentTheme = nullptr;
|
|
|
|
ITheme* GetCurrentTheme()
|
|
{
|
|
return currentTheme;
|
|
}
|
|
|
|
void InitializeTheme()
|
|
{
|
|
CHECK_ERROR(currentTheme == nullptr, L"vl::presentation::theme::InitializeTheme()#Theme has already been initialized");
|
|
currentTheme = new Theme;
|
|
}
|
|
|
|
void FinalizeTheme()
|
|
{
|
|
CHECK_ERROR(currentTheme != nullptr, L"vl::presentation::theme::FinalizeTheme()#Theme has not been initialized");
|
|
delete currentTheme;
|
|
currentTheme = nullptr;
|
|
}
|
|
|
|
bool RegisterTheme(Ptr<ThemeTemplates> theme)
|
|
{
|
|
CHECK_ERROR(currentTheme != nullptr, L"vl::presentation::theme::RegisterTheme(const WString&, Ptr<ThemeTemplates>)#Theme has already been initialized");
|
|
return currentTheme->RegisterTheme(theme->Name, theme);
|
|
}
|
|
|
|
Ptr<ThemeTemplates> UnregisterTheme(const WString& name)
|
|
{
|
|
CHECK_ERROR(currentTheme != nullptr, L"vl::presentation::theme::UnregisterTheme(const WString&)#Theme has already been initialized");
|
|
return currentTheme->UnregisterTheme(name);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\CONTROLS\TEXTEDITORPACKAGE\GUIDOCUMENTVIEWER.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace controls
|
|
{
|
|
using namespace collections;
|
|
using namespace elements;
|
|
using namespace compositions;
|
|
|
|
/***********************************************************************
|
|
GuiDocumentItem
|
|
***********************************************************************/
|
|
|
|
GuiDocumentItem::GuiDocumentItem(const WString& _name)
|
|
:name(_name)
|
|
{
|
|
container = new GuiBoundsComposition;
|
|
container->SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
container->SetAssociatedCursor(GetCurrentController()->ResourceService()->GetSystemCursor(INativeCursor::Arrow));
|
|
}
|
|
|
|
GuiDocumentItem::~GuiDocumentItem()
|
|
{
|
|
if (!owned)
|
|
{
|
|
SafeDeleteComposition(container);
|
|
}
|
|
}
|
|
|
|
compositions::GuiGraphicsComposition* GuiDocumentItem::GetContainer()
|
|
{
|
|
return container;
|
|
}
|
|
|
|
WString GuiDocumentItem::GetName()
|
|
{
|
|
return name;
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiDocumentCommonInterface
|
|
***********************************************************************/
|
|
|
|
void GuiDocumentCommonInterface::InvokeUndoRedoChanged()
|
|
{
|
|
UndoRedoChanged.Execute(documentControl->GetNotifyEventArguments());
|
|
}
|
|
|
|
void GuiDocumentCommonInterface::InvokeModifiedChanged()
|
|
{
|
|
ModifiedChanged.Execute(documentControl->GetNotifyEventArguments());
|
|
}
|
|
|
|
void GuiDocumentCommonInterface::UpdateCaretPoint()
|
|
{
|
|
GuiGraphicsHost* host=documentComposition->GetRelatedGraphicsHost();
|
|
if(host)
|
|
{
|
|
Rect caret=documentElement->GetCaretBounds(documentElement->GetCaretEnd(), documentElement->IsCaretEndPreferFrontSide());
|
|
Point view=GetDocumentViewPosition();
|
|
vint x=caret.x1-view.x;
|
|
vint y=caret.y2-view.y;
|
|
host->SetCaretPoint(Point(x, y), documentComposition);
|
|
}
|
|
}
|
|
|
|
void GuiDocumentCommonInterface::Move(TextPos caret, bool shift, bool frontSide)
|
|
{
|
|
TextPos begin=documentElement->GetCaretBegin();
|
|
TextPos end=documentElement->GetCaretEnd();
|
|
|
|
TextPos newBegin=shift?begin:caret;
|
|
TextPos newEnd=caret;
|
|
documentElement->SetCaret(newBegin, newEnd, frontSide);
|
|
documentElement->SetCaretVisible(true);
|
|
|
|
Rect bounds=documentElement->GetCaretBounds(newEnd, frontSide);
|
|
if(bounds!=Rect())
|
|
{
|
|
bounds.x1-=15;
|
|
bounds.y1-=15;
|
|
bounds.x2+=15;
|
|
bounds.y2+=15;
|
|
EnsureRectVisible(bounds);
|
|
}
|
|
UpdateCaretPoint();
|
|
SelectionChanged.Execute(documentControl->GetNotifyEventArguments());
|
|
}
|
|
|
|
bool GuiDocumentCommonInterface::ProcessKey(VKEY code, bool shift, bool ctrl)
|
|
{
|
|
if(IGuiShortcutKeyItem* item=internalShortcutKeyManager->TryGetShortcut(ctrl, shift, false, code))
|
|
{
|
|
GuiEventArgs arguments(documentControl->GetBoundsComposition());
|
|
item->Executed.Execute(arguments);
|
|
return true;
|
|
}
|
|
|
|
TextPos currentCaret=documentElement->GetCaretEnd();
|
|
bool frontSide=documentElement->IsCaretEndPreferFrontSide();
|
|
TextPos begin=documentElement->GetCaretBegin();
|
|
TextPos end=documentElement->GetCaretEnd();
|
|
|
|
switch(code)
|
|
{
|
|
case VKEY::_UP:
|
|
{
|
|
TextPos newCaret=documentElement->CalculateCaret(currentCaret, IGuiGraphicsParagraph::CaretMoveUp, frontSide);
|
|
Move(newCaret, shift, frontSide);
|
|
}
|
|
break;
|
|
case VKEY::_DOWN:
|
|
{
|
|
TextPos newCaret=documentElement->CalculateCaret(currentCaret, IGuiGraphicsParagraph::CaretMoveDown, frontSide);
|
|
Move(newCaret, shift, frontSide);
|
|
}
|
|
break;
|
|
case VKEY::_LEFT:
|
|
{
|
|
TextPos newCaret=documentElement->CalculateCaret(currentCaret, IGuiGraphicsParagraph::CaretMoveLeft, frontSide);
|
|
Move(newCaret, shift, frontSide);
|
|
}
|
|
break;
|
|
case VKEY::_RIGHT:
|
|
{
|
|
TextPos newCaret=documentElement->CalculateCaret(currentCaret, IGuiGraphicsParagraph::CaretMoveRight, frontSide);
|
|
Move(newCaret, shift, frontSide);
|
|
}
|
|
break;
|
|
case VKEY::_HOME:
|
|
{
|
|
TextPos newCaret=documentElement->CalculateCaret(currentCaret, IGuiGraphicsParagraph::CaretLineFirst, frontSide);
|
|
if(newCaret==currentCaret)
|
|
{
|
|
newCaret=documentElement->CalculateCaret(currentCaret, IGuiGraphicsParagraph::CaretFirst, frontSide);
|
|
}
|
|
Move(newCaret, shift, frontSide);
|
|
}
|
|
break;
|
|
case VKEY::_END:
|
|
{
|
|
TextPos newCaret=documentElement->CalculateCaret(currentCaret, IGuiGraphicsParagraph::CaretLineLast, frontSide);
|
|
if(newCaret==currentCaret)
|
|
{
|
|
newCaret=documentElement->CalculateCaret(currentCaret, IGuiGraphicsParagraph::CaretLast, frontSide);
|
|
}
|
|
Move(newCaret, shift, frontSide);
|
|
}
|
|
break;
|
|
case VKEY::_PRIOR:
|
|
{
|
|
}
|
|
break;
|
|
case VKEY::_NEXT:
|
|
{
|
|
}
|
|
break;
|
|
case VKEY::_BACK:
|
|
if(editMode==Editable)
|
|
{
|
|
if(begin==end)
|
|
{
|
|
ProcessKey(VKEY::_LEFT, true, false);
|
|
}
|
|
Array<WString> text;
|
|
EditText(documentElement->GetCaretBegin(), documentElement->GetCaretEnd(), documentElement->IsCaretEndPreferFrontSide(), text);
|
|
return true;
|
|
}
|
|
break;
|
|
case VKEY::_DELETE:
|
|
if(editMode==Editable)
|
|
{
|
|
if(begin==end)
|
|
{
|
|
ProcessKey(VKEY::_RIGHT, true, false);
|
|
}
|
|
Array<WString> text;
|
|
EditText(documentElement->GetCaretBegin(), documentElement->GetCaretEnd(), documentElement->IsCaretEndPreferFrontSide(), text);
|
|
return true;
|
|
}
|
|
break;
|
|
case VKEY::_RETURN:
|
|
if(editMode==Editable)
|
|
{
|
|
if(ctrl)
|
|
{
|
|
Array<WString> text(1);
|
|
text[0]=L"\r\n";
|
|
EditText(documentElement->GetCaretBegin(), documentElement->GetCaretEnd(), documentElement->IsCaretEndPreferFrontSide(), text);
|
|
}
|
|
else
|
|
{
|
|
Array<WString> text(2);
|
|
EditText(documentElement->GetCaretBegin(), documentElement->GetCaretEnd(), documentElement->IsCaretEndPreferFrontSide(), text);
|
|
}
|
|
return true;
|
|
}
|
|
break;
|
|
default:;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void GuiDocumentCommonInterface::InstallDocumentViewer(
|
|
GuiControl* _sender,
|
|
compositions::GuiGraphicsComposition* _container,
|
|
compositions::GuiGraphicsComposition* eventComposition,
|
|
compositions::GuiGraphicsComposition* focusableComposition
|
|
)
|
|
{
|
|
documentControl=_sender;
|
|
|
|
documentElement=GuiDocumentElement::Create();
|
|
documentElement->SetCallback(this);
|
|
|
|
documentComposition=new GuiBoundsComposition;
|
|
documentComposition->SetOwnedElement(documentElement);
|
|
documentComposition->SetMinSizeLimitation(GuiGraphicsComposition::LimitToElement);
|
|
documentComposition->SetAlignmentToParent(Margin(5, 5, 5, 5));
|
|
_container->AddChild(documentComposition);
|
|
|
|
documentComposition->GetEventReceiver()->mouseMove.AttachMethod(this, &GuiDocumentCommonInterface::OnMouseMove);
|
|
documentComposition->GetEventReceiver()->leftButtonDown.AttachMethod(this, &GuiDocumentCommonInterface::OnMouseDown);
|
|
documentComposition->GetEventReceiver()->leftButtonUp.AttachMethod(this, &GuiDocumentCommonInterface::OnMouseUp);
|
|
documentComposition->GetEventReceiver()->mouseLeave.AttachMethod(this, &GuiDocumentCommonInterface::OnMouseLeave);
|
|
|
|
focusableComposition->GetEventReceiver()->caretNotify.AttachMethod(this, &GuiDocumentCommonInterface::OnCaretNotify);
|
|
focusableComposition->GetEventReceiver()->gotFocus.AttachMethod(this, &GuiDocumentCommonInterface::OnGotFocus);
|
|
focusableComposition->GetEventReceiver()->lostFocus.AttachMethod(this, &GuiDocumentCommonInterface::OnLostFocus);
|
|
focusableComposition->GetEventReceiver()->keyDown.AttachMethod(this, &GuiDocumentCommonInterface::OnKeyDown);
|
|
focusableComposition->GetEventReceiver()->charInput.AttachMethod(this, &GuiDocumentCommonInterface::OnCharInput);
|
|
|
|
undoRedoProcessor->Setup(documentElement, documentComposition);
|
|
ActiveHyperlinkChanged.SetAssociatedComposition(eventComposition);
|
|
ActiveHyperlinkExecuted.SetAssociatedComposition(eventComposition);
|
|
SelectionChanged.SetAssociatedComposition(eventComposition);
|
|
UndoRedoChanged.SetAssociatedComposition(eventComposition);
|
|
ModifiedChanged.SetAssociatedComposition(eventComposition);
|
|
|
|
undoRedoProcessor->UndoRedoChanged.Add(this, &GuiDocumentCommonInterface::InvokeUndoRedoChanged);
|
|
undoRedoProcessor->ModifiedChanged.Add(this, &GuiDocumentCommonInterface::InvokeModifiedChanged);
|
|
SetDocument(new DocumentModel);
|
|
}
|
|
|
|
void GuiDocumentCommonInterface::SetActiveHyperlink(Ptr<DocumentHyperlinkRun::Package> package)
|
|
{
|
|
ActivateActiveHyperlink(false);
|
|
activeHyperlinks =
|
|
!package ? nullptr :
|
|
package->hyperlinks.Count() == 0 ? nullptr :
|
|
package;
|
|
ActivateActiveHyperlink(true);
|
|
ActiveHyperlinkChanged.Execute(documentControl->GetNotifyEventArguments());
|
|
}
|
|
|
|
void GuiDocumentCommonInterface::ActivateActiveHyperlink(bool activate)
|
|
{
|
|
if (activeHyperlinks)
|
|
{
|
|
FOREACH(Ptr<DocumentHyperlinkRun>, run, activeHyperlinks->hyperlinks)
|
|
{
|
|
run->styleName = activate ? run->activeStyleName : run->normalStyleName;
|
|
}
|
|
documentElement->NotifyParagraphUpdated(activeHyperlinks->row, 1, 1, false);
|
|
}
|
|
}
|
|
|
|
void GuiDocumentCommonInterface::AddShortcutCommand(VKEY key, const Func<void()>& eventHandler)
|
|
{
|
|
IGuiShortcutKeyItem* item=internalShortcutKeyManager->CreateShortcut(true, false, false, key);
|
|
item->Executed.AttachLambda([=](GuiGraphicsComposition* sender, GuiEventArgs& arguments)
|
|
{
|
|
eventHandler();
|
|
});
|
|
}
|
|
|
|
void GuiDocumentCommonInterface::EditTextInternal(TextPos begin, TextPos end, const Func<void(TextPos, TextPos, vint&, vint&)>& editor)
|
|
{
|
|
// save run before editing
|
|
if(begin>end)
|
|
{
|
|
TextPos temp=begin;
|
|
begin=end;
|
|
end=temp;
|
|
}
|
|
Ptr<DocumentModel> originalModel=documentElement->GetDocument()->CopyDocument(begin, end, true);
|
|
if(originalModel)
|
|
{
|
|
// edit
|
|
vint paragraphCount=0;
|
|
vint lastParagraphLength=0;
|
|
editor(begin, end, paragraphCount, lastParagraphLength);
|
|
|
|
// calculate new caret
|
|
TextPos caret;
|
|
if(paragraphCount==0)
|
|
{
|
|
caret=begin;
|
|
}
|
|
else if(paragraphCount==1)
|
|
{
|
|
caret=TextPos(begin.row, begin.column+lastParagraphLength);
|
|
}
|
|
else
|
|
{
|
|
caret=TextPos(begin.row+paragraphCount-1, lastParagraphLength);
|
|
}
|
|
documentElement->SetCaret(caret, caret, true);
|
|
documentControl->TextChanged.Execute(documentControl->GetNotifyEventArguments());
|
|
UpdateCaretPoint();
|
|
SelectionChanged.Execute(documentControl->GetNotifyEventArguments());
|
|
|
|
// save run after editing
|
|
Ptr<DocumentModel> inputModel=documentElement->GetDocument()->CopyDocument(begin, caret, true);
|
|
|
|
// submit redo-undo
|
|
GuiDocumentUndoRedoProcessor::ReplaceModelStruct arguments;
|
|
arguments.originalStart=begin;
|
|
arguments.originalEnd=end;
|
|
arguments.originalModel=originalModel;
|
|
arguments.inputStart=begin;
|
|
arguments.inputEnd=caret;
|
|
arguments.inputModel=inputModel;
|
|
undoRedoProcessor->OnReplaceModel(arguments);
|
|
}
|
|
}
|
|
|
|
void GuiDocumentCommonInterface::EditStyleInternal(TextPos begin, TextPos end, const Func<void(TextPos, TextPos)>& editor)
|
|
{
|
|
// save run before editing
|
|
if(begin>end)
|
|
{
|
|
TextPos temp=begin;
|
|
begin=end;
|
|
end=temp;
|
|
}
|
|
Ptr<DocumentModel> originalModel=documentElement->GetDocument()->CopyDocument(begin, end, true);
|
|
if(originalModel)
|
|
{
|
|
// edit
|
|
editor(begin, end);
|
|
|
|
// save run after editing
|
|
Ptr<DocumentModel> inputModel=documentElement->GetDocument()->CopyDocument(begin, end, true);
|
|
|
|
// submit redo-undo
|
|
GuiDocumentUndoRedoProcessor::ReplaceModelStruct arguments;
|
|
arguments.originalStart=begin;
|
|
arguments.originalEnd=end;
|
|
arguments.originalModel=originalModel;
|
|
arguments.inputStart=begin;
|
|
arguments.inputEnd=end;
|
|
arguments.inputModel=inputModel;
|
|
undoRedoProcessor->OnReplaceModel(arguments);
|
|
}
|
|
}
|
|
|
|
void GuiDocumentCommonInterface::MergeBaselineAndDefaultFont(Ptr<DocumentModel> document)
|
|
{
|
|
document->MergeDefaultFont(documentControl->GetDisplayFont());
|
|
if (baselineDocument)
|
|
{
|
|
document->MergeBaselineStyles(baselineDocument);
|
|
}
|
|
}
|
|
|
|
void GuiDocumentCommonInterface::OnFontChanged()
|
|
{
|
|
auto document = documentElement->GetDocument();
|
|
MergeBaselineAndDefaultFont(document);
|
|
documentElement->SetDocument(document);
|
|
}
|
|
|
|
void GuiDocumentCommonInterface::OnCaretNotify(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
if(documentControl->GetVisuallyEnabled())
|
|
{
|
|
if(editMode!=ViewOnly)
|
|
{
|
|
documentElement->SetCaretVisible(!documentElement->GetCaretVisible());
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiDocumentCommonInterface::OnGotFocus(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
if(documentControl->GetVisuallyEnabled())
|
|
{
|
|
if(editMode!=ViewOnly)
|
|
{
|
|
documentElement->SetCaretVisible(true);
|
|
UpdateCaretPoint();
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiDocumentCommonInterface::OnLostFocus(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
if(documentControl->GetVisuallyEnabled())
|
|
{
|
|
documentElement->SetCaretVisible(false);
|
|
}
|
|
}
|
|
|
|
void GuiDocumentCommonInterface::OnKeyDown(compositions::GuiGraphicsComposition* sender, compositions::GuiKeyEventArgs& arguments)
|
|
{
|
|
if(documentControl->GetVisuallyEnabled())
|
|
{
|
|
if(editMode!=ViewOnly)
|
|
{
|
|
if(ProcessKey(arguments.code, arguments.shift, arguments.ctrl))
|
|
{
|
|
arguments.handled=true;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiDocumentCommonInterface::OnCharInput(compositions::GuiGraphicsComposition* sender, compositions::GuiCharEventArgs& arguments)
|
|
{
|
|
if (documentControl->GetVisuallyEnabled())
|
|
{
|
|
if (editMode == Editable &&
|
|
arguments.code != (wchar_t)VKEY::_ESCAPE &&
|
|
arguments.code != (wchar_t)VKEY::_BACK &&
|
|
arguments.code != (wchar_t)VKEY::_RETURN &&
|
|
(arguments.code != (wchar_t)VKEY::_TAB || documentControl->GetAcceptTabInput()) &&
|
|
!arguments.ctrl)
|
|
{
|
|
Array<WString> text(1);
|
|
text[0] = WString(arguments.code);
|
|
EditText(documentElement->GetCaretBegin(), documentElement->GetCaretEnd(), documentElement->IsCaretEndPreferFrontSide(), text);
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiDocumentCommonInterface::OnMouseMove(compositions::GuiGraphicsComposition* sender, compositions::GuiMouseEventArgs& arguments)
|
|
{
|
|
if(documentControl->GetVisuallyEnabled())
|
|
{
|
|
switch(editMode)
|
|
{
|
|
case ViewOnly:
|
|
{
|
|
auto package = documentElement->GetHyperlinkFromPoint({ arguments.x, arguments.y });
|
|
bool handCursor = false;
|
|
|
|
if(dragging)
|
|
{
|
|
if(activeHyperlinks)
|
|
{
|
|
if (package && CompareEnumerable(activeHyperlinks->hyperlinks, package->hyperlinks) == 0)
|
|
{
|
|
ActivateActiveHyperlink(true);
|
|
handCursor = true;
|
|
}
|
|
else
|
|
{
|
|
ActivateActiveHyperlink(false);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
SetActiveHyperlink(package);
|
|
handCursor = activeHyperlinks;
|
|
}
|
|
|
|
if(handCursor)
|
|
{
|
|
auto cursor = GetCurrentController()->ResourceService()->GetSystemCursor(INativeCursor::Hand);
|
|
documentComposition->SetAssociatedCursor(cursor);
|
|
}
|
|
else
|
|
{
|
|
documentComposition->SetAssociatedCursor(nullptr);
|
|
}
|
|
}
|
|
break;
|
|
case Selectable:
|
|
case Editable:
|
|
if(dragging)
|
|
{
|
|
TextPos caret=documentElement->CalculateCaretFromPoint(Point(arguments.x, arguments.y));
|
|
TextPos oldCaret=documentElement->GetCaretBegin();
|
|
Move(caret, true, (oldCaret==caret?documentElement->IsCaretEndPreferFrontSide():caret<oldCaret));
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiDocumentCommonInterface::OnMouseDown(compositions::GuiGraphicsComposition* sender, compositions::GuiMouseEventArgs& arguments)
|
|
{
|
|
if(documentControl->GetVisuallyEnabled())
|
|
{
|
|
switch(editMode)
|
|
{
|
|
case ViewOnly:
|
|
SetActiveHyperlink(documentElement->GetHyperlinkFromPoint({ arguments.x, arguments.y }));
|
|
break;
|
|
case Selectable:
|
|
case Editable:
|
|
{
|
|
documentControl->SetFocus();
|
|
TextPos caret=documentElement->CalculateCaretFromPoint(Point(arguments.x, arguments.y));
|
|
TextPos oldCaret=documentElement->GetCaretEnd();
|
|
if(caret!=oldCaret)
|
|
{
|
|
Move(caret, arguments.shift, caret<oldCaret);
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
dragging=true;
|
|
}
|
|
}
|
|
|
|
void GuiDocumentCommonInterface::OnMouseUp(compositions::GuiGraphicsComposition* sender, compositions::GuiMouseEventArgs& arguments)
|
|
{
|
|
if(documentControl->GetVisuallyEnabled())
|
|
{
|
|
dragging=false;
|
|
switch(editMode)
|
|
{
|
|
case ViewOnly:
|
|
{
|
|
auto package = documentElement->GetHyperlinkFromPoint({ arguments.x, arguments.y });
|
|
if(activeHyperlinks)
|
|
{
|
|
if (package && CompareEnumerable(activeHyperlinks->hyperlinks, package->hyperlinks) == 0)
|
|
{
|
|
ActiveHyperlinkExecuted.Execute(documentControl->GetNotifyEventArguments());
|
|
}
|
|
else
|
|
{
|
|
SetActiveHyperlink(nullptr);
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
default:;
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiDocumentCommonInterface::OnMouseLeave(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
SetActiveHyperlink(0);
|
|
}
|
|
|
|
Point GuiDocumentCommonInterface::GetDocumentViewPosition()
|
|
{
|
|
return Point(0, 0);
|
|
}
|
|
|
|
void GuiDocumentCommonInterface::EnsureRectVisible(Rect bounds)
|
|
{
|
|
}
|
|
|
|
//================ callback
|
|
|
|
void GuiDocumentCommonInterface::OnStartRender()
|
|
{
|
|
FOREACH(Ptr<GuiDocumentItem>, item, documentItems.Values())
|
|
{
|
|
item->visible = false;
|
|
}
|
|
}
|
|
|
|
void GuiDocumentCommonInterface::OnFinishRender()
|
|
{
|
|
FOREACH(Ptr<GuiDocumentItem>, item, documentItems.Values())
|
|
{
|
|
if (item->container->GetVisible() != item->visible)
|
|
{
|
|
item->container->SetVisible(item->visible);
|
|
}
|
|
}
|
|
}
|
|
|
|
Size GuiDocumentCommonInterface::OnRenderEmbeddedObject(const WString& name, const Rect& location)
|
|
{
|
|
vint index = documentItems.Keys().IndexOf(name);
|
|
if (index != -1)
|
|
{
|
|
auto item = documentItems.Values()[index];
|
|
auto size = item->container->GetBounds().GetSize();
|
|
item->container->SetBounds(Rect(location.LeftTop(), Size(0, 0)));
|
|
item->visible = true;
|
|
return size;
|
|
}
|
|
return Size();
|
|
}
|
|
|
|
//================ basic
|
|
|
|
GuiDocumentCommonInterface::GuiDocumentCommonInterface()
|
|
{
|
|
undoRedoProcessor=new GuiDocumentUndoRedoProcessor;
|
|
|
|
internalShortcutKeyManager=new GuiShortcutKeyManager;
|
|
AddShortcutCommand(VKEY::_Z, Func<bool()>(this, &GuiDocumentCommonInterface::Undo));
|
|
AddShortcutCommand(VKEY::_Y, Func<bool()>(this, &GuiDocumentCommonInterface::Redo));
|
|
AddShortcutCommand(VKEY::_A, Func<void()>(this, &GuiDocumentCommonInterface::SelectAll));
|
|
AddShortcutCommand(VKEY::_X, Func<bool()>(this, &GuiDocumentCommonInterface::Cut));
|
|
AddShortcutCommand(VKEY::_C, Func<bool()>(this, &GuiDocumentCommonInterface::Copy));
|
|
AddShortcutCommand(VKEY::_V, Func<bool()>(this, &GuiDocumentCommonInterface::Paste));
|
|
}
|
|
|
|
GuiDocumentCommonInterface::~GuiDocumentCommonInterface()
|
|
{
|
|
}
|
|
|
|
Ptr<DocumentModel> GuiDocumentCommonInterface::GetDocument()
|
|
{
|
|
return documentElement->GetDocument();
|
|
}
|
|
|
|
void GuiDocumentCommonInterface::SetDocument(Ptr<DocumentModel> value)
|
|
{
|
|
SetActiveHyperlink(0);
|
|
ClearUndoRedo();
|
|
NotifyModificationSaved();
|
|
|
|
if (value)
|
|
{
|
|
if (value->paragraphs.Count() == 0)
|
|
{
|
|
value->paragraphs.Add(new DocumentParagraphRun);
|
|
}
|
|
MergeBaselineAndDefaultFont(value);
|
|
}
|
|
|
|
documentElement->SetDocument(value);
|
|
}
|
|
|
|
//================ document items
|
|
|
|
bool GuiDocumentCommonInterface::AddDocumentItem(Ptr<GuiDocumentItem> value)
|
|
{
|
|
if (documentItems.Keys().Contains(value->GetName()))
|
|
{
|
|
return false;
|
|
}
|
|
documentItems.Add(value->GetName(), value);
|
|
documentComposition->AddChild(value->container);
|
|
value->visible = false;
|
|
value->owned = true;
|
|
value->container->SetVisible(false);
|
|
return true;
|
|
}
|
|
|
|
bool GuiDocumentCommonInterface::RemoveDocumentItem(Ptr<GuiDocumentItem> value)
|
|
{
|
|
vint index = documentItems.Keys().IndexOf(value->GetName());
|
|
if (index == -1)
|
|
{
|
|
return false;
|
|
}
|
|
if (documentItems.Values()[index] != value)
|
|
{
|
|
return false;
|
|
}
|
|
value->owned = false;
|
|
documentComposition->RemoveChild(value->container);
|
|
documentItems.Remove(value->GetName());
|
|
return true;
|
|
}
|
|
|
|
const GuiDocumentCommonInterface::DocumentItemMap& GuiDocumentCommonInterface::GetDocumentItems()
|
|
{
|
|
return documentItems;
|
|
}
|
|
|
|
//================ caret operations
|
|
|
|
TextPos GuiDocumentCommonInterface::GetCaretBegin()
|
|
{
|
|
return documentElement->GetCaretBegin();
|
|
}
|
|
|
|
TextPos GuiDocumentCommonInterface::GetCaretEnd()
|
|
{
|
|
return documentElement->GetCaretEnd();
|
|
}
|
|
|
|
void GuiDocumentCommonInterface::SetCaret(TextPos begin, TextPos end)
|
|
{
|
|
documentElement->SetCaret(begin, end, end>=begin);
|
|
UpdateCaretPoint();
|
|
SelectionChanged.Execute(documentControl->GetNotifyEventArguments());
|
|
}
|
|
|
|
TextPos GuiDocumentCommonInterface::CalculateCaretFromPoint(Point point)
|
|
{
|
|
return documentElement->CalculateCaretFromPoint(point);
|
|
}
|
|
|
|
Rect GuiDocumentCommonInterface::GetCaretBounds(TextPos caret, bool frontSide)
|
|
{
|
|
return documentElement->GetCaretBounds(caret, frontSide);
|
|
}
|
|
|
|
//================ editing operations
|
|
|
|
void GuiDocumentCommonInterface::NotifyParagraphUpdated(vint index, vint oldCount, vint newCount, bool updatedText)
|
|
{
|
|
documentElement->NotifyParagraphUpdated(index, oldCount, newCount, updatedText);
|
|
}
|
|
|
|
void GuiDocumentCommonInterface::EditRun(TextPos begin, TextPos end, Ptr<DocumentModel> model, bool copy)
|
|
{
|
|
EditTextInternal(begin, end, [=](TextPos begin, TextPos end, vint& paragraphCount, vint& lastParagraphLength)
|
|
{
|
|
documentElement->EditRun(begin, end, model, copy);
|
|
paragraphCount=model->paragraphs.Count();
|
|
lastParagraphLength=paragraphCount==0?0:model->paragraphs[paragraphCount-1]->GetText(false).Length();
|
|
});
|
|
}
|
|
|
|
void GuiDocumentCommonInterface::EditText(TextPos begin, TextPos end, bool frontSide, const collections::Array<WString>& text)
|
|
{
|
|
EditTextInternal(begin, end, [=, &text](TextPos begin, TextPos end, vint& paragraphCount, vint& lastParagraphLength)
|
|
{
|
|
documentElement->EditText(begin, end, frontSide, text);
|
|
paragraphCount=text.Count();
|
|
lastParagraphLength=paragraphCount==0?0:text[paragraphCount-1].Length();
|
|
});
|
|
}
|
|
|
|
void GuiDocumentCommonInterface::EditStyle(TextPos begin, TextPos end, Ptr<DocumentStyleProperties> style)
|
|
{
|
|
EditStyleInternal(begin, end, [=](TextPos begin, TextPos end)
|
|
{
|
|
documentElement->EditStyle(begin, end, style);
|
|
});
|
|
}
|
|
|
|
void GuiDocumentCommonInterface::EditImage(TextPos begin, TextPos end, Ptr<GuiImageData> image)
|
|
{
|
|
EditTextInternal(begin, end, [=](TextPos begin, TextPos end, vint& paragraphCount, vint& lastParagraphLength)
|
|
{
|
|
documentElement->EditImage(begin, end, image);
|
|
paragraphCount=1;
|
|
lastParagraphLength=wcslen(DocumentImageRun::RepresentationText);
|
|
});
|
|
}
|
|
|
|
void GuiDocumentCommonInterface::EditHyperlink(vint paragraphIndex, vint begin, vint end, const WString& reference, const WString& normalStyleName, const WString& activeStyleName)
|
|
{
|
|
EditStyleInternal(TextPos(paragraphIndex, begin), TextPos(paragraphIndex, end), [=](TextPos begin, TextPos end)
|
|
{
|
|
documentElement->EditHyperlink(begin.row, begin.column, end.column, reference, normalStyleName, activeStyleName);
|
|
});
|
|
}
|
|
|
|
void GuiDocumentCommonInterface::RemoveHyperlink(vint paragraphIndex, vint begin, vint end)
|
|
{
|
|
EditStyleInternal(TextPos(paragraphIndex, begin), TextPos(paragraphIndex, end), [=](TextPos begin, TextPos end)
|
|
{
|
|
documentElement->RemoveHyperlink(begin.row, begin.column, end.column);
|
|
});
|
|
}
|
|
|
|
void GuiDocumentCommonInterface::EditStyleName(TextPos begin, TextPos end, const WString& styleName)
|
|
{
|
|
EditStyleInternal(begin, end, [=](TextPos begin, TextPos end)
|
|
{
|
|
documentElement->EditStyleName(begin, end, styleName);
|
|
});
|
|
}
|
|
|
|
void GuiDocumentCommonInterface::RemoveStyleName(TextPos begin, TextPos end)
|
|
{
|
|
EditStyleInternal(begin, end, [=](TextPos begin, TextPos end)
|
|
{
|
|
documentElement->RemoveStyleName(begin, end);
|
|
});
|
|
}
|
|
|
|
void GuiDocumentCommonInterface::RenameStyle(const WString& oldStyleName, const WString& newStyleName)
|
|
{
|
|
documentElement->RenameStyle(oldStyleName, newStyleName);
|
|
|
|
// submit redo-undo
|
|
GuiDocumentUndoRedoProcessor::RenameStyleStruct arguments;
|
|
arguments.oldStyleName=oldStyleName;
|
|
arguments.newStyleName=newStyleName;
|
|
undoRedoProcessor->OnRenameStyle(arguments);
|
|
}
|
|
|
|
void GuiDocumentCommonInterface::ClearStyle(TextPos begin, TextPos end)
|
|
{
|
|
EditStyleInternal(begin, end, [=](TextPos begin, TextPos end)
|
|
{
|
|
documentElement->ClearStyle(begin, end);
|
|
});
|
|
}
|
|
|
|
Ptr<DocumentStyleProperties> GuiDocumentCommonInterface::SummarizeStyle(TextPos begin, TextPos end)
|
|
{
|
|
if (begin>end)
|
|
{
|
|
TextPos temp = begin;
|
|
begin = end;
|
|
end = temp;
|
|
}
|
|
return documentElement->SummarizeStyle(begin, end);
|
|
}
|
|
|
|
Nullable<WString> GuiDocumentCommonInterface::SummarizeStyleName(TextPos begin, TextPos end)
|
|
{
|
|
if (begin>end)
|
|
{
|
|
TextPos temp = begin;
|
|
begin = end;
|
|
end = temp;
|
|
}
|
|
return documentElement->SummarizeStyleName(begin, end);
|
|
}
|
|
|
|
void GuiDocumentCommonInterface::SetParagraphAlignments(TextPos begin, TextPos end, const collections::Array<Nullable<Alignment>>& alignments)
|
|
{
|
|
vint first = begin.row;
|
|
vint last = end.row;
|
|
if (first > last)
|
|
{
|
|
vint temp = first;
|
|
first = last;
|
|
last = temp;
|
|
}
|
|
|
|
Ptr<DocumentModel> document = documentElement->GetDocument();
|
|
if (0 <= first && first < document->paragraphs.Count() && 0 <= last && last < document->paragraphs.Count() && last - first + 1 == alignments.Count())
|
|
{
|
|
Ptr<GuiDocumentUndoRedoProcessor::SetAlignmentStruct> arguments = new GuiDocumentUndoRedoProcessor::SetAlignmentStruct;
|
|
arguments->start = first;
|
|
arguments->end = last;
|
|
arguments->originalAlignments.Resize(alignments.Count());
|
|
arguments->inputAlignments.Resize(alignments.Count());
|
|
for (vint i = first; i <= last; i++)
|
|
{
|
|
arguments->originalAlignments[i - first] = document->paragraphs[i]->alignment;
|
|
arguments->inputAlignments[i - first] = alignments[i - first];
|
|
}
|
|
documentElement->SetParagraphAlignment(begin, end, alignments);
|
|
undoRedoProcessor->OnSetAlignment(arguments);
|
|
}
|
|
}
|
|
|
|
void GuiDocumentCommonInterface::SetParagraphAlignment(TextPos begin, TextPos end, Nullable<Alignment> alignment)
|
|
{
|
|
#if defined VCZH_GCC && defined VCZH_64
|
|
#define abs labs
|
|
#endif
|
|
Array<Nullable<Alignment>> alignments(abs(begin.row - end.row) + 1);
|
|
#if defined VCZH_GCC && defined VCZH_64
|
|
#undef abs
|
|
#endif
|
|
for (vint i = 0; i < alignments.Count(); i++)
|
|
{
|
|
alignments[i] = alignment;
|
|
}
|
|
SetParagraphAlignments(begin, end, alignments);
|
|
}
|
|
|
|
Nullable<Alignment> GuiDocumentCommonInterface::SummarizeParagraphAlignment(TextPos begin, TextPos end)
|
|
{
|
|
if (begin>end)
|
|
{
|
|
TextPos temp = begin;
|
|
begin = end;
|
|
end = temp;
|
|
}
|
|
return documentElement->SummarizeParagraphAlignment(begin, end);
|
|
}
|
|
|
|
//================ editing control
|
|
|
|
WString GuiDocumentCommonInterface::GetActiveHyperlinkReference()
|
|
{
|
|
return activeHyperlinks ? activeHyperlinks->hyperlinks[0]->reference : L"";
|
|
}
|
|
|
|
GuiDocumentCommonInterface::EditMode GuiDocumentCommonInterface::GetEditMode()
|
|
{
|
|
return editMode;
|
|
}
|
|
|
|
void GuiDocumentCommonInterface::SetEditMode(EditMode value)
|
|
{
|
|
if(activeHyperlinks)
|
|
{
|
|
SetActiveHyperlink(nullptr);
|
|
}
|
|
|
|
editMode=value;
|
|
if(editMode==ViewOnly)
|
|
{
|
|
documentComposition->SetAssociatedCursor(0);
|
|
}
|
|
else
|
|
{
|
|
INativeCursor* cursor=GetCurrentController()->ResourceService()->GetSystemCursor(INativeCursor::IBeam);
|
|
documentComposition->SetAssociatedCursor(cursor);
|
|
}
|
|
}
|
|
|
|
//================ selection operations
|
|
|
|
void GuiDocumentCommonInterface::SelectAll()
|
|
{
|
|
vint lastIndex=documentElement->GetDocument()->paragraphs.Count()-1;
|
|
Ptr<DocumentParagraphRun> lastParagraph=documentElement->GetDocument()->paragraphs[lastIndex];
|
|
|
|
TextPos begin(0, 0);
|
|
TextPos end(lastIndex, lastParagraph->GetText(false).Length());
|
|
SetCaret(begin, end);
|
|
}
|
|
|
|
WString GuiDocumentCommonInterface::GetSelectionText()
|
|
{
|
|
TextPos begin=documentElement->GetCaretBegin();
|
|
TextPos end=documentElement->GetCaretEnd();
|
|
if(begin>end)
|
|
{
|
|
TextPos temp=begin;
|
|
begin=end;
|
|
end=temp;
|
|
}
|
|
|
|
Ptr<DocumentModel> model=documentElement->GetDocument()->CopyDocument(begin, end, false);
|
|
return model->GetText(true);
|
|
}
|
|
|
|
void GuiDocumentCommonInterface::SetSelectionText(const WString& value)
|
|
{
|
|
List<WString> paragraphs;
|
|
{
|
|
stream::StringReader reader(value);
|
|
WString paragraph;
|
|
bool empty=true;
|
|
|
|
while(!reader.IsEnd())
|
|
{
|
|
WString line=reader.ReadLine();
|
|
if(empty)
|
|
{
|
|
paragraph+=line;
|
|
empty=false;
|
|
}
|
|
else if(line!=L"")
|
|
{
|
|
paragraph+=L"\r\n"+line;
|
|
}
|
|
else
|
|
{
|
|
paragraphs.Add(paragraph);
|
|
paragraph=L"";
|
|
empty=true;
|
|
}
|
|
}
|
|
|
|
if(!empty)
|
|
{
|
|
paragraphs.Add(paragraph);
|
|
}
|
|
}
|
|
|
|
TextPos begin=documentElement->GetCaretBegin();
|
|
TextPos end=documentElement->GetCaretEnd();
|
|
if(begin>end)
|
|
{
|
|
TextPos temp=begin;
|
|
begin=end;
|
|
end=temp;
|
|
}
|
|
|
|
Array<WString> text;
|
|
CopyFrom(text, paragraphs);
|
|
EditText(begin, end, documentElement->IsCaretEndPreferFrontSide(), text);
|
|
}
|
|
|
|
Ptr<DocumentModel> GuiDocumentCommonInterface::GetSelectionModel()
|
|
{
|
|
TextPos begin=documentElement->GetCaretBegin();
|
|
TextPos end=documentElement->GetCaretEnd();
|
|
if(begin>end)
|
|
{
|
|
TextPos temp=begin;
|
|
begin=end;
|
|
end=temp;
|
|
}
|
|
|
|
Ptr<DocumentModel> model=documentElement->GetDocument()->CopyDocument(begin, end, true);
|
|
return model;
|
|
}
|
|
|
|
void GuiDocumentCommonInterface::SetSelectionModel(Ptr<DocumentModel> value)
|
|
{
|
|
TextPos begin=documentElement->GetCaretBegin();
|
|
TextPos end=documentElement->GetCaretEnd();
|
|
if(begin>end)
|
|
{
|
|
TextPos temp=begin;
|
|
begin=end;
|
|
end=temp;
|
|
}
|
|
|
|
EditRun(begin, end, value, true);
|
|
}
|
|
|
|
//================ clipboard operations
|
|
|
|
bool GuiDocumentCommonInterface::CanCut()
|
|
{
|
|
return editMode==Editable && documentElement->GetCaretBegin()!=documentElement->GetCaretEnd();
|
|
}
|
|
|
|
bool GuiDocumentCommonInterface::CanCopy()
|
|
{
|
|
return documentElement->GetCaretBegin()!=documentElement->GetCaretEnd();
|
|
}
|
|
|
|
bool GuiDocumentCommonInterface::CanPaste()
|
|
{
|
|
if (editMode == Editable)
|
|
{
|
|
auto reader = GetCurrentController()->ClipboardService()->ReadClipboard();
|
|
return reader->ContainsText() || reader->ContainsDocument() || reader->ContainsImage();
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool GuiDocumentCommonInterface::Cut()
|
|
{
|
|
if (!CanCut())return false;
|
|
auto writer = GetCurrentController()->ClipboardService()->WriteClipboard();
|
|
auto model = GetSelectionModel();
|
|
writer->SetDocument(model);
|
|
writer->Submit();
|
|
SetSelectionText(L"");
|
|
return true;
|
|
}
|
|
|
|
bool GuiDocumentCommonInterface::Copy()
|
|
{
|
|
if (!CanCopy()) return false;
|
|
auto writer = GetCurrentController()->ClipboardService()->WriteClipboard();
|
|
auto model = GetSelectionModel();
|
|
writer->SetDocument(model);
|
|
writer->Submit();
|
|
return true;
|
|
}
|
|
|
|
bool GuiDocumentCommonInterface::Paste()
|
|
{
|
|
if (!CanPaste()) return false;
|
|
auto reader = GetCurrentController()->ClipboardService()->ReadClipboard();
|
|
if (reader->ContainsDocument())
|
|
{
|
|
if (auto document = reader->GetDocument())
|
|
{
|
|
SetSelectionModel(document);
|
|
return true;
|
|
}
|
|
}
|
|
if (reader->ContainsText())
|
|
{
|
|
SetSelectionText(reader->GetText());
|
|
return true;
|
|
}
|
|
if (reader->ContainsImage())
|
|
{
|
|
if (auto image = reader->GetImage())
|
|
{
|
|
auto imageData = MakePtr<GuiImageData>(image, 0);
|
|
EditImage(GetCaretBegin(), GetCaretEnd(), imageData);
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
//================ undo redo control
|
|
|
|
bool GuiDocumentCommonInterface::CanUndo()
|
|
{
|
|
return editMode==Editable && undoRedoProcessor->CanUndo();
|
|
}
|
|
|
|
bool GuiDocumentCommonInterface::CanRedo()
|
|
{
|
|
return editMode==Editable && undoRedoProcessor->CanRedo();
|
|
}
|
|
|
|
void GuiDocumentCommonInterface::ClearUndoRedo()
|
|
{
|
|
undoRedoProcessor->ClearUndoRedo();
|
|
}
|
|
|
|
bool GuiDocumentCommonInterface::GetModified()
|
|
{
|
|
return undoRedoProcessor->GetModified();
|
|
}
|
|
|
|
void GuiDocumentCommonInterface::NotifyModificationSaved()
|
|
{
|
|
undoRedoProcessor->NotifyModificationSaved();
|
|
}
|
|
|
|
bool GuiDocumentCommonInterface::Undo()
|
|
{
|
|
if(CanUndo())
|
|
{
|
|
return undoRedoProcessor->Undo();
|
|
}
|
|
else
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
bool GuiDocumentCommonInterface::Redo()
|
|
{
|
|
if(CanRedo())
|
|
{
|
|
return undoRedoProcessor->Redo();
|
|
}
|
|
else
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiDocumentViewer
|
|
***********************************************************************/
|
|
|
|
void GuiDocumentViewer::BeforeControlTemplateUninstalled_()
|
|
{
|
|
}
|
|
|
|
void GuiDocumentViewer::AfterControlTemplateInstalled_(bool initialize)
|
|
{
|
|
auto ct = GetControlTemplateObject(true);
|
|
baselineDocument = ct->GetBaselineDocument();
|
|
if (documentElement)
|
|
{
|
|
documentElement->SetCaretColor(ct->GetCaretColor());
|
|
SetDocument(GetDocument());
|
|
}
|
|
}
|
|
|
|
void GuiDocumentViewer::UpdateDisplayFont()
|
|
{
|
|
GuiScrollContainer::UpdateDisplayFont();
|
|
OnFontChanged();
|
|
}
|
|
|
|
Point GuiDocumentViewer::GetDocumentViewPosition()
|
|
{
|
|
return GetViewBounds().LeftTop();
|
|
}
|
|
|
|
void GuiDocumentViewer::EnsureRectVisible(Rect bounds)
|
|
{
|
|
Rect viewBounds=GetViewBounds();
|
|
vint offset=0;
|
|
if(bounds.y1<viewBounds.y1)
|
|
{
|
|
offset=bounds.y1-viewBounds.y1;
|
|
}
|
|
else if(bounds.y2>viewBounds.y2)
|
|
{
|
|
offset=bounds.y2-viewBounds.y2;
|
|
}
|
|
|
|
if (auto scroll = GetVerticalScroll())
|
|
{
|
|
scroll->SetPosition(viewBounds.y1 + offset);
|
|
}
|
|
}
|
|
|
|
GuiDocumentViewer::GuiDocumentViewer(theme::ThemeName themeName)
|
|
:GuiScrollContainer(themeName)
|
|
{
|
|
SetAcceptTabInput(true);
|
|
SetFocusableComposition(boundsComposition);
|
|
InstallDocumentViewer(this, containerComposition, boundsComposition, focusableComposition);
|
|
|
|
SetExtendToFullWidth(true);
|
|
SetHorizontalAlwaysVisible(false);
|
|
}
|
|
|
|
GuiDocumentViewer::~GuiDocumentViewer()
|
|
{
|
|
}
|
|
|
|
const WString& GuiDocumentViewer::GetText()
|
|
{
|
|
text=documentElement->GetDocument()->GetText(true);
|
|
return text;
|
|
}
|
|
|
|
void GuiDocumentViewer::SetText(const WString& value)
|
|
{
|
|
SelectAll();
|
|
SetSelectionText(value);
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiDocumentLabel
|
|
***********************************************************************/
|
|
|
|
void GuiDocumentLabel::BeforeControlTemplateUninstalled_()
|
|
{
|
|
}
|
|
|
|
void GuiDocumentLabel::AfterControlTemplateInstalled_(bool initialize)
|
|
{
|
|
auto ct = GetControlTemplateObject(true);
|
|
baselineDocument = ct->GetBaselineDocument();
|
|
if (documentElement)
|
|
{
|
|
documentElement->SetCaretColor(ct->GetCaretColor());
|
|
SetDocument(GetDocument());
|
|
}
|
|
}
|
|
|
|
void GuiDocumentLabel::UpdateDisplayFont()
|
|
{
|
|
GuiControl::UpdateDisplayFont();
|
|
OnFontChanged();
|
|
}
|
|
|
|
GuiDocumentLabel::GuiDocumentLabel(theme::ThemeName themeName)
|
|
:GuiControl(themeName)
|
|
{
|
|
SetAcceptTabInput(true);
|
|
SetFocusableComposition(boundsComposition);
|
|
InstallDocumentViewer(this, containerComposition, boundsComposition, focusableComposition);
|
|
}
|
|
|
|
GuiDocumentLabel::~GuiDocumentLabel()
|
|
{
|
|
}
|
|
|
|
const WString& GuiDocumentLabel::GetText()
|
|
{
|
|
text=documentElement->GetDocument()->GetText(true);
|
|
return text;
|
|
}
|
|
|
|
void GuiDocumentLabel::SetText(const WString& value)
|
|
{
|
|
SelectAll();
|
|
SetSelectionText(value);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
.\CONTROLS\TEXTEDITORPACKAGE\GUITEXTCOMMONINTERFACE.CPP
|
|
***********************************************************************/
|
|
#include <math.h>
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace controls
|
|
{
|
|
using namespace elements;
|
|
using namespace elements::text;
|
|
using namespace compositions;
|
|
|
|
/***********************************************************************
|
|
GuiTextBoxCommonInterface::DefaultCallback
|
|
***********************************************************************/
|
|
|
|
GuiTextBoxCommonInterface::DefaultCallback::DefaultCallback(elements::GuiColorizedTextElement* _textElement, compositions::GuiGraphicsComposition* _textComposition)
|
|
:textElement(_textElement)
|
|
,textComposition(_textComposition)
|
|
{
|
|
}
|
|
|
|
GuiTextBoxCommonInterface::DefaultCallback::~DefaultCallback()
|
|
{
|
|
}
|
|
|
|
TextPos GuiTextBoxCommonInterface::DefaultCallback::GetLeftWord(TextPos pos)
|
|
{
|
|
return pos;
|
|
}
|
|
|
|
TextPos GuiTextBoxCommonInterface::DefaultCallback::GetRightWord(TextPos pos)
|
|
{
|
|
return pos;
|
|
}
|
|
|
|
void GuiTextBoxCommonInterface::DefaultCallback::GetWord(TextPos pos, TextPos& begin, TextPos& end)
|
|
{
|
|
begin=pos;
|
|
end=pos;
|
|
}
|
|
|
|
vint GuiTextBoxCommonInterface::DefaultCallback::GetPageRows()
|
|
{
|
|
return textComposition->GetBounds().Height()/textElement->GetLines().GetRowHeight();
|
|
}
|
|
|
|
bool GuiTextBoxCommonInterface::DefaultCallback::BeforeModify(TextPos start, TextPos end, const WString& originalText, WString& inputText)
|
|
{
|
|
return true;
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiTextBoxCommonInterface
|
|
***********************************************************************/
|
|
|
|
void GuiTextBoxCommonInterface::InvokeUndoRedoChanged()
|
|
{
|
|
UndoRedoChanged.Execute(textControl->GetNotifyEventArguments());
|
|
}
|
|
|
|
void GuiTextBoxCommonInterface::InvokeModifiedChanged()
|
|
{
|
|
ModifiedChanged.Execute(textControl->GetNotifyEventArguments());
|
|
}
|
|
|
|
void GuiTextBoxCommonInterface::UpdateCaretPoint()
|
|
{
|
|
GuiGraphicsHost* host=textComposition->GetRelatedGraphicsHost();
|
|
if(host)
|
|
{
|
|
Rect caret=textElement->GetLines().GetRectFromTextPos(textElement->GetCaretEnd());
|
|
Point view=textElement->GetViewPosition();
|
|
vint x=caret.x1-view.x;
|
|
vint y=caret.y2-view.y;
|
|
host->SetCaretPoint(Point(x, y), textComposition);
|
|
}
|
|
}
|
|
|
|
void GuiTextBoxCommonInterface::Move(TextPos pos, bool shift)
|
|
{
|
|
TextPos oldBegin = textElement->GetCaretBegin();
|
|
TextPos oldEnd = textElement->GetCaretEnd();
|
|
|
|
#if defined VCZH_MSVC
|
|
if (0 <= pos.row && pos.row < textElement->GetLines().GetCount())
|
|
{
|
|
TextLine& line = textElement->GetLines().GetLine(pos.row);
|
|
if (pos.column > 0 && UTF16SPFirst(line.text[pos.column - 1]) && UTF16SPSecond(line.text[pos.column]))
|
|
{
|
|
if (pos < oldBegin)
|
|
{
|
|
pos.column--;
|
|
}
|
|
else if (pos > oldBegin)
|
|
{
|
|
pos.column++;
|
|
}
|
|
}
|
|
}
|
|
#endif
|
|
|
|
pos = textElement->GetLines().Normalize(pos);
|
|
if (!shift)
|
|
{
|
|
textElement->SetCaretBegin(pos);
|
|
}
|
|
textElement->SetCaretEnd(pos);
|
|
if (textControl)
|
|
{
|
|
GuiGraphicsHost* host = textComposition->GetRelatedGraphicsHost();
|
|
if (host)
|
|
{
|
|
if (host->GetFocusedComposition() == textControl->GetFocusableComposition())
|
|
{
|
|
textElement->SetCaretVisible(true);
|
|
}
|
|
}
|
|
}
|
|
|
|
Rect bounds = textElement->GetLines().GetRectFromTextPos(pos);
|
|
Rect view = Rect(textElement->GetViewPosition(), textComposition->GetBounds().GetSize());
|
|
Point viewPoint = view.LeftTop();
|
|
|
|
if (view.x2 > view.x1 && view.y2 > view.y1)
|
|
{
|
|
if (bounds.x1 < view.x1)
|
|
{
|
|
viewPoint.x = bounds.x1;
|
|
}
|
|
else if (bounds.x2 > view.x2)
|
|
{
|
|
viewPoint.x = bounds.x2 - view.Width();
|
|
}
|
|
if (bounds.y1 < view.y1)
|
|
{
|
|
viewPoint.y = bounds.y1;
|
|
}
|
|
else if (bounds.y2 > view.y2)
|
|
{
|
|
viewPoint.y = bounds.y2 - view.Height();
|
|
}
|
|
}
|
|
|
|
callback->ScrollToView(viewPoint);
|
|
UpdateCaretPoint();
|
|
|
|
TextPos newBegin = textElement->GetCaretBegin();
|
|
TextPos newEnd = textElement->GetCaretEnd();
|
|
if (oldBegin != newBegin || oldEnd != newEnd)
|
|
{
|
|
ICommonTextEditCallback::TextCaretChangedStruct arguments;
|
|
arguments.oldBegin = oldBegin;
|
|
arguments.oldEnd = oldEnd;
|
|
arguments.newBegin = newBegin;
|
|
arguments.newEnd = newEnd;
|
|
arguments.editVersion = editVersion;
|
|
for (vint i = 0; i < textEditCallbacks.Count(); i++)
|
|
{
|
|
textEditCallbacks[i]->TextCaretChanged(arguments);
|
|
}
|
|
SelectionChanged.Execute(textControl->GetNotifyEventArguments());
|
|
}
|
|
}
|
|
|
|
void GuiTextBoxCommonInterface::Modify(TextPos start, TextPos end, const WString& input, bool asKeyInput)
|
|
{
|
|
if(start>end)
|
|
{
|
|
TextPos temp=start;
|
|
start=end;
|
|
end=temp;
|
|
}
|
|
TextPos originalStart=start;
|
|
TextPos originalEnd=end;
|
|
WString originalText=textElement->GetLines().GetText(start, end);
|
|
WString inputText=input;
|
|
if(callback->BeforeModify(start, end, originalText, inputText))
|
|
{
|
|
{
|
|
ICommonTextEditCallback::TextEditPreviewStruct arguments;
|
|
arguments.originalStart=originalStart;
|
|
arguments.originalEnd=originalEnd;
|
|
arguments.originalText=originalText;
|
|
arguments.inputText=inputText;
|
|
arguments.editVersion=editVersion;
|
|
arguments.keyInput=asKeyInput;
|
|
for(vint i=0;i<textEditCallbacks.Count();i++)
|
|
{
|
|
textEditCallbacks[i]->TextEditPreview(arguments);
|
|
}
|
|
|
|
inputText=arguments.inputText;
|
|
if(originalStart!=arguments.originalStart || originalEnd!=arguments.originalEnd)
|
|
{
|
|
originalStart=arguments.originalStart;
|
|
originalEnd=arguments.originalEnd;
|
|
originalText=textElement->GetLines().GetText(originalStart, originalEnd);
|
|
start=originalStart;
|
|
end=originalEnd;
|
|
}
|
|
}
|
|
|
|
SPIN_LOCK(elementModifyLock)
|
|
{
|
|
end=textElement->GetLines().Modify(start, end, inputText);
|
|
}
|
|
callback->AfterModify(originalStart, originalEnd, originalText, start, end, inputText);
|
|
|
|
editVersion++;
|
|
{
|
|
ICommonTextEditCallback::TextEditNotifyStruct arguments;
|
|
arguments.originalStart=originalStart;
|
|
arguments.originalEnd=originalEnd;
|
|
arguments.originalText=originalText;
|
|
arguments.inputStart=start;
|
|
arguments.inputEnd=end;
|
|
arguments.inputText=inputText;
|
|
arguments.editVersion=editVersion;
|
|
arguments.keyInput=asKeyInput;
|
|
for(vint i=0;i<textEditCallbacks.Count();i++)
|
|
{
|
|
textEditCallbacks[i]->TextEditNotify(arguments);
|
|
}
|
|
}
|
|
|
|
Move(end, false);
|
|
|
|
for(vint i=0;i<textEditCallbacks.Count();i++)
|
|
{
|
|
textEditCallbacks[i]->TextEditFinished(editVersion);
|
|
}
|
|
|
|
textControl->TextChanged.Execute(textControl->GetNotifyEventArguments());
|
|
}
|
|
}
|
|
|
|
bool GuiTextBoxCommonInterface::ProcessKey(VKEY code, bool shift, bool ctrl)
|
|
{
|
|
if(IGuiShortcutKeyItem* item=internalShortcutKeyManager->TryGetShortcut(ctrl, shift, false, code))
|
|
{
|
|
GuiEventArgs arguments(textControl->GetBoundsComposition());
|
|
item->Executed.Execute(arguments);
|
|
return true;
|
|
}
|
|
|
|
TextPos begin=textElement->GetCaretBegin();
|
|
TextPos end=textElement->GetCaretEnd();
|
|
switch(code)
|
|
{
|
|
case VKEY::_ESCAPE:
|
|
if(autoComplete && autoComplete->IsListOpening() && !shift && !ctrl)
|
|
{
|
|
autoComplete->CloseList();
|
|
return true;
|
|
}
|
|
break;
|
|
case VKEY::_RETURN:
|
|
if(autoComplete && autoComplete->IsListOpening() && !shift && !ctrl)
|
|
{
|
|
if(autoComplete->ApplySelectedListItem())
|
|
{
|
|
preventEnterDueToAutoComplete=true;
|
|
return true;
|
|
}
|
|
}
|
|
break;
|
|
case VKEY::_UP:
|
|
if(autoComplete && autoComplete->IsListOpening() && !shift && !ctrl)
|
|
{
|
|
autoComplete->SelectPreviousListItem();
|
|
}
|
|
else
|
|
{
|
|
end.row--;
|
|
Move(end, shift);
|
|
}
|
|
return true;
|
|
case VKEY::_DOWN:
|
|
if(autoComplete && autoComplete->IsListOpening() && !shift && !ctrl)
|
|
{
|
|
autoComplete->SelectNextListItem();
|
|
}
|
|
else
|
|
{
|
|
end.row++;
|
|
Move(end, shift);
|
|
}
|
|
return true;
|
|
case VKEY::_LEFT:
|
|
{
|
|
if(ctrl)
|
|
{
|
|
Move(callback->GetLeftWord(end), shift);
|
|
}
|
|
else
|
|
{
|
|
if(end.column==0)
|
|
{
|
|
if(end.row>0)
|
|
{
|
|
end.row--;
|
|
end=textElement->GetLines().Normalize(end);
|
|
end.column=textElement->GetLines().GetLine(end.row).dataLength;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
end.column--;
|
|
}
|
|
Move(end, shift);
|
|
}
|
|
}
|
|
return true;
|
|
case VKEY::_RIGHT:
|
|
{
|
|
if(ctrl)
|
|
{
|
|
Move(callback->GetRightWord(end), shift);
|
|
}
|
|
else
|
|
{
|
|
if(end.column==textElement->GetLines().GetLine(end.row).dataLength)
|
|
{
|
|
if(end.row<textElement->GetLines().GetCount()-1)
|
|
{
|
|
end.row++;
|
|
end.column=0;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
end.column++;
|
|
}
|
|
Move(end, shift);
|
|
}
|
|
}
|
|
return true;
|
|
case VKEY::_HOME:
|
|
{
|
|
if(ctrl)
|
|
{
|
|
Move(TextPos(0, 0), shift);
|
|
}
|
|
else
|
|
{
|
|
end.column=0;
|
|
Move(end, shift);
|
|
}
|
|
}
|
|
return true;
|
|
case VKEY::_END:
|
|
{
|
|
if(ctrl)
|
|
{
|
|
end.row=textElement->GetLines().GetCount()-1;
|
|
}
|
|
end.column=textElement->GetLines().GetLine(end.row).dataLength;
|
|
Move(end, shift);
|
|
}
|
|
return true;
|
|
case VKEY::_PRIOR:
|
|
{
|
|
end.row-=callback->GetPageRows();
|
|
Move(end, shift);
|
|
}
|
|
return true;
|
|
case VKEY::_NEXT:
|
|
{
|
|
end.row+=callback->GetPageRows();
|
|
Move(end, shift);
|
|
}
|
|
return true;
|
|
case VKEY::_BACK:
|
|
if(!readonly)
|
|
{
|
|
if(ctrl && !shift)
|
|
{
|
|
ProcessKey(VKEY::_LEFT, true, true);
|
|
ProcessKey(VKEY::_BACK, false, false);
|
|
}
|
|
else if(!ctrl && shift)
|
|
{
|
|
ProcessKey(VKEY::_UP, true, false);
|
|
ProcessKey(VKEY::_BACK, false, false);
|
|
}
|
|
else
|
|
{
|
|
if(begin==end)
|
|
{
|
|
ProcessKey(VKEY::_LEFT, true, false);
|
|
}
|
|
SetSelectionTextAsKeyInput(L"");
|
|
}
|
|
return true;
|
|
}
|
|
break;
|
|
case VKEY::_DELETE:
|
|
if(!readonly)
|
|
{
|
|
if(ctrl && !shift)
|
|
{
|
|
ProcessKey(VKEY::_RIGHT, true, true);
|
|
ProcessKey(VKEY::_DELETE, false, false);
|
|
}
|
|
else if(!ctrl && shift)
|
|
{
|
|
ProcessKey(VKEY::_DOWN, true, false);
|
|
ProcessKey(VKEY::_DELETE, false, false);
|
|
}
|
|
else
|
|
{
|
|
if(begin==end)
|
|
{
|
|
ProcessKey(VKEY::_RIGHT, true, false);
|
|
}
|
|
SetSelectionTextAsKeyInput(L"");
|
|
}
|
|
return true;
|
|
}
|
|
break;
|
|
default:;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void GuiTextBoxCommonInterface::OnGotFocus(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
textElement->SetFocused(true);
|
|
textElement->SetCaretVisible(true);
|
|
UpdateCaretPoint();
|
|
}
|
|
|
|
void GuiTextBoxCommonInterface::OnLostFocus(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
textElement->SetFocused(false);
|
|
textElement->SetCaretVisible(false);
|
|
}
|
|
|
|
void GuiTextBoxCommonInterface::OnCaretNotify(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
textElement->SetCaretVisible(!textElement->GetCaretVisible());
|
|
}
|
|
|
|
void GuiTextBoxCommonInterface::OnLeftButtonDown(compositions::GuiGraphicsComposition* sender, compositions::GuiMouseEventArgs& arguments)
|
|
{
|
|
if(textControl->GetVisuallyEnabled() && arguments.compositionSource==arguments.eventSource)
|
|
{
|
|
dragging=true;
|
|
TextPos pos=GetNearestTextPos(Point(arguments.x, arguments.y));
|
|
Move(pos, arguments.shift);
|
|
}
|
|
}
|
|
|
|
void GuiTextBoxCommonInterface::OnLeftButtonUp(compositions::GuiGraphicsComposition* sender, compositions::GuiMouseEventArgs& arguments)
|
|
{
|
|
if(textControl->GetVisuallyEnabled() && arguments.compositionSource==arguments.eventSource)
|
|
{
|
|
dragging=false;
|
|
}
|
|
}
|
|
|
|
void GuiTextBoxCommonInterface::OnMouseMove(compositions::GuiGraphicsComposition* sender, compositions::GuiMouseEventArgs& arguments)
|
|
{
|
|
if(textControl->GetVisuallyEnabled() && arguments.compositionSource==arguments.eventSource)
|
|
{
|
|
if(dragging)
|
|
{
|
|
TextPos pos=GetNearestTextPos(Point(arguments.x, arguments.y));
|
|
Move(pos, true);
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiTextBoxCommonInterface::OnKeyDown(compositions::GuiGraphicsComposition* sender, compositions::GuiKeyEventArgs& arguments)
|
|
{
|
|
if(textControl->GetVisuallyEnabled() && arguments.compositionSource==arguments.eventSource)
|
|
{
|
|
if(ProcessKey(arguments.code, arguments.shift, arguments.ctrl))
|
|
{
|
|
arguments.handled=true;
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiTextBoxCommonInterface::OnCharInput(compositions::GuiGraphicsComposition* sender, compositions::GuiCharEventArgs& arguments)
|
|
{
|
|
if (preventEnterDueToAutoComplete)
|
|
{
|
|
preventEnterDueToAutoComplete = false;
|
|
if (arguments.code == (wchar_t)VKEY::_RETURN)
|
|
{
|
|
return;
|
|
}
|
|
}
|
|
if (textControl->GetVisuallyEnabled() && arguments.compositionSource == arguments.eventSource)
|
|
{
|
|
if (!readonly &&
|
|
arguments.code != (wchar_t)VKEY::_ESCAPE &&
|
|
arguments.code != (wchar_t)VKEY::_BACK &&
|
|
(arguments.code != (wchar_t)VKEY::_TAB || textControl->GetAcceptTabInput()) &&
|
|
!arguments.ctrl)
|
|
{
|
|
SetSelectionTextAsKeyInput(WString(arguments.code));
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiTextBoxCommonInterface::Install(
|
|
elements::GuiColorizedTextElement* _textElement,
|
|
compositions::GuiGraphicsComposition* _textComposition,
|
|
GuiControl* _textControl,
|
|
compositions::GuiGraphicsComposition* eventComposition,
|
|
compositions::GuiGraphicsComposition* focusableComposition
|
|
)
|
|
{
|
|
textElement=_textElement;
|
|
textComposition=_textComposition;
|
|
textControl=_textControl;
|
|
textComposition->SetAssociatedCursor(GetCurrentController()->ResourceService()->GetSystemCursor(INativeCursor::IBeam));
|
|
SelectionChanged.SetAssociatedComposition(eventComposition);
|
|
UndoRedoChanged.SetAssociatedComposition(eventComposition);
|
|
ModifiedChanged.SetAssociatedComposition(eventComposition);
|
|
|
|
undoRedoProcessor->UndoRedoChanged.Add(this, &GuiTextBoxCommonInterface::InvokeUndoRedoChanged);
|
|
undoRedoProcessor->ModifiedChanged.Add(this, &GuiTextBoxCommonInterface::InvokeModifiedChanged);
|
|
|
|
focusableComposition->GetEventReceiver()->gotFocus.AttachMethod(this, &GuiTextBoxCommonInterface::OnGotFocus);
|
|
focusableComposition->GetEventReceiver()->lostFocus.AttachMethod(this, &GuiTextBoxCommonInterface::OnLostFocus);
|
|
focusableComposition->GetEventReceiver()->caretNotify.AttachMethod(this, &GuiTextBoxCommonInterface::OnCaretNotify);
|
|
textComposition->GetEventReceiver()->leftButtonDown.AttachMethod(this, &GuiTextBoxCommonInterface::OnLeftButtonDown);
|
|
textComposition->GetEventReceiver()->leftButtonUp.AttachMethod(this, &GuiTextBoxCommonInterface::OnLeftButtonUp);
|
|
textComposition->GetEventReceiver()->mouseMove.AttachMethod(this, &GuiTextBoxCommonInterface::OnMouseMove);
|
|
focusableComposition->GetEventReceiver()->keyDown.AttachMethod(this, &GuiTextBoxCommonInterface::OnKeyDown);
|
|
focusableComposition->GetEventReceiver()->charInput.AttachMethod(this, &GuiTextBoxCommonInterface::OnCharInput);
|
|
|
|
for(vint i=0;i<textEditCallbacks.Count();i++)
|
|
{
|
|
textEditCallbacks[i]->Attach(textElement, elementModifyLock, textComposition ,editVersion);
|
|
}
|
|
}
|
|
|
|
GuiTextBoxCommonInterface::ICallback* GuiTextBoxCommonInterface::GetCallback()
|
|
{
|
|
return callback;
|
|
}
|
|
|
|
void GuiTextBoxCommonInterface::SetCallback(ICallback* value)
|
|
{
|
|
callback=value;
|
|
}
|
|
|
|
bool GuiTextBoxCommonInterface::AttachTextEditCallback(Ptr<ICommonTextEditCallback> value)
|
|
{
|
|
if(textEditCallbacks.Contains(value.Obj()))
|
|
{
|
|
return false;
|
|
}
|
|
else
|
|
{
|
|
textEditCallbacks.Add(value);
|
|
if(textElement)
|
|
{
|
|
value->Attach(textElement, elementModifyLock, textComposition, editVersion);
|
|
}
|
|
return true;
|
|
}
|
|
}
|
|
|
|
bool GuiTextBoxCommonInterface::DetachTextEditCallback(Ptr<ICommonTextEditCallback> value)
|
|
{
|
|
if(textEditCallbacks.Remove(value.Obj()))
|
|
{
|
|
value->Detach();
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
void GuiTextBoxCommonInterface::AddShortcutCommand(VKEY key, const Func<void()>& eventHandler)
|
|
{
|
|
IGuiShortcutKeyItem* item=internalShortcutKeyManager->CreateShortcut(true, false, false, key);
|
|
item->Executed.AttachLambda([=](GuiGraphicsComposition* sender, GuiEventArgs& arguments)
|
|
{
|
|
eventHandler();
|
|
});
|
|
}
|
|
|
|
elements::GuiColorizedTextElement* GuiTextBoxCommonInterface::GetTextElement()
|
|
{
|
|
return textElement;
|
|
}
|
|
|
|
void GuiTextBoxCommonInterface::UnsafeSetText(const WString& value)
|
|
{
|
|
if(textElement)
|
|
{
|
|
TextPos end;
|
|
if(textElement->GetLines().GetCount()>0)
|
|
{
|
|
end.row=textElement->GetLines().GetCount()-1;
|
|
end.column=textElement->GetLines().GetLine(end.row).dataLength;
|
|
}
|
|
Modify(TextPos(), end, value, false);
|
|
}
|
|
}
|
|
|
|
GuiTextBoxCommonInterface::GuiTextBoxCommonInterface()
|
|
:textElement(0)
|
|
,textComposition(0)
|
|
,editVersion(0)
|
|
,textControl(0)
|
|
,callback(0)
|
|
,dragging(false)
|
|
,readonly(false)
|
|
,preventEnterDueToAutoComplete(false)
|
|
{
|
|
undoRedoProcessor=new GuiTextBoxUndoRedoProcessor;
|
|
AttachTextEditCallback(undoRedoProcessor);
|
|
|
|
internalShortcutKeyManager=new GuiShortcutKeyManager;
|
|
AddShortcutCommand(VKEY::_Z, Func<bool()>(this, &GuiTextBoxCommonInterface::Undo));
|
|
AddShortcutCommand(VKEY::_Y, Func<bool()>(this, &GuiTextBoxCommonInterface::Redo));
|
|
AddShortcutCommand(VKEY::_A, Func<void()>(this, &GuiTextBoxCommonInterface::SelectAll));
|
|
AddShortcutCommand(VKEY::_X, Func<bool()>(this, &GuiTextBoxCommonInterface::Cut));
|
|
AddShortcutCommand(VKEY::_C, Func<bool()>(this, &GuiTextBoxCommonInterface::Copy));
|
|
AddShortcutCommand(VKEY::_V, Func<bool()>(this, &GuiTextBoxCommonInterface::Paste));
|
|
}
|
|
|
|
GuiTextBoxCommonInterface::~GuiTextBoxCommonInterface()
|
|
{
|
|
if(colorizer)
|
|
{
|
|
DetachTextEditCallback(colorizer);
|
|
colorizer=0;
|
|
}
|
|
if(undoRedoProcessor)
|
|
{
|
|
DetachTextEditCallback(undoRedoProcessor);
|
|
undoRedoProcessor=0;
|
|
}
|
|
|
|
for(vint i=0;i<textEditCallbacks.Count();i++)
|
|
{
|
|
textEditCallbacks[i]->Detach();
|
|
}
|
|
textEditCallbacks.Clear();
|
|
}
|
|
|
|
//================ clipboard operations
|
|
|
|
bool GuiTextBoxCommonInterface::CanCut()
|
|
{
|
|
return !readonly && textElement->GetCaretBegin()!=textElement->GetCaretEnd() && textElement->GetPasswordChar()==L'\0';
|
|
}
|
|
|
|
bool GuiTextBoxCommonInterface::CanCopy()
|
|
{
|
|
return textElement->GetCaretBegin()!=textElement->GetCaretEnd() && textElement->GetPasswordChar()==L'\0';
|
|
}
|
|
|
|
bool GuiTextBoxCommonInterface::CanPaste()
|
|
{
|
|
if (!readonly && textElement->GetPasswordChar() == L'\0')
|
|
{
|
|
auto reader = GetCurrentController()->ClipboardService()->ReadClipboard();
|
|
return reader->ContainsText();
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool GuiTextBoxCommonInterface::Cut()
|
|
{
|
|
if (!CanCut()) return false;
|
|
auto writer = GetCurrentController()->ClipboardService()->WriteClipboard();
|
|
writer->SetText(GetSelectionText());
|
|
writer->Submit();
|
|
SetSelectionText(L"");
|
|
return true;
|
|
}
|
|
|
|
bool GuiTextBoxCommonInterface::Copy()
|
|
{
|
|
if (!CanCopy()) return false;
|
|
auto writer = GetCurrentController()->ClipboardService()->WriteClipboard();
|
|
writer->SetText(GetSelectionText());
|
|
writer->Submit();
|
|
return true;
|
|
}
|
|
|
|
bool GuiTextBoxCommonInterface::Paste()
|
|
{
|
|
if (!CanPaste()) return false;
|
|
auto reader = GetCurrentController()->ClipboardService()->ReadClipboard();
|
|
SetSelectionText(reader->GetText());
|
|
return true;
|
|
}
|
|
|
|
//================ editing control
|
|
|
|
bool GuiTextBoxCommonInterface::GetReadonly()
|
|
{
|
|
return readonly;
|
|
}
|
|
|
|
void GuiTextBoxCommonInterface::SetReadonly(bool value)
|
|
{
|
|
readonly=value;
|
|
}
|
|
|
|
//================ text operations
|
|
|
|
void GuiTextBoxCommonInterface::Select(TextPos begin, TextPos end)
|
|
{
|
|
Move(begin, false);
|
|
Move(end, true);
|
|
}
|
|
|
|
void GuiTextBoxCommonInterface::SelectAll()
|
|
{
|
|
vint row=textElement->GetLines().GetCount()-1;
|
|
Move(TextPos(0, 0), false);
|
|
Move(TextPos(row, textElement->GetLines().GetLine(row).dataLength), true);
|
|
}
|
|
|
|
WString GuiTextBoxCommonInterface::GetSelectionText()
|
|
{
|
|
TextPos selectionBegin=textElement->GetCaretBegin()<textElement->GetCaretEnd()?textElement->GetCaretBegin():textElement->GetCaretEnd();
|
|
TextPos selectionEnd=textElement->GetCaretBegin()>textElement->GetCaretEnd()?textElement->GetCaretBegin():textElement->GetCaretEnd();
|
|
return textElement->GetLines().GetText(selectionBegin, selectionEnd);
|
|
}
|
|
|
|
void GuiTextBoxCommonInterface::SetSelectionText(const WString& value)
|
|
{
|
|
Modify(textElement->GetCaretBegin(), textElement->GetCaretEnd(), value, false);
|
|
}
|
|
|
|
void GuiTextBoxCommonInterface::SetSelectionTextAsKeyInput(const WString& value)
|
|
{
|
|
Modify(textElement->GetCaretBegin(), textElement->GetCaretEnd(), value, true);
|
|
}
|
|
|
|
WString GuiTextBoxCommonInterface::GetRowText(vint row)
|
|
{
|
|
TextPos start=textElement->GetLines().Normalize(TextPos(row, 0));
|
|
TextPos end=TextPos(start.row, textElement->GetLines().GetLine(start.row).dataLength);
|
|
return GetFragmentText(start, end);
|
|
}
|
|
|
|
vint GuiTextBoxCommonInterface::GetRowCount()
|
|
{
|
|
return textElement->GetLines().GetCount();
|
|
}
|
|
|
|
WString GuiTextBoxCommonInterface::GetFragmentText(TextPos start, TextPos end)
|
|
{
|
|
start=textElement->GetLines().Normalize(start);
|
|
end=textElement->GetLines().Normalize(end);
|
|
return textElement->GetLines().GetText(start, end);
|
|
}
|
|
|
|
TextPos GuiTextBoxCommonInterface::GetCaretBegin()
|
|
{
|
|
return textElement->GetCaretBegin();
|
|
}
|
|
|
|
TextPos GuiTextBoxCommonInterface::GetCaretEnd()
|
|
{
|
|
return textElement->GetCaretEnd();
|
|
}
|
|
|
|
TextPos GuiTextBoxCommonInterface::GetCaretSmall()
|
|
{
|
|
TextPos c1=GetCaretBegin();
|
|
TextPos c2=GetCaretEnd();
|
|
return c1<c2?c1:c2;
|
|
}
|
|
|
|
TextPos GuiTextBoxCommonInterface::GetCaretLarge()
|
|
{
|
|
TextPos c1=GetCaretBegin();
|
|
TextPos c2=GetCaretEnd();
|
|
return c1>c2?c1:c2;
|
|
}
|
|
|
|
//================ position query
|
|
|
|
vint GuiTextBoxCommonInterface::GetRowWidth(vint row)
|
|
{
|
|
return textElement->GetLines().GetRowWidth(row);
|
|
}
|
|
|
|
vint GuiTextBoxCommonInterface::GetRowHeight()
|
|
{
|
|
return textElement->GetLines().GetRowHeight();
|
|
}
|
|
|
|
vint GuiTextBoxCommonInterface::GetMaxWidth()
|
|
{
|
|
return textElement->GetLines().GetMaxWidth();
|
|
}
|
|
|
|
vint GuiTextBoxCommonInterface::GetMaxHeight()
|
|
{
|
|
return textElement->GetLines().GetMaxHeight();
|
|
}
|
|
|
|
TextPos GuiTextBoxCommonInterface::GetTextPosFromPoint(Point point)
|
|
{
|
|
Point view=textElement->GetViewPosition();
|
|
return textElement->GetLines().GetTextPosFromPoint(Point(point.x+view.x, point.y+view.y));
|
|
}
|
|
|
|
Point GuiTextBoxCommonInterface::GetPointFromTextPos(TextPos pos)
|
|
{
|
|
Point view=textElement->GetViewPosition();
|
|
Point result=textElement->GetLines().GetPointFromTextPos(pos);
|
|
return Point(result.x-view.x, result.y-view.y);
|
|
}
|
|
|
|
Rect GuiTextBoxCommonInterface::GetRectFromTextPos(TextPos pos)
|
|
{
|
|
Point view=textElement->GetViewPosition();
|
|
Rect result=textElement->GetLines().GetRectFromTextPos(pos);
|
|
return Rect(Point(result.x1-view.x, result.y1-view.y), result.GetSize());
|
|
}
|
|
|
|
TextPos GuiTextBoxCommonInterface::GetNearestTextPos(Point point)
|
|
{
|
|
Point viewPosition=textElement->GetViewPosition();
|
|
Point mousePosition=Point(point.x+viewPosition.x, point.y+viewPosition.y);
|
|
TextPos pos=textElement->GetLines().GetTextPosFromPoint(mousePosition);
|
|
if(pos.column<textElement->GetLines().GetLine(pos.row).dataLength)
|
|
{
|
|
Rect rect=textElement->GetLines().GetRectFromTextPos(pos);
|
|
if(abs((int)(rect.x1-mousePosition.x))>=abs((int)(rect.x2-1-mousePosition.x)))
|
|
{
|
|
pos.column++;
|
|
}
|
|
}
|
|
return pos;
|
|
}
|
|
|
|
//================ colorizing
|
|
|
|
Ptr<GuiTextBoxColorizerBase> GuiTextBoxCommonInterface::GetColorizer()
|
|
{
|
|
return colorizer;
|
|
}
|
|
|
|
void GuiTextBoxCommonInterface::SetColorizer(Ptr<GuiTextBoxColorizerBase> value)
|
|
{
|
|
if (!filledDefaultColors)
|
|
{
|
|
filledDefaultColors = true;
|
|
CopyFrom(defaultColors, GetTextElement()->GetColors());
|
|
}
|
|
|
|
if(colorizer)
|
|
{
|
|
DetachTextEditCallback(colorizer);
|
|
}
|
|
colorizer=value;
|
|
if(colorizer)
|
|
{
|
|
AttachTextEditCallback(colorizer);
|
|
GetTextElement()->SetColors(colorizer->GetColors());
|
|
}
|
|
else
|
|
{
|
|
GetTextElement()->SetColors(defaultColors);
|
|
GetTextElement()->ResetTextColorIndex(0);
|
|
}
|
|
}
|
|
|
|
//================ auto complete
|
|
|
|
Ptr<GuiTextBoxAutoCompleteBase> GuiTextBoxCommonInterface::GetAutoComplete()
|
|
{
|
|
return autoComplete;
|
|
}
|
|
|
|
void GuiTextBoxCommonInterface::SetAutoComplete(Ptr<GuiTextBoxAutoCompleteBase> value)
|
|
{
|
|
if(autoComplete)
|
|
{
|
|
DetachTextEditCallback(autoComplete);
|
|
}
|
|
autoComplete=value;
|
|
if(autoComplete)
|
|
{
|
|
AttachTextEditCallback(autoComplete);
|
|
}
|
|
}
|
|
|
|
//================ undo redo control
|
|
|
|
vuint GuiTextBoxCommonInterface::GetEditVersion()
|
|
{
|
|
return editVersion;
|
|
}
|
|
|
|
bool GuiTextBoxCommonInterface::CanUndo()
|
|
{
|
|
return !readonly && undoRedoProcessor->CanUndo();
|
|
}
|
|
|
|
bool GuiTextBoxCommonInterface::CanRedo()
|
|
{
|
|
return !readonly && undoRedoProcessor->CanRedo();
|
|
}
|
|
|
|
void GuiTextBoxCommonInterface::ClearUndoRedo()
|
|
{
|
|
undoRedoProcessor->ClearUndoRedo();
|
|
}
|
|
|
|
bool GuiTextBoxCommonInterface::GetModified()
|
|
{
|
|
return undoRedoProcessor->GetModified();
|
|
}
|
|
|
|
void GuiTextBoxCommonInterface::NotifyModificationSaved()
|
|
{
|
|
undoRedoProcessor->NotifyModificationSaved();
|
|
}
|
|
|
|
bool GuiTextBoxCommonInterface::Undo()
|
|
{
|
|
if(CanUndo())
|
|
{
|
|
return undoRedoProcessor->Undo();
|
|
}
|
|
else
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
bool GuiTextBoxCommonInterface::Redo()
|
|
{
|
|
if(CanRedo())
|
|
{
|
|
return undoRedoProcessor->Redo();
|
|
}
|
|
else
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\CONTROLS\TEXTEDITORPACKAGE\GUITEXTCONTROLS.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace controls
|
|
{
|
|
using namespace elements;
|
|
using namespace elements::text;
|
|
using namespace compositions;
|
|
using namespace collections;
|
|
|
|
/***********************************************************************
|
|
GuiMultilineTextBox::DefaultTextElementOperatorCallback
|
|
***********************************************************************/
|
|
|
|
GuiMultilineTextBox::TextElementOperatorCallback::TextElementOperatorCallback(GuiMultilineTextBox* _textControl)
|
|
:GuiTextBoxCommonInterface::DefaultCallback(
|
|
_textControl->textElement,
|
|
_textControl->textComposition
|
|
)
|
|
,textControl(_textControl)
|
|
{
|
|
}
|
|
|
|
void GuiMultilineTextBox::TextElementOperatorCallback::AfterModify(TextPos originalStart, TextPos originalEnd, const WString& originalText, TextPos inputStart, TextPos inputEnd, const WString& inputText)
|
|
{
|
|
textControl->CalculateView();
|
|
}
|
|
|
|
void GuiMultilineTextBox::TextElementOperatorCallback::ScrollToView(Point point)
|
|
{
|
|
point.x+=TextMargin;
|
|
point.y+=TextMargin;
|
|
Point oldPoint = textControl->GetViewPosition();
|
|
vint marginX=0;
|
|
vint marginY=0;
|
|
if(oldPoint.x<point.x)
|
|
{
|
|
marginX=TextMargin;
|
|
}
|
|
else if(oldPoint.x>point.x)
|
|
{
|
|
marginX=-TextMargin;
|
|
}
|
|
if(oldPoint.y<point.y)
|
|
{
|
|
marginY=TextMargin;
|
|
}
|
|
else if(oldPoint.y>point.y)
|
|
{
|
|
marginY=-TextMargin;
|
|
}
|
|
textControl->SetViewPosition(Point(point.x + marginX, point.y + marginY));
|
|
}
|
|
|
|
vint GuiMultilineTextBox::TextElementOperatorCallback::GetTextMargin()
|
|
{
|
|
return TextMargin;
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiMultilineTextBox::CommandExecutor
|
|
***********************************************************************/
|
|
|
|
GuiMultilineTextBox::CommandExecutor::CommandExecutor(GuiMultilineTextBox* _textBox)
|
|
:textBox(_textBox)
|
|
{
|
|
}
|
|
|
|
GuiMultilineTextBox::CommandExecutor::~CommandExecutor()
|
|
{
|
|
}
|
|
|
|
void GuiMultilineTextBox::CommandExecutor::UnsafeSetText(const WString& value)
|
|
{
|
|
textBox->UnsafeSetText(value);
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiMultilineTextBox
|
|
***********************************************************************/
|
|
|
|
void GuiMultilineTextBox::BeforeControlTemplateUninstalled_()
|
|
{
|
|
auto ct = GetControlTemplateObject(false);
|
|
if (!ct) return;
|
|
|
|
ct->SetCommands(nullptr);
|
|
}
|
|
|
|
void GuiMultilineTextBox::AfterControlTemplateInstalled_(bool initialize)
|
|
{
|
|
auto ct = GetControlTemplateObject(true);
|
|
Array<ColorEntry> colors(1);
|
|
colors[0] = ct->GetTextColor();
|
|
textElement->SetColors(colors);
|
|
textElement->SetCaretColor(ct->GetCaretColor());
|
|
ct->SetCommands(commandExecutor.Obj());
|
|
}
|
|
|
|
void GuiMultilineTextBox::UpdateVisuallyEnabled()
|
|
{
|
|
GuiControl::UpdateVisuallyEnabled();
|
|
textElement->SetVisuallyEnabled(GetVisuallyEnabled());
|
|
}
|
|
|
|
void GuiMultilineTextBox::UpdateDisplayFont()
|
|
{
|
|
GuiControl::UpdateDisplayFont();
|
|
textElement->SetFont(GetDisplayFont());
|
|
CalculateViewAndSetScroll();
|
|
}
|
|
|
|
void GuiMultilineTextBox::OnRenderTargetChanged(elements::IGuiGraphicsRenderTarget* renderTarget)
|
|
{
|
|
CalculateViewAndSetScroll();
|
|
GuiScrollView::OnRenderTargetChanged(renderTarget);
|
|
}
|
|
|
|
Size GuiMultilineTextBox::QueryFullSize()
|
|
{
|
|
TextLines& lines = textElement->GetLines();
|
|
return Size(lines.GetMaxWidth() + TextMargin * 2, lines.GetMaxHeight() + TextMargin * 2);
|
|
}
|
|
|
|
void GuiMultilineTextBox::UpdateView(Rect viewBounds)
|
|
{
|
|
textElement->SetViewPosition(viewBounds.LeftTop() - Size(TextMargin, TextMargin));
|
|
}
|
|
|
|
void GuiMultilineTextBox::CalculateViewAndSetScroll()
|
|
{
|
|
auto ct = GetControlTemplateObject(true);
|
|
CalculateView();
|
|
vint smallMove = textElement->GetLines().GetRowHeight();
|
|
vint bigMove = smallMove * 5;
|
|
|
|
if (auto scroll = ct->GetHorizontalScroll())
|
|
{
|
|
scroll->SetSmallMove(smallMove);
|
|
scroll->SetBigMove(bigMove);
|
|
}
|
|
|
|
if (auto scroll = ct->GetVerticalScroll())
|
|
{
|
|
scroll->SetSmallMove(smallMove);
|
|
scroll->SetBigMove(bigMove);
|
|
}
|
|
}
|
|
|
|
void GuiMultilineTextBox::OnBoundsMouseButtonDown(compositions::GuiGraphicsComposition* sender, compositions::GuiMouseEventArgs& arguments)
|
|
{
|
|
if(GetVisuallyEnabled())
|
|
{
|
|
SetFocus();
|
|
}
|
|
}
|
|
|
|
GuiMultilineTextBox::GuiMultilineTextBox(theme::ThemeName themeName)
|
|
:GuiScrollView(themeName)
|
|
{
|
|
textElement = GuiColorizedTextElement::Create();
|
|
textElement->SetFont(GetDisplayFont());
|
|
|
|
textComposition = new GuiBoundsComposition;
|
|
textComposition->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
textComposition->SetOwnedElement(textElement);
|
|
containerComposition->AddChild(textComposition);
|
|
|
|
callback = new TextElementOperatorCallback(this);
|
|
commandExecutor = new CommandExecutor(this);
|
|
|
|
SetAcceptTabInput(true);
|
|
SetFocusableComposition(boundsComposition);
|
|
Install(textElement, textComposition, this, boundsComposition, focusableComposition);
|
|
SetCallback(callback.Obj());
|
|
|
|
boundsComposition->GetEventReceiver()->leftButtonDown.AttachMethod(this, &GuiMultilineTextBox::OnBoundsMouseButtonDown);
|
|
boundsComposition->GetEventReceiver()->middleButtonDown.AttachMethod(this, &GuiMultilineTextBox::OnBoundsMouseButtonDown);
|
|
boundsComposition->GetEventReceiver()->rightButtonDown.AttachMethod(this, &GuiMultilineTextBox::OnBoundsMouseButtonDown);
|
|
}
|
|
|
|
GuiMultilineTextBox::~GuiMultilineTextBox()
|
|
{
|
|
}
|
|
|
|
const WString& GuiMultilineTextBox::GetText()
|
|
{
|
|
text = textElement->GetLines().GetText();
|
|
return text;
|
|
}
|
|
|
|
void GuiMultilineTextBox::SetText(const WString& value)
|
|
{
|
|
UnsafeSetText(value);
|
|
textElement->SetCaretBegin(TextPos(0, 0));
|
|
textElement->SetCaretEnd(TextPos(0, 0));
|
|
CalculateView();
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiSinglelineTextBox::DefaultTextElementOperatorCallback
|
|
***********************************************************************/
|
|
|
|
GuiSinglelineTextBox::TextElementOperatorCallback::TextElementOperatorCallback(GuiSinglelineTextBox* _textControl)
|
|
:GuiTextBoxCommonInterface::DefaultCallback(
|
|
_textControl->textElement,
|
|
_textControl->textComposition
|
|
)
|
|
{
|
|
}
|
|
|
|
bool GuiSinglelineTextBox::TextElementOperatorCallback::BeforeModify(TextPos start, TextPos end, const WString& originalText, WString& inputText)
|
|
{
|
|
vint length=inputText.Length();
|
|
const wchar_t* input=inputText.Buffer();
|
|
for(vint i=0;i<length;i++)
|
|
{
|
|
if(*input==0 || *input==L'\r' || *input==L'\n')
|
|
{
|
|
length=i;
|
|
break;
|
|
}
|
|
}
|
|
if(length!=inputText.Length())
|
|
{
|
|
inputText=inputText.Left(length);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
void GuiSinglelineTextBox::TextElementOperatorCallback::AfterModify(TextPos originalStart, TextPos originalEnd, const WString& originalText, TextPos inputStart, TextPos inputEnd, const WString& inputText)
|
|
{
|
|
}
|
|
|
|
void GuiSinglelineTextBox::TextElementOperatorCallback::ScrollToView(Point point)
|
|
{
|
|
vint newX=point.x;
|
|
vint oldX=textElement->GetViewPosition().x;
|
|
vint marginX=0;
|
|
if(oldX<newX)
|
|
{
|
|
marginX=TextMargin;
|
|
}
|
|
else if(oldX>newX)
|
|
{
|
|
marginX=-TextMargin;
|
|
}
|
|
|
|
newX+=marginX;
|
|
vint minX=-TextMargin;
|
|
vint maxX=textElement->GetLines().GetMaxWidth()+TextMargin-textComposition->GetBounds().Width();
|
|
if(newX>=maxX)
|
|
{
|
|
newX=maxX-1;
|
|
}
|
|
if(newX<minX)
|
|
{
|
|
newX=minX;
|
|
}
|
|
textElement->SetViewPosition(Point(newX, -TextMargin));
|
|
}
|
|
|
|
vint GuiSinglelineTextBox::TextElementOperatorCallback::GetTextMargin()
|
|
{
|
|
return TextMargin;
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiSinglelineTextBox
|
|
***********************************************************************/
|
|
|
|
void GuiSinglelineTextBox::BeforeControlTemplateUninstalled_()
|
|
{
|
|
}
|
|
|
|
void GuiSinglelineTextBox::AfterControlTemplateInstalled_(bool initialize)
|
|
{
|
|
auto ct = GetControlTemplateObject(true);
|
|
Array<ColorEntry> colors(1);
|
|
colors[0] = ct->GetTextColor();
|
|
textElement->SetColors(colors);
|
|
textElement->SetCaretColor(ct->GetCaretColor());
|
|
}
|
|
|
|
void GuiSinglelineTextBox::UpdateVisuallyEnabled()
|
|
{
|
|
GuiControl::UpdateVisuallyEnabled();
|
|
textElement->SetVisuallyEnabled(GetVisuallyEnabled());
|
|
}
|
|
|
|
void GuiSinglelineTextBox::UpdateDisplayFont()
|
|
{
|
|
GuiControl::UpdateDisplayFont();
|
|
textElement->SetFont(GetDisplayFont());
|
|
RearrangeTextElement();
|
|
}
|
|
|
|
void GuiSinglelineTextBox::RearrangeTextElement()
|
|
{
|
|
textCompositionTable->SetRowOption(
|
|
1,
|
|
GuiCellOption::AbsoluteOption(
|
|
textElement->GetLines().GetRowHeight() + 2 * TextMargin)
|
|
);
|
|
}
|
|
|
|
void GuiSinglelineTextBox::OnRenderTargetChanged(elements::IGuiGraphicsRenderTarget* renderTarget)
|
|
{
|
|
GuiControl::OnRenderTargetChanged(renderTarget);
|
|
RearrangeTextElement();
|
|
}
|
|
|
|
void GuiSinglelineTextBox::OnBoundsMouseButtonDown(compositions::GuiGraphicsComposition* sender, compositions::GuiMouseEventArgs& arguments)
|
|
{
|
|
if(GetVisuallyEnabled())
|
|
{
|
|
SetFocus();
|
|
}
|
|
}
|
|
|
|
GuiSinglelineTextBox::GuiSinglelineTextBox(theme::ThemeName themeName)
|
|
:GuiControl(themeName)
|
|
{
|
|
textElement = GuiColorizedTextElement::Create();
|
|
textElement->SetFont(GetDisplayFont());
|
|
textElement->SetViewPosition(Point(-GuiSinglelineTextBox::TextMargin, -GuiSinglelineTextBox::TextMargin));
|
|
|
|
textCompositionTable = new GuiTableComposition;
|
|
textCompositionTable->SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
textCompositionTable->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
textCompositionTable->SetRowsAndColumns(3, 1);
|
|
textCompositionTable->SetRowOption(0, GuiCellOption::PercentageOption(0.5));
|
|
textCompositionTable->SetRowOption(1, GuiCellOption::AbsoluteOption(0));
|
|
textCompositionTable->SetRowOption(2, GuiCellOption::PercentageOption(0.5));
|
|
textCompositionTable->SetColumnOption(0, GuiCellOption::PercentageOption(1.0));
|
|
containerComposition->AddChild(textCompositionTable);
|
|
|
|
textComposition = new GuiCellComposition;
|
|
textComposition->SetOwnedElement(textElement);
|
|
textCompositionTable->AddChild(textComposition);
|
|
textComposition->SetSite(1, 0, 1, 1);
|
|
|
|
callback = new TextElementOperatorCallback(this);
|
|
SetAcceptTabInput(true);
|
|
SetFocusableComposition(boundsComposition);
|
|
Install(textElement, textComposition, this, boundsComposition, focusableComposition);
|
|
SetCallback(callback.Obj());
|
|
|
|
boundsComposition->GetEventReceiver()->leftButtonDown.AttachMethod(this, &GuiSinglelineTextBox::OnBoundsMouseButtonDown);
|
|
boundsComposition->GetEventReceiver()->middleButtonDown.AttachMethod(this, &GuiSinglelineTextBox::OnBoundsMouseButtonDown);
|
|
boundsComposition->GetEventReceiver()->rightButtonDown.AttachMethod(this, &GuiSinglelineTextBox::OnBoundsMouseButtonDown);
|
|
}
|
|
|
|
GuiSinglelineTextBox::~GuiSinglelineTextBox()
|
|
{
|
|
}
|
|
|
|
const WString& GuiSinglelineTextBox::GetText()
|
|
{
|
|
text = textElement->GetLines().GetText();
|
|
return text;
|
|
}
|
|
|
|
void GuiSinglelineTextBox::SetText(const WString& value)
|
|
{
|
|
UnsafeSetText(value);
|
|
textElement->SetCaretBegin(TextPos(0, 0));
|
|
textElement->SetCaretEnd(TextPos(0, 0));
|
|
}
|
|
|
|
wchar_t GuiSinglelineTextBox::GetPasswordChar()
|
|
{
|
|
return textElement->GetPasswordChar();
|
|
}
|
|
|
|
void GuiSinglelineTextBox::SetPasswordChar(wchar_t value)
|
|
{
|
|
textElement->SetPasswordChar(value);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\CONTROLS\TEXTEDITORPACKAGE\EDITORCALLBACK\GUITEXTAUTOCOMPLETE.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace controls
|
|
{
|
|
using namespace collections;
|
|
|
|
/***********************************************************************
|
|
GuiTextBoxAutoCompleteBase::TextListControlProvider
|
|
***********************************************************************/
|
|
|
|
GuiTextBoxAutoCompleteBase::TextListControlProvider::TextListControlProvider(TemplateProperty<templates::GuiTextListTemplate> controlTemplate)
|
|
{
|
|
autoCompleteList = new GuiTextList(theme::ThemeName::TextList);
|
|
if (controlTemplate)
|
|
{
|
|
autoCompleteList->SetControlTemplate(controlTemplate);
|
|
}
|
|
autoCompleteList->SetHorizontalAlwaysVisible(false);
|
|
autoCompleteList->SetVerticalAlwaysVisible(false);
|
|
}
|
|
|
|
GuiTextBoxAutoCompleteBase::TextListControlProvider::~TextListControlProvider()
|
|
{
|
|
}
|
|
|
|
GuiControl* GuiTextBoxAutoCompleteBase::TextListControlProvider::GetAutoCompleteControl()
|
|
{
|
|
return autoCompleteList;
|
|
}
|
|
|
|
GuiSelectableListControl* GuiTextBoxAutoCompleteBase::TextListControlProvider::GetListControl()
|
|
{
|
|
return autoCompleteList;
|
|
}
|
|
|
|
void GuiTextBoxAutoCompleteBase::TextListControlProvider::SetSortedContent(const collections::List<AutoCompleteItem>& items)
|
|
{
|
|
autoCompleteList->GetItems().Clear();
|
|
FOREACH(AutoCompleteItem, item, items)
|
|
{
|
|
autoCompleteList->GetItems().Add(new list::TextItem(item.text));
|
|
}
|
|
}
|
|
|
|
vint GuiTextBoxAutoCompleteBase::TextListControlProvider::GetItemCount()
|
|
{
|
|
return autoCompleteList->GetItems().Count();
|
|
}
|
|
|
|
WString GuiTextBoxAutoCompleteBase::TextListControlProvider::GetItemText(vint index)
|
|
{
|
|
return autoCompleteList->GetItems()[index]->GetText();
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiTextBoxAutoCompleteBase
|
|
***********************************************************************/
|
|
|
|
bool GuiTextBoxAutoCompleteBase::IsPrefix(const WString& prefix, const WString& candidate)
|
|
{
|
|
if(candidate.Length()>=prefix.Length())
|
|
{
|
|
if(INVLOC.Compare(prefix, candidate.Sub(0, prefix.Length()), Locale::IgnoreCase)==0)
|
|
{
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
GuiTextBoxAutoCompleteBase::GuiTextBoxAutoCompleteBase(Ptr<IAutoCompleteControlProvider> _autoCompleteControlProvider)
|
|
:element(0)
|
|
, elementModifyLock(0)
|
|
, ownerComposition(0)
|
|
, autoCompleteControlProvider(_autoCompleteControlProvider)
|
|
{
|
|
if (!autoCompleteControlProvider)
|
|
{
|
|
autoCompleteControlProvider = new TextListControlProvider;
|
|
}
|
|
autoCompleteControlProvider->GetAutoCompleteControl()->GetBoundsComposition()->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
|
|
autoCompletePopup = new GuiPopup(theme::ThemeName::Menu);
|
|
autoCompletePopup->AddChild(autoCompleteControlProvider->GetAutoCompleteControl());
|
|
}
|
|
|
|
GuiTextBoxAutoCompleteBase::~GuiTextBoxAutoCompleteBase()
|
|
{
|
|
delete autoCompletePopup;
|
|
}
|
|
|
|
void GuiTextBoxAutoCompleteBase::Attach(elements::GuiColorizedTextElement* _element, SpinLock& _elementModifyLock, compositions::GuiGraphicsComposition* _ownerComposition, vuint editVersion)
|
|
{
|
|
if(_element)
|
|
{
|
|
SPIN_LOCK(_elementModifyLock)
|
|
{
|
|
element=_element;
|
|
elementModifyLock=&_elementModifyLock;
|
|
ownerComposition=_ownerComposition;
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiTextBoxAutoCompleteBase::Detach()
|
|
{
|
|
if(element && elementModifyLock)
|
|
{
|
|
SPIN_LOCK(*elementModifyLock)
|
|
{
|
|
element=0;
|
|
elementModifyLock=0;
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiTextBoxAutoCompleteBase::TextEditPreview(TextEditPreviewStruct& arguments)
|
|
{
|
|
}
|
|
|
|
void GuiTextBoxAutoCompleteBase::TextEditNotify(const TextEditNotifyStruct& arguments)
|
|
{
|
|
if(element && elementModifyLock)
|
|
{
|
|
if(IsListOpening())
|
|
{
|
|
TextPos begin=GetListStartPosition();
|
|
TextPos end=arguments.inputEnd;
|
|
WString editingText=element->GetLines().GetText(begin, end);
|
|
HighlightList(editingText);
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiTextBoxAutoCompleteBase::TextCaretChanged(const TextCaretChangedStruct& arguments)
|
|
{
|
|
}
|
|
|
|
void GuiTextBoxAutoCompleteBase::TextEditFinished(vuint editVersion)
|
|
{
|
|
}
|
|
|
|
bool GuiTextBoxAutoCompleteBase::IsListOpening()
|
|
{
|
|
return autoCompletePopup->GetOpening();
|
|
}
|
|
|
|
void GuiTextBoxAutoCompleteBase::OpenList(TextPos startPosition)
|
|
{
|
|
if(element && elementModifyLock)
|
|
{
|
|
autoCompleteStartPosition=startPosition;
|
|
Rect bounds=element->GetLines().GetRectFromTextPos(startPosition);
|
|
Point viewPosition=element->GetViewPosition();
|
|
GuiControl* ownerControl=ownerComposition->GetRelatedControl();
|
|
Rect compositionBounds=ownerComposition->GetGlobalBounds();
|
|
Rect controlBounds=ownerControl->GetBoundsComposition()->GetGlobalBounds();
|
|
vint px=compositionBounds.x1-controlBounds.x1-viewPosition.x;
|
|
vint py=compositionBounds.y1-controlBounds.y1-viewPosition.y;
|
|
bounds.x1+=px;
|
|
bounds.x2+=px;
|
|
bounds.y1+=py+5;
|
|
bounds.y2+=py+5;
|
|
autoCompletePopup->ShowPopup(ownerControl, bounds, true);
|
|
}
|
|
}
|
|
|
|
void GuiTextBoxAutoCompleteBase::CloseList()
|
|
{
|
|
autoCompletePopup->Close();
|
|
}
|
|
|
|
void GuiTextBoxAutoCompleteBase::SetListContent(const collections::List<AutoCompleteItem>& items)
|
|
{
|
|
if(items.Count()==0)
|
|
{
|
|
CloseList();
|
|
}
|
|
|
|
List<AutoCompleteItem> sortedItems;
|
|
CopyFrom(
|
|
sortedItems,
|
|
From(items)
|
|
.OrderBy([](const AutoCompleteItem& a, const AutoCompleteItem& b)
|
|
{
|
|
return INVLOC.Compare(a.text, b.text, Locale::IgnoreCase);
|
|
})
|
|
);
|
|
|
|
autoCompleteControlProvider->SetSortedContent(sortedItems);
|
|
autoCompleteControlProvider->GetAutoCompleteControl()->GetBoundsComposition()->SetPreferredMinSize(Size(200, 200));
|
|
}
|
|
|
|
TextPos GuiTextBoxAutoCompleteBase::GetListStartPosition()
|
|
{
|
|
return autoCompleteStartPosition;
|
|
}
|
|
|
|
bool GuiTextBoxAutoCompleteBase::SelectPreviousListItem()
|
|
{
|
|
if(!IsListOpening()) return false;
|
|
if(autoCompleteControlProvider->GetListControl()->GetSelectedItems().Count()==0)
|
|
{
|
|
autoCompleteControlProvider->GetListControl()->SetSelected(0, true);
|
|
}
|
|
else
|
|
{
|
|
vint index=autoCompleteControlProvider->GetListControl()->GetSelectedItems()[0];
|
|
if (index > 0) index--;
|
|
autoCompleteControlProvider->GetListControl()->SetSelected(index, true);
|
|
autoCompleteControlProvider->GetListControl()->EnsureItemVisible(index);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool GuiTextBoxAutoCompleteBase::SelectNextListItem()
|
|
{
|
|
if(!IsListOpening()) return false;
|
|
if (autoCompleteControlProvider->GetListControl()->GetSelectedItems().Count() == 0)
|
|
{
|
|
autoCompleteControlProvider->GetListControl()->SetSelected(0, true);
|
|
}
|
|
else
|
|
{
|
|
vint index = autoCompleteControlProvider->GetListControl()->GetSelectedItems()[0];
|
|
if (index < autoCompleteControlProvider->GetItemCount() - 1) index++;
|
|
autoCompleteControlProvider->GetListControl()->SetSelected(index, true);
|
|
autoCompleteControlProvider->GetListControl()->EnsureItemVisible(index);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool GuiTextBoxAutoCompleteBase::ApplySelectedListItem()
|
|
{
|
|
if(!IsListOpening()) return false;
|
|
if(!ownerComposition) return false;
|
|
const auto& selectedItems = autoCompleteControlProvider->GetListControl()->GetSelectedItems();
|
|
if (selectedItems.Count() == 0) return false;
|
|
GuiTextBoxCommonInterface* ci=dynamic_cast<GuiTextBoxCommonInterface*>(ownerComposition->GetRelatedControl());
|
|
if(!ci) return false;
|
|
|
|
vint index = selectedItems[0];
|
|
WString selectedItem = autoCompleteControlProvider->GetItemText(index);
|
|
TextPos begin = autoCompleteStartPosition;
|
|
TextPos end = ci->GetCaretEnd();
|
|
ci->Select(begin, end);
|
|
ci->SetSelectionText(selectedItem);
|
|
CloseList();
|
|
return true;
|
|
}
|
|
|
|
WString GuiTextBoxAutoCompleteBase::GetSelectedListItem()
|
|
{
|
|
if(!IsListOpening()) return L"";
|
|
const auto& selectedItems = autoCompleteControlProvider->GetListControl()->GetSelectedItems();
|
|
if (selectedItems.Count() == 0) return L"";
|
|
vint index = selectedItems[0];
|
|
return autoCompleteControlProvider->GetItemText(index);
|
|
}
|
|
|
|
void GuiTextBoxAutoCompleteBase::HighlightList(const WString& editingText)
|
|
{
|
|
if(IsListOpening())
|
|
{
|
|
vint first=0;
|
|
vint last = autoCompleteControlProvider->GetItemCount() - 1;
|
|
vint selected=-1;
|
|
|
|
while (first <= last)
|
|
{
|
|
vint middle = (first + last) / 2;
|
|
WString text = autoCompleteControlProvider->GetItemText(middle);
|
|
if (IsPrefix(editingText, text))
|
|
{
|
|
selected = middle;
|
|
break;
|
|
}
|
|
|
|
vint result = INVLOC.Compare(editingText, text, Locale::IgnoreCase);
|
|
if (result <= 0)
|
|
{
|
|
last = middle - 1;
|
|
}
|
|
else
|
|
{
|
|
first = middle + 1;
|
|
}
|
|
}
|
|
|
|
while(selected>0)
|
|
{
|
|
WString text = autoCompleteControlProvider->GetItemText(selected - 1);
|
|
if (IsPrefix(editingText, text))
|
|
{
|
|
selected--;
|
|
}
|
|
else
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
if(selected!=-1)
|
|
{
|
|
autoCompleteControlProvider->GetListControl()->SetSelected(selected, true);
|
|
autoCompleteControlProvider->GetListControl()->EnsureItemVisible(selected);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\CONTROLS\TEXTEDITORPACKAGE\EDITORCALLBACK\GUITEXTCOLORIZER.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace controls
|
|
{
|
|
using namespace elements;
|
|
using namespace elements::text;
|
|
|
|
/***********************************************************************
|
|
GuiTextBoxColorizerBase
|
|
***********************************************************************/
|
|
|
|
void GuiTextBoxColorizerBase::ColorizerThreadProc(void* argument)
|
|
{
|
|
GuiTextBoxColorizerBase* colorizer=(GuiTextBoxColorizerBase*)argument;
|
|
while(!colorizer->isFinalizing)
|
|
{
|
|
vint lineIndex=-1;
|
|
wchar_t* text=0;
|
|
vuint32_t* colors=0;
|
|
vint length=0;
|
|
vint lexerState=-1;
|
|
vint contextState=-1;
|
|
|
|
SPIN_LOCK(*colorizer->elementModifyLock)
|
|
{
|
|
if(colorizer->colorizedLineCount>=colorizer->element->GetLines().GetCount())
|
|
{
|
|
colorizer->isColorizerRunning=false;
|
|
goto CANCEL_COLORIZING;
|
|
}
|
|
|
|
lineIndex=colorizer->colorizedLineCount++;
|
|
TextLine& line=colorizer->element->GetLines().GetLine(lineIndex);
|
|
length=line.dataLength;
|
|
text=new wchar_t[length+2];
|
|
colors=new vuint32_t[length+2];
|
|
memcpy(text, line.text, sizeof(wchar_t)*length);
|
|
text[length]=L'\r';
|
|
text[length+1]=L'\n';
|
|
lexerState=lineIndex==0?colorizer->GetLexerStartState():colorizer->element->GetLines().GetLine(lineIndex-1).lexerFinalState;
|
|
contextState=lineIndex==0?colorizer->GetContextStartState():colorizer->element->GetLines().GetLine(lineIndex-1).contextFinalState;
|
|
}
|
|
|
|
colorizer->ColorizeLineWithCRLF(lineIndex, text, colors, length+2, lexerState, contextState);
|
|
|
|
SPIN_LOCK(*colorizer->elementModifyLock)
|
|
{
|
|
if(lineIndex<colorizer->colorizedLineCount && lineIndex<colorizer->element->GetLines().GetCount())
|
|
{
|
|
TextLine& line=colorizer->element->GetLines().GetLine(lineIndex);
|
|
line.lexerFinalState=lexerState;
|
|
line.contextFinalState=contextState;
|
|
for(vint i=0;i<length;i++)
|
|
{
|
|
line.att[i].colorIndex=colors[i];
|
|
}
|
|
}
|
|
delete[] text;
|
|
delete[] colors;
|
|
}
|
|
}
|
|
CANCEL_COLORIZING:
|
|
colorizer->colorizerRunningEvent.Leave();
|
|
}
|
|
|
|
void GuiTextBoxColorizerBase::StartColorizer()
|
|
{
|
|
if(!isColorizerRunning)
|
|
{
|
|
isColorizerRunning=true;
|
|
colorizerRunningEvent.Enter();
|
|
ThreadPoolLite::Queue(&GuiTextBoxColorizerBase::ColorizerThreadProc, this);
|
|
}
|
|
}
|
|
|
|
void GuiTextBoxColorizerBase::StopColorizer(bool forever)
|
|
{
|
|
isFinalizing=true;
|
|
colorizerRunningEvent.Enter();
|
|
colorizerRunningEvent.Leave();
|
|
colorizedLineCount=0;
|
|
if(!forever)
|
|
{
|
|
isFinalizing=false;
|
|
}
|
|
}
|
|
|
|
void GuiTextBoxColorizerBase::StopColorizerForever()
|
|
{
|
|
StopColorizer(true);
|
|
}
|
|
|
|
GuiTextBoxColorizerBase::GuiTextBoxColorizerBase()
|
|
:element(0)
|
|
,elementModifyLock(0)
|
|
,colorizedLineCount(0)
|
|
,isColorizerRunning(false)
|
|
,isFinalizing(false)
|
|
{
|
|
}
|
|
|
|
GuiTextBoxColorizerBase::~GuiTextBoxColorizerBase()
|
|
{
|
|
StopColorizerForever();
|
|
}
|
|
|
|
void GuiTextBoxColorizerBase::Attach(elements::GuiColorizedTextElement* _element, SpinLock& _elementModifyLock, compositions::GuiGraphicsComposition* _ownerComposition, vuint editVersion)
|
|
{
|
|
if(_element)
|
|
{
|
|
SPIN_LOCK(_elementModifyLock)
|
|
{
|
|
element=_element;
|
|
elementModifyLock=&_elementModifyLock;
|
|
StartColorizer();
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiTextBoxColorizerBase::Detach()
|
|
{
|
|
if(element && elementModifyLock)
|
|
{
|
|
StopColorizer(false);
|
|
SPIN_LOCK(*elementModifyLock)
|
|
{
|
|
element=0;
|
|
elementModifyLock=0;
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiTextBoxColorizerBase::TextEditPreview(TextEditPreviewStruct& arguments)
|
|
{
|
|
}
|
|
|
|
void GuiTextBoxColorizerBase::TextEditNotify(const TextEditNotifyStruct& arguments)
|
|
{
|
|
if(element && elementModifyLock)
|
|
{
|
|
SPIN_LOCK(*elementModifyLock)
|
|
{
|
|
vint line
|
|
=arguments.originalStart.row<arguments.originalEnd.row
|
|
?arguments.originalStart.row
|
|
:arguments.originalEnd.row;
|
|
if(colorizedLineCount>line)
|
|
{
|
|
colorizedLineCount=line;
|
|
}
|
|
StartColorizer();
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiTextBoxColorizerBase::TextCaretChanged(const TextCaretChangedStruct& arguments)
|
|
{
|
|
}
|
|
|
|
void GuiTextBoxColorizerBase::TextEditFinished(vuint editVersion)
|
|
{
|
|
}
|
|
|
|
void GuiTextBoxColorizerBase::RestartColorizer()
|
|
{
|
|
if(element && elementModifyLock)
|
|
{
|
|
SPIN_LOCK(*elementModifyLock)
|
|
{
|
|
colorizedLineCount=0;
|
|
StartColorizer();
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiTextBoxRegexColorizer
|
|
***********************************************************************/
|
|
|
|
struct GuiTextBoxRegexColorizerProcData
|
|
{
|
|
GuiTextBoxRegexColorizer* colorizer;
|
|
vint lineIndex;
|
|
const wchar_t* text;
|
|
vuint32_t* colors;
|
|
vint contextState;
|
|
};
|
|
|
|
void GuiTextBoxRegexColorizer::ColorizerProc(void* argument, vint start, vint length, vint token)
|
|
{
|
|
GuiTextBoxRegexColorizerProcData& data=**(GuiTextBoxRegexColorizerProcData**)argument;
|
|
data.colorizer->ColorizeTokenContextSensitive(data.lineIndex, data.text, start, length, token, data.contextState);
|
|
for(vint i=0;i<length;i++)
|
|
{
|
|
data.colors[start+i]=(int)token+1;
|
|
}
|
|
}
|
|
|
|
GuiTextBoxRegexColorizer::GuiTextBoxRegexColorizer()
|
|
{
|
|
colors.Resize(1);
|
|
}
|
|
|
|
GuiTextBoxRegexColorizer::~GuiTextBoxRegexColorizer()
|
|
{
|
|
StopColorizerForever();
|
|
}
|
|
|
|
elements::text::ColorEntry GuiTextBoxRegexColorizer::GetDefaultColor()
|
|
{
|
|
return defaultColor;
|
|
}
|
|
|
|
collections::List<WString>& GuiTextBoxRegexColorizer::GetTokenRegexes()
|
|
{
|
|
return tokenRegexes;
|
|
}
|
|
|
|
collections::List<elements::text::ColorEntry>& GuiTextBoxRegexColorizer::GetTokenColors()
|
|
{
|
|
return tokenColors;
|
|
}
|
|
|
|
collections::List<elements::text::ColorEntry>& GuiTextBoxRegexColorizer::GetExtraTokenColors()
|
|
{
|
|
return extraTokenColors;
|
|
}
|
|
|
|
vint GuiTextBoxRegexColorizer::GetExtraTokenIndexStart()
|
|
{
|
|
if(lexer)
|
|
{
|
|
return tokenColors.Count();
|
|
}
|
|
else
|
|
{
|
|
return -1;
|
|
}
|
|
}
|
|
|
|
bool GuiTextBoxRegexColorizer::SetDefaultColor(elements::text::ColorEntry value)
|
|
{
|
|
if(lexer)
|
|
{
|
|
return false;
|
|
}
|
|
else
|
|
{
|
|
defaultColor=value;
|
|
return true;
|
|
}
|
|
}
|
|
|
|
vint GuiTextBoxRegexColorizer::AddToken(const WString& regex, elements::text::ColorEntry color)
|
|
{
|
|
if(lexer)
|
|
{
|
|
return -1;
|
|
}
|
|
else
|
|
{
|
|
tokenRegexes.Add(regex);
|
|
tokenColors.Add(color);
|
|
return tokenColors.Count()-1;
|
|
}
|
|
}
|
|
|
|
vint GuiTextBoxRegexColorizer::AddExtraToken(elements::text::ColorEntry color)
|
|
{
|
|
if(lexer)
|
|
{
|
|
return -1;
|
|
}
|
|
else
|
|
{
|
|
extraTokenColors.Add(color);
|
|
return extraTokenColors.Count()-1;
|
|
}
|
|
}
|
|
|
|
void GuiTextBoxRegexColorizer::ClearTokens()
|
|
{
|
|
tokenRegexes.Clear();
|
|
tokenColors.Clear();
|
|
extraTokenColors.Clear();
|
|
lexer=0;
|
|
}
|
|
|
|
void GuiTextBoxRegexColorizer::Setup()
|
|
{
|
|
if (lexer || tokenRegexes.Count() == 0)
|
|
{
|
|
colors.Resize(1);
|
|
colors[0] = defaultColor;
|
|
}
|
|
else
|
|
{
|
|
{
|
|
regex::RegexProc proc;
|
|
proc.colorizeProc = &GuiTextBoxRegexColorizer::ColorizerProc;
|
|
proc.argument = colorizerArgument;
|
|
lexer = new regex::RegexLexer(tokenRegexes, proc);
|
|
}
|
|
colors.Resize(1 + tokenRegexes.Count() + extraTokenColors.Count());
|
|
colors[0] = defaultColor;
|
|
for (vint i = 0; i < tokenColors.Count(); i++)
|
|
{
|
|
colors[i + 1] = tokenColors[i];
|
|
}
|
|
for (vint i = 0; i < extraTokenColors.Count(); i++)
|
|
{
|
|
colors[i + 1 + tokenColors.Count()] = extraTokenColors[i];
|
|
}
|
|
colorizer = new regex::RegexLexerColorizer(lexer->Colorize());
|
|
}
|
|
}
|
|
|
|
void GuiTextBoxRegexColorizer::ColorizeTokenContextSensitive(vint lineIndex, const wchar_t* text, vint start, vint length, vint& token, vint& contextState)
|
|
{
|
|
}
|
|
|
|
vint GuiTextBoxRegexColorizer::GetLexerStartState()
|
|
{
|
|
return lexer?colorizer->GetStartState():-1;
|
|
}
|
|
|
|
vint GuiTextBoxRegexColorizer::GetContextStartState()
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
void GuiTextBoxRegexColorizer::ColorizeLineWithCRLF(vint lineIndex, const wchar_t* text, vuint32_t* colors, vint length, vint& lexerState, vint& contextState)
|
|
{
|
|
memset(colors, 0, sizeof(*colors)*length);
|
|
if (lexer)
|
|
{
|
|
GuiTextBoxRegexColorizerProcData data;
|
|
data.colorizer = this;
|
|
data.lineIndex = lineIndex;
|
|
data.text = text;
|
|
data.colors = colors;
|
|
data.contextState = contextState;
|
|
|
|
regex::RegexLexerColorizer::InternalState internalState;
|
|
internalState.currentState = lexerState;
|
|
colorizer->SetInternalState(internalState);
|
|
colorizerArgument[0] = &data;
|
|
colorizer->Colorize(text, length);
|
|
|
|
lexerState = colorizer->GetInternalState().currentState;
|
|
contextState = data.contextState;
|
|
}
|
|
else
|
|
{
|
|
lexerState = -1;
|
|
contextState = -1;
|
|
}
|
|
}
|
|
|
|
const GuiTextBoxRegexColorizer::ColorArray& GuiTextBoxRegexColorizer::GetColors()
|
|
{
|
|
return colors;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\CONTROLS\TEXTEDITORPACKAGE\EDITORCALLBACK\GUITEXTUNDOREDO.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace controls
|
|
{
|
|
using namespace elements;
|
|
using namespace elements::text;
|
|
using namespace compositions;
|
|
|
|
/***********************************************************************
|
|
GuiGeneralUndoRedoProcessor
|
|
***********************************************************************/
|
|
|
|
GuiGeneralUndoRedoProcessor::GuiGeneralUndoRedoProcessor()
|
|
:firstFutureStep(0)
|
|
,savedStep(0)
|
|
,performingUndoRedo(false)
|
|
{
|
|
}
|
|
|
|
GuiGeneralUndoRedoProcessor::~GuiGeneralUndoRedoProcessor()
|
|
{
|
|
}
|
|
|
|
void GuiGeneralUndoRedoProcessor::PushStep(Ptr<IEditStep> step)
|
|
{
|
|
if(!performingUndoRedo)
|
|
{
|
|
if(firstFutureStep<savedStep)
|
|
{
|
|
savedStep=-1;
|
|
}
|
|
|
|
vint count=steps.Count()-firstFutureStep;
|
|
if(count>0)
|
|
{
|
|
steps.RemoveRange(firstFutureStep, count);
|
|
}
|
|
|
|
steps.Add(step);
|
|
firstFutureStep=steps.Count();
|
|
UndoRedoChanged();
|
|
ModifiedChanged();
|
|
}
|
|
}
|
|
|
|
bool GuiGeneralUndoRedoProcessor::CanUndo()
|
|
{
|
|
return firstFutureStep>0;
|
|
}
|
|
|
|
bool GuiGeneralUndoRedoProcessor::CanRedo()
|
|
{
|
|
return steps.Count()>firstFutureStep;
|
|
}
|
|
|
|
void GuiGeneralUndoRedoProcessor::ClearUndoRedo()
|
|
{
|
|
if(!performingUndoRedo)
|
|
{
|
|
steps.Clear();
|
|
firstFutureStep=0;
|
|
savedStep=0;
|
|
}
|
|
}
|
|
|
|
bool GuiGeneralUndoRedoProcessor::GetModified()
|
|
{
|
|
return firstFutureStep!=savedStep;
|
|
}
|
|
|
|
void GuiGeneralUndoRedoProcessor::NotifyModificationSaved()
|
|
{
|
|
if(!performingUndoRedo)
|
|
{
|
|
savedStep=firstFutureStep;
|
|
ModifiedChanged();
|
|
}
|
|
}
|
|
|
|
bool GuiGeneralUndoRedoProcessor::Undo()
|
|
{
|
|
if(!CanUndo()) return false;
|
|
performingUndoRedo=true;
|
|
firstFutureStep--;
|
|
steps[firstFutureStep]->Undo();
|
|
performingUndoRedo=false;
|
|
UndoRedoChanged();
|
|
ModifiedChanged();
|
|
return true;
|
|
}
|
|
|
|
bool GuiGeneralUndoRedoProcessor::Redo()
|
|
{
|
|
if(!CanRedo()) return false;
|
|
performingUndoRedo=true;
|
|
firstFutureStep++;
|
|
steps[firstFutureStep-1]->Redo();
|
|
performingUndoRedo=false;
|
|
UndoRedoChanged();
|
|
ModifiedChanged();
|
|
return true;
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiTextBoxUndoRedoProcessor::EditStep
|
|
***********************************************************************/
|
|
|
|
void GuiTextBoxUndoRedoProcessor::EditStep::Undo()
|
|
{
|
|
GuiTextBoxCommonInterface* ci=dynamic_cast<GuiTextBoxCommonInterface*>(processor->ownerComposition->GetRelatedControl());
|
|
if(ci)
|
|
{
|
|
ci->Select(arguments.inputStart, arguments.inputEnd);
|
|
ci->SetSelectionText(arguments.originalText);
|
|
ci->Select(arguments.originalStart, arguments.originalEnd);
|
|
}
|
|
}
|
|
|
|
void GuiTextBoxUndoRedoProcessor::EditStep::Redo()
|
|
{
|
|
GuiTextBoxCommonInterface* ci=dynamic_cast<GuiTextBoxCommonInterface*>(processor->ownerComposition->GetRelatedControl());
|
|
if(ci)
|
|
{
|
|
ci->Select(arguments.originalStart, arguments.originalEnd);
|
|
ci->SetSelectionText(arguments.inputText);
|
|
ci->Select(arguments.inputStart, arguments.inputEnd);
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiTextBoxUndoRedoProcessor
|
|
***********************************************************************/
|
|
|
|
GuiTextBoxUndoRedoProcessor::GuiTextBoxUndoRedoProcessor()
|
|
:ownerComposition(0)
|
|
{
|
|
}
|
|
|
|
GuiTextBoxUndoRedoProcessor::~GuiTextBoxUndoRedoProcessor()
|
|
{
|
|
}
|
|
|
|
void GuiTextBoxUndoRedoProcessor::Attach(elements::GuiColorizedTextElement* element, SpinLock& elementModifyLock, compositions::GuiGraphicsComposition* _ownerComposition, vuint editVersion)
|
|
{
|
|
ownerComposition=_ownerComposition;
|
|
}
|
|
|
|
void GuiTextBoxUndoRedoProcessor::Detach()
|
|
{
|
|
ClearUndoRedo();
|
|
}
|
|
|
|
void GuiTextBoxUndoRedoProcessor::TextEditPreview(TextEditPreviewStruct& arguments)
|
|
{
|
|
}
|
|
|
|
void GuiTextBoxUndoRedoProcessor::TextEditNotify(const TextEditNotifyStruct& arguments)
|
|
{
|
|
Ptr<EditStep> step=new EditStep;
|
|
step->processor=this;
|
|
step->arguments=arguments;
|
|
PushStep(step);
|
|
}
|
|
|
|
void GuiTextBoxUndoRedoProcessor::TextCaretChanged(const TextCaretChangedStruct& arguments)
|
|
{
|
|
}
|
|
|
|
void GuiTextBoxUndoRedoProcessor::TextEditFinished(vuint editVersion)
|
|
{
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiDocumentUndoRedoProcessor::ReplaceModelStep
|
|
***********************************************************************/
|
|
|
|
void GuiDocumentUndoRedoProcessor::ReplaceModelStep::Undo()
|
|
{
|
|
GuiDocumentCommonInterface* ci=dynamic_cast<GuiDocumentCommonInterface*>(processor->ownerComposition->GetRelatedControl());
|
|
if(ci)
|
|
{
|
|
ci->EditRun(arguments.inputStart, arguments.inputEnd, arguments.originalModel, true);
|
|
ci->SetCaret(arguments.originalStart, arguments.originalEnd);
|
|
}
|
|
}
|
|
|
|
void GuiDocumentUndoRedoProcessor::ReplaceModelStep::Redo()
|
|
{
|
|
GuiDocumentCommonInterface* ci=dynamic_cast<GuiDocumentCommonInterface*>(processor->ownerComposition->GetRelatedControl());
|
|
if(ci)
|
|
{
|
|
ci->EditRun(arguments.originalStart, arguments.originalEnd, arguments.inputModel, true);
|
|
ci->SetCaret(arguments.inputStart, arguments.inputEnd);
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiDocumentUndoRedoProcessor::RenameStyleStep
|
|
***********************************************************************/
|
|
|
|
void GuiDocumentUndoRedoProcessor::RenameStyleStep::Undo()
|
|
{
|
|
GuiDocumentCommonInterface* ci=dynamic_cast<GuiDocumentCommonInterface*>(processor->ownerComposition->GetRelatedControl());
|
|
if(ci)
|
|
{
|
|
ci->RenameStyle(arguments.newStyleName, arguments.oldStyleName);
|
|
}
|
|
}
|
|
|
|
void GuiDocumentUndoRedoProcessor::RenameStyleStep::Redo()
|
|
{
|
|
GuiDocumentCommonInterface* ci=dynamic_cast<GuiDocumentCommonInterface*>(processor->ownerComposition->GetRelatedControl());
|
|
if(ci)
|
|
{
|
|
ci->RenameStyle(arguments.oldStyleName, arguments.newStyleName);
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiDocumentUndoRedoProcessor::SetAlignmentStep
|
|
***********************************************************************/
|
|
|
|
void GuiDocumentUndoRedoProcessor::SetAlignmentStep::Undo()
|
|
{
|
|
GuiDocumentCommonInterface* ci=dynamic_cast<GuiDocumentCommonInterface*>(processor->ownerComposition->GetRelatedControl());
|
|
if(ci)
|
|
{
|
|
ci->SetParagraphAlignments(TextPos(arguments->start, 0), TextPos(arguments->end, 0), arguments->originalAlignments);
|
|
}
|
|
}
|
|
|
|
void GuiDocumentUndoRedoProcessor::SetAlignmentStep::Redo()
|
|
{
|
|
GuiDocumentCommonInterface* ci=dynamic_cast<GuiDocumentCommonInterface*>(processor->ownerComposition->GetRelatedControl());
|
|
if(ci)
|
|
{
|
|
ci->SetParagraphAlignments(TextPos(arguments->start, 0), TextPos(arguments->end, 0), arguments->inputAlignments);
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiDocumentUndoRedoProcessor
|
|
***********************************************************************/
|
|
|
|
GuiDocumentUndoRedoProcessor::GuiDocumentUndoRedoProcessor()
|
|
:element(0)
|
|
,ownerComposition(0)
|
|
{
|
|
}
|
|
|
|
GuiDocumentUndoRedoProcessor::~GuiDocumentUndoRedoProcessor()
|
|
{
|
|
}
|
|
|
|
void GuiDocumentUndoRedoProcessor::Setup(elements::GuiDocumentElement* _element, compositions::GuiGraphicsComposition* _ownerComposition)
|
|
{
|
|
element=_element;
|
|
ownerComposition=_ownerComposition;
|
|
}
|
|
|
|
void GuiDocumentUndoRedoProcessor::OnReplaceModel(const ReplaceModelStruct& arguments)
|
|
{
|
|
Ptr<ReplaceModelStep> step=new ReplaceModelStep;
|
|
step->processor=this;
|
|
step->arguments=arguments;
|
|
PushStep(step);
|
|
}
|
|
|
|
void GuiDocumentUndoRedoProcessor::OnRenameStyle(const RenameStyleStruct& arguments)
|
|
{
|
|
Ptr<RenameStyleStep> step=new RenameStyleStep;
|
|
step->processor=this;
|
|
step->arguments=arguments;
|
|
PushStep(step);
|
|
}
|
|
|
|
void GuiDocumentUndoRedoProcessor::OnSetAlignment(Ptr<SetAlignmentStruct> arguments)
|
|
{
|
|
Ptr<SetAlignmentStep> step=new SetAlignmentStep;
|
|
step->processor=this;
|
|
step->arguments=arguments;
|
|
PushStep(step);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\CONTROLS\TEXTEDITORPACKAGE\LANGUAGESERVICE\GUILANGUAGEAUTOCOMPLETE.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace controls
|
|
{
|
|
using namespace elements;
|
|
using namespace regex;
|
|
using namespace parsing;
|
|
using namespace parsing::tabling;
|
|
using namespace collections;
|
|
|
|
/***********************************************************************
|
|
GuiGrammarAutoComplete
|
|
***********************************************************************/
|
|
|
|
void GuiGrammarAutoComplete::Attach(elements::GuiColorizedTextElement* _element, SpinLock& _elementModifyLock, compositions::GuiGraphicsComposition* _ownerComposition, vuint editVersion)
|
|
{
|
|
GuiTextBoxAutoCompleteBase::Attach(_element, _elementModifyLock, _ownerComposition, editVersion);
|
|
RepeatingParsingExecutor::CallbackBase::Attach(_element, _elementModifyLock, _ownerComposition, editVersion);
|
|
}
|
|
|
|
void GuiGrammarAutoComplete::Detach()
|
|
{
|
|
GuiTextBoxAutoCompleteBase::Detach();
|
|
RepeatingParsingExecutor::CallbackBase::Detach();
|
|
if(element && elementModifyLock)
|
|
{
|
|
EnsureAutoCompleteFinished();
|
|
}
|
|
}
|
|
|
|
void GuiGrammarAutoComplete::TextEditPreview(TextEditPreviewStruct& arguments)
|
|
{
|
|
GuiTextBoxAutoCompleteBase::TextEditPreview(arguments);
|
|
RepeatingParsingExecutor::CallbackBase::TextEditPreview(arguments);
|
|
|
|
if(element && elementModifyLock)
|
|
{
|
|
if(IsListOpening() && arguments.keyInput && arguments.originalText==L"" && arguments.inputText!=L"")
|
|
{
|
|
WString selectedItem=GetSelectedListItem();
|
|
if(selectedItem!=L"")
|
|
{
|
|
TextPos begin=GetListStartPosition();
|
|
TextPos end=arguments.originalStart;
|
|
WString editingText=element->GetLines().GetText(begin, end);
|
|
editingText+=arguments.inputText;
|
|
if(grammarParser->GetTable()->GetLexer().Walk().IsClosedToken(editingText))
|
|
{
|
|
arguments.originalStart=begin;
|
|
arguments.inputText=selectedItem+arguments.inputText;
|
|
CloseList();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiGrammarAutoComplete::TextEditNotify(const TextEditNotifyStruct& arguments)
|
|
{
|
|
GuiTextBoxAutoCompleteBase::TextEditNotify(arguments);
|
|
RepeatingParsingExecutor::CallbackBase::TextEditNotify(arguments);
|
|
if(element && elementModifyLock)
|
|
{
|
|
editing=true;
|
|
SPIN_LOCK(editTraceLock)
|
|
{
|
|
editTrace.Add(arguments);
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiGrammarAutoComplete::TextCaretChanged(const TextCaretChangedStruct& arguments)
|
|
{
|
|
GuiTextBoxAutoCompleteBase::TextCaretChanged(arguments);
|
|
RepeatingParsingExecutor::CallbackBase::TextCaretChanged(arguments);
|
|
if(element && elementModifyLock)
|
|
{
|
|
SPIN_LOCK(editTraceLock)
|
|
{
|
|
// queue a fake TextEditNotifyStruct
|
|
// a fake struct can be detected by (trace.originalText==L"" && trace.inputText==L"")
|
|
TextEditNotifyStruct trace;
|
|
trace.editVersion=arguments.editVersion;
|
|
trace.originalStart=arguments.oldBegin;
|
|
trace.originalEnd=arguments.oldEnd;
|
|
trace.inputStart=arguments.newBegin;
|
|
trace.inputEnd=arguments.newEnd;
|
|
|
|
// ensure trace.originalStart<=trace.originalEnd
|
|
if(trace.originalStart>trace.originalEnd)
|
|
{
|
|
TextPos temp=trace.originalStart;
|
|
trace.originalStart=trace.originalEnd;
|
|
trace.originalEnd=temp;
|
|
}
|
|
// ensure trace.inputStart<=trace.inputEnd
|
|
if(trace.inputStart>trace.inputEnd)
|
|
{
|
|
TextPos temp=trace.inputStart;
|
|
trace.inputStart=trace.inputEnd;
|
|
trace.inputEnd=temp;
|
|
}
|
|
editTrace.Add(trace);
|
|
}
|
|
|
|
SPIN_LOCK(contextLock)
|
|
{
|
|
if(context.input.node)
|
|
{
|
|
if(editing)
|
|
{
|
|
// if the current caret changing is caused by editing
|
|
// submit a task with valid editVersion and invalid node and code
|
|
RepeatingParsingOutput input;
|
|
input.editVersion=context.input.editVersion;
|
|
SubmitTask(input);
|
|
}
|
|
else if(context.input.editVersion == arguments.editVersion)
|
|
{
|
|
// if the current caret changing is not caused by editing
|
|
// submit a task with the previous input
|
|
SubmitTask(context.input);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiGrammarAutoComplete::TextEditFinished(vuint editVersion)
|
|
{
|
|
GuiTextBoxAutoCompleteBase::TextEditFinished(editVersion);
|
|
RepeatingParsingExecutor::CallbackBase::TextEditFinished(editVersion);
|
|
if(element && elementModifyLock)
|
|
{
|
|
editing=false;
|
|
}
|
|
}
|
|
|
|
void GuiGrammarAutoComplete::OnParsingFinishedAsync(const RepeatingParsingOutput& arguments)
|
|
{
|
|
if(element && elementModifyLock)
|
|
{
|
|
GetApplication()->InvokeInMainThread(ownerComposition->GetRelatedControlHost(), [=]()
|
|
{
|
|
// submit a task if the RepeatingParsingExecutor notices a new parsing result
|
|
SubmitTask(arguments);
|
|
});
|
|
}
|
|
}
|
|
|
|
void GuiGrammarAutoComplete::CollectLeftRecursiveRules()
|
|
{
|
|
leftRecursiveRules.Clear();
|
|
Ptr<ParsingGeneralParser> parser=parsingExecutor->GetParser();
|
|
Ptr<ParsingTable> table=parser->GetTable();
|
|
vint stateCount=table->GetStateCount();
|
|
vint tokenCount=table->GetTokenCount();
|
|
for(vint i=0;i<stateCount;i++)
|
|
{
|
|
for(vint j=0;j<tokenCount;j++)
|
|
{
|
|
Ptr<ParsingTable::TransitionBag> bag=table->GetTransitionBag(i, j);
|
|
if(bag)
|
|
{
|
|
FOREACH(Ptr<ParsingTable::TransitionItem>, item, bag->transitionItems)
|
|
{
|
|
FOREACH(ParsingTable::Instruction, ins, item->instructions)
|
|
{
|
|
if(ins.instructionType==ParsingTable::Instruction::LeftRecursiveReduce)
|
|
{
|
|
if(!leftRecursiveRules.Contains(ins.creatorRule))
|
|
{
|
|
leftRecursiveRules.Add(ins.creatorRule);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
vint GuiGrammarAutoComplete::UnsafeGetEditTraceIndex(vuint editVersion)
|
|
{
|
|
// get the index of the latest TextEditNotifyStruct of a specified edit version
|
|
// this function should be called inside SPIN_LOCK(editTraceLock)
|
|
// perform a binary search
|
|
vint start = 0;
|
|
vint end = editTrace.Count() - 1;
|
|
while (start <= end)
|
|
{
|
|
vint middle = (start + end) / 2;
|
|
TextEditNotifyStruct& trace = editTrace[middle];
|
|
|
|
if (editVersion<trace.editVersion)
|
|
{
|
|
end = middle - 1;
|
|
}
|
|
else if (editVersion>trace.editVersion)
|
|
{
|
|
start = middle + 1;
|
|
}
|
|
else
|
|
{
|
|
// if multiple TextEditNotifyStruct is found, choose the latest one
|
|
while (middle < editTrace.Count() - 1)
|
|
{
|
|
if (editTrace[middle + 1].editVersion == editTrace[middle].editVersion)
|
|
{
|
|
middle++;
|
|
}
|
|
else
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
return middle;
|
|
}
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
TextPos GuiGrammarAutoComplete::ChooseCorrectTextPos(TextPos pos, const regex::RegexTokens& tokens)
|
|
{
|
|
Ptr<ParsingTable> table=grammarParser->GetTable();
|
|
RegexToken lastToken;
|
|
lastToken.reading=0;
|
|
|
|
FOREACH(RegexToken, token, tokens)
|
|
{
|
|
// we treat "class| Name" as editing the first token
|
|
if(TextPos(token.rowEnd, token.columnEnd+1)>=pos)
|
|
{
|
|
if(table->GetTableTokenIndex(token.token)!=-1 && lastToken.reading)
|
|
{
|
|
pos=TextPos(lastToken.rowStart, lastToken.columnStart);
|
|
}
|
|
break;
|
|
}
|
|
lastToken=token;
|
|
}
|
|
return pos;
|
|
}
|
|
|
|
void GuiGrammarAutoComplete::ExecuteRefresh(AutoCompleteContext& newContext)
|
|
{
|
|
// process the input of a task is submitted not by text editing
|
|
// find the text selection by the edit version of the input
|
|
TextPos startPos, endPos;
|
|
{
|
|
SPIN_LOCK(editTraceLock)
|
|
{
|
|
vint traceIndex = UnsafeGetEditTraceIndex(newContext.input.editVersion);
|
|
if (traceIndex == -1) return;
|
|
|
|
TextEditNotifyStruct& trace = editTrace[traceIndex];
|
|
startPos = trace.inputStart;
|
|
endPos = trace.inputEnd;
|
|
}
|
|
|
|
const RegexLexer& lexer = grammarParser->GetTable()->GetLexer();
|
|
RegexTokens tokens = lexer.Parse(newContext.input.code);
|
|
startPos = ChooseCorrectTextPos(startPos, tokens);
|
|
}
|
|
|
|
// locate the deepest node using the text selection
|
|
ParsingTextPos start(startPos.row, startPos.column);
|
|
ParsingTextPos end(endPos.row, endPos.column);
|
|
ParsingTextRange range(start, end);
|
|
ParsingTreeNode* found = newContext.input.node->FindDeepestNode(range);
|
|
ParsingTreeObject* selectedNode = 0;
|
|
|
|
// if the location failed, choose the root node
|
|
if (!found || startPos == TextPos(0, 0))
|
|
{
|
|
found = newContext.input.node.Obj();
|
|
}
|
|
|
|
if (!selectedNode)
|
|
{
|
|
// from the deepest node, traverse towards the root node
|
|
// find the deepest node whose created rule is a left recursive rule and whose parent is not
|
|
ParsingTreeObject* lrec = 0;
|
|
ParsingTreeNode* current = found;
|
|
while (current)
|
|
{
|
|
ParsingTreeObject* obj = dynamic_cast<ParsingTreeObject*>(current);
|
|
if (obj)
|
|
{
|
|
FOREACH(WString, rule, obj->GetCreatorRules())
|
|
{
|
|
if (leftRecursiveRules.Contains(rule))
|
|
{
|
|
lrec = obj;
|
|
break;
|
|
}
|
|
}
|
|
if (obj && lrec && lrec != obj)
|
|
{
|
|
selectedNode = lrec;
|
|
break;
|
|
}
|
|
}
|
|
current = current->GetParent();
|
|
}
|
|
}
|
|
|
|
if (!selectedNode)
|
|
{
|
|
// if there is no left recursive rule that creates the deepest node and all indirect parents
|
|
// choose the deepest ParsingTreeObject
|
|
ParsingTreeNode* current = found;
|
|
while (current)
|
|
{
|
|
ParsingTreeObject* obj = dynamic_cast<ParsingTreeObject*>(current);
|
|
if (obj)
|
|
{
|
|
selectedNode = obj;
|
|
break;
|
|
}
|
|
current = current->GetParent();
|
|
}
|
|
}
|
|
|
|
if (selectedNode)
|
|
{
|
|
// get the code range of the selected node
|
|
start = selectedNode->GetCodeRange().start;
|
|
end = selectedNode->GetCodeRange().end;
|
|
|
|
// get all properties from the selected node
|
|
newContext.rule = selectedNode->GetCreatorRules()[selectedNode->GetCreatorRules().Count() - 1];
|
|
newContext.originalRange = selectedNode->GetCodeRange();
|
|
newContext.originalNode = dynamic_cast<ParsingTreeObject*>(selectedNode);
|
|
newContext.modifiedNode = newContext.originalNode;
|
|
newContext.modifiedEditVersion = newContext.input.editVersion;
|
|
|
|
// get the corresponding code of the selected node
|
|
if (start.index >= 0 && end.index >= 0)
|
|
{
|
|
newContext.modifiedCode = newContext.input.code.Sub(start.index, end.index - start.index + 1).Buffer();
|
|
}
|
|
}
|
|
}
|
|
|
|
bool GuiGrammarAutoComplete::NormalizeTextPos(AutoCompleteContext& newContext, elements::text::TextLines& lines, TextPos& pos)
|
|
{
|
|
// get the start position
|
|
TextPos start(newContext.originalRange.start.row, newContext.originalRange.start.column);
|
|
|
|
// get the end position of the end of lines
|
|
TextPos end
|
|
= lines.GetCount() <= 1
|
|
? TextPos(start.row, start.column + lines.GetLine(0).dataLength)
|
|
: TextPos(start.row + lines.GetCount() - 1, lines.GetLine(lines.GetCount() - 1).dataLength)
|
|
;
|
|
|
|
if (start <= pos && pos <= end)
|
|
{
|
|
// if the pos is inside the range
|
|
// normalize the pos to a new coordinate that the beginning position of lines is (row=0, column=0)
|
|
pos.row -= start.row;
|
|
if (pos.row == 0)
|
|
{
|
|
pos.column -= start.column;
|
|
}
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
void GuiGrammarAutoComplete::ExecuteEdit(AutoCompleteContext& newContext)
|
|
{
|
|
// process the input of a task that is submitted by text editing
|
|
// this function make an approximiation to the context if the RepeatingParsingExecutor is not fast enough
|
|
// copy all TextEditNotifyStruct that is caused by a text editing before (and including) the edit version of the input
|
|
List<TextEditNotifyStruct> usedTrace;
|
|
{
|
|
SPIN_LOCK(editTraceLock)
|
|
{
|
|
CopyFrom(
|
|
usedTrace,
|
|
From(editTrace)
|
|
.Where([&newContext](const TextEditNotifyStruct& value)
|
|
{
|
|
return (value.originalText != L"" || value.inputText != L"") && value.editVersion > newContext.modifiedEditVersion;
|
|
})
|
|
);
|
|
}
|
|
}
|
|
|
|
// apply all modification to get the new modifiedCode
|
|
bool failed = false;
|
|
if (usedTrace.Count() > 0)
|
|
{
|
|
if (usedTrace[0].editVersion != newContext.modifiedEditVersion + 1)
|
|
{
|
|
// failed if any TextEditNotifyStruct is missing
|
|
failed = true;
|
|
}
|
|
else
|
|
{
|
|
// initialize a TextLines with the latest modifiedCode
|
|
text::TextLines lines(nullptr);
|
|
lines.SetText(newContext.modifiedCode);
|
|
FOREACH(TextEditNotifyStruct, trace, usedTrace)
|
|
{
|
|
// apply a modification to lines
|
|
TextPos start = trace.originalStart;
|
|
TextPos end = trace.originalEnd;
|
|
|
|
// only if the modification is meaningful
|
|
if (NormalizeTextPos(newContext, lines, start) && NormalizeTextPos(newContext, lines, end))
|
|
{
|
|
lines.Modify(start, end, trace.inputText);
|
|
}
|
|
else
|
|
{
|
|
// otherwise, failed
|
|
failed = true;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!failed)
|
|
{
|
|
newContext.modifiedCode = lines.GetText();
|
|
}
|
|
}
|
|
}
|
|
|
|
if (failed)
|
|
{
|
|
// clear originalNode to notify that the current context goes wrong
|
|
newContext.originalNode = 0;
|
|
}
|
|
|
|
if (usedTrace.Count() > 0)
|
|
{
|
|
// update the edit version
|
|
newContext.modifiedEditVersion = usedTrace[usedTrace.Count() - 1].editVersion;
|
|
}
|
|
}
|
|
|
|
void GuiGrammarAutoComplete::DeleteFutures(collections::List<parsing::tabling::ParsingState::Future*>& futures)
|
|
{
|
|
// delete all futures and clear the list
|
|
FOREACH(ParsingState::Future*, future, futures)
|
|
{
|
|
delete future;
|
|
}
|
|
futures.Clear();
|
|
}
|
|
|
|
regex::RegexToken* GuiGrammarAutoComplete::TraverseTransitions(
|
|
parsing::tabling::ParsingState& state,
|
|
parsing::tabling::ParsingTransitionCollector& transitionCollector,
|
|
TextPos stopPosition,
|
|
collections::List<parsing::tabling::ParsingState::Future*>& nonRecoveryFutures,
|
|
collections::List<parsing::tabling::ParsingState::Future*>& recoveryFutures
|
|
)
|
|
{
|
|
const List<ParsingState::TransitionResult>& transitions = transitionCollector.GetTransitions();
|
|
for (vint index = 0; index < transitions.Count(); index++)
|
|
{
|
|
const ParsingState::TransitionResult& transition = transitions[index];
|
|
switch (transition.transitionType)
|
|
{
|
|
case ParsingState::TransitionResult::AmbiguityBegin:
|
|
break;
|
|
case ParsingState::TransitionResult::AmbiguityBranch:
|
|
// ambiguity branches are not nested
|
|
// tokens in different braches are the same
|
|
// so we only need to run one branch, and skip the others
|
|
index = transitionCollector.GetAmbiguityEndFromBegin(transitionCollector.GetAmbiguityBeginFromBranch(index));
|
|
break;
|
|
case ParsingState::TransitionResult::AmbiguityEnd:
|
|
break;
|
|
case ParsingState::TransitionResult::ExecuteInstructions:
|
|
{
|
|
// test does the token reach the stop position
|
|
if (transition.token)
|
|
{
|
|
// we treat "A|B" as editing A if token A is endless, otherwise treated as editing B
|
|
TextPos tokenEnd(transition.token->rowEnd, transition.token->columnEnd + 1);
|
|
|
|
// if the caret is not at the end of the token
|
|
if (tokenEnd > stopPosition)
|
|
{
|
|
// stop the traversing and return the editing token
|
|
return transition.token;
|
|
}
|
|
else if (tokenEnd == stopPosition)
|
|
{
|
|
// if the caret is at the end of the token, and it is a closed token
|
|
// e.g. identifier is not a closed token, string is a closed token
|
|
if (!grammarParser->GetTable()->GetLexer().Walk().IsClosedToken(transition.token->reading, transition.token->length))
|
|
{
|
|
// stop the traversing and return the editing token
|
|
return transition.token;
|
|
}
|
|
}
|
|
}
|
|
|
|
// traverse the PDA using the token specified in the current transition
|
|
vint tableTokenIndex = transition.tableTokenIndex;
|
|
List<ParsingState::Future*> possibilities;
|
|
if (recoveryFutures.Count() > 0)
|
|
{
|
|
FOREACH(ParsingState::Future*, future, recoveryFutures)
|
|
{
|
|
state.Explore(tableTokenIndex, future, possibilities);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
FOREACH(ParsingState::Future*, future, nonRecoveryFutures)
|
|
{
|
|
state.Explore(tableTokenIndex, future, possibilities);
|
|
}
|
|
}
|
|
|
|
// delete duplicated futures
|
|
List<ParsingState::Future*> selectedPossibilities;
|
|
for (vint i = 0; i < possibilities.Count(); i++)
|
|
{
|
|
ParsingState::Future* candidateFuture = possibilities[i];
|
|
bool duplicated = false;
|
|
FOREACH(ParsingState::Future*, future, selectedPossibilities)
|
|
{
|
|
if (
|
|
candidateFuture->currentState == future->currentState &&
|
|
candidateFuture->reduceStateCount == future->reduceStateCount &&
|
|
candidateFuture->shiftStates.Count() == future->shiftStates.Count()
|
|
)
|
|
{
|
|
bool same = true;
|
|
for (vint j = 0; j < future->shiftStates.Count(); j++)
|
|
{
|
|
if (candidateFuture->shiftStates[i] != future->shiftStates[i])
|
|
{
|
|
same = false;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if ((duplicated = same))
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (duplicated)
|
|
{
|
|
delete candidateFuture;
|
|
}
|
|
else
|
|
{
|
|
selectedPossibilities.Add(candidateFuture);
|
|
}
|
|
}
|
|
|
|
// step forward
|
|
if (transition.token || transition.tableTokenIndex == ParsingTable::TokenBegin)
|
|
{
|
|
DeleteFutures(nonRecoveryFutures);
|
|
DeleteFutures(recoveryFutures);
|
|
CopyFrom(nonRecoveryFutures, selectedPossibilities);
|
|
}
|
|
else
|
|
{
|
|
DeleteFutures(recoveryFutures);
|
|
CopyFrom(recoveryFutures, selectedPossibilities);
|
|
}
|
|
}
|
|
break;
|
|
default:;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
regex::RegexToken* GuiGrammarAutoComplete::SearchValidInputToken(
|
|
parsing::tabling::ParsingState& state,
|
|
parsing::tabling::ParsingTransitionCollector& transitionCollector,
|
|
TextPos stopPosition,
|
|
AutoCompleteContext& newContext,
|
|
collections::SortedList<vint>& tableTokenIndices
|
|
)
|
|
{
|
|
// initialize the PDA state
|
|
state.Reset(newContext.rule);
|
|
List<ParsingState::Future*> nonRecoveryFutures, recoveryFutures;
|
|
nonRecoveryFutures.Add(state.ExploreCreateRootFuture());
|
|
|
|
// traverse the PDA until it reach the stop position
|
|
// nonRecoveryFutures store the state when the last token (existing) is reached
|
|
// recoveryFutures store the state when the last token (inserted by error recovery) is reached
|
|
RegexToken* token = TraverseTransitions(state, transitionCollector, stopPosition, nonRecoveryFutures, recoveryFutures);
|
|
|
|
// explore all possibilities from the last token before the stop position
|
|
List<ParsingState::Future*> possibilities;
|
|
for (vint i = 0; i < nonRecoveryFutures.Count(); i++)
|
|
{
|
|
state.Explore(ParsingTable::NormalReduce, nonRecoveryFutures[i], nonRecoveryFutures);
|
|
state.Explore(ParsingTable::LeftRecursiveReduce, nonRecoveryFutures[i], nonRecoveryFutures);
|
|
}
|
|
FOREACH(ParsingState::Future*, future, nonRecoveryFutures)
|
|
{
|
|
vint count = state.GetTable()->GetTokenCount();
|
|
for (vint i = ParsingTable::UserTokenStart; i < count; i++)
|
|
{
|
|
state.Explore(i, future, possibilities);
|
|
}
|
|
}
|
|
|
|
// get all possible tokens that marked using @AutoCompleteCandidate
|
|
FOREACH(ParsingState::Future*, future, possibilities)
|
|
{
|
|
if (!tableTokenIndices.Contains(future->selectedToken))
|
|
{
|
|
tableTokenIndices.Add(future->selectedToken);
|
|
}
|
|
}
|
|
|
|
// release all data
|
|
DeleteFutures(possibilities);
|
|
DeleteFutures(nonRecoveryFutures);
|
|
DeleteFutures(recoveryFutures);
|
|
|
|
// return the editing token
|
|
return token;
|
|
}
|
|
|
|
TextPos GuiGrammarAutoComplete::GlobalTextPosToModifiedTextPos(AutoCompleteContext& newContext, TextPos pos)
|
|
{
|
|
pos.row-=newContext.originalRange.start.row;
|
|
if(pos.row==0)
|
|
{
|
|
pos.column-=newContext.originalRange.start.column;
|
|
}
|
|
return pos;
|
|
}
|
|
|
|
TextPos GuiGrammarAutoComplete::ModifiedTextPosToGlobalTextPos(AutoCompleteContext& newContext, TextPos pos)
|
|
{
|
|
if(pos.row==0)
|
|
{
|
|
pos.column+=newContext.originalRange.start.column;
|
|
}
|
|
pos.row+=newContext.originalRange.start.row;
|
|
return pos;
|
|
}
|
|
|
|
void GuiGrammarAutoComplete::ExecuteCalculateList(AutoCompleteContext& newContext)
|
|
{
|
|
// calcuate the content of the auto complete list
|
|
// it is sad that, because the parser's algorithm is too complex
|
|
// we need to reparse and track the internal state of the PDA(push-down automaton) here.
|
|
// initialize the PDA
|
|
ParsingState state(newContext.modifiedCode, grammarParser->GetTable());
|
|
state.Reset(newContext.rule);
|
|
|
|
// prepare to get all transitions
|
|
ParsingTransitionCollector collector;
|
|
List<Ptr<ParsingError>> errors;
|
|
|
|
// reparse and get all transitions during parsing
|
|
if (grammarParser->Parse(state, collector, errors))
|
|
{
|
|
// if modifiedNode is not prepared (the task is submitted because of text editing)
|
|
// use the transition to build the syntax tree
|
|
if (!newContext.modifiedNode)
|
|
{
|
|
ParsingTreeBuilder builder;
|
|
builder.Reset();
|
|
bool succeeded = true;
|
|
FOREACH(ParsingState::TransitionResult, transition, collector.GetTransitions())
|
|
{
|
|
if (!(succeeded = builder.Run(transition)))
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (succeeded)
|
|
{
|
|
Ptr<ParsingTreeNode> parsedNode = builder.GetNode();
|
|
newContext.modifiedNode = parsedNode.Cast<ParsingTreeObject>();
|
|
newContext.modifiedNode->InitializeQueryCache();
|
|
}
|
|
}
|
|
|
|
if (newContext.modifiedNode)
|
|
{
|
|
// get the latest text editing trace
|
|
TextEditNotifyStruct trace;
|
|
SPIN_LOCK(editTraceLock)
|
|
{
|
|
vint index = UnsafeGetEditTraceIndex(newContext.modifiedEditVersion);
|
|
if (index == -1)
|
|
{
|
|
return;
|
|
}
|
|
else
|
|
{
|
|
trace = editTrace[index];
|
|
}
|
|
}
|
|
|
|
// calculate the stop position for PDA traversing
|
|
TextPos stopPosition = GlobalTextPosToModifiedTextPos(newContext, trace.inputStart);
|
|
|
|
// find all possible token before the current caret using the PDA
|
|
Ptr<AutoCompleteData> autoComplete = new AutoCompleteData;
|
|
SortedList<vint> tableTokenIndices;
|
|
RegexToken* editingToken = SearchValidInputToken(state, collector, stopPosition, newContext, tableTokenIndices);
|
|
|
|
// collect all auto complete types
|
|
{
|
|
// collect all keywords that can be put into the auto complete list
|
|
FOREACH(vint, token, tableTokenIndices)
|
|
{
|
|
vint regexToken = token - ParsingTable::UserTokenStart;
|
|
if (regexToken >= 0)
|
|
{
|
|
autoComplete->candidates.Add(regexToken);
|
|
if (parsingExecutor->GetTokenMetaData(regexToken).isCandidate)
|
|
{
|
|
autoComplete->shownCandidates.Add(regexToken);
|
|
}
|
|
}
|
|
}
|
|
|
|
// calculate the arranged stopPosition
|
|
if (editingToken)
|
|
{
|
|
TextPos tokenPos(editingToken->rowStart, editingToken->columnStart);
|
|
if (tokenPos < stopPosition)
|
|
{
|
|
stopPosition = tokenPos;
|
|
}
|
|
}
|
|
|
|
// calculate the start/end position for PDA traversing
|
|
TextPos startPos, endPos;
|
|
{
|
|
startPos = ModifiedTextPosToGlobalTextPos(newContext, stopPosition);
|
|
autoComplete->startPosition = startPos;
|
|
endPos = trace.inputEnd;
|
|
if (newContext.modifiedNode != newContext.originalNode)
|
|
{
|
|
startPos = GlobalTextPosToModifiedTextPos(newContext, startPos);
|
|
endPos = GlobalTextPosToModifiedTextPos(newContext, endPos);
|
|
}
|
|
if (startPos<endPos && endPos.column>0)
|
|
{
|
|
endPos.column--;
|
|
}
|
|
}
|
|
|
|
// calculate the auto complete type
|
|
if (editingToken && parsingExecutor->GetTokenMetaData(editingToken->token).hasAutoComplete)
|
|
{
|
|
ParsingTextRange range(ParsingTextPos(startPos.row, startPos.column), ParsingTextPos(endPos.row, endPos.column));
|
|
AutoCompleteData::RetriveContext(*autoComplete.Obj(), range, newContext.modifiedNode.Obj(), parsingExecutor.Obj());
|
|
}
|
|
}
|
|
newContext.autoComplete = autoComplete;
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiGrammarAutoComplete::Execute(const RepeatingParsingOutput& input)
|
|
{
|
|
SPIN_LOCK(contextLock)
|
|
{
|
|
if(input.editVersion<context.input.editVersion)
|
|
{
|
|
return;
|
|
}
|
|
}
|
|
AutoCompleteContext newContext;
|
|
bool byGlobalCorrection=false;
|
|
|
|
if(input.node)
|
|
{
|
|
newContext.input=input;
|
|
ExecuteRefresh(newContext);
|
|
byGlobalCorrection=true;
|
|
}
|
|
else
|
|
{
|
|
SPIN_LOCK(contextLock)
|
|
{
|
|
newContext=context;
|
|
newContext.modifiedNode=0;
|
|
newContext.autoComplete=0;
|
|
}
|
|
if(newContext.originalNode)
|
|
{
|
|
ExecuteEdit(newContext);
|
|
}
|
|
}
|
|
|
|
if(newContext.originalNode)
|
|
{
|
|
ExecuteCalculateList(newContext);
|
|
}
|
|
|
|
SPIN_LOCK(contextLock)
|
|
{
|
|
context=newContext;
|
|
}
|
|
if(newContext.modifiedNode)
|
|
{
|
|
OnContextFinishedAsync(context);
|
|
GetApplication()->InvokeInMainThread(ownerComposition->GetRelatedControlHost(), [=]()
|
|
{
|
|
PostList(newContext, byGlobalCorrection);
|
|
});
|
|
}
|
|
}
|
|
|
|
void GuiGrammarAutoComplete::PostList(const AutoCompleteContext& newContext, bool byGlobalCorrection)
|
|
{
|
|
bool openList = true; // true: make the list visible
|
|
bool keepListState = false; // true: don't change the list visibility
|
|
Ptr<AutoCompleteData> autoComplete = newContext.autoComplete;
|
|
|
|
// if failed to get the auto complete list, close
|
|
if (!autoComplete)
|
|
{
|
|
openList = false;
|
|
}
|
|
if (openList)
|
|
{
|
|
if (autoComplete->shownCandidates.Count() + autoComplete->candidateItems.Count() == 0)
|
|
{
|
|
openList = false;
|
|
}
|
|
}
|
|
|
|
TextPos startPosition, endPosition;
|
|
WString editingText;
|
|
if (openList)
|
|
{
|
|
SPIN_LOCK(editTraceLock)
|
|
{
|
|
// if the edit version is invalid, cancel
|
|
vint traceIndex = UnsafeGetEditTraceIndex(newContext.modifiedEditVersion);
|
|
if (traceIndex == -1)
|
|
{
|
|
return;
|
|
}
|
|
// an edit version has two trace at most, for text change and caret change, here we peak the text change
|
|
if (traceIndex > 0 && editTrace[traceIndex - 1].editVersion == context.modifiedEditVersion)
|
|
{
|
|
traceIndex--;
|
|
}
|
|
// if the edit version is not created by keyboard input, close
|
|
if (traceIndex >= 0)
|
|
{
|
|
TextEditNotifyStruct& trace = editTrace[traceIndex];
|
|
if (!trace.keyInput)
|
|
{
|
|
openList = false;
|
|
}
|
|
}
|
|
|
|
// scan all traces from the calculation's edit version until now
|
|
if (openList)
|
|
{
|
|
keepListState = true;
|
|
startPosition = autoComplete->startPosition;
|
|
endPosition = editTrace[editTrace.Count() - 1].inputEnd;
|
|
for (vint i = traceIndex; i < editTrace.Count(); i++)
|
|
{
|
|
TextEditNotifyStruct& trace = editTrace[i];
|
|
// if there are no text change trace until now, don't change the list
|
|
if (trace.originalText != L"" || trace.inputText != L"")
|
|
{
|
|
keepListState = false;
|
|
}
|
|
// if the edit position goes before the start position of the auto complete, refresh
|
|
if (trace.inputEnd <= startPosition)
|
|
{
|
|
openList = false;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (traceIndex > 0)
|
|
{
|
|
editTrace.RemoveRange(0, traceIndex);
|
|
}
|
|
}
|
|
}
|
|
|
|
// if there is a global correction send to the UI thread but the list is not opening, cancel
|
|
if (byGlobalCorrection && !IsListOpening())
|
|
{
|
|
return;
|
|
}
|
|
|
|
// if the input text from the start position to the current position crosses a token, close
|
|
if (openList && element)
|
|
{
|
|
editingText = element->GetLines().GetText(startPosition, endPosition);
|
|
if (grammarParser->GetTable()->GetLexer().Walk().IsClosedToken(editingText))
|
|
{
|
|
openList = false;
|
|
}
|
|
}
|
|
|
|
// calculate the content of the list
|
|
if (autoComplete && ((!keepListState && openList) || IsListOpening()))
|
|
{
|
|
SortedList<WString> itemKeys;
|
|
List<ParsingCandidateItem> itemValues;
|
|
|
|
// copy all candidate keywords
|
|
FOREACH(vint, token, autoComplete->shownCandidates)
|
|
{
|
|
WString literal = parsingExecutor->GetTokenMetaData(token).unescapedRegexText;
|
|
if (literal != L"" && !itemKeys.Contains(literal))
|
|
{
|
|
ParsingCandidateItem item;
|
|
item.name = literal;
|
|
item.semanticId = -1;
|
|
itemValues.Insert(itemKeys.Add(literal), item);
|
|
}
|
|
}
|
|
|
|
// copy all candidate symbols
|
|
if (autoComplete->acceptableSemanticIds)
|
|
{
|
|
FOREACH(ParsingCandidateItem, item, autoComplete->candidateItems)
|
|
{
|
|
if (autoComplete->acceptableSemanticIds->Contains(item.semanticId))
|
|
{
|
|
// add all acceptable display of a symbol
|
|
// because a symbol can has multiple representation in different places
|
|
if (item.name != L"" && !itemKeys.Contains(item.name))
|
|
{
|
|
itemValues.Insert(itemKeys.Add(item.name), item);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// fill the list
|
|
List<GuiTextBoxAutoCompleteBase::AutoCompleteItem> candidateItems;
|
|
for (vint i = 0; i < itemValues.Count(); i++)
|
|
{
|
|
auto& item = itemValues[i];
|
|
if (item.tag.IsNull())
|
|
{
|
|
if (auto analyzer = parsingExecutor->GetAnalyzer())
|
|
{
|
|
item.tag = analyzer->CreateTagForCandidateItem(item);
|
|
}
|
|
}
|
|
|
|
GuiTextBoxAutoCompleteBase::AutoCompleteItem candidateItem;
|
|
candidateItem.text = item.name;
|
|
candidateItem.tag = item.tag;
|
|
candidateItems.Add(candidateItem);
|
|
}
|
|
SetListContent(candidateItems);
|
|
}
|
|
|
|
// set the list state
|
|
if (!keepListState)
|
|
{
|
|
if (openList)
|
|
{
|
|
OpenList(startPosition);
|
|
}
|
|
else
|
|
{
|
|
CloseList();
|
|
}
|
|
}
|
|
|
|
if (IsListOpening())
|
|
{
|
|
HighlightList(editingText);
|
|
}
|
|
}
|
|
|
|
void GuiGrammarAutoComplete::Initialize()
|
|
{
|
|
grammarParser=CreateAutoRecoverParser(parsingExecutor->GetParser()->GetTable());
|
|
CollectLeftRecursiveRules();
|
|
parsingExecutor->AttachCallback(this);
|
|
}
|
|
|
|
void GuiGrammarAutoComplete::OnContextFinishedAsync(AutoCompleteContext& context)
|
|
{
|
|
if (auto analyzer = parsingExecutor->GetAnalyzer())
|
|
{
|
|
if (context.autoComplete && context.autoComplete->acceptableSemanticIds)
|
|
{
|
|
analyzer->GetCandidateItemsAsync(*context.autoComplete.Obj(), context, context.autoComplete->candidateItems);
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiGrammarAutoComplete::EnsureAutoCompleteFinished()
|
|
{
|
|
parsingExecutor->EnsureTaskFinished();
|
|
SPIN_LOCK(contextLock)
|
|
{
|
|
context = AutoCompleteContext();
|
|
}
|
|
}
|
|
|
|
GuiGrammarAutoComplete::GuiGrammarAutoComplete(Ptr<RepeatingParsingExecutor> _parsingExecutor)
|
|
:RepeatingParsingExecutor::CallbackBase(_parsingExecutor)
|
|
,editing(false)
|
|
{
|
|
Initialize();
|
|
}
|
|
|
|
GuiGrammarAutoComplete::GuiGrammarAutoComplete(Ptr<parsing::tabling::ParsingGeneralParser> _grammarParser, const WString& _grammarRule)
|
|
:RepeatingParsingExecutor::CallbackBase(new RepeatingParsingExecutor(_grammarParser, _grammarRule))
|
|
,editing(false)
|
|
{
|
|
Initialize();
|
|
}
|
|
|
|
GuiGrammarAutoComplete::~GuiGrammarAutoComplete()
|
|
{
|
|
EnsureAutoCompleteFinished();
|
|
parsingExecutor->DetachCallback(this);
|
|
}
|
|
|
|
Ptr<RepeatingParsingExecutor> GuiGrammarAutoComplete::GetParsingExecutor()
|
|
{
|
|
return parsingExecutor;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\CONTROLS\TEXTEDITORPACKAGE\LANGUAGESERVICE\GUILANGUAGECOLORIZER.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace controls
|
|
{
|
|
using namespace elements;
|
|
using namespace parsing;
|
|
using namespace parsing::tabling;
|
|
using namespace collections;
|
|
using namespace theme;
|
|
|
|
/***********************************************************************
|
|
GuiGrammarColorizer
|
|
***********************************************************************/
|
|
|
|
void GuiGrammarColorizer::OnParsingFinishedAsync(const RepeatingParsingOutput& output)
|
|
{
|
|
SPIN_LOCK(contextLock)
|
|
{
|
|
context=output;
|
|
OnContextFinishedAsync(context);
|
|
}
|
|
RestartColorizer();
|
|
}
|
|
|
|
void GuiGrammarColorizer::OnContextFinishedAsync(const RepeatingParsingOutput& context)
|
|
{
|
|
}
|
|
|
|
void GuiGrammarColorizer::Attach(elements::GuiColorizedTextElement* _element, SpinLock& _elementModifyLock, compositions::GuiGraphicsComposition* _ownerComposition, vuint editVersion)
|
|
{
|
|
GuiTextBoxRegexColorizer::Attach(_element, _elementModifyLock, _ownerComposition, editVersion);
|
|
RepeatingParsingExecutor::CallbackBase::Attach(_element, _elementModifyLock, _ownerComposition, editVersion);
|
|
}
|
|
|
|
void GuiGrammarColorizer::Detach()
|
|
{
|
|
GuiTextBoxRegexColorizer::Detach();
|
|
RepeatingParsingExecutor::CallbackBase::Detach();
|
|
if(element && elementModifyLock)
|
|
{
|
|
parsingExecutor->EnsureTaskFinished();
|
|
StopColorizer(false);
|
|
}
|
|
}
|
|
|
|
void GuiGrammarColorizer::TextEditPreview(TextEditPreviewStruct& arguments)
|
|
{
|
|
GuiTextBoxRegexColorizer::TextEditPreview(arguments);
|
|
RepeatingParsingExecutor::CallbackBase::TextEditPreview(arguments);
|
|
}
|
|
|
|
void GuiGrammarColorizer::TextEditNotify(const TextEditNotifyStruct& arguments)
|
|
{
|
|
GuiTextBoxRegexColorizer::TextEditNotify(arguments);
|
|
RepeatingParsingExecutor::CallbackBase::TextEditNotify(arguments);
|
|
}
|
|
|
|
void GuiGrammarColorizer::TextCaretChanged(const TextCaretChangedStruct& arguments)
|
|
{
|
|
GuiTextBoxRegexColorizer::TextCaretChanged(arguments);
|
|
RepeatingParsingExecutor::CallbackBase::TextCaretChanged(arguments);
|
|
}
|
|
|
|
void GuiGrammarColorizer::TextEditFinished(vuint editVersion)
|
|
{
|
|
GuiTextBoxRegexColorizer::TextEditFinished(editVersion);
|
|
RepeatingParsingExecutor::CallbackBase::TextEditFinished(editVersion);
|
|
}
|
|
|
|
void GuiGrammarColorizer::OnSemanticColorize(SemanticColorizeContext& context, const RepeatingParsingOutput& input)
|
|
{
|
|
if (auto analyzer = parsingExecutor->GetAnalyzer())
|
|
{
|
|
auto semanticId = analyzer->GetSemanticIdForTokenAsync(context, input);
|
|
if(semanticId!=-1)
|
|
{
|
|
context.semanticId=semanticId;
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiGrammarColorizer::EnsureColorizerFinished()
|
|
{
|
|
parsingExecutor->EnsureTaskFinished();
|
|
StopColorizerForever();
|
|
SPIN_LOCK(contextLock)
|
|
{
|
|
context=RepeatingParsingOutput();
|
|
}
|
|
}
|
|
|
|
GuiGrammarColorizer::GuiGrammarColorizer(Ptr<RepeatingParsingExecutor> _parsingExecutor)
|
|
:RepeatingParsingExecutor::CallbackBase(_parsingExecutor)
|
|
{
|
|
parsingExecutor->AttachCallback(this);
|
|
BeginSetColors();
|
|
}
|
|
|
|
GuiGrammarColorizer::GuiGrammarColorizer(Ptr<parsing::tabling::ParsingGeneralParser> _grammarParser, const WString& _grammarRule)
|
|
:RepeatingParsingExecutor::CallbackBase(new RepeatingParsingExecutor(_grammarParser, _grammarRule))
|
|
{
|
|
parsingExecutor->AttachCallback(this);
|
|
BeginSetColors();
|
|
}
|
|
|
|
GuiGrammarColorizer::~GuiGrammarColorizer()
|
|
{
|
|
EnsureColorizerFinished();
|
|
parsingExecutor->DetachCallback(this);
|
|
}
|
|
|
|
void GuiGrammarColorizer::BeginSetColors()
|
|
{
|
|
ClearTokens();
|
|
colorSettings.Clear();
|
|
|
|
text::ColorEntry entry;
|
|
{
|
|
entry.normal.text = Color(0, 0, 0);
|
|
entry.normal.background = Color(0, 0, 0, 0);
|
|
entry.selectedFocused.text = Color(255, 255, 255);
|
|
entry.selectedFocused.background = Color(51, 153, 255);
|
|
entry.selectedUnfocused.text = Color(255, 255, 255);
|
|
entry.selectedUnfocused.background = Color(51, 153, 255);
|
|
}
|
|
|
|
SetDefaultColor(entry);
|
|
colorSettings.Add(L"Default", entry);
|
|
}
|
|
|
|
const collections::SortedList<WString>& GuiGrammarColorizer::GetColorNames()
|
|
{
|
|
return colorSettings.Keys();
|
|
}
|
|
|
|
GuiGrammarColorizer::ColorEntry GuiGrammarColorizer::GetColor(const WString& name)
|
|
{
|
|
vint index=colorSettings.Keys().IndexOf(name);
|
|
return index==-1?GetDefaultColor():colorSettings.Values().Get(index);
|
|
}
|
|
|
|
void GuiGrammarColorizer::SetColor(const WString& name, const ColorEntry& entry)
|
|
{
|
|
colorSettings.Set(name, entry);
|
|
}
|
|
|
|
void GuiGrammarColorizer::SetColor(const WString& name, const Color& color)
|
|
{
|
|
text::ColorEntry entry=GetDefaultColor();
|
|
entry.normal.text=color;
|
|
SetColor(name, entry);
|
|
}
|
|
|
|
void GuiGrammarColorizer::EndSetColors()
|
|
{
|
|
SortedList<WString> tokenColors;
|
|
Ptr<ParsingTable> table=parsingExecutor->GetParser()->GetTable();
|
|
semanticColorMap.Clear();
|
|
|
|
vint tokenCount=table->GetTokenCount();
|
|
for(vint token=ParsingTable::UserTokenStart;token<tokenCount;token++)
|
|
{
|
|
const ParsingTable::TokenInfo& tokenInfo=table->GetTokenInfo(token);
|
|
const RepeatingParsingExecutor::TokenMetaData& md=parsingExecutor->GetTokenMetaData(token-ParsingTable::UserTokenStart);
|
|
if(md.defaultColorIndex==-1)
|
|
{
|
|
AddToken(tokenInfo.regex, GetDefaultColor());
|
|
}
|
|
else
|
|
{
|
|
WString name=parsingExecutor->GetSemanticName(md.defaultColorIndex);
|
|
vint color=AddToken(tokenInfo.regex, GetColor(name));
|
|
semanticColorMap.Set(md.defaultColorIndex, color);
|
|
tokenColors.Add(name);
|
|
}
|
|
}
|
|
|
|
FOREACH_INDEXER(WString, color, index, colorSettings.Keys())
|
|
{
|
|
if(!tokenColors.Contains(color))
|
|
{
|
|
vint semanticId=parsingExecutor->GetSemanticId(color);
|
|
if(semanticId!=-1)
|
|
{
|
|
vint tokenId=AddExtraToken(colorSettings.Values().Get(index));
|
|
vint color=tokenId+tokenCount-ParsingTable::UserTokenStart;
|
|
semanticColorMap.Set(semanticId, color);
|
|
}
|
|
}
|
|
}
|
|
Setup();
|
|
}
|
|
|
|
void GuiGrammarColorizer::ColorizeTokenContextSensitive(vint lineIndex, const wchar_t* text, vint start, vint length, vint& token, vint& contextState)
|
|
{
|
|
SPIN_LOCK(contextLock)
|
|
{
|
|
ParsingTreeObject* node=context.node.Obj();
|
|
if(node && token!=-1 && parsingExecutor->GetTokenMetaData(token).hasContextColor)
|
|
{
|
|
ParsingTextPos pos(lineIndex, start);
|
|
SemanticColorizeContext scContext;
|
|
if(SemanticColorizeContext::RetriveContext(scContext, pos, node, parsingExecutor.Obj()))
|
|
{
|
|
const RepeatingParsingExecutor::FieldMetaData& md=parsingExecutor->GetFieldMetaData(scContext.type, scContext.field);
|
|
vint semantic=md.colorIndex;
|
|
scContext.semanticId=-1;
|
|
|
|
if(scContext.acceptableSemanticIds)
|
|
{
|
|
OnSemanticColorize(scContext, context);
|
|
if(md.semantics->Contains(scContext.semanticId))
|
|
{
|
|
semantic=scContext.semanticId;
|
|
}
|
|
}
|
|
|
|
if(semantic!=-1)
|
|
{
|
|
vint index=semanticColorMap.Keys().IndexOf(semantic);
|
|
if(index!=-1)
|
|
{
|
|
token=semanticColorMap.Values()[index];
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
Ptr<RepeatingParsingExecutor> GuiGrammarColorizer::GetParsingExecutor()
|
|
{
|
|
return parsingExecutor;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\CONTROLS\TEXTEDITORPACKAGE\LANGUAGESERVICE\GUILANGUAGEOPERATIONS.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace controls
|
|
{
|
|
using namespace collections;
|
|
using namespace parsing;
|
|
using namespace parsing::tabling;
|
|
using namespace regex_internal;
|
|
|
|
/***********************************************************************
|
|
ParsingContext
|
|
***********************************************************************/
|
|
|
|
bool ParsingTokenContext::RetriveContext(ParsingTokenContext& output, parsing::ParsingTreeNode* foundNode, RepeatingParsingExecutor* executor)
|
|
{
|
|
ParsingTreeToken* foundToken=dynamic_cast<ParsingTreeToken*>(foundNode);
|
|
if(!foundToken) return false;
|
|
ParsingTreeObject* tokenParent=dynamic_cast<ParsingTreeObject*>(foundNode->GetParent());
|
|
if(!tokenParent) return false;
|
|
vint index=tokenParent->GetMembers().Values().IndexOf(foundNode);
|
|
if(index==-1) return false;
|
|
|
|
WString type=tokenParent->GetType();
|
|
WString field=tokenParent->GetMembers().Keys().Get(index);
|
|
const RepeatingParsingExecutor::FieldMetaData& md=executor->GetFieldMetaData(type, field);
|
|
|
|
output.foundToken=foundToken;
|
|
output.tokenParent=tokenParent;
|
|
output.type=type;
|
|
output.field=field;
|
|
output.acceptableSemanticIds=md.semantics;
|
|
return true;
|
|
}
|
|
|
|
bool ParsingTokenContext::RetriveContext(ParsingTokenContext& output, parsing::ParsingTextPos pos, parsing::ParsingTreeObject* rootNode, RepeatingParsingExecutor* executor)
|
|
{
|
|
ParsingTreeNode* foundNode=rootNode->FindDeepestNode(pos);
|
|
if(!foundNode) return false;
|
|
return RetriveContext(output, foundNode, executor);
|
|
}
|
|
|
|
bool ParsingTokenContext::RetriveContext(ParsingTokenContext& output, parsing::ParsingTextRange range, ParsingTreeObject* rootNode, RepeatingParsingExecutor* executor)
|
|
{
|
|
ParsingTreeNode* foundNode=rootNode->FindDeepestNode(range);
|
|
if(!foundNode) return false;
|
|
return RetriveContext(output, foundNode, executor);
|
|
}
|
|
|
|
/***********************************************************************
|
|
RepeatingParsingExecutor::IParsingAnalyzer
|
|
***********************************************************************/
|
|
|
|
parsing::ParsingTreeNode* RepeatingParsingExecutor::IParsingAnalyzer::ToParent(parsing::ParsingTreeNode* node, const RepeatingPartialParsingOutput* output)
|
|
{
|
|
if (!output || !output->modifiedNode) return node;
|
|
return node == output->modifiedNode.Obj()
|
|
? output->originalNode.Obj()
|
|
: node;
|
|
}
|
|
|
|
parsing::ParsingTreeObject* RepeatingParsingExecutor::IParsingAnalyzer::ToChild(parsing::ParsingTreeObject* node, const RepeatingPartialParsingOutput* output)
|
|
{
|
|
if (!output || !output->modifiedNode) return node;
|
|
return node == output->originalNode.Obj()
|
|
? output->modifiedNode.Obj()
|
|
: node;
|
|
}
|
|
|
|
Ptr<parsing::ParsingTreeNode> RepeatingParsingExecutor::IParsingAnalyzer::ToChild(Ptr<parsing::ParsingTreeNode> node, const RepeatingPartialParsingOutput* output)
|
|
{
|
|
if (!output) return node;
|
|
return node == output->originalNode
|
|
? output->modifiedNode.Cast<ParsingTreeNode>()
|
|
: node;
|
|
}
|
|
|
|
parsing::ParsingTreeNode* RepeatingParsingExecutor::IParsingAnalyzer::GetParent(parsing::ParsingTreeNode* node, const RepeatingPartialParsingOutput* output)
|
|
{
|
|
return ToParent(node, output)->GetParent();
|
|
}
|
|
|
|
Ptr<parsing::ParsingTreeNode> RepeatingParsingExecutor::IParsingAnalyzer::GetMember(parsing::ParsingTreeObject* node, const WString& name, const RepeatingPartialParsingOutput* output)
|
|
{
|
|
return ToChild(ToChild(node, output)->GetMember(name), output);
|
|
}
|
|
|
|
Ptr<parsing::ParsingTreeNode> RepeatingParsingExecutor::IParsingAnalyzer::GetItem(parsing::ParsingTreeArray* node, vint index, const RepeatingPartialParsingOutput* output)
|
|
{
|
|
return ToChild(node->GetItem(index), output);
|
|
}
|
|
|
|
/***********************************************************************
|
|
RepeatingParsingExecutor::CallbackBase
|
|
***********************************************************************/
|
|
|
|
RepeatingParsingExecutor::CallbackBase::CallbackBase(Ptr<RepeatingParsingExecutor> _parsingExecutor)
|
|
:parsingExecutor(_parsingExecutor)
|
|
,callbackAutoPushing(false)
|
|
,callbackElement(0)
|
|
,callbackElementModifyLock(0)
|
|
{
|
|
}
|
|
|
|
RepeatingParsingExecutor::CallbackBase::~CallbackBase()
|
|
{
|
|
}
|
|
|
|
void RepeatingParsingExecutor::CallbackBase::RequireAutoSubmitTask(bool enabled)
|
|
{
|
|
callbackAutoPushing=enabled;
|
|
}
|
|
|
|
void RepeatingParsingExecutor::CallbackBase::Attach(elements::GuiColorizedTextElement* _element, SpinLock& _elementModifyLock, compositions::GuiGraphicsComposition* _ownerComposition, vuint editVersion)
|
|
{
|
|
if(_element)
|
|
{
|
|
SPIN_LOCK(_elementModifyLock)
|
|
{
|
|
callbackElement=_element;
|
|
callbackElementModifyLock=&_elementModifyLock;
|
|
}
|
|
}
|
|
|
|
parsingExecutor->ActivateCallback(this);
|
|
if(callbackElement && callbackElementModifyLock && callbackAutoPushing)
|
|
{
|
|
SPIN_LOCK(*callbackElementModifyLock)
|
|
{
|
|
RepeatingParsingInput input;
|
|
input.editVersion=editVersion;
|
|
input.code=callbackElement->GetLines().GetText();
|
|
parsingExecutor->SubmitTask(input);
|
|
}
|
|
}
|
|
}
|
|
|
|
void RepeatingParsingExecutor::CallbackBase::Detach()
|
|
{
|
|
if(callbackElement && callbackElementModifyLock)
|
|
{
|
|
SPIN_LOCK(*callbackElementModifyLock)
|
|
{
|
|
callbackElement=0;
|
|
callbackElementModifyLock=0;
|
|
}
|
|
}
|
|
|
|
parsingExecutor->DeactivateCallback(this);
|
|
}
|
|
|
|
void RepeatingParsingExecutor::CallbackBase::TextEditPreview(TextEditPreviewStruct& arguments)
|
|
{
|
|
}
|
|
|
|
void RepeatingParsingExecutor::CallbackBase::TextEditNotify(const TextEditNotifyStruct& arguments)
|
|
{
|
|
}
|
|
|
|
void RepeatingParsingExecutor::CallbackBase::TextCaretChanged(const TextCaretChangedStruct& arguments)
|
|
{
|
|
}
|
|
|
|
void RepeatingParsingExecutor::CallbackBase::TextEditFinished(vuint editVersion)
|
|
{
|
|
if(callbackElement && callbackElementModifyLock && callbackAutoPushing)
|
|
{
|
|
SPIN_LOCK(*callbackElementModifyLock)
|
|
{
|
|
RepeatingParsingInput input;
|
|
input.editVersion=editVersion;
|
|
input.code=callbackElement->GetLines().GetText();
|
|
parsingExecutor->SubmitTask(input);
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
RepeatingParsingExecutor
|
|
***********************************************************************/
|
|
|
|
void RepeatingParsingExecutor::Execute(const RepeatingParsingInput& input)
|
|
{
|
|
List<Ptr<ParsingError>> errors;
|
|
Ptr<ParsingTreeObject> node=grammarParser->Parse(input.code, grammarRule, errors).Cast<ParsingTreeObject>();
|
|
if(node)
|
|
{
|
|
node->InitializeQueryCache();
|
|
}
|
|
|
|
RepeatingParsingOutput result;
|
|
result.node=node;
|
|
result.editVersion=input.editVersion;
|
|
result.code=input.code;
|
|
if(node)
|
|
{
|
|
OnContextFinishedAsync(result);
|
|
FOREACH(ICallback*, callback, callbacks)
|
|
{
|
|
callback->OnParsingFinishedAsync(result);
|
|
}
|
|
}
|
|
}
|
|
|
|
void RepeatingParsingExecutor::PrepareMetaData()
|
|
{
|
|
Ptr<ParsingTable> table=grammarParser->GetTable();
|
|
tokenIndexMap.Clear();
|
|
semanticIndexMap.Clear();
|
|
tokenMetaDatas.Clear();
|
|
fieldMetaDatas.Clear();
|
|
|
|
Dictionary<vint, Ptr<ParsingTable::AttributeInfo>> tokenColorAtts, tokenContextColorAtts, tokenCandidateAtts, tokenAutoCompleteAtts;
|
|
Dictionary<FieldDesc, Ptr<ParsingTable::AttributeInfo>> fieldColorAtts, fieldSemanticAtts;
|
|
|
|
{
|
|
vint tokenCount=table->GetTokenCount();
|
|
for(vint token=ParsingTable::UserTokenStart;token<tokenCount;token++)
|
|
{
|
|
const ParsingTable::TokenInfo& tokenInfo=table->GetTokenInfo(token);
|
|
vint tokenIndex=token-ParsingTable::UserTokenStart;
|
|
tokenIndexMap.Add(tokenInfo.name, tokenIndex);
|
|
|
|
if(Ptr<ParsingTable::AttributeInfo> att=GetColorAttribute(tokenInfo.attributeIndex))
|
|
{
|
|
tokenColorAtts.Add(tokenIndex, att);
|
|
}
|
|
if(Ptr<ParsingTable::AttributeInfo> att=GetContextColorAttribute(tokenInfo.attributeIndex))
|
|
{
|
|
tokenContextColorAtts.Add(tokenIndex, att);
|
|
}
|
|
if(Ptr<ParsingTable::AttributeInfo> att=GetCandidateAttribute(tokenInfo.attributeIndex))
|
|
{
|
|
tokenCandidateAtts.Add(tokenIndex, att);
|
|
}
|
|
if(Ptr<ParsingTable::AttributeInfo> att=GetAutoCompleteAttribute(tokenInfo.attributeIndex))
|
|
{
|
|
tokenAutoCompleteAtts.Add(tokenIndex, att);
|
|
}
|
|
}
|
|
}
|
|
{
|
|
vint fieldCount=table->GetTreeFieldInfoCount();
|
|
for(vint field=0;field<fieldCount;field++)
|
|
{
|
|
const ParsingTable::TreeFieldInfo& fieldInfo=table->GetTreeFieldInfo(field);
|
|
FieldDesc fieldDesc(fieldInfo.type, fieldInfo.field);
|
|
|
|
if(Ptr<ParsingTable::AttributeInfo> att=GetColorAttribute(fieldInfo.attributeIndex))
|
|
{
|
|
fieldColorAtts.Add(fieldDesc, att);
|
|
}
|
|
if(Ptr<ParsingTable::AttributeInfo> att=GetSemanticAttribute(fieldInfo.attributeIndex))
|
|
{
|
|
fieldSemanticAtts.Add(fieldDesc, att);
|
|
}
|
|
}
|
|
}
|
|
|
|
FOREACH(Ptr<ParsingTable::AttributeInfo>, att,
|
|
From(tokenColorAtts.Values())
|
|
.Concat(tokenContextColorAtts.Values())
|
|
.Concat(fieldColorAtts.Values())
|
|
.Concat(fieldSemanticAtts.Values())
|
|
)
|
|
{
|
|
FOREACH(WString, argument, att->arguments)
|
|
{
|
|
if(!semanticIndexMap.Contains(argument))
|
|
{
|
|
semanticIndexMap.Add(argument);
|
|
}
|
|
}
|
|
}
|
|
|
|
vint index=0;
|
|
FOREACH(vint, tokenIndex, tokenIndexMap.Values())
|
|
{
|
|
TokenMetaData md;
|
|
md.tableTokenIndex=tokenIndex+ParsingTable::UserTokenStart;
|
|
md.lexerTokenIndex=tokenIndex;
|
|
md.defaultColorIndex=-1;
|
|
md.hasContextColor=false;
|
|
md.hasAutoComplete=false;
|
|
md.isCandidate=false;
|
|
|
|
if((index=tokenColorAtts.Keys().IndexOf(tokenIndex))!=-1)
|
|
{
|
|
md.defaultColorIndex=semanticIndexMap.IndexOf(tokenColorAtts.Values()[index]->arguments[0]);
|
|
}
|
|
md.hasContextColor=tokenContextColorAtts.Keys().Contains(tokenIndex);
|
|
md.hasAutoComplete=tokenAutoCompleteAtts.Keys().Contains(tokenIndex);
|
|
if((md.isCandidate=tokenCandidateAtts.Keys().Contains(tokenIndex)))
|
|
{
|
|
const ParsingTable::TokenInfo& tokenInfo=table->GetTokenInfo(md.tableTokenIndex);
|
|
if(IsRegexEscapedLiteralString(tokenInfo.regex))
|
|
{
|
|
md.unescapedRegexText=UnescapeTextForRegex(tokenInfo.regex);
|
|
}
|
|
else
|
|
{
|
|
md.isCandidate=false;
|
|
}
|
|
}
|
|
|
|
tokenMetaDatas.Add(tokenIndex, md);
|
|
}
|
|
{
|
|
vint fieldCount=table->GetTreeFieldInfoCount();
|
|
for(vint field=0;field<fieldCount;field++)
|
|
{
|
|
const ParsingTable::TreeFieldInfo& fieldInfo=table->GetTreeFieldInfo(field);
|
|
FieldDesc fieldDesc(fieldInfo.type, fieldInfo.field);
|
|
|
|
FieldMetaData md;
|
|
md.colorIndex=-1;
|
|
|
|
if((index=fieldColorAtts.Keys().IndexOf(fieldDesc))!=-1)
|
|
{
|
|
md.colorIndex=semanticIndexMap.IndexOf(fieldColorAtts.Values()[index]->arguments[0]);
|
|
}
|
|
if((index=fieldSemanticAtts.Keys().IndexOf(fieldDesc))!=-1)
|
|
{
|
|
md.semantics=new List<vint>;
|
|
FOREACH(WString, argument, fieldSemanticAtts.Values()[index]->arguments)
|
|
{
|
|
md.semantics->Add(semanticIndexMap.IndexOf(argument));
|
|
}
|
|
}
|
|
|
|
fieldMetaDatas.Add(fieldDesc, md);
|
|
}
|
|
}
|
|
}
|
|
|
|
void RepeatingParsingExecutor::OnContextFinishedAsync(RepeatingParsingOutput& context)
|
|
{
|
|
if(analyzer)
|
|
{
|
|
context.cache = analyzer->CreateCacheAsync(context);
|
|
}
|
|
}
|
|
|
|
RepeatingParsingExecutor::RepeatingParsingExecutor(Ptr<parsing::tabling::ParsingGeneralParser> _grammarParser, const WString& _grammarRule, Ptr<IParsingAnalyzer> _analyzer)
|
|
:grammarParser(_grammarParser)
|
|
,grammarRule(_grammarRule)
|
|
,analyzer(_analyzer)
|
|
,autoPushingCallback(0)
|
|
{
|
|
PrepareMetaData();
|
|
if (analyzer)
|
|
{
|
|
analyzer->Attach(this);
|
|
}
|
|
}
|
|
|
|
RepeatingParsingExecutor::~RepeatingParsingExecutor()
|
|
{
|
|
EnsureTaskFinished();
|
|
if (analyzer)
|
|
{
|
|
analyzer->Detach(this);
|
|
}
|
|
}
|
|
|
|
Ptr<parsing::tabling::ParsingGeneralParser> RepeatingParsingExecutor::GetParser()
|
|
{
|
|
return grammarParser;
|
|
}
|
|
|
|
bool RepeatingParsingExecutor::AttachCallback(ICallback* value)
|
|
{
|
|
if(!value) return false;
|
|
if(callbacks.Contains(value)) return false;
|
|
callbacks.Add(value);
|
|
return true;
|
|
}
|
|
|
|
bool RepeatingParsingExecutor::DetachCallback(ICallback* value)
|
|
{
|
|
if(!value) return false;
|
|
if(!callbacks.Contains(value)) return false;
|
|
DeactivateCallback(value);
|
|
callbacks.Remove(value);
|
|
return true;
|
|
}
|
|
|
|
bool RepeatingParsingExecutor::ActivateCallback(ICallback* value)
|
|
{
|
|
if(!value) return false;
|
|
if(!callbacks.Contains(value)) return false;
|
|
if(activatedCallbacks.Contains(value)) return false;
|
|
activatedCallbacks.Add(value);
|
|
|
|
if(!autoPushingCallback)
|
|
{
|
|
autoPushingCallback=value;
|
|
autoPushingCallback->RequireAutoSubmitTask(true);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool RepeatingParsingExecutor::DeactivateCallback(ICallback* value)
|
|
{
|
|
if(!value) return false;
|
|
if(!callbacks.Contains(value)) return false;
|
|
if(!activatedCallbacks.Contains(value)) return false;
|
|
|
|
if(autoPushingCallback==value)
|
|
{
|
|
autoPushingCallback->RequireAutoSubmitTask(false);
|
|
autoPushingCallback=0;
|
|
}
|
|
activatedCallbacks.Remove(value);
|
|
if(!autoPushingCallback && activatedCallbacks.Count()>0)
|
|
{
|
|
autoPushingCallback=activatedCallbacks[0];
|
|
autoPushingCallback->RequireAutoSubmitTask(true);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
Ptr<RepeatingParsingExecutor::IParsingAnalyzer> RepeatingParsingExecutor::GetAnalyzer()
|
|
{
|
|
return analyzer;
|
|
}
|
|
|
|
vint RepeatingParsingExecutor::GetTokenIndex(const WString& tokenName)
|
|
{
|
|
vint index=tokenIndexMap.Keys().IndexOf(tokenName);
|
|
return index==-1?-1:tokenIndexMap.Values()[index];
|
|
}
|
|
|
|
vint RepeatingParsingExecutor::GetSemanticId(const WString& name)
|
|
{
|
|
return semanticIndexMap.IndexOf(name);
|
|
}
|
|
|
|
WString RepeatingParsingExecutor::GetSemanticName(vint id)
|
|
{
|
|
return 0<=id&&id<semanticIndexMap.Count()?semanticIndexMap[id]:L"";
|
|
}
|
|
|
|
const RepeatingParsingExecutor::TokenMetaData& RepeatingParsingExecutor::GetTokenMetaData(vint regexTokenIndex)
|
|
{
|
|
return tokenMetaDatas[regexTokenIndex];
|
|
}
|
|
|
|
const RepeatingParsingExecutor::FieldMetaData& RepeatingParsingExecutor::GetFieldMetaData(const WString& type, const WString& field)
|
|
{
|
|
return fieldMetaDatas[FieldDesc(type, field)];
|
|
}
|
|
|
|
Ptr<parsing::tabling::ParsingTable::AttributeInfo> RepeatingParsingExecutor::GetAttribute(vint index, const WString& name, vint argumentCount)
|
|
{
|
|
if(index!=-1)
|
|
{
|
|
Ptr<ParsingTable::AttributeInfo> att=grammarParser->GetTable()->GetAttributeInfo(index)->FindFirst(name);
|
|
if(att && (argumentCount==-1 || att->arguments.Count()==argumentCount))
|
|
{
|
|
return att;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
Ptr<parsing::tabling::ParsingTable::AttributeInfo> RepeatingParsingExecutor::GetColorAttribute(vint index)
|
|
{
|
|
return GetAttribute(index, L"Color", 1);
|
|
}
|
|
|
|
Ptr<parsing::tabling::ParsingTable::AttributeInfo> RepeatingParsingExecutor::GetContextColorAttribute(vint index)
|
|
{
|
|
return GetAttribute(index, L"ContextColor", 0);
|
|
}
|
|
|
|
Ptr<parsing::tabling::ParsingTable::AttributeInfo> RepeatingParsingExecutor::GetSemanticAttribute(vint index)
|
|
{
|
|
return GetAttribute(index, L"Semantic", -1);
|
|
}
|
|
|
|
Ptr<parsing::tabling::ParsingTable::AttributeInfo> RepeatingParsingExecutor::GetCandidateAttribute(vint index)
|
|
{
|
|
return GetAttribute(index, L"Candidate", 0);
|
|
}
|
|
|
|
Ptr<parsing::tabling::ParsingTable::AttributeInfo> RepeatingParsingExecutor::GetAutoCompleteAttribute(vint index)
|
|
{
|
|
return GetAttribute(index, L"AutoComplete", 0);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\CONTROLS\TOOLSTRIPPACKAGE\GUIMENUCONTROLS.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace controls
|
|
{
|
|
using namespace compositions;
|
|
|
|
/***********************************************************************
|
|
IGuiMenuService
|
|
***********************************************************************/
|
|
|
|
const wchar_t* const IGuiMenuService::Identifier = L"vl::presentation::controls::IGuiMenuService";
|
|
const wchar_t* const IGuiMenuDropdownProvider::Identifier = L"vl::presentation::controls::IGuiMenuDropdownProvider";
|
|
|
|
IGuiMenuService::IGuiMenuService()
|
|
:openingMenu(0)
|
|
{
|
|
}
|
|
|
|
void IGuiMenuService::MenuItemExecuted()
|
|
{
|
|
if(openingMenu)
|
|
{
|
|
openingMenu->Hide();
|
|
}
|
|
if(GetParentMenuService())
|
|
{
|
|
GetParentMenuService()->MenuItemExecuted();
|
|
}
|
|
}
|
|
|
|
GuiMenu* IGuiMenuService::GetOpeningMenu()
|
|
{
|
|
return openingMenu;
|
|
}
|
|
|
|
void IGuiMenuService::MenuOpened(GuiMenu* menu)
|
|
{
|
|
if(openingMenu!=menu && openingMenu)
|
|
{
|
|
openingMenu->Hide();
|
|
}
|
|
openingMenu=menu;
|
|
}
|
|
|
|
void IGuiMenuService::MenuClosed(GuiMenu* menu)
|
|
{
|
|
if(openingMenu==menu)
|
|
{
|
|
openingMenu=0;
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiMenu
|
|
***********************************************************************/
|
|
|
|
void GuiMenu::BeforeControlTemplateUninstalled_()
|
|
{
|
|
}
|
|
|
|
void GuiMenu::AfterControlTemplateInstalled_(bool initialize)
|
|
{
|
|
}
|
|
|
|
IGuiMenuService* GuiMenu::GetParentMenuService()
|
|
{
|
|
return parentMenuService;
|
|
}
|
|
|
|
IGuiMenuService::Direction GuiMenu::GetPreferredDirection()
|
|
{
|
|
return IGuiMenuService::Vertical;
|
|
}
|
|
|
|
bool GuiMenu::IsActiveState()
|
|
{
|
|
return true;
|
|
}
|
|
|
|
bool GuiMenu::IsSubMenuActivatedByMouseDown()
|
|
{
|
|
return false;
|
|
}
|
|
|
|
void GuiMenu::MenuItemExecuted()
|
|
{
|
|
IGuiMenuService::MenuItemExecuted();
|
|
Hide();
|
|
}
|
|
|
|
void GuiMenu::OnWindowOpened(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
if(parentMenuService)
|
|
{
|
|
parentMenuService->MenuOpened(this);
|
|
}
|
|
}
|
|
|
|
void GuiMenu::OnDeactivatedAltHost()
|
|
{
|
|
if(hideOnDeactivateAltHost)
|
|
{
|
|
Hide();
|
|
}
|
|
GuiPopup::OnDeactivatedAltHost();
|
|
}
|
|
|
|
void GuiMenu::MouseClickedOnOtherWindow(GuiWindow* window)
|
|
{
|
|
GuiMenu* targetMenu=dynamic_cast<GuiMenu*>(window);
|
|
if(!targetMenu)
|
|
{
|
|
Hide();
|
|
}
|
|
}
|
|
|
|
void GuiMenu::OnWindowClosed(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
if(parentMenuService)
|
|
{
|
|
parentMenuService->MenuClosed(this);
|
|
GuiMenu* openingSubMenu=GetOpeningMenu();
|
|
if(openingSubMenu)
|
|
{
|
|
openingSubMenu->Hide();
|
|
}
|
|
}
|
|
}
|
|
|
|
GuiMenu::GuiMenu(theme::ThemeName themeName, GuiControl* _owner)
|
|
:GuiPopup(themeName)
|
|
, owner(_owner)
|
|
, parentMenuService(0)
|
|
{
|
|
GetNativeWindow()->SetAlwaysPassFocusToParent(true);
|
|
UpdateMenuService();
|
|
WindowOpened.AttachMethod(this, &GuiMenu::OnWindowOpened);
|
|
WindowClosed.AttachMethod(this, &GuiMenu::OnWindowClosed);
|
|
}
|
|
|
|
GuiMenu::~GuiMenu()
|
|
{
|
|
}
|
|
|
|
void GuiMenu::UpdateMenuService()
|
|
{
|
|
if(owner)
|
|
{
|
|
parentMenuService=owner->QueryTypedService<IGuiMenuService>();
|
|
}
|
|
}
|
|
|
|
IDescriptable* GuiMenu::QueryService(const WString& identifier)
|
|
{
|
|
if(identifier==IGuiMenuService::Identifier)
|
|
{
|
|
return (IGuiMenuService*)this;
|
|
}
|
|
else
|
|
{
|
|
return GuiPopup::QueryService(identifier);
|
|
}
|
|
}
|
|
|
|
bool GuiMenu::GetHideOnDeactivateAltHost()
|
|
{
|
|
return hideOnDeactivateAltHost;
|
|
}
|
|
|
|
void GuiMenu::SetHideOnDeactivateAltHost(bool value)
|
|
{
|
|
hideOnDeactivateAltHost = value;
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiMenuBar
|
|
***********************************************************************/
|
|
|
|
IGuiMenuService* GuiMenuBar::GetParentMenuService()
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
IGuiMenuService::Direction GuiMenuBar::GetPreferredDirection()
|
|
{
|
|
return IGuiMenuService::Horizontal;
|
|
}
|
|
|
|
bool GuiMenuBar::IsActiveState()
|
|
{
|
|
return GetOpeningMenu()!=0;
|
|
}
|
|
|
|
bool GuiMenuBar::IsSubMenuActivatedByMouseDown()
|
|
{
|
|
return true;
|
|
}
|
|
|
|
GuiMenuBar::GuiMenuBar(theme::ThemeName themeName)
|
|
:GuiControl(themeName)
|
|
{
|
|
}
|
|
|
|
GuiMenuBar::~GuiMenuBar()
|
|
{
|
|
}
|
|
|
|
IDescriptable* GuiMenuBar::QueryService(const WString& identifier)
|
|
{
|
|
if(identifier==IGuiMenuService::Identifier)
|
|
{
|
|
return (IGuiMenuService*)this;
|
|
}
|
|
else
|
|
{
|
|
return GuiControl::QueryService(identifier);
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiMenuButton
|
|
***********************************************************************/
|
|
|
|
void GuiMenuButton::BeforeControlTemplateUninstalled_()
|
|
{
|
|
auto host = GetSubMenuHost();
|
|
host->Clicked.Detach(hostClickedHandler);
|
|
host->GetBoundsComposition()->GetEventReceiver()->mouseEnter.Detach(hostMouseEnterHandler);
|
|
|
|
hostClickedHandler = nullptr;
|
|
hostMouseEnterHandler = nullptr;
|
|
}
|
|
|
|
void GuiMenuButton::AfterControlTemplateInstalled_(bool initialize)
|
|
{
|
|
auto ct = GetControlTemplateObject(true);
|
|
auto host = GetSubMenuHost();
|
|
|
|
ct->SetSubMenuOpening(GetSubMenuOpening());
|
|
ct->SetLargeImage(largeImage);
|
|
ct->SetImage(image);
|
|
ct->SetShortcutText(shortcutText);
|
|
ct->SetSubMenuExisting(subMenu != nullptr);
|
|
|
|
hostClickedHandler = host->Clicked.AttachMethod(this, &GuiMenuButton::OnClicked);
|
|
hostMouseEnterHandler = host->GetBoundsComposition()->GetEventReceiver()->mouseEnter.AttachMethod(this, &GuiMenuButton::OnMouseEnter);
|
|
}
|
|
|
|
GuiButton* GuiMenuButton::GetSubMenuHost()
|
|
{
|
|
GuiButton* button = GetControlTemplateObject(true)->GetSubMenuHost();
|
|
return button ? button : this;
|
|
}
|
|
|
|
bool GuiMenuButton::OpenSubMenuInternal()
|
|
{
|
|
if (!GetSubMenuOpening())
|
|
{
|
|
if (ownerMenuService)
|
|
{
|
|
GuiMenu* openingSiblingMenu = ownerMenuService->GetOpeningMenu();
|
|
if (openingSiblingMenu)
|
|
{
|
|
openingSiblingMenu->Hide();
|
|
}
|
|
}
|
|
|
|
BeforeSubMenuOpening.Execute(GetNotifyEventArguments());
|
|
if (subMenu)
|
|
{
|
|
subMenu->SetClientSize(preferredMenuClientSize);
|
|
IGuiMenuService::Direction direction = GetSubMenuDirection();
|
|
subMenu->ShowPopup(GetSubMenuHost(), direction == IGuiMenuService::Horizontal);
|
|
AfterSubMenuOpening.Execute(GetNotifyEventArguments());
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void GuiMenuButton::OnParentLineChanged()
|
|
{
|
|
GuiButton::OnParentLineChanged();
|
|
ownerMenuService=QueryTypedService<IGuiMenuService>();
|
|
if(ownerMenuService)
|
|
{
|
|
SetClickOnMouseUp(!ownerMenuService->IsSubMenuActivatedByMouseDown());
|
|
}
|
|
if(subMenu)
|
|
{
|
|
subMenu->UpdateMenuService();
|
|
}
|
|
}
|
|
|
|
compositions::IGuiAltActionHost* GuiMenuButton::GetActivatingAltHost()
|
|
{
|
|
if (subMenu)
|
|
{
|
|
return subMenu->QueryTypedService<IGuiAltActionHost>();
|
|
}
|
|
else
|
|
{
|
|
return GuiSelectableButton::GetActivatingAltHost();
|
|
}
|
|
}
|
|
|
|
void GuiMenuButton::OnSubMenuWindowOpened(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
SubMenuOpeningChanged.Execute(GetNotifyEventArguments());
|
|
GetControlTemplateObject(true)->SetSubMenuOpening(true);
|
|
}
|
|
|
|
void GuiMenuButton::OnSubMenuWindowClosed(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
SubMenuOpeningChanged.Execute(GetNotifyEventArguments());
|
|
GetControlTemplateObject(true)->SetSubMenuOpening(false);
|
|
}
|
|
|
|
void GuiMenuButton::OnMouseEnter(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
if(GetVisuallyEnabled())
|
|
{
|
|
if(cascadeAction && ownerMenuService && ownerMenuService->IsActiveState())
|
|
{
|
|
OpenSubMenuInternal();
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiMenuButton::OnClicked(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
if(GetVisuallyEnabled())
|
|
{
|
|
if(!OpenSubMenuInternal() && ownerMenuService)
|
|
{
|
|
ownerMenuService->MenuItemExecuted();
|
|
}
|
|
}
|
|
}
|
|
|
|
IGuiMenuService::Direction GuiMenuButton::GetSubMenuDirection()
|
|
{
|
|
return ownerMenuService?ownerMenuService->GetPreferredDirection():IGuiMenuService::Horizontal;
|
|
}
|
|
|
|
void GuiMenuButton::DetachSubMenu()
|
|
{
|
|
if (subMenu)
|
|
{
|
|
subMenu->WindowOpened.Detach(subMenuWindowOpenedHandler);
|
|
subMenu->WindowClosed.Detach(subMenuWindowClosedHandler);
|
|
if (ownedSubMenu)
|
|
{
|
|
delete subMenu;
|
|
}
|
|
}
|
|
}
|
|
|
|
GuiMenu* GuiMenuButton::ProvideDropdownMenu()
|
|
{
|
|
return GetSubMenu();
|
|
}
|
|
|
|
GuiMenuButton::GuiMenuButton(theme::ThemeName themeName)
|
|
:GuiSelectableButton(themeName)
|
|
,subMenu(0)
|
|
,ownedSubMenu(false)
|
|
,ownerMenuService(0)
|
|
,cascadeAction(true)
|
|
{
|
|
SetAutoSelection(false);
|
|
BeforeSubMenuOpening.SetAssociatedComposition(boundsComposition);
|
|
SubMenuOpeningChanged.SetAssociatedComposition(boundsComposition);
|
|
LargeImageChanged.SetAssociatedComposition(boundsComposition);
|
|
ImageChanged.SetAssociatedComposition(boundsComposition);
|
|
ShortcutTextChanged.SetAssociatedComposition(boundsComposition);
|
|
}
|
|
|
|
GuiMenuButton::~GuiMenuButton()
|
|
{
|
|
DetachSubMenu();
|
|
}
|
|
|
|
Ptr<GuiImageData> GuiMenuButton::GetLargeImage()
|
|
{
|
|
return largeImage;
|
|
}
|
|
|
|
void GuiMenuButton::SetLargeImage(Ptr<GuiImageData> value)
|
|
{
|
|
if (largeImage != value)
|
|
{
|
|
largeImage = value;
|
|
GetControlTemplateObject(true)->SetLargeImage(largeImage);
|
|
LargeImageChanged.Execute(GetNotifyEventArguments());
|
|
}
|
|
}
|
|
|
|
Ptr<GuiImageData> GuiMenuButton::GetImage()
|
|
{
|
|
return image;
|
|
}
|
|
|
|
void GuiMenuButton::SetImage(Ptr<GuiImageData> value)
|
|
{
|
|
if (image != value)
|
|
{
|
|
image = value;
|
|
GetControlTemplateObject(true)->SetImage(image);
|
|
ImageChanged.Execute(GetNotifyEventArguments());
|
|
}
|
|
}
|
|
|
|
const WString& GuiMenuButton::GetShortcutText()
|
|
{
|
|
return shortcutText;
|
|
}
|
|
|
|
void GuiMenuButton::SetShortcutText(const WString& value)
|
|
{
|
|
if (shortcutText != value)
|
|
{
|
|
shortcutText = value;
|
|
GetControlTemplateObject(true)->SetShortcutText(shortcutText);
|
|
ShortcutTextChanged.Execute(GetNotifyEventArguments());
|
|
}
|
|
}
|
|
|
|
bool GuiMenuButton::IsSubMenuExists()
|
|
{
|
|
return subMenu!=0;
|
|
}
|
|
|
|
GuiMenu* GuiMenuButton::GetSubMenu()
|
|
{
|
|
return subMenu;
|
|
}
|
|
|
|
GuiMenu* GuiMenuButton::CreateSubMenu(TemplateProperty<templates::GuiMenuTemplate> subMenuTemplate)
|
|
{
|
|
if (!subMenu)
|
|
{
|
|
GuiMenu* newSubMenu = new GuiMenu(theme::ThemeName::Menu, this);
|
|
newSubMenu->SetControlTemplate(subMenuTemplate ? subMenuTemplate : GetControlTemplateObject(true)->GetSubMenuTemplate());
|
|
SetSubMenu(newSubMenu, true);
|
|
}
|
|
return subMenu;
|
|
}
|
|
|
|
void GuiMenuButton::SetSubMenu(GuiMenu* value, bool owned)
|
|
{
|
|
if(subMenu)
|
|
{
|
|
DetachSubMenu();
|
|
}
|
|
subMenu=value;
|
|
ownedSubMenu=owned;
|
|
if(subMenu)
|
|
{
|
|
subMenuWindowOpenedHandler = subMenu->WindowOpened.AttachMethod(this, &GuiMenuButton::OnSubMenuWindowOpened);
|
|
subMenuWindowClosedHandler = subMenu->WindowClosed.AttachMethod(this, &GuiMenuButton::OnSubMenuWindowClosed);
|
|
}
|
|
GetControlTemplateObject(true)->SetSubMenuExisting(subMenu != nullptr);
|
|
}
|
|
|
|
void GuiMenuButton::DestroySubMenu()
|
|
{
|
|
if(subMenu)
|
|
{
|
|
DetachSubMenu();
|
|
subMenu=0;
|
|
ownedSubMenu=false;
|
|
GetControlTemplateObject(true)->SetSubMenuExisting(false);
|
|
}
|
|
}
|
|
|
|
bool GuiMenuButton::GetOwnedSubMenu()
|
|
{
|
|
return subMenu && ownedSubMenu;
|
|
}
|
|
|
|
bool GuiMenuButton::GetSubMenuOpening()
|
|
{
|
|
if(subMenu)
|
|
{
|
|
return subMenu->GetOpening();
|
|
}
|
|
else
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
void GuiMenuButton::SetSubMenuOpening(bool value)
|
|
{
|
|
if (subMenu && subMenu->GetOpening() != value)
|
|
{
|
|
if (value)
|
|
{
|
|
OpenSubMenuInternal();
|
|
}
|
|
else
|
|
{
|
|
subMenu->Close();
|
|
}
|
|
}
|
|
}
|
|
|
|
Size GuiMenuButton::GetPreferredMenuClientSize()
|
|
{
|
|
return preferredMenuClientSize;
|
|
}
|
|
|
|
void GuiMenuButton::SetPreferredMenuClientSize(Size value)
|
|
{
|
|
preferredMenuClientSize=value;
|
|
}
|
|
|
|
bool GuiMenuButton::GetCascadeAction()
|
|
{
|
|
return cascadeAction;
|
|
}
|
|
|
|
void GuiMenuButton::SetCascadeAction(bool value)
|
|
{
|
|
cascadeAction=value;
|
|
}
|
|
|
|
IDescriptable* GuiMenuButton::QueryService(const WString& identifier)
|
|
{
|
|
if (identifier == IGuiMenuDropdownProvider::Identifier)
|
|
{
|
|
return (IGuiMenuDropdownProvider*)this;
|
|
}
|
|
else
|
|
{
|
|
return GuiSelectableButton::QueryService(identifier);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\CONTROLS\TOOLSTRIPPACKAGE\GUIRIBBONCONTROLS.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace controls
|
|
{
|
|
using namespace reflection::description;
|
|
using namespace collections;
|
|
using namespace compositions;
|
|
using namespace theme;
|
|
using namespace templates;
|
|
|
|
/***********************************************************************
|
|
GuiRibbonTab
|
|
***********************************************************************/
|
|
|
|
void GuiRibbonTab::BeforeControlTemplateUninstalled_()
|
|
{
|
|
auto ct = GetControlTemplateObject(false);
|
|
if (!ct) return;
|
|
|
|
if (auto bhc = ct->GetBeforeHeadersContainer())
|
|
{
|
|
bhc->RemoveChild(beforeHeaders);
|
|
}
|
|
if (auto ahc = ct->GetAfterHeadersContainer())
|
|
{
|
|
ahc->RemoveChild(afterHeaders);
|
|
}
|
|
}
|
|
|
|
void GuiRibbonTab::AfterControlTemplateInstalled_(bool initialize)
|
|
{
|
|
auto ct = GetControlTemplateObject(true);
|
|
if (auto bhc = ct->GetBeforeHeadersContainer())
|
|
{
|
|
bhc->AddChild(beforeHeaders);
|
|
}
|
|
if (auto ahc = ct->GetAfterHeadersContainer())
|
|
{
|
|
ahc->AddChild(afterHeaders);
|
|
}
|
|
}
|
|
|
|
GuiRibbonTab::GuiRibbonTab(theme::ThemeName themeName)
|
|
:GuiTab(themeName)
|
|
{
|
|
beforeHeaders = new GuiBoundsComposition();
|
|
beforeHeaders->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
beforeHeaders->SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
|
|
afterHeaders = new GuiBoundsComposition();
|
|
afterHeaders->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
afterHeaders->SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
}
|
|
|
|
GuiRibbonTab::~GuiRibbonTab()
|
|
{
|
|
if (!beforeHeaders->GetParent())
|
|
{
|
|
SafeDeleteComposition(beforeHeaders);
|
|
}
|
|
if (!afterHeaders->GetParent())
|
|
{
|
|
SafeDeleteComposition(afterHeaders);
|
|
}
|
|
}
|
|
|
|
compositions::GuiGraphicsComposition* GuiRibbonTab::GetBeforeHeaders()
|
|
{
|
|
return beforeHeaders;
|
|
}
|
|
|
|
compositions::GuiGraphicsComposition* GuiRibbonTab::GetAfterHeaders()
|
|
{
|
|
return afterHeaders;
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiRibbonGroupCollection
|
|
***********************************************************************/
|
|
|
|
bool GuiRibbonGroupCollection::QueryInsert(vint index, GuiRibbonGroup* const& value)
|
|
{
|
|
return !value->GetBoundsComposition()->GetParent();
|
|
}
|
|
|
|
void GuiRibbonGroupCollection::AfterInsert(vint index, GuiRibbonGroup* const& value)
|
|
{
|
|
value->GetBoundsComposition()->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
|
|
auto item = new GuiStackItemComposition();
|
|
item->AddChild(value->GetBoundsComposition());
|
|
|
|
tabPage->stack->InsertStackItem(index, item);
|
|
}
|
|
|
|
void GuiRibbonGroupCollection::AfterRemove(vint index, vint count)
|
|
{
|
|
for (vint i = 0; i < count; i++)
|
|
{
|
|
auto item = tabPage->stack->GetStackItems()[index];
|
|
tabPage->stack->RemoveChild(item);
|
|
|
|
item->RemoveChild(item->Children()[0]);
|
|
delete item;
|
|
}
|
|
}
|
|
|
|
GuiRibbonGroupCollection::GuiRibbonGroupCollection(GuiRibbonTabPage* _tabPage)
|
|
:tabPage(_tabPage)
|
|
{
|
|
}
|
|
|
|
GuiRibbonGroupCollection::~GuiRibbonGroupCollection()
|
|
{
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiRibbonTabPage
|
|
***********************************************************************/
|
|
|
|
GuiRibbonTabPage::GuiRibbonTabPage(theme::ThemeName themeName)
|
|
:GuiTabPage(themeName)
|
|
, groups(this)
|
|
{
|
|
stack = new GuiStackComposition();
|
|
stack->SetDirection(GuiStackComposition::Horizontal);
|
|
stack->SetAlignmentToParent(Margin(2, 2, 2, 2));
|
|
stack->SetPadding(2);
|
|
stack->SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
|
|
responsiveStack = new GuiResponsiveStackComposition();
|
|
responsiveStack->SetDirection(ResponsiveDirection::Horizontal);
|
|
responsiveStack->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
responsiveStack->AddChild(stack);
|
|
|
|
responsiveContainer = new GuiResponsiveContainerComposition();
|
|
responsiveContainer->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
responsiveContainer->SetResponsiveTarget(responsiveStack);
|
|
|
|
containerComposition->AddChild(responsiveContainer);
|
|
|
|
HighlightedChanged.SetAssociatedComposition(boundsComposition);
|
|
}
|
|
|
|
GuiRibbonTabPage::~GuiRibbonTabPage()
|
|
{
|
|
}
|
|
|
|
bool GuiRibbonTabPage::GetHighlighted()
|
|
{
|
|
return highlighted;
|
|
}
|
|
|
|
void GuiRibbonTabPage::SetHighlighted(bool value)
|
|
{
|
|
if (highlighted != value)
|
|
{
|
|
highlighted = value;
|
|
HighlightedChanged.Execute(GetNotifyEventArguments());
|
|
}
|
|
}
|
|
|
|
collections::ObservableListBase<GuiRibbonGroup*>& GuiRibbonTabPage::GetGroups()
|
|
{
|
|
return groups;
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiRibbonGroupItemCollection
|
|
***********************************************************************/
|
|
|
|
bool GuiRibbonGroupItemCollection::QueryInsert(vint index, GuiControl* const& value)
|
|
{
|
|
return !value->GetBoundsComposition()->GetParent();
|
|
}
|
|
|
|
void GuiRibbonGroupItemCollection::AfterInsert(vint index, GuiControl* const& value)
|
|
{
|
|
value->GetBoundsComposition()->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
|
|
auto item = new GuiStackItemComposition();
|
|
item->AddChild(value->GetBoundsComposition());
|
|
|
|
group->stack->InsertStackItem(index, item);
|
|
}
|
|
|
|
void GuiRibbonGroupItemCollection::AfterRemove(vint index, vint count)
|
|
{
|
|
for (vint i = 0; i < count; i++)
|
|
{
|
|
auto item = group->stack->GetStackItems()[index];
|
|
group->stack->RemoveChild(item);
|
|
|
|
item->RemoveChild(item->Children()[0]);
|
|
delete item;
|
|
}
|
|
}
|
|
|
|
GuiRibbonGroupItemCollection::GuiRibbonGroupItemCollection(GuiRibbonGroup* _group)
|
|
:group(_group)
|
|
{
|
|
}
|
|
|
|
GuiRibbonGroupItemCollection::~GuiRibbonGroupItemCollection()
|
|
{
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiRibbonGroup::CommandExecutor
|
|
***********************************************************************/
|
|
|
|
GuiRibbonGroup::CommandExecutor::CommandExecutor(GuiRibbonGroup* _group)
|
|
:group(_group)
|
|
{
|
|
}
|
|
|
|
GuiRibbonGroup::CommandExecutor::~CommandExecutor()
|
|
{
|
|
}
|
|
|
|
void GuiRibbonGroup::CommandExecutor::NotifyExpandButtonClicked()
|
|
{
|
|
group->ExpandButtonClicked.Execute(group->GetNotifyEventArguments());
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiRibbonGroupMenu
|
|
***********************************************************************/
|
|
|
|
class GuiRibbonGroupMenu : public GuiMenu, public Description<GuiRibbonGroupMenu>
|
|
{
|
|
private:
|
|
IGuiMenuService::Direction GetPreferredDirection()override
|
|
{
|
|
return IGuiMenuService::Horizontal;
|
|
}
|
|
|
|
bool IsActiveState()override
|
|
{
|
|
return false;
|
|
}
|
|
|
|
public:
|
|
GuiRibbonGroupMenu(theme::ThemeName themeName, GuiControl* _owner)
|
|
:GuiMenu(themeName, _owner)
|
|
{
|
|
}
|
|
};
|
|
|
|
/***********************************************************************
|
|
GuiRibbonGroup
|
|
***********************************************************************/
|
|
|
|
void GuiRibbonGroup::BeforeControlTemplateUninstalled_()
|
|
{
|
|
auto ct = GetControlTemplateObject(false);
|
|
if (!ct) return;
|
|
|
|
ct->SetCommands(nullptr);
|
|
}
|
|
|
|
void GuiRibbonGroup::AfterControlTemplateInstalled_(bool initialize)
|
|
{
|
|
auto ct = GetControlTemplateObject(true);
|
|
ct->SetExpandable(expandable);
|
|
ct->SetCollapsed(responsiveView->GetCurrentView() == responsiveFixedButton);
|
|
ct->SetCommands(commandExecutor.Obj());
|
|
dropdownButton->SetControlTemplate(ct->GetLargeDropdownButtonTemplate());
|
|
dropdownMenu->SetControlTemplate(ct->GetSubMenuTemplate());
|
|
}
|
|
|
|
void GuiRibbonGroup::OnBoundsChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
dropdownMenu->GetBoundsComposition()->SetPreferredMinSize(Size(0, containerComposition->GetBounds().Height()));
|
|
}
|
|
|
|
void GuiRibbonGroup::OnTextChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
dropdownButton->SetText(GetText());
|
|
}
|
|
|
|
void GuiRibbonGroup::OnBeforeSubMenuOpening(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
if (responsiveView->GetViews().Contains(responsiveFixedButton))
|
|
{
|
|
auto currentDropdown = dropdownMenu;
|
|
if (items.Count() == 1)
|
|
{
|
|
if (auto provider = items[0]->QueryTypedService<IGuiMenuDropdownProvider>())
|
|
{
|
|
if (auto menu = provider->ProvideDropdownMenu())
|
|
{
|
|
currentDropdown = menu;
|
|
}
|
|
}
|
|
}
|
|
dropdownButton->SetSubMenu(currentDropdown, false);
|
|
}
|
|
}
|
|
|
|
void GuiRibbonGroup::OnBeforeSwitchingView(compositions::GuiGraphicsComposition* sender, compositions::GuiItemEventArgs& arguments)
|
|
{
|
|
if (auto ct = GetControlTemplateObject(false))
|
|
{
|
|
ct->SetCollapsed(arguments.itemIndex == 1);
|
|
}
|
|
|
|
if (arguments.itemIndex == 0)
|
|
{
|
|
while (responsiveStack->LevelDown());
|
|
dropdownMenu->GetContainerComposition()->RemoveChild(stack);
|
|
responsiveStack->AddChild(stack);
|
|
dropdownButton->SetSubMenu(nullptr, false);
|
|
}
|
|
else
|
|
{
|
|
while (responsiveStack->LevelUp());
|
|
responsiveStack->RemoveChild(stack);
|
|
dropdownMenu->GetContainerComposition()->AddChild(stack);
|
|
}
|
|
}
|
|
|
|
GuiRibbonGroup::GuiRibbonGroup(theme::ThemeName themeName)
|
|
:GuiControl(themeName)
|
|
, items(this)
|
|
{
|
|
commandExecutor = new CommandExecutor(this);
|
|
{
|
|
stack = new GuiStackComposition();
|
|
stack->SetDirection(GuiStackComposition::Horizontal);
|
|
stack->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
stack->SetPadding(2);
|
|
stack->SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
|
|
responsiveStack = new GuiResponsiveStackComposition();
|
|
responsiveStack->SetDirection(ResponsiveDirection::Horizontal);
|
|
responsiveStack->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
responsiveStack->AddChild(stack);
|
|
}
|
|
{
|
|
dropdownButton = new GuiToolstripButton(theme::ThemeName::RibbonLargeDropdownButton);
|
|
dropdownButton->GetBoundsComposition()->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
|
|
responsiveFixedButton = new GuiResponsiveFixedComposition();
|
|
responsiveFixedButton->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
responsiveFixedButton->AddChild(dropdownButton->GetBoundsComposition());
|
|
|
|
dropdownMenu = new GuiRibbonGroupMenu(theme::ThemeName::Menu, dropdownButton);
|
|
}
|
|
|
|
responsiveView = new GuiResponsiveViewComposition();
|
|
responsiveView->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
responsiveView->GetViews().Add(responsiveStack);
|
|
|
|
containerComposition->AddChild(responsiveView);
|
|
|
|
ExpandableChanged.SetAssociatedComposition(boundsComposition);
|
|
ExpandButtonClicked.SetAssociatedComposition(boundsComposition);
|
|
LargeImageChanged.SetAssociatedComposition(boundsComposition);
|
|
|
|
TextChanged.AttachMethod(this, &GuiRibbonGroup::OnTextChanged);
|
|
boundsComposition->BoundsChanged.AttachMethod(this, &GuiRibbonGroup::OnBoundsChanged);
|
|
responsiveView->BeforeSwitchingView.AttachMethod(this, &GuiRibbonGroup::OnBeforeSwitchingView);
|
|
dropdownButton->BeforeSubMenuOpening.AttachMethod(this, &GuiRibbonGroup::OnBeforeSubMenuOpening);
|
|
}
|
|
|
|
GuiRibbonGroup::~GuiRibbonGroup()
|
|
{
|
|
if (!responsiveView->GetViews().Contains(responsiveFixedButton))
|
|
{
|
|
SafeDeleteComposition(responsiveFixedButton);
|
|
}
|
|
delete dropdownMenu;
|
|
}
|
|
|
|
bool GuiRibbonGroup::GetExpandable()
|
|
{
|
|
return expandable;
|
|
}
|
|
|
|
void GuiRibbonGroup::SetExpandable(bool value)
|
|
{
|
|
if (expandable != value)
|
|
{
|
|
expandable = value;
|
|
GetControlTemplateObject(true)->SetExpandable(expandable);
|
|
ExpandableChanged.Execute(GetNotifyEventArguments());
|
|
}
|
|
}
|
|
|
|
Ptr<GuiImageData> GuiRibbonGroup::GetLargeImage()
|
|
{
|
|
return largeImage;
|
|
}
|
|
|
|
void GuiRibbonGroup::SetLargeImage(Ptr<GuiImageData> value)
|
|
{
|
|
if (largeImage != value)
|
|
{
|
|
largeImage = value;
|
|
dropdownButton->SetLargeImage(value);
|
|
LargeImageChanged.Execute(GetNotifyEventArguments());
|
|
|
|
if (value)
|
|
{
|
|
if (!responsiveView->GetViews().Contains(responsiveFixedButton))
|
|
{
|
|
responsiveView->GetViews().Add(responsiveFixedButton);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (responsiveView->GetViews().Contains(responsiveFixedButton))
|
|
{
|
|
responsiveView->GetViews().Remove(responsiveFixedButton);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
collections::ObservableListBase<GuiControl*>& GuiRibbonGroup::GetItems()
|
|
{
|
|
return items;
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiRibbonIconLabel
|
|
***********************************************************************/
|
|
|
|
void GuiRibbonIconLabel::BeforeControlTemplateUninstalled_()
|
|
{
|
|
}
|
|
|
|
void GuiRibbonIconLabel::AfterControlTemplateInstalled_(bool initialize)
|
|
{
|
|
auto ct = GetControlTemplateObject(true);
|
|
ct->SetImage(image);
|
|
}
|
|
|
|
GuiRibbonIconLabel::GuiRibbonIconLabel(theme::ThemeName themeName)
|
|
:GuiControl(themeName)
|
|
{
|
|
ImageChanged.SetAssociatedComposition(boundsComposition);
|
|
}
|
|
|
|
GuiRibbonIconLabel::~GuiRibbonIconLabel()
|
|
{
|
|
}
|
|
|
|
Ptr<GuiImageData> GuiRibbonIconLabel::GetImage()
|
|
{
|
|
return image;
|
|
}
|
|
|
|
void GuiRibbonIconLabel::SetImage(Ptr<GuiImageData> value)
|
|
{
|
|
if (image != value)
|
|
{
|
|
image = value;
|
|
GetControlTemplateObject(true)->SetImage(image);
|
|
ImageChanged.Execute(GetNotifyEventArguments());
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiRibbonButtonsItemCollection
|
|
***********************************************************************/
|
|
|
|
bool GuiRibbonButtonsItemCollection::QueryInsert(vint index, GuiControl* const& value)
|
|
{
|
|
return !value->GetBoundsComposition()->GetParent();
|
|
}
|
|
|
|
void GuiRibbonButtonsItemCollection::AfterInsert(vint index, GuiControl* const& value)
|
|
{
|
|
buttons->responsiveView->GetSharedControls().Add(value);
|
|
buttons->SetButtonThemeName(buttons->responsiveView->GetCurrentView(), value);
|
|
|
|
for (vint i = 0; i < sizeof(buttons->views) / sizeof(*buttons->views); i++)
|
|
{
|
|
if (auto view = buttons->views[i])
|
|
{
|
|
auto stack = dynamic_cast<GuiStackComposition*>(view->Children()[0]);
|
|
|
|
auto shared = new GuiResponsiveSharedComposition();
|
|
shared->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
shared->SetShared(value);
|
|
|
|
auto item = new GuiStackItemComposition();
|
|
item->AddChild(shared);
|
|
|
|
stack->InsertStackItem(index, item);
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiRibbonButtonsItemCollection::BeforeRemove(vint index, GuiControl* const& value)
|
|
{
|
|
CHECK_FAIL(L"GuiRibbonButtonsItemCollection::BeforeRemove(vint, GuiControl* const&)#Controls are not allowed to be removed from GuiRibbonButtons.");
|
|
}
|
|
|
|
GuiRibbonButtonsItemCollection::GuiRibbonButtonsItemCollection(GuiRibbonButtons* _buttons)
|
|
:buttons(_buttons)
|
|
{
|
|
}
|
|
|
|
GuiRibbonButtonsItemCollection::~GuiRibbonButtonsItemCollection()
|
|
{
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiRibbonButtons
|
|
***********************************************************************/
|
|
|
|
void GuiRibbonButtons::BeforeControlTemplateUninstalled_()
|
|
{
|
|
}
|
|
|
|
void GuiRibbonButtons::AfterControlTemplateInstalled_(bool initialize)
|
|
{
|
|
FOREACH(GuiControl*, button, buttons)
|
|
{
|
|
SetButtonThemeName(responsiveView->GetCurrentView(), button);
|
|
}
|
|
}
|
|
|
|
void GuiRibbonButtons::OnBeforeSwitchingView(compositions::GuiGraphicsComposition* sender, compositions::GuiItemEventArgs& arguments)
|
|
{
|
|
FOREACH(GuiControl*, button, buttons)
|
|
{
|
|
SetButtonThemeName(responsiveView->GetViews()[arguments.itemIndex], button);
|
|
}
|
|
}
|
|
|
|
void GuiRibbonButtons::SetButtonThemeName(compositions::GuiResponsiveCompositionBase* fixed, GuiControl* button)
|
|
{
|
|
if (fixed && button)
|
|
{
|
|
auto themeName = button->GetControlThemeName();
|
|
vint type = -1;
|
|
switch (themeName)
|
|
{
|
|
case ThemeName::RibbonLargeButton:
|
|
case ThemeName::RibbonSmallButton:
|
|
case ThemeName::ToolstripButton:
|
|
type = 0;
|
|
break;
|
|
case ThemeName::RibbonLargeDropdownButton:
|
|
case ThemeName::RibbonSmallDropdownButton:
|
|
case ThemeName::ToolstripDropdownButton:
|
|
type = 1;
|
|
break;
|
|
case ThemeName::RibbonLargeSplitButton:
|
|
case ThemeName::RibbonSmallSplitButton:
|
|
case ThemeName::ToolstripSplitButton:
|
|
type = 2;
|
|
break;
|
|
case ThemeName::RibbonSmallIconLabel:
|
|
case ThemeName::RibbonIconLabel:
|
|
type = 3;
|
|
break;
|
|
default:;
|
|
}
|
|
|
|
if (type != -1)
|
|
{
|
|
ThemeName themeName = ThemeName::Unknown;
|
|
TemplateProperty<GuiToolstripButtonTemplate> controlTemplate;
|
|
|
|
if (fixed == views[(vint)RibbonButtonSize::Large])
|
|
{
|
|
switch (type)
|
|
{
|
|
case 0: themeName = ThemeName::RibbonLargeButton; break;
|
|
case 1: themeName = ThemeName::RibbonLargeDropdownButton; break;
|
|
case 2: themeName = ThemeName::RibbonLargeSplitButton; break;
|
|
case 3: themeName = ThemeName::RibbonSmallIconLabel; break;
|
|
}
|
|
}
|
|
else if (fixed == views[(vint)RibbonButtonSize::Small])
|
|
{
|
|
switch (type)
|
|
{
|
|
case 0: themeName = ThemeName::RibbonSmallButton; break;
|
|
case 1: themeName = ThemeName::RibbonSmallDropdownButton; break;
|
|
case 2: themeName = ThemeName::RibbonSmallSplitButton; break;
|
|
case 3: themeName = ThemeName::RibbonSmallIconLabel; break;
|
|
}
|
|
}
|
|
else if (fixed == views[(vint)RibbonButtonSize::Icon])
|
|
{
|
|
switch (type)
|
|
{
|
|
case 0: themeName = ThemeName::ToolstripButton; break;
|
|
case 1: themeName = ThemeName::ToolstripDropdownButton; break;
|
|
case 2: themeName = ThemeName::ToolstripSplitButton; break;
|
|
case 3: themeName = ThemeName::RibbonIconLabel; break;
|
|
}
|
|
}
|
|
|
|
if (auto ct = GetControlTemplateObject(false))
|
|
{
|
|
if (fixed == views[(vint)RibbonButtonSize::Large])
|
|
{
|
|
switch (type)
|
|
{
|
|
case 0: controlTemplate = ct->GetLargeButtonTemplate(); break;
|
|
case 1: controlTemplate = ct->GetLargeDropdownButtonTemplate(); break;
|
|
case 2: controlTemplate = ct->GetLargeSplitButtonTemplate(); break;
|
|
case 3: controlTemplate = ct->GetSmallIconLabelTemplate(); break;
|
|
}
|
|
}
|
|
else if (fixed == views[(vint)RibbonButtonSize::Small])
|
|
{
|
|
switch (type)
|
|
{
|
|
case 0: controlTemplate = ct->GetSmallButtonTemplate(); break;
|
|
case 1: controlTemplate = ct->GetSmallDropdownButtonTemplate(); break;
|
|
case 2: controlTemplate = ct->GetSmallSplitButtonTemplate(); break;
|
|
case 3: controlTemplate = ct->GetSmallIconLabelTemplate(); break;
|
|
}
|
|
}
|
|
else if (fixed == views[(vint)RibbonButtonSize::Icon])
|
|
{
|
|
switch (type)
|
|
{
|
|
case 0: controlTemplate = ct->GetIconButtonTemplate(); break;
|
|
case 1: controlTemplate = ct->GetIconDropdownButtonTemplate(); break;
|
|
case 2: controlTemplate = ct->GetIconSplitButtonTemplate(); break;
|
|
case 3: controlTemplate = ct->GetIconLabelTemplate(); break;
|
|
}
|
|
}
|
|
}
|
|
|
|
button->SetControlThemeNameAndTemplate(themeName, controlTemplate);
|
|
}
|
|
}
|
|
}
|
|
|
|
GuiRibbonButtons::GuiRibbonButtons(theme::ThemeName themeName, RibbonButtonSize _maxSize, RibbonButtonSize _minSize)
|
|
:GuiControl(themeName)
|
|
, maxSize(_maxSize)
|
|
, minSize(_minSize)
|
|
, buttons(this)
|
|
{
|
|
responsiveView = new GuiResponsiveViewComposition();
|
|
responsiveView->SetDirection(ResponsiveDirection::Horizontal);
|
|
responsiveView->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
responsiveView->BeforeSwitchingView.AttachMethod(this, &GuiRibbonButtons::OnBeforeSwitchingView);
|
|
|
|
auto installButton = [&](GuiTableComposition* table, vint row, vint column, GuiControl* buttonContainer)
|
|
{
|
|
auto shared = new GuiResponsiveSharedComposition();
|
|
shared->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
shared->SetShared(buttonContainer);
|
|
|
|
auto cell = new GuiCellComposition();
|
|
cell->SetSite(row, column, 1, 1);
|
|
cell->AddChild(shared);
|
|
|
|
table->AddChild(cell);
|
|
};
|
|
|
|
for (vint i = 0; i < sizeof(views) / sizeof(*views); i++)
|
|
{
|
|
if ((vint)maxSize <= i && i <= (vint)minSize)
|
|
{
|
|
auto stack = new GuiStackComposition();
|
|
stack->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
stack->SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
stack->SetDirection(i == 0 ? GuiStackComposition::Horizontal : GuiStackComposition::Vertical);
|
|
|
|
views[i] = new GuiResponsiveFixedComposition();
|
|
views[i]->AddChild(stack);
|
|
responsiveView->GetViews().Add(views[i]);
|
|
}
|
|
}
|
|
|
|
auto sharedSizeRootComposition = new GuiSharedSizeRootComposition();
|
|
sharedSizeRootComposition->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
sharedSizeRootComposition->SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
sharedSizeRootComposition->AddChild(responsiveView);
|
|
|
|
containerComposition->AddChild(sharedSizeRootComposition);
|
|
}
|
|
|
|
GuiRibbonButtons::~GuiRibbonButtons()
|
|
{
|
|
}
|
|
|
|
collections::ObservableListBase<GuiControl*>& GuiRibbonButtons::GetButtons()
|
|
{
|
|
return buttons;
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiRibbonToolstripsGroupCollection
|
|
***********************************************************************/
|
|
|
|
bool GuiRibbonToolstripsGroupCollection::QueryInsert(vint index, GuiToolstripGroup* const& value)
|
|
{
|
|
return !value->GetBoundsComposition()->GetParent();
|
|
}
|
|
|
|
void GuiRibbonToolstripsGroupCollection::AfterInsert(vint index, GuiToolstripGroup* const& value)
|
|
{
|
|
toolstrips->RearrangeToolstripGroups();
|
|
}
|
|
|
|
void GuiRibbonToolstripsGroupCollection::AfterRemove(vint index, vint count)
|
|
{
|
|
toolstrips->RearrangeToolstripGroups();
|
|
}
|
|
|
|
GuiRibbonToolstripsGroupCollection::GuiRibbonToolstripsGroupCollection(GuiRibbonToolstrips* _toolstrips)
|
|
:toolstrips(_toolstrips)
|
|
{
|
|
}
|
|
|
|
GuiRibbonToolstripsGroupCollection::~GuiRibbonToolstripsGroupCollection()
|
|
{
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiRibbonToolstrips
|
|
***********************************************************************/
|
|
|
|
#define ARRLEN(X) sizeof(X) / sizeof(*X)
|
|
|
|
void GuiRibbonToolstrips::BeforeControlTemplateUninstalled_()
|
|
{
|
|
}
|
|
|
|
void GuiRibbonToolstrips::AfterControlTemplateInstalled_(bool initialize)
|
|
{
|
|
auto ct = GetControlTemplateObject(true);
|
|
for (vint i = 0; i < ARRLEN(toolbars); i++)
|
|
{
|
|
toolbars[i]->SetControlTemplate(ct->GetToolbarTemplate());
|
|
}
|
|
}
|
|
|
|
void GuiRibbonToolstrips::OnBeforeSwitchingView(compositions::GuiGraphicsComposition* sender, compositions::GuiItemEventArgs& arguments)
|
|
{
|
|
RearrangeToolstripGroups(arguments.itemIndex);
|
|
}
|
|
|
|
void GuiRibbonToolstrips::RearrangeToolstripGroups(vint viewIndex)
|
|
{
|
|
static_assert(ARRLEN(longContainers) == 2, "");
|
|
static_assert(ARRLEN(shortContainers) == 3, "");
|
|
|
|
if (viewIndex == -1)
|
|
{
|
|
viewIndex = responsiveView->GetViews().IndexOf(responsiveView->GetCurrentView());
|
|
}
|
|
|
|
for (vint i = 0; i < ARRLEN(longContainers); i++)
|
|
{
|
|
longContainers[i]->GetToolstripItems().Clear();
|
|
}
|
|
for (vint i = 0; i < ARRLEN(shortContainers); i++)
|
|
{
|
|
shortContainers[i]->GetToolstripItems().Clear();
|
|
}
|
|
|
|
vint count = viewIndex == 0 ? 2 : 3;
|
|
|
|
if (groups.Count() <= count)
|
|
{
|
|
auto containers = viewIndex == 0 ? longContainers : shortContainers;
|
|
for (vint i = 0; i < groups.Count(); i++)
|
|
{
|
|
containers[i]->GetToolstripItems().Add(groups[i]);
|
|
}
|
|
}
|
|
else if (count == 3)
|
|
{
|
|
#define DELTA(POSTFIX) (abs(count1##POSTFIX - count2##POSTFIX) + abs(count2##POSTFIX - count3##POSTFIX) + abs(count3##POSTFIX - count1##POSTFIX))
|
|
#define DEFINE_COUNT(POSTFIX, OFFSET_FIRST, OFFSET_LAST) \
|
|
vint count1##POSTFIX = count1_o + (OFFSET_FIRST); \
|
|
vint count2##POSTFIX = count2_o - (OFFSET_FIRST) - (OFFSET_LAST); \
|
|
vint count3##POSTFIX = count3_o + (OFFSET_LAST)
|
|
#define MIN(a, b) (a)<(b)?(a):(b)
|
|
|
|
vint firstGroupCount = 0;
|
|
vint lastGroupCount = 0;
|
|
|
|
vint count1_o = 0;
|
|
vint count2_o = From(groups)
|
|
.Select([](GuiToolstripGroup* group) {return group->GetToolstripItems().Count(); })
|
|
.Aggregate([](vint a, vint b) {return a + b; });
|
|
vint count3_o = 0;
|
|
vint delta_o = DELTA(_o);
|
|
|
|
while (firstGroupCount + lastGroupCount < groups.Count())
|
|
{
|
|
auto newFirstGroup = groups[firstGroupCount];
|
|
auto newLastGroup = groups[groups.Count() - lastGroupCount - 1];
|
|
|
|
DEFINE_COUNT(_f, newFirstGroup->GetToolstripItems().Count(), 0);
|
|
vint delta_f = DELTA(_f);
|
|
|
|
DEFINE_COUNT(_l, 0, newLastGroup->GetToolstripItems().Count());
|
|
vint delta_l = DELTA(_l);
|
|
|
|
vint delta = MIN(delta_o, MIN(delta_f, delta_l));
|
|
if (delta == delta_f)
|
|
{
|
|
firstGroupCount++;
|
|
count1_o = count1_f;
|
|
count2_o = count2_f;
|
|
count3_o = count3_f;
|
|
delta_o = delta_f;
|
|
}
|
|
else if (delta == delta_l)
|
|
{
|
|
lastGroupCount++;
|
|
count1_o = count1_l;
|
|
count2_o = count2_l;
|
|
count3_o = count3_l;
|
|
delta_o = delta_l;
|
|
}
|
|
else
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
vint minMiddle = firstGroupCount;
|
|
vint maxMiddle = groups.Count() - lastGroupCount - 1;
|
|
for (vint j = 0; j < groups.Count(); j++)
|
|
{
|
|
shortContainers[
|
|
j < minMiddle ? 0 :
|
|
j>maxMiddle ? 2 :
|
|
1
|
|
]->GetToolstripItems().Add(groups[j]);
|
|
}
|
|
|
|
#undef MIN
|
|
#undef DEFINE_COUNT
|
|
#undef DELTA
|
|
}
|
|
else if (count == 2)
|
|
{
|
|
vint firstGroupCount = groups.Count();
|
|
{
|
|
vint count1 = 0;
|
|
vint count2 = From(groups)
|
|
.Select([](GuiToolstripGroup* group) {return group->GetToolstripItems().Count(); })
|
|
.Aggregate([](vint a, vint b) {return a + b; });
|
|
vint delta = abs(count2 - count1);
|
|
|
|
for (vint i = 0; i < groups.Count(); i++)
|
|
{
|
|
auto groupCount = groups[i]->GetToolstripItems().Count();
|
|
vint count1_2 = count1 + groupCount;
|
|
vint count2_2 = count2 - groupCount;
|
|
vint delta_2 = abs(count2_2 - count1_2);
|
|
|
|
if (delta < delta_2)
|
|
{
|
|
firstGroupCount = i;
|
|
break;
|
|
}
|
|
|
|
count1 = count1_2;
|
|
count2 = count2_2;
|
|
delta = delta_2;
|
|
}
|
|
}
|
|
|
|
for (vint j = 0; j < groups.Count(); j++)
|
|
{
|
|
longContainers[j < firstGroupCount ? 0 : 1]->GetToolstripItems().Add(groups[j]);
|
|
}
|
|
}
|
|
}
|
|
|
|
GuiRibbonToolstrips::GuiRibbonToolstrips(theme::ThemeName themeName)
|
|
:GuiControl(themeName)
|
|
, groups(this)
|
|
{
|
|
responsiveView = new GuiResponsiveViewComposition();
|
|
responsiveView->SetDirection(ResponsiveDirection::Horizontal);
|
|
responsiveView->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
responsiveView->BeforeSwitchingView.AttachMethod(this, &GuiRibbonToolstrips::OnBeforeSwitchingView);
|
|
|
|
vint toolbarIndex = 0;
|
|
for (vint i = 0; i < sizeof(views) / sizeof(*views); i++)
|
|
{
|
|
auto containers = i == 0 ? longContainers : shortContainers;
|
|
vint count = i == 0 ? 2 : 3;
|
|
|
|
auto table = new GuiTableComposition();
|
|
table->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
table->SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
|
|
table->SetRowsAndColumns(count * 2 + 1, 1);
|
|
table->SetColumnOption(0, GuiCellOption::MinSizeOption());
|
|
table->SetRowOption(0, GuiCellOption::PercentageOption(1.0));
|
|
for (vint j = 0; j < count; j++)
|
|
{
|
|
table->SetRowOption(j * 2 + 1, GuiCellOption::MinSizeOption());
|
|
table->SetRowOption(j * 2 + 2, GuiCellOption::PercentageOption(1.0));
|
|
}
|
|
|
|
for (vint j = 0; j < count; j++)
|
|
{
|
|
auto toolbar = new GuiToolstripToolBar(theme::ThemeName::ToolstripToolBar);
|
|
toolbar->GetBoundsComposition()->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
toolbars[toolbarIndex++] = toolbar;
|
|
|
|
auto cell = new GuiCellComposition();
|
|
cell->SetSite(j * 2 + 1, 0, 1, 1);
|
|
cell->AddChild(toolbar->GetBoundsComposition());
|
|
table->AddChild(cell);
|
|
|
|
auto container = new GuiToolstripGroupContainer(theme::ThemeName::CustomControl);
|
|
toolbar->GetToolstripItems().Add(container);
|
|
containers[j] = container;
|
|
}
|
|
|
|
views[i] = new GuiResponsiveFixedComposition();
|
|
views[i]->AddChild(table);
|
|
responsiveView->GetViews().Add(views[i]);
|
|
}
|
|
|
|
containerComposition->AddChild(responsiveView);
|
|
}
|
|
|
|
GuiRibbonToolstrips::~GuiRibbonToolstrips()
|
|
{
|
|
}
|
|
|
|
collections::ObservableListBase<GuiToolstripGroup*>& GuiRibbonToolstrips::GetGroups()
|
|
{
|
|
return groups;
|
|
}
|
|
|
|
#undef ARRLEN
|
|
|
|
/***********************************************************************
|
|
GuiRibbonGallery::CommandExecutor
|
|
***********************************************************************/
|
|
|
|
GuiRibbonGallery::CommandExecutor::CommandExecutor(GuiRibbonGallery* _gallery)
|
|
:gallery(_gallery)
|
|
{
|
|
}
|
|
|
|
GuiRibbonGallery::CommandExecutor::~CommandExecutor()
|
|
{
|
|
}
|
|
|
|
void GuiRibbonGallery::CommandExecutor::NotifyScrollUp()
|
|
{
|
|
gallery->RequestedScrollUp.Execute(gallery->GetNotifyEventArguments());
|
|
}
|
|
|
|
void GuiRibbonGallery::CommandExecutor::NotifyScrollDown()
|
|
{
|
|
gallery->RequestedScrollDown.Execute(gallery->GetNotifyEventArguments());
|
|
}
|
|
|
|
void GuiRibbonGallery::CommandExecutor::NotifyDropdown()
|
|
{
|
|
gallery->RequestedDropdown.Execute(gallery->GetNotifyEventArguments());
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiRibbonGallery
|
|
***********************************************************************/
|
|
|
|
void GuiRibbonGallery::BeforeControlTemplateUninstalled_()
|
|
{
|
|
auto ct = GetControlTemplateObject(false);
|
|
if (!ct) return;
|
|
|
|
ct->SetCommands(nullptr);
|
|
}
|
|
|
|
void GuiRibbonGallery::AfterControlTemplateInstalled_(bool initialize)
|
|
{
|
|
auto ct = GetControlTemplateObject(true);
|
|
ct->SetCommands(commandExecutor.Obj());
|
|
ct->SetScrollUpEnabled(scrollUpEnabled);
|
|
ct->SetScrollDownEnabled(scrollDownEnabled);
|
|
}
|
|
|
|
GuiRibbonGallery::GuiRibbonGallery(theme::ThemeName themeName)
|
|
:GuiControl(themeName)
|
|
{
|
|
commandExecutor = new CommandExecutor(this);
|
|
|
|
ScrollUpEnabledChanged.SetAssociatedComposition(boundsComposition);
|
|
ScrollDownEnabledChanged.SetAssociatedComposition(boundsComposition);
|
|
RequestedScrollUp.SetAssociatedComposition(boundsComposition);
|
|
RequestedScrollDown.SetAssociatedComposition(boundsComposition);
|
|
RequestedDropdown.SetAssociatedComposition(boundsComposition);
|
|
}
|
|
|
|
GuiRibbonGallery::~GuiRibbonGallery()
|
|
{
|
|
}
|
|
|
|
bool GuiRibbonGallery::GetScrollUpEnabled()
|
|
{
|
|
return scrollUpEnabled;
|
|
}
|
|
|
|
void GuiRibbonGallery::SetScrollUpEnabled(bool value)
|
|
{
|
|
if (scrollUpEnabled != value)
|
|
{
|
|
scrollUpEnabled = value;
|
|
GetControlTemplateObject(true)->SetScrollUpEnabled(value);
|
|
}
|
|
}
|
|
|
|
bool GuiRibbonGallery::GetScrollDownEnabled()
|
|
{
|
|
return scrollDownEnabled;
|
|
}
|
|
|
|
void GuiRibbonGallery::SetScrollDownEnabled(bool value)
|
|
{
|
|
if (scrollDownEnabled != value)
|
|
{
|
|
scrollDownEnabled = value;
|
|
GetControlTemplateObject(true)->SetScrollDownEnabled(value);
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiRibbonToolstripMenu
|
|
***********************************************************************/
|
|
|
|
void GuiRibbonToolstripMenu::BeforeControlTemplateUninstalled_()
|
|
{
|
|
auto ct = GetControlTemplateObject(false);
|
|
if (!ct) return;
|
|
|
|
if (auto cc = ct->GetContentComposition())
|
|
{
|
|
cc->RemoveChild(contentComposition);
|
|
}
|
|
}
|
|
|
|
void GuiRibbonToolstripMenu::AfterControlTemplateInstalled_(bool initialize)
|
|
{
|
|
auto ct = GetControlTemplateObject(true);
|
|
if (auto cc = ct->GetContentComposition())
|
|
{
|
|
cc->AddChild(contentComposition);
|
|
}
|
|
}
|
|
|
|
GuiRibbonToolstripMenu::GuiRibbonToolstripMenu(theme::ThemeName themeName, GuiControl* owner)
|
|
:GuiToolstripMenu(themeName, owner)
|
|
{
|
|
contentComposition = new GuiBoundsComposition();
|
|
contentComposition->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
contentComposition->SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
}
|
|
|
|
GuiRibbonToolstripMenu::~GuiRibbonToolstripMenu()
|
|
{
|
|
if (!contentComposition->GetParent())
|
|
{
|
|
SafeDeleteComposition(contentComposition);
|
|
}
|
|
}
|
|
|
|
compositions::GuiGraphicsComposition* GuiRibbonToolstripMenu::GetContentComposition()
|
|
{
|
|
return contentComposition;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
.\CONTROLS\TOOLSTRIPPACKAGE\GUIRIBBONGALLERYLIST.CPP
|
|
***********************************************************************/
|
|
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace controls
|
|
{
|
|
using namespace reflection::description;
|
|
using namespace collections;
|
|
using namespace compositions;
|
|
using namespace templates;
|
|
|
|
namespace list
|
|
{
|
|
|
|
/***********************************************************************
|
|
list::GalleryGroup
|
|
***********************************************************************/
|
|
|
|
GalleryGroup::GalleryGroup()
|
|
{
|
|
}
|
|
|
|
GalleryGroup::~GalleryGroup()
|
|
{
|
|
if (eventHandler)
|
|
{
|
|
itemValues.Cast<IValueObservableList>()->ItemChanged.Remove(eventHandler);
|
|
}
|
|
}
|
|
|
|
WString GalleryGroup::GetName()
|
|
{
|
|
return name;
|
|
}
|
|
|
|
Ptr<IValueList> GalleryGroup::GetItemValues()
|
|
{
|
|
return itemValues;
|
|
}
|
|
|
|
/***********************************************************************
|
|
list::GroupedDataSource
|
|
***********************************************************************/
|
|
|
|
void GroupedDataSource::RebuildItemSource()
|
|
{
|
|
ignoreGroupChanged = true;
|
|
joinedItemSource.Clear();
|
|
groupedItemSource.Clear();
|
|
cachedGroupItemCounts.Clear();
|
|
ignoreGroupChanged = false;
|
|
|
|
if (itemSource)
|
|
{
|
|
if (GetGroupEnabled())
|
|
{
|
|
FOREACH_INDEXER(Value, groupValue, index, GetLazyList<Value>(itemSource))
|
|
{
|
|
auto group = MakePtr<GalleryGroup>();
|
|
group->name = titleProperty(groupValue);
|
|
group->itemValues = GetChildren(childrenProperty(groupValue));
|
|
AttachGroupChanged(group, index);
|
|
groupedItemSource.Add(group);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
auto group = MakePtr<GalleryGroup>();
|
|
group->itemValues = GetChildren(itemSource);
|
|
AttachGroupChanged(group, 0);
|
|
groupedItemSource.Add(group);
|
|
}
|
|
}
|
|
}
|
|
|
|
Ptr<IValueList> GroupedDataSource::GetChildren(Ptr<IValueEnumerable> children)
|
|
{
|
|
if (!children)
|
|
{
|
|
return nullptr;
|
|
}
|
|
else if (auto list = children.Cast<IValueList>())
|
|
{
|
|
return list;
|
|
}
|
|
else
|
|
{
|
|
return IValueList::Create(GetLazyList<Value>(children));
|
|
}
|
|
}
|
|
|
|
void GroupedDataSource::AttachGroupChanged(Ptr<GalleryGroup> group, vint index)
|
|
{
|
|
if (auto observable = group->itemValues.Cast<IValueObservableList>())
|
|
{
|
|
group->eventHandler = observable->ItemChanged.Add([this, index](vint start, vint oldCount, vint newCount)
|
|
{
|
|
OnGroupItemChanged(index, start, oldCount, newCount);
|
|
});
|
|
}
|
|
}
|
|
|
|
void GroupedDataSource::OnGroupChanged(vint start, vint oldCount, vint newCount)
|
|
{
|
|
if (!ignoreGroupChanged)
|
|
{
|
|
for (vint i = 0; i < oldCount; i++)
|
|
{
|
|
RemoveGroupFromJoined(start);
|
|
}
|
|
for (vint i = 0; i < newCount; i++)
|
|
{
|
|
InsertGroupToJoined(start + i);
|
|
}
|
|
}
|
|
}
|
|
|
|
void GroupedDataSource::OnGroupItemChanged(vint index, vint start, vint oldCount, vint newCount)
|
|
{
|
|
if (!ignoreGroupChanged)
|
|
{
|
|
vint countBeforeGroup = GetCountBeforeGroup(index);
|
|
vint joinedIndex = countBeforeGroup + start;
|
|
vint minCount = oldCount < newCount ? oldCount : newCount;
|
|
auto itemValues = groupedItemSource[index]->itemValues;
|
|
|
|
for (vint i = 0; i < minCount; i++)
|
|
{
|
|
joinedItemSource.Set(joinedIndex + i, itemValues->Get(start + i));
|
|
}
|
|
|
|
if (oldCount < newCount)
|
|
{
|
|
for (vint i = minCount; i < newCount; i++)
|
|
{
|
|
joinedItemSource.Insert(joinedIndex + i, itemValues->Get(start + i));
|
|
}
|
|
}
|
|
else if (oldCount > newCount)
|
|
{
|
|
for (vint i = minCount; i < oldCount; i++)
|
|
{
|
|
joinedItemSource.RemoveAt(joinedIndex + i);
|
|
}
|
|
}
|
|
|
|
cachedGroupItemCounts[index] += newCount - oldCount;
|
|
}
|
|
}
|
|
|
|
vint GroupedDataSource::GetCountBeforeGroup(vint index)
|
|
{
|
|
vint count = 0;
|
|
for (vint i = 0; i < index; i++)
|
|
{
|
|
count += cachedGroupItemCounts[i];
|
|
}
|
|
return count;
|
|
}
|
|
|
|
void GroupedDataSource::InsertGroupToJoined(vint index)
|
|
{
|
|
vint countBeforeGroup = GetCountBeforeGroup(index);
|
|
auto group = groupedItemSource[index];
|
|
vint itemCount = group->itemValues ? group->itemValues->GetCount() : 0;
|
|
cachedGroupItemCounts.Insert(index, itemCount);
|
|
|
|
if (itemCount > 0)
|
|
{
|
|
for (vint i = 0; i < itemCount; i++)
|
|
{
|
|
joinedItemSource.Insert(countBeforeGroup + i, group->itemValues->Get(i));
|
|
}
|
|
}
|
|
}
|
|
|
|
void GroupedDataSource::RemoveGroupFromJoined(vint index)
|
|
{
|
|
vint countBeforeGroup = GetCountBeforeGroup(index);
|
|
joinedItemSource.RemoveRange(countBeforeGroup, cachedGroupItemCounts[index]);
|
|
cachedGroupItemCounts.RemoveAt(index);
|
|
}
|
|
|
|
GroupedDataSource::GroupedDataSource(compositions::GuiGraphicsComposition* _associatedComposition)
|
|
:associatedComposition(_associatedComposition)
|
|
{
|
|
GroupEnabledChanged.SetAssociatedComposition(associatedComposition);
|
|
GroupTitlePropertyChanged.SetAssociatedComposition(associatedComposition);
|
|
GroupChildrenPropertyChanged.SetAssociatedComposition(associatedComposition);
|
|
|
|
groupChangedHandler = groupedItemSource.GetWrapper()->ItemChanged.Add(this, &GroupedDataSource::OnGroupChanged);
|
|
}
|
|
|
|
GroupedDataSource::~GroupedDataSource()
|
|
{
|
|
joinedItemSource.GetWrapper()->ItemChanged.Remove(groupChangedHandler);
|
|
}
|
|
|
|
Ptr<IValueEnumerable> GroupedDataSource::GetItemSource()
|
|
{
|
|
return itemSource;
|
|
}
|
|
|
|
void GroupedDataSource::SetItemSource(Ptr<IValueEnumerable> value)
|
|
{
|
|
if (itemSource != value)
|
|
{
|
|
itemSource = value;
|
|
RebuildItemSource();
|
|
}
|
|
}
|
|
|
|
bool GroupedDataSource::GetGroupEnabled()
|
|
{
|
|
return titleProperty && childrenProperty;
|
|
}
|
|
|
|
ItemProperty<WString> GroupedDataSource::GetGroupTitleProperty()
|
|
{
|
|
return titleProperty;
|
|
}
|
|
|
|
void GroupedDataSource::SetGroupTitleProperty(const ItemProperty<WString>& value)
|
|
{
|
|
if (titleProperty != value)
|
|
{
|
|
titleProperty = value;
|
|
GroupTitlePropertyChanged.Execute(GuiEventArgs(associatedComposition));
|
|
GroupEnabledChanged.Execute(GuiEventArgs(associatedComposition));
|
|
RebuildItemSource();
|
|
}
|
|
}
|
|
|
|
ItemProperty<Ptr<IValueEnumerable>> GroupedDataSource::GetGroupChildrenProperty()
|
|
{
|
|
return childrenProperty;
|
|
}
|
|
|
|
void GroupedDataSource::SetGroupChildrenProperty(const ItemProperty<Ptr<IValueEnumerable>>& value)
|
|
{
|
|
if (childrenProperty != value)
|
|
{
|
|
childrenProperty = value;
|
|
GroupChildrenPropertyChanged.Execute(GuiEventArgs(associatedComposition));
|
|
GroupEnabledChanged.Execute(GuiEventArgs(associatedComposition));
|
|
RebuildItemSource();
|
|
}
|
|
}
|
|
|
|
const GroupedDataSource::GalleryGroupList& GroupedDataSource::GetGroups()
|
|
{
|
|
return groupedItemSource;
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiBindableRibbonGalleryList
|
|
***********************************************************************/
|
|
|
|
void GuiBindableRibbonGalleryList::BeforeControlTemplateUninstalled_()
|
|
{
|
|
}
|
|
|
|
void GuiBindableRibbonGalleryList::AfterControlTemplateInstalled_(bool initialize)
|
|
{
|
|
auto ct = GetControlTemplateObject(true);
|
|
itemList->SetControlTemplate(ct->GetItemListTemplate());
|
|
subMenu->SetControlTemplate(ct->GetMenuTemplate());
|
|
groupContainer->SetControlTemplate(ct->GetGroupContainerTemplate());
|
|
MenuResetGroupTemplate();
|
|
UpdateLayoutSizeOffset();
|
|
}
|
|
|
|
void GuiBindableRibbonGalleryList::UpdateLayoutSizeOffset()
|
|
{
|
|
auto cSize = itemList->GetContainerComposition()->GetBounds();
|
|
auto bSize = itemList->GetBoundsComposition()->GetBounds();
|
|
layout->SetSizeOffset(Size(bSize.Width() - cSize.Width(), bSize.Height() - cSize.Height()));
|
|
|
|
if (layout->GetItemWidth() > 0)
|
|
{
|
|
vint columns = layout->GetVisibleItemCount();
|
|
if (columns == 0) columns = 1;
|
|
vint rows = (visibleItemCount + columns - 1) / columns;
|
|
vint height = (vint)(layout->GetBounds().Height()*(rows + 0.5));
|
|
groupContainer->GetBoundsComposition()->SetPreferredMinSize(Size(0, height));
|
|
}
|
|
else
|
|
{
|
|
groupContainer->GetBoundsComposition()->SetPreferredMinSize(Size(0, 0));
|
|
}
|
|
}
|
|
|
|
void GuiBindableRibbonGalleryList::OnItemListSelectionChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
auto pos = IndexToGalleryPos(itemList->GetSelectedItemIndex());
|
|
if (pos.group != -1 && pos.item != -1)
|
|
{
|
|
for (vint i = 0; i < groupedItemSource.Count(); i++)
|
|
{
|
|
auto group = groupedItemSource[i];
|
|
if (group->GetItemValues())
|
|
{
|
|
vint count = group->GetItemValues()->GetCount();
|
|
for (vint j = 0; j < count; j++)
|
|
{
|
|
auto background = MenuGetGroupItemBackground(i, j);
|
|
background->SetSelected(pos.group == i && pos.item == j);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!skipItemAppliedEvent && itemList->GetSelectedItemIndex() != -1)
|
|
{
|
|
GuiItemEventArgs itemAppliedArgs(boundsComposition);
|
|
itemAppliedArgs.itemIndex = itemList->GetSelectedItemIndex();
|
|
ItemApplied.Execute(itemAppliedArgs);
|
|
}
|
|
SelectionChanged.Execute(GetNotifyEventArguments());
|
|
}
|
|
|
|
void GuiBindableRibbonGalleryList::OnItemListItemMouseEnter(compositions::GuiGraphicsComposition* sender, compositions::GuiItemEventArgs& arguments)
|
|
{
|
|
StartPreview(arguments.itemIndex);
|
|
}
|
|
|
|
void GuiBindableRibbonGalleryList::OnItemListItemMouseLeave(compositions::GuiGraphicsComposition* sender, compositions::GuiItemEventArgs& arguments)
|
|
{
|
|
StopPreview(arguments.itemIndex);
|
|
}
|
|
|
|
void GuiBindableRibbonGalleryList::OnBoundsChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
UpdateLayoutSizeOffset();
|
|
|
|
auto bounds = boundsComposition->GetBounds();
|
|
subMenu->GetBoundsComposition()->SetPreferredMinSize(Size(bounds.Width() + 20, 1));
|
|
|
|
for (vint i = 0; i < groupedItemSource.Count(); i++)
|
|
{
|
|
auto group = groupedItemSource[i];
|
|
if (group->GetItemValues())
|
|
{
|
|
vint count = group->GetItemValues()->GetCount();
|
|
for (vint j = 0; j < count; j++)
|
|
{
|
|
auto background = MenuGetGroupItemBackground(i, j);
|
|
background->GetBoundsComposition()->SetPreferredMinSize(Size(0, bounds.Height()));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiBindableRibbonGalleryList::OnRequestedDropdown(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
subMenu->ShowPopup(this, Point(0, 0));
|
|
}
|
|
|
|
void GuiBindableRibbonGalleryList::OnRequestedScrollUp(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
itemListArranger->ScrollUp();
|
|
}
|
|
|
|
void GuiBindableRibbonGalleryList::OnRequestedScrollDown(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
itemListArranger->ScrollDown();
|
|
}
|
|
|
|
void GuiBindableRibbonGalleryList::MenuResetGroupTemplate()
|
|
{
|
|
groupStack->SetItemTemplate([this](const Value& groupValue)->GuiTemplate*
|
|
{
|
|
auto group = UnboxValue<Ptr<list::GalleryGroup>>(groupValue);
|
|
|
|
auto groupTemplate = new GuiTemplate;
|
|
groupTemplate->SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
|
|
auto groupContentStack = new GuiStackComposition;
|
|
groupContentStack->SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
groupContentStack->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
groupContentStack->SetDirection(GuiStackComposition::Vertical);
|
|
{
|
|
auto item = new GuiStackItemComposition;
|
|
groupContentStack->AddChild(item);
|
|
|
|
auto header = new GuiControl(theme::ThemeName::RibbonToolstripHeader);
|
|
header->SetControlTemplate(GetControlTemplateObject(true)->GetHeaderTemplate());
|
|
header->GetBoundsComposition()->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
header->SetText(group->GetName());
|
|
item->AddChild(header->GetBoundsComposition());
|
|
}
|
|
if (itemStyle)
|
|
{
|
|
auto item = new GuiStackItemComposition;
|
|
item->SetPreferredMinSize(Size(1, 1));
|
|
groupContentStack->AddChild(item);
|
|
|
|
auto groupItemFlow = new GuiRepeatFlowComposition();
|
|
groupItemFlow->SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
groupItemFlow->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
groupItemFlow->SetItemSource(group->GetItemValues());
|
|
groupItemFlow->SetItemTemplate([=](const Value& groupItemValue)->GuiTemplate*
|
|
{
|
|
auto groupItemTemplate = new GuiTemplate;
|
|
groupItemTemplate->SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
|
|
auto backgroundButton = new GuiSelectableButton(theme::ThemeName::ListItemBackground);
|
|
if (auto style = GetControlTemplateObject(true)->GetBackgroundTemplate())
|
|
{
|
|
backgroundButton->SetControlTemplate(style);
|
|
}
|
|
backgroundButton->SetAutoFocus(false);
|
|
backgroundButton->SetAutoSelection(false);
|
|
backgroundButton->Clicked.AttachLambda([=](GuiGraphicsComposition* sender, GuiEventArgs& arguments)
|
|
{
|
|
auto groupIndex = groupStack->GetStackItems().IndexOf(dynamic_cast<GuiStackItemComposition*>(groupTemplate->GetParent()));
|
|
auto itemIndex = groupItemFlow->GetFlowItems().IndexOf(dynamic_cast<GuiFlowItemComposition*>(groupItemTemplate->GetParent()));
|
|
auto index = GalleryPosToIndex({ groupIndex,itemIndex });
|
|
itemList->SetSelected(index, true);
|
|
itemList->EnsureItemVisible(index);
|
|
subMenu->Close();
|
|
});
|
|
backgroundButton->GetBoundsComposition()->GetEventReceiver()->mouseEnter.AttachLambda([=](GuiGraphicsComposition* sender, GuiEventArgs& arguments)
|
|
{
|
|
auto groupIndex = groupStack->GetStackItems().IndexOf(dynamic_cast<GuiStackItemComposition*>(groupTemplate->GetParent()));
|
|
auto itemIndex = groupItemFlow->GetFlowItems().IndexOf(dynamic_cast<GuiFlowItemComposition*>(groupItemTemplate->GetParent()));
|
|
auto index = GalleryPosToIndex({ groupIndex,itemIndex });
|
|
StartPreview(index);
|
|
});
|
|
backgroundButton->GetBoundsComposition()->GetEventReceiver()->mouseLeave.AttachLambda([=](GuiGraphicsComposition* sender, GuiEventArgs& arguments)
|
|
{
|
|
auto groupIndex = groupStack->GetStackItems().IndexOf(dynamic_cast<GuiStackItemComposition*>(groupTemplate->GetParent()));
|
|
auto itemIndex = groupItemFlow->GetFlowItems().IndexOf(dynamic_cast<GuiFlowItemComposition*>(groupItemTemplate->GetParent()));
|
|
auto index = GalleryPosToIndex({ groupIndex,itemIndex });
|
|
StopPreview(index);
|
|
});
|
|
groupItemTemplate->AddChild(backgroundButton->GetBoundsComposition());
|
|
|
|
auto itemTemplate = itemStyle(groupItemValue);
|
|
itemTemplate->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
backgroundButton->GetContainerComposition()->AddChild(itemTemplate);
|
|
|
|
return groupItemTemplate;
|
|
});
|
|
item->AddChild(groupItemFlow);
|
|
}
|
|
groupTemplate->AddChild(groupContentStack);
|
|
|
|
return groupTemplate;
|
|
});
|
|
}
|
|
|
|
GuiControl* GuiBindableRibbonGalleryList::MenuGetGroupHeader(vint groupIndex)
|
|
{
|
|
CHECK_ERROR(0 <= groupIndex && groupIndex < groupedItemSource.Count(), L"GuiBindableRibbonGalleryList::MenuGetGroupHeader(vint)#Group index out of range");
|
|
auto stackItem = groupStack->GetStackItems()[groupIndex];
|
|
auto groupTemplate = stackItem->Children()[0];
|
|
auto groupContentStack = dynamic_cast<GuiStackComposition*>(groupTemplate->Children()[0]);
|
|
auto groupHeaderItem = groupContentStack->GetStackItems()[0];
|
|
auto groupHeader = groupHeaderItem->Children()[0]->GetAssociatedControl();
|
|
CHECK_ERROR(groupHeader, L"GuiBindableRibbonGalleryList::MenuGetGroupHeader(vint)#Internal error.");
|
|
return groupHeader;
|
|
}
|
|
|
|
compositions::GuiRepeatFlowComposition* GuiBindableRibbonGalleryList::MenuGetGroupFlow(vint groupIndex)
|
|
{
|
|
CHECK_ERROR(0 <= groupIndex && groupIndex < groupedItemSource.Count(), L"GuiBindableRibbonGalleryList::MenuGetGroupFlow(vint)#Group index out of range");
|
|
if (!itemStyle) return nullptr;
|
|
auto stackItem = groupStack->GetStackItems()[groupIndex];
|
|
auto groupTemplate = stackItem->Children()[0];
|
|
auto groupContentStack = dynamic_cast<GuiStackComposition*>(groupTemplate->Children()[0]);
|
|
auto groupContentItem = groupContentStack->GetStackItems()[1];
|
|
auto groupFlow = dynamic_cast<GuiRepeatFlowComposition*>(groupContentItem->Children()[0]);
|
|
CHECK_ERROR(groupFlow, L"GuiBindableRibbonGalleryList::MenuGetGroupHeader(vint)#Internal error.");
|
|
return groupFlow;
|
|
}
|
|
|
|
GuiSelectableButton* GuiBindableRibbonGalleryList::MenuGetGroupItemBackground(vint groupIndex, vint itemIndex)
|
|
{
|
|
CHECK_ERROR(0 <= groupIndex && groupIndex < groupedItemSource.Count(), L"GuiBindableRibbonGalleryList::MenuGetGroupItemBackground(vint, vint)#Group index out of range");
|
|
auto group = groupedItemSource[groupIndex];
|
|
CHECK_ERROR(group->GetItemValues() && 0 <= itemIndex && itemIndex < group->GetItemValues()->GetCount(), L"GuiBindableRibbonGalleryList::MenuGetGroupHeader(vint, vint)#Item index out of range");
|
|
|
|
auto groupFlow = MenuGetGroupFlow(groupIndex);
|
|
auto groupFlowItem = groupFlow->GetFlowItems()[itemIndex];
|
|
auto groupItemTemplate = groupFlowItem->Children()[0];
|
|
auto groupItemBackground = dynamic_cast<GuiSelectableButton*>(groupItemTemplate->Children()[0]->GetAssociatedControl());
|
|
CHECK_ERROR(groupItemBackground, L"GuiBindableRibbonGalleryList::MenuGetGroupHeader(vint, vint)#Internal error.");
|
|
return groupItemBackground;
|
|
}
|
|
|
|
void GuiBindableRibbonGalleryList::StartPreview(vint index)
|
|
{
|
|
if (index != itemList->GetSelectedItemIndex())
|
|
{
|
|
GuiItemEventArgs previewArgs(boundsComposition);
|
|
previewArgs.itemIndex = index;
|
|
PreviewStarted.Execute(previewArgs);
|
|
}
|
|
}
|
|
|
|
void GuiBindableRibbonGalleryList::StopPreview(vint index)
|
|
{
|
|
if (index != itemList->GetSelectedItemIndex())
|
|
{
|
|
GuiItemEventArgs previewArgs(boundsComposition);
|
|
previewArgs.itemIndex = index;
|
|
PreviewStopped.Execute(previewArgs);
|
|
}
|
|
}
|
|
|
|
GuiMenu* GuiBindableRibbonGalleryList::ProvideDropdownMenu()
|
|
{
|
|
return GetSubMenu();
|
|
}
|
|
|
|
GuiBindableRibbonGalleryList::GuiBindableRibbonGalleryList(theme::ThemeName themeName)
|
|
:GuiRibbonGallery(themeName)
|
|
, GroupedDataSource(boundsComposition)
|
|
{
|
|
ItemTemplateChanged.SetAssociatedComposition(boundsComposition);
|
|
SelectionChanged.SetAssociatedComposition(boundsComposition);
|
|
PreviewStarted.SetAssociatedComposition(boundsComposition);
|
|
PreviewStopped.SetAssociatedComposition(boundsComposition);
|
|
ItemApplied.SetAssociatedComposition(boundsComposition);
|
|
subMenu = new GuiRibbonToolstripMenu(theme::ThemeName::RibbonToolstripMenu, this);
|
|
|
|
{
|
|
layout = new ribbon_impl::GalleryResponsiveLayout;
|
|
layout->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
containerComposition->AddChild(layout);
|
|
|
|
itemListArranger = new ribbon_impl::GalleryItemArranger(this);
|
|
itemList = new GuiBindableTextList(theme::ThemeName::RibbonGalleryItemList);
|
|
itemList->GetBoundsComposition()->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
itemList->SetArranger(itemListArranger);
|
|
itemList->SetItemSource(joinedItemSource.GetWrapper());
|
|
itemList->SelectionChanged.AttachMethod(this, &GuiBindableRibbonGalleryList::OnItemListSelectionChanged);
|
|
itemList->ItemMouseEnter.AttachMethod(this, &GuiBindableRibbonGalleryList::OnItemListItemMouseEnter);
|
|
itemList->ItemMouseLeave.AttachMethod(this, &GuiBindableRibbonGalleryList::OnItemListItemMouseLeave);
|
|
layout->AddChild(itemList->GetBoundsComposition());
|
|
}
|
|
{
|
|
groupContainer = new GuiScrollContainer(theme::ThemeName::ScrollView);
|
|
groupContainer->SetHorizontalAlwaysVisible(false);
|
|
groupContainer->SetVerticalAlwaysVisible(false);
|
|
groupContainer->SetExtendToFullWidth(true);
|
|
groupContainer->GetBoundsComposition()->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
subMenu->GetContentComposition()->AddChild(groupContainer->GetBoundsComposition());
|
|
|
|
groupStack = new GuiRepeatStackComposition();
|
|
groupStack->SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
groupStack->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
groupStack->SetDirection(GuiStackComposition::Vertical);
|
|
groupStack->SetItemSource(groupedItemSource.GetWrapper());
|
|
groupContainer->GetContainerComposition()->AddChild(groupStack);
|
|
MenuResetGroupTemplate();
|
|
}
|
|
|
|
RequestedScrollUp.AttachMethod(this, &GuiBindableRibbonGalleryList::OnRequestedScrollUp);
|
|
RequestedScrollDown.AttachMethod(this, &GuiBindableRibbonGalleryList::OnRequestedScrollDown);
|
|
RequestedDropdown.AttachMethod(this, &GuiBindableRibbonGalleryList::OnRequestedDropdown);
|
|
boundsComposition->BoundsChanged.AttachMethod(this, &GuiBindableRibbonGalleryList::OnBoundsChanged);
|
|
itemListArranger->UnblockScrollUpdate();
|
|
}
|
|
|
|
GuiBindableRibbonGalleryList::~GuiBindableRibbonGalleryList()
|
|
{
|
|
delete subMenu;
|
|
}
|
|
|
|
GuiBindableRibbonGalleryList::ItemStyleProperty GuiBindableRibbonGalleryList::GetItemTemplate()
|
|
{
|
|
return itemStyle;
|
|
}
|
|
|
|
void GuiBindableRibbonGalleryList::SetItemTemplate(ItemStyleProperty value)
|
|
{
|
|
if (itemStyle != value)
|
|
{
|
|
itemStyle = value;
|
|
itemList->SetItemTemplate(value);
|
|
ItemTemplateChanged.Execute(GetNotifyEventArguments());
|
|
}
|
|
}
|
|
|
|
GalleryPos GuiBindableRibbonGalleryList::IndexToGalleryPos(vint index)
|
|
{
|
|
if (0 <= index && index < joinedItemSource.Count())
|
|
{
|
|
FOREACH_INDEXER(Ptr<list::GalleryGroup>, group, groupIndex, groupedItemSource)
|
|
{
|
|
auto itemValues = group->GetItemValues();
|
|
vint itemCount = itemValues ? itemValues->GetCount() : 0;
|
|
if (index >= itemCount)
|
|
{
|
|
index -= itemCount;
|
|
}
|
|
else
|
|
{
|
|
return GalleryPos(groupIndex, index);
|
|
}
|
|
}
|
|
}
|
|
return {};
|
|
}
|
|
|
|
vint GuiBindableRibbonGalleryList::GalleryPosToIndex(GalleryPos pos)
|
|
{
|
|
if (0 <= pos.group && pos.group < groupedItemSource.Count())
|
|
{
|
|
auto countBeforeGroup = GetCountBeforeGroup(pos.group);
|
|
auto itemValues = groupedItemSource[pos.group]->GetItemValues();
|
|
vint itemCount = itemValues ? itemValues->GetCount() : 0;
|
|
if (0 <= pos.item && pos.item < itemCount)
|
|
{
|
|
return countBeforeGroup + pos.item;
|
|
}
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
vint GuiBindableRibbonGalleryList::GetMinCount()
|
|
{
|
|
return layout->GetMinCount();
|
|
}
|
|
|
|
void GuiBindableRibbonGalleryList::SetMinCount(vint value)
|
|
{
|
|
layout->SetMinCount(value);
|
|
}
|
|
|
|
vint GuiBindableRibbonGalleryList::GetMaxCount()
|
|
{
|
|
return layout->GetMaxCount();
|
|
}
|
|
|
|
void GuiBindableRibbonGalleryList::SetMaxCount(vint value)
|
|
{
|
|
layout->SetMaxCount(value);
|
|
}
|
|
|
|
vint GuiBindableRibbonGalleryList::GetSelectedIndex()
|
|
{
|
|
return itemList->GetSelectedItemIndex();
|
|
}
|
|
|
|
description::Value GuiBindableRibbonGalleryList::GetSelectedItem()
|
|
{
|
|
vint index = itemList->GetSelectedItemIndex();
|
|
if (index == -1) return Value();
|
|
|
|
auto pos = IndexToGalleryPos(index);
|
|
return groupedItemSource[pos.group]->GetItemValues()->Get(pos.item);
|
|
}
|
|
|
|
void GuiBindableRibbonGalleryList::ApplyItem(vint index)
|
|
{
|
|
if (index == -1)
|
|
{
|
|
itemList->ClearSelection();
|
|
}
|
|
else
|
|
{
|
|
itemList->SetSelected(index, true);
|
|
itemList->EnsureItemVisible(index);
|
|
}
|
|
}
|
|
|
|
void GuiBindableRibbonGalleryList::SelectItem(vint index)
|
|
{
|
|
skipItemAppliedEvent = true;
|
|
ApplyItem(index);
|
|
skipItemAppliedEvent = false;
|
|
}
|
|
|
|
vint GuiBindableRibbonGalleryList::GetVisibleItemCount()
|
|
{
|
|
return visibleItemCount;
|
|
}
|
|
|
|
void GuiBindableRibbonGalleryList::SetVisibleItemCount(vint value)
|
|
{
|
|
if (visibleItemCount != value)
|
|
{
|
|
visibleItemCount = value;
|
|
UpdateLayoutSizeOffset();
|
|
}
|
|
}
|
|
|
|
GuiToolstripMenu* GuiBindableRibbonGalleryList::GetSubMenu()
|
|
{
|
|
return subMenu;
|
|
}
|
|
|
|
IDescriptable* GuiBindableRibbonGalleryList::QueryService(const WString& identifier)
|
|
{
|
|
if (identifier == IGuiMenuDropdownProvider::Identifier)
|
|
{
|
|
return (IGuiMenuDropdownProvider*)this;
|
|
}
|
|
else
|
|
{
|
|
return GuiRibbonGallery::QueryService(identifier);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\CONTROLS\TOOLSTRIPPACKAGE\GUIRIBBONIMPL.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace controls
|
|
{
|
|
using namespace compositions;
|
|
|
|
/***********************************************************************
|
|
GalleryItemArranger
|
|
***********************************************************************/
|
|
|
|
namespace ribbon_impl
|
|
{
|
|
void GalleryItemArranger::BeginPlaceItem(bool forMoving, Rect newBounds, vint& newStartIndex)
|
|
{
|
|
if (forMoving)
|
|
{
|
|
pim_itemWidth = itemWidth;
|
|
newStartIndex = firstIndex;
|
|
}
|
|
}
|
|
|
|
void GalleryItemArranger::PlaceItem(bool forMoving, bool newCreatedStyle, vint index, ItemStyleRecord style, Rect viewBounds, Rect& bounds, Margin& alignmentToParent)
|
|
{
|
|
alignmentToParent = Margin(-1, 0, -1, 0);
|
|
bounds = Rect(Point((index - firstIndex) * itemWidth, 0), Size(itemWidth, 0));
|
|
|
|
if (forMoving)
|
|
{
|
|
vint styleWidth = callback->GetStylePreferredSize(GetStyleBounds(style)).x;
|
|
if (pim_itemWidth < styleWidth)
|
|
{
|
|
pim_itemWidth = styleWidth;
|
|
}
|
|
}
|
|
}
|
|
|
|
bool GalleryItemArranger::IsItemOutOfViewBounds(vint index, ItemStyleRecord style, Rect bounds, Rect viewBounds)
|
|
{
|
|
return bounds.Right() + pim_itemWidth > viewBounds.Right();
|
|
}
|
|
|
|
bool GalleryItemArranger::EndPlaceItem(bool forMoving, Rect newBounds, vint newStartIndex)
|
|
{
|
|
bool result = false;
|
|
if (forMoving)
|
|
{
|
|
if (pim_itemWidth != itemWidth)
|
|
{
|
|
itemWidth = pim_itemWidth;
|
|
result = true;
|
|
}
|
|
}
|
|
|
|
if (!blockScrollUpdate)
|
|
{
|
|
UnblockScrollUpdate();
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
void GalleryItemArranger::InvalidateItemSizeCache()
|
|
{
|
|
itemWidth = 1;
|
|
}
|
|
|
|
Size GalleryItemArranger::OnCalculateTotalSize()
|
|
{
|
|
return Size(1, 1);
|
|
}
|
|
|
|
GalleryItemArranger::GalleryItemArranger(GuiBindableRibbonGalleryList* _owner)
|
|
:owner(_owner)
|
|
{
|
|
}
|
|
|
|
GalleryItemArranger::~GalleryItemArranger()
|
|
{
|
|
}
|
|
|
|
vint GalleryItemArranger::FindItem(vint itemIndex, compositions::KeyDirection key)
|
|
{
|
|
vint count = itemProvider->Count();
|
|
vint groupCount = viewBounds.Width() / itemWidth;
|
|
|
|
switch (key)
|
|
{
|
|
case KeyDirection::Left:
|
|
itemIndex--;
|
|
break;
|
|
case KeyDirection::Right:
|
|
itemIndex++;
|
|
break;
|
|
case KeyDirection::Home:
|
|
itemIndex = 0;
|
|
break;
|
|
case KeyDirection::End:
|
|
itemIndex = count;
|
|
break;
|
|
case KeyDirection::PageUp:
|
|
itemIndex -= groupCount;
|
|
break;
|
|
case KeyDirection::PageDown:
|
|
itemIndex += groupCount;
|
|
break;
|
|
default:
|
|
return -1;
|
|
}
|
|
|
|
if (itemIndex < 0) return 0;
|
|
else if (itemIndex >= count) return count - 1;
|
|
else return itemIndex;
|
|
}
|
|
|
|
GuiListControl::EnsureItemVisibleResult GalleryItemArranger::EnsureItemVisible(vint itemIndex)
|
|
{
|
|
if (callback)
|
|
{
|
|
if (0 <= itemIndex && itemIndex < itemProvider->Count())
|
|
{
|
|
vint groupCount = viewBounds.Width() / itemWidth;
|
|
if (itemIndex < firstIndex)
|
|
{
|
|
firstIndex = itemIndex;
|
|
callback->OnTotalSizeChanged();
|
|
}
|
|
else if (itemIndex >= firstIndex + groupCount)
|
|
{
|
|
firstIndex = itemIndex - groupCount + 1;
|
|
callback->OnTotalSizeChanged();
|
|
}
|
|
return GuiListControl::EnsureItemVisibleResult::NotMoved;
|
|
}
|
|
else
|
|
{
|
|
return GuiListControl::EnsureItemVisibleResult::ItemNotExists;
|
|
}
|
|
}
|
|
return GuiListControl::EnsureItemVisibleResult::NotMoved;
|
|
}
|
|
|
|
Size GalleryItemArranger::GetAdoptedSize(Size expectedSize)
|
|
{
|
|
return Size(1, 1);
|
|
}
|
|
|
|
void GalleryItemArranger::ScrollUp()
|
|
{
|
|
vint count = itemProvider->Count();
|
|
vint groupCount = viewBounds.Width() / itemWidth;
|
|
if (count > groupCount)
|
|
{
|
|
firstIndex -= groupCount;
|
|
if (firstIndex < 0)
|
|
{
|
|
firstIndex = 0;
|
|
}
|
|
|
|
if (callback)
|
|
{
|
|
callback->OnTotalSizeChanged();
|
|
}
|
|
}
|
|
}
|
|
|
|
void GalleryItemArranger::ScrollDown()
|
|
{
|
|
vint count = itemProvider->Count();
|
|
vint groupCount = viewBounds.Width() / itemWidth;
|
|
if (count > groupCount)
|
|
{
|
|
firstIndex += groupCount;
|
|
if (firstIndex > count - groupCount)
|
|
{
|
|
firstIndex = count - groupCount;
|
|
}
|
|
|
|
if (callback)
|
|
{
|
|
callback->OnTotalSizeChanged();
|
|
}
|
|
}
|
|
}
|
|
|
|
void GalleryItemArranger::UnblockScrollUpdate()
|
|
{
|
|
blockScrollUpdate = false;
|
|
|
|
vint count = itemProvider->Count();
|
|
vint groupCount = viewBounds.Width() / pim_itemWidth;
|
|
owner->SetScrollUpEnabled(firstIndex > 0);
|
|
owner->SetScrollDownEnabled(firstIndex + groupCount < count);
|
|
if (owner->layout->GetItemWidth() != pim_itemWidth)
|
|
{
|
|
owner->layout->SetItemWidth(pim_itemWidth);
|
|
owner->UpdateLayoutSizeOffset();
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GalleryResponsiveLayout
|
|
***********************************************************************/
|
|
|
|
void GalleryResponsiveLayout::UpdateMinSize()
|
|
{
|
|
SetPreferredMinSize(Size(itemCount * itemWidth + sizeOffset.x, sizeOffset.y));
|
|
}
|
|
|
|
GalleryResponsiveLayout::GalleryResponsiveLayout()
|
|
{
|
|
SetDirection(ResponsiveDirection::Horizontal);
|
|
}
|
|
|
|
GalleryResponsiveLayout::~GalleryResponsiveLayout()
|
|
{
|
|
}
|
|
|
|
vint GalleryResponsiveLayout::GetMinCount()
|
|
{
|
|
return minCount;
|
|
}
|
|
|
|
vint GalleryResponsiveLayout::GetMaxCount()
|
|
{
|
|
return maxCount;
|
|
}
|
|
|
|
vint GalleryResponsiveLayout::GetItemWidth()
|
|
{
|
|
return itemWidth;
|
|
}
|
|
|
|
Size GalleryResponsiveLayout::GetSizeOffset()
|
|
{
|
|
return sizeOffset;
|
|
}
|
|
|
|
vint GalleryResponsiveLayout::GetVisibleItemCount()
|
|
{
|
|
return itemCount;
|
|
}
|
|
|
|
void GalleryResponsiveLayout::SetMinCount(vint value)
|
|
{
|
|
vint oldCount = GetLevelCount();
|
|
vint oldLevel = GetCurrentLevel();
|
|
|
|
if (minCount != value)
|
|
{
|
|
if (value < 0) value = 0;
|
|
minCount = value;
|
|
if (maxCount < minCount) maxCount = minCount;
|
|
if (itemCount < minCount) itemCount = minCount;
|
|
UpdateMinSize();
|
|
}
|
|
|
|
bool countChanged = oldCount != GetLevelCount();
|
|
bool levelChanged = oldLevel != GetCurrentLevel();
|
|
if (countChanged) LevelCountChanged.Execute(GuiEventArgs(this));
|
|
if (levelChanged) CurrentLevelChanged.Execute(GuiEventArgs(this));
|
|
if (countChanged || levelChanged) OnResponsiveChildLevelUpdated();
|
|
}
|
|
|
|
void GalleryResponsiveLayout::SetMaxCount(vint value)
|
|
{
|
|
vint oldCount = GetLevelCount();
|
|
vint oldLevel = GetCurrentLevel();
|
|
|
|
if (maxCount != value)
|
|
{
|
|
if (value < 0) value = 0;
|
|
maxCount = value;
|
|
if (minCount > maxCount) minCount = maxCount;
|
|
if (itemCount > maxCount) itemCount = maxCount;
|
|
UpdateMinSize();
|
|
}
|
|
|
|
if (oldCount != GetLevelCount()) LevelCountChanged.Execute(GuiEventArgs(this));
|
|
if (oldLevel != GetCurrentLevel()) CurrentLevelChanged.Execute(GuiEventArgs(this));
|
|
}
|
|
|
|
void GalleryResponsiveLayout::SetItemWidth(vint value)
|
|
{
|
|
if (itemWidth != value)
|
|
{
|
|
itemWidth = value;
|
|
UpdateMinSize();
|
|
}
|
|
}
|
|
|
|
void GalleryResponsiveLayout::SetSizeOffset(Size value)
|
|
{
|
|
if (sizeOffset != value)
|
|
{
|
|
sizeOffset = value;
|
|
UpdateMinSize();
|
|
}
|
|
}
|
|
|
|
vint GalleryResponsiveLayout::GetLevelCount()
|
|
{
|
|
return maxCount - minCount + 1;
|
|
}
|
|
|
|
vint GalleryResponsiveLayout::GetCurrentLevel()
|
|
{
|
|
return itemCount - minCount;
|
|
}
|
|
|
|
bool GalleryResponsiveLayout::LevelDown()
|
|
{
|
|
if (itemCount > minCount)
|
|
{
|
|
itemCount--;
|
|
UpdateMinSize();
|
|
CurrentLevelChanged.Execute(GuiEventArgs(this));
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool GalleryResponsiveLayout::LevelUp()
|
|
{
|
|
if (itemCount < maxCount)
|
|
{
|
|
itemCount++;
|
|
UpdateMinSize();
|
|
CurrentLevelChanged.Execute(GuiEventArgs(this));
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\CONTROLS\TOOLSTRIPPACKAGE\GUITOOLSTRIPCOMMAND.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace controls
|
|
{
|
|
using namespace collections;
|
|
using namespace compositions;
|
|
using namespace regex;
|
|
using namespace parsing;
|
|
|
|
/***********************************************************************
|
|
GuiToolstripCommand
|
|
***********************************************************************/
|
|
|
|
void GuiToolstripCommand::OnShortcutKeyItemExecuted(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
Executed.ExecuteWithNewSender(arguments, sender);
|
|
}
|
|
|
|
void GuiToolstripCommand::OnRenderTargetChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
UpdateShortcutOwner();
|
|
}
|
|
|
|
void GuiToolstripCommand::InvokeDescriptionChanged()
|
|
{
|
|
GuiEventArgs arguments;
|
|
DescriptionChanged.Execute(arguments);
|
|
}
|
|
|
|
void GuiToolstripCommand::ReplaceShortcut(compositions::IGuiShortcutKeyItem* value, Ptr<ShortcutBuilder> builder)
|
|
{
|
|
if (shortcutKeyItem != value)
|
|
{
|
|
if (shortcutKeyItem)
|
|
{
|
|
shortcutKeyItem->Executed.Detach(shortcutKeyItemExecutedHandler);
|
|
if (shortcutBuilder)
|
|
{
|
|
auto manager = dynamic_cast<GuiShortcutKeyManager*>(shortcutOwner->GetShortcutKeyManager());
|
|
if (manager)
|
|
{
|
|
manager->DestroyShortcut(shortcutBuilder->ctrl, shortcutBuilder->shift, shortcutBuilder->alt, shortcutBuilder->key);
|
|
}
|
|
}
|
|
}
|
|
shortcutKeyItem = nullptr;
|
|
shortcutKeyItemExecutedHandler = nullptr;
|
|
shortcutBuilder = value ? builder : nullptr;
|
|
if (value)
|
|
{
|
|
shortcutKeyItem = value;
|
|
shortcutKeyItemExecutedHandler = shortcutKeyItem->Executed.AttachMethod(this, &GuiToolstripCommand::OnShortcutKeyItemExecuted);
|
|
}
|
|
InvokeDescriptionChanged();
|
|
}
|
|
}
|
|
|
|
void GuiToolstripCommand::BuildShortcut(const WString& builderText)
|
|
{
|
|
List<Ptr<ParsingError>> errors;
|
|
if (auto parser = GetParserManager()->GetParser<ShortcutBuilder>(L"SHORTCUT"))
|
|
{
|
|
if (Ptr<ShortcutBuilder> builder = parser->ParseInternal(builderText, errors))
|
|
{
|
|
if (shortcutOwner)
|
|
{
|
|
if (!shortcutOwner->GetShortcutKeyManager())
|
|
{
|
|
shortcutOwner->SetShortcutKeyManager(new GuiShortcutKeyManager);
|
|
}
|
|
if (auto manager = dynamic_cast<GuiShortcutKeyManager*>(shortcutOwner->GetShortcutKeyManager()))
|
|
{
|
|
IGuiShortcutKeyItem* item = manager->TryGetShortcut(builder->ctrl, builder->shift, builder->alt, builder->key);
|
|
if (!item)
|
|
{
|
|
item = manager->CreateShortcut(builder->ctrl, builder->shift, builder->alt, builder->key);
|
|
if (item)
|
|
{
|
|
ReplaceShortcut(item, builder);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
shortcutBuilder = builder;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiToolstripCommand::UpdateShortcutOwner()
|
|
{
|
|
GuiControlHost* host = nullptr;
|
|
if (auto control = dynamic_cast<GuiControl*>(attachedRootObject))
|
|
{
|
|
host = control->GetRelatedControlHost();
|
|
}
|
|
else if (auto composition = dynamic_cast<GuiGraphicsComposition*>(attachedRootObject))
|
|
{
|
|
host = composition->GetRelatedControlHost();
|
|
}
|
|
|
|
if (shortcutOwner != host)
|
|
{
|
|
if (shortcutOwner)
|
|
{
|
|
ReplaceShortcut(nullptr, nullptr);
|
|
shortcutOwner = nullptr;
|
|
}
|
|
shortcutOwner = host;
|
|
if (shortcutBuilder && !shortcutKeyItem)
|
|
{
|
|
BuildShortcut(shortcutBuilder->text);
|
|
}
|
|
}
|
|
}
|
|
|
|
GuiToolstripCommand::GuiToolstripCommand()
|
|
{
|
|
}
|
|
|
|
GuiToolstripCommand::~GuiToolstripCommand()
|
|
{
|
|
}
|
|
|
|
void GuiToolstripCommand::Attach(GuiInstanceRootObject* rootObject)
|
|
{
|
|
GuiGraphicsComposition* rootComposition = nullptr;
|
|
|
|
if (attachedRootObject != rootObject)
|
|
{
|
|
if (attachedRootObject)
|
|
{
|
|
if (auto control = dynamic_cast<GuiControl*>(attachedRootObject))
|
|
{
|
|
control->ControlSignalTrigerred.Detach(renderTargetChangedHandler);
|
|
}
|
|
else if (auto composition = dynamic_cast<GuiGraphicsComposition*>(attachedRootObject))
|
|
{
|
|
composition->GetEventReceiver()->renderTargetChanged.Detach(renderTargetChangedHandler);
|
|
}
|
|
renderTargetChangedHandler = nullptr;
|
|
}
|
|
|
|
attachedRootObject = rootObject;
|
|
if (attachedRootObject)
|
|
{
|
|
if (auto control = dynamic_cast<GuiControl*>(attachedRootObject))
|
|
{
|
|
renderTargetChangedHandler = control->ControlSignalTrigerred.AttachLambda(
|
|
[=](GuiGraphicsComposition* sender, GuiControlSignalEventArgs& arguments)
|
|
{
|
|
OnRenderTargetChanged(sender, arguments);
|
|
});
|
|
}
|
|
else if (auto composition = dynamic_cast<GuiGraphicsComposition*>(attachedRootObject))
|
|
{
|
|
renderTargetChangedHandler = composition->GetEventReceiver()->renderTargetChanged.AttachMethod(this, &GuiToolstripCommand::OnRenderTargetChanged);
|
|
}
|
|
}
|
|
UpdateShortcutOwner();
|
|
}
|
|
}
|
|
|
|
void GuiToolstripCommand::Detach(GuiInstanceRootObject* rootObject)
|
|
{
|
|
Attach(nullptr);
|
|
}
|
|
|
|
Ptr<GuiImageData> GuiToolstripCommand::GetLargeImage()
|
|
{
|
|
return largeImage;
|
|
}
|
|
|
|
void GuiToolstripCommand::SetLargeImage(Ptr<GuiImageData> value)
|
|
{
|
|
if (largeImage != value)
|
|
{
|
|
largeImage = value;
|
|
InvokeDescriptionChanged();
|
|
}
|
|
}
|
|
|
|
Ptr<GuiImageData> GuiToolstripCommand::GetImage()
|
|
{
|
|
return image;
|
|
}
|
|
|
|
void GuiToolstripCommand::SetImage(Ptr<GuiImageData> value)
|
|
{
|
|
if(image!=value)
|
|
{
|
|
image=value;
|
|
InvokeDescriptionChanged();
|
|
}
|
|
}
|
|
|
|
const WString& GuiToolstripCommand::GetText()
|
|
{
|
|
return text;
|
|
}
|
|
|
|
void GuiToolstripCommand::SetText(const WString& value)
|
|
{
|
|
if(text!=value)
|
|
{
|
|
text=value;
|
|
InvokeDescriptionChanged();
|
|
}
|
|
}
|
|
|
|
compositions::IGuiShortcutKeyItem* GuiToolstripCommand::GetShortcut()
|
|
{
|
|
return shortcutKeyItem;
|
|
}
|
|
|
|
void GuiToolstripCommand::SetShortcut(compositions::IGuiShortcutKeyItem* value)
|
|
{
|
|
ReplaceShortcut(value, 0);
|
|
}
|
|
|
|
WString GuiToolstripCommand::GetShortcutBuilder()
|
|
{
|
|
return shortcutBuilder ? shortcutBuilder->text : L"";
|
|
}
|
|
|
|
void GuiToolstripCommand::SetShortcutBuilder(const WString& value)
|
|
{
|
|
BuildShortcut(value);
|
|
}
|
|
|
|
bool GuiToolstripCommand::GetEnabled()
|
|
{
|
|
return enabled;
|
|
}
|
|
|
|
void GuiToolstripCommand::SetEnabled(bool value)
|
|
{
|
|
if(enabled!=value)
|
|
{
|
|
enabled=value;
|
|
InvokeDescriptionChanged();
|
|
}
|
|
}
|
|
|
|
bool GuiToolstripCommand::GetSelected()
|
|
{
|
|
return selected;
|
|
}
|
|
|
|
void GuiToolstripCommand::SetSelected(bool value)
|
|
{
|
|
if(selected!=value)
|
|
{
|
|
selected=value;
|
|
InvokeDescriptionChanged();
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiToolstripCommand::ShortcutBuilder Parser
|
|
***********************************************************************/
|
|
|
|
class GuiToolstripCommandShortcutParser : public Object, public IGuiParser<GuiToolstripCommand::ShortcutBuilder>
|
|
{
|
|
typedef GuiToolstripCommand::ShortcutBuilder ShortcutBuilder;
|
|
public:
|
|
Regex regexShortcut;
|
|
|
|
GuiToolstripCommandShortcutParser()
|
|
:regexShortcut(L"((<ctrl>Ctrl)/+|(<shift>Shift)/+|(<alt>Alt)/+)*(<key>/.+)")
|
|
{
|
|
}
|
|
|
|
Ptr<ShortcutBuilder> ParseInternal(const WString& text, collections::List<Ptr<ParsingError>>& errors)override
|
|
{
|
|
Ptr<RegexMatch> match=regexShortcut.MatchHead(text);
|
|
if (match && match->Result().Length() != text.Length())
|
|
{
|
|
errors.Add(new ParsingError(L"Failed to parse a shortcut \"" + text + L"\"."));
|
|
return 0;
|
|
}
|
|
|
|
Ptr<ShortcutBuilder> builder = new ShortcutBuilder;
|
|
builder->text = text;
|
|
builder->ctrl = match->Groups().Contains(L"ctrl");
|
|
builder->shift = match->Groups().Contains(L"shift");
|
|
builder->alt = match->Groups().Contains(L"alt");
|
|
|
|
WString name = match->Groups()[L"key"][0].Value();
|
|
builder->key = GetCurrentController()->InputService()->GetKey(name);
|
|
|
|
return builder->key == VKEY::_UNKNOWN ? nullptr : builder;
|
|
}
|
|
};
|
|
|
|
/***********************************************************************
|
|
GuiToolstripCommandPlugin
|
|
***********************************************************************/
|
|
|
|
class GuiToolstripCommandPlugin : public Object, public IGuiPlugin
|
|
{
|
|
public:
|
|
|
|
GUI_PLUGIN_NAME(GacUI_Compiler_ShortcutParser)
|
|
{
|
|
GUI_PLUGIN_DEPEND(GacUI_Parser);
|
|
}
|
|
|
|
void Load()override
|
|
{
|
|
IGuiParserManager* manager=GetParserManager();
|
|
manager->SetParser(L"SHORTCUT", new GuiToolstripCommandShortcutParser);
|
|
}
|
|
|
|
void Unload()override
|
|
{
|
|
}
|
|
};
|
|
GUI_REGISTER_PLUGIN(GuiToolstripCommandPlugin)
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\CONTROLS\TOOLSTRIPPACKAGE\GUITOOLSTRIPMENU.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace controls
|
|
{
|
|
using namespace collections;
|
|
using namespace compositions;
|
|
|
|
/***********************************************************************
|
|
GuiToolstripCollectionBase
|
|
***********************************************************************/
|
|
|
|
const wchar_t* const IToolstripUpdateLayoutInvoker::Identifier = L"vl::presentation::controls::IToolstripUpdateLayoutInvoker";
|
|
|
|
void GuiToolstripCollectionBase::InvokeUpdateLayout()
|
|
{
|
|
if(contentCallback)
|
|
{
|
|
contentCallback->UpdateLayout();
|
|
}
|
|
}
|
|
|
|
bool GuiToolstripCollectionBase::QueryInsert(vint index, GuiControl* const& child)
|
|
{
|
|
return !items.Contains(child) && !child->GetBoundsComposition()->GetParent();
|
|
}
|
|
|
|
void GuiToolstripCollectionBase::BeforeRemove(vint index, GuiControl* const& child)
|
|
{
|
|
if (auto invoker = child->QueryTypedService<IToolstripUpdateLayoutInvoker>())
|
|
{
|
|
invoker->SetCallback(nullptr);
|
|
}
|
|
InvokeUpdateLayout();
|
|
}
|
|
|
|
void GuiToolstripCollectionBase::AfterInsert(vint index, GuiControl* const& child)
|
|
{
|
|
if (auto invoker = child->QueryTypedService<IToolstripUpdateLayoutInvoker>())
|
|
{
|
|
invoker->SetCallback(contentCallback);
|
|
}
|
|
InvokeUpdateLayout();
|
|
}
|
|
|
|
void GuiToolstripCollectionBase::AfterRemove(vint index, vint count)
|
|
{
|
|
InvokeUpdateLayout();
|
|
}
|
|
|
|
GuiToolstripCollectionBase::GuiToolstripCollectionBase(IToolstripUpdateLayout* _contentCallback)
|
|
:contentCallback(_contentCallback)
|
|
{
|
|
}
|
|
|
|
GuiToolstripCollectionBase::~GuiToolstripCollectionBase()
|
|
{
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiToolstripCollection
|
|
***********************************************************************/
|
|
|
|
void GuiToolstripCollection::UpdateItemVisibility(vint index, GuiControl* child)
|
|
{
|
|
auto stackItem = stackComposition->GetStackItems()[index];
|
|
if (child->GetVisible())
|
|
{
|
|
stackItem->SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
child->GetBoundsComposition()->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
}
|
|
else
|
|
{
|
|
stackItem->SetMinSizeLimitation(GuiGraphicsComposition::NoLimit);
|
|
child->GetBoundsComposition()->SetAlignmentToParent(Margin(-1, -1, -1, -1));
|
|
}
|
|
}
|
|
|
|
void GuiToolstripCollection::OnItemVisibleChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
auto child = sender->GetRelatedControl();
|
|
vint index = IndexOf(child);
|
|
UpdateItemVisibility(index, child);
|
|
InvokeUpdateLayout();
|
|
}
|
|
|
|
void GuiToolstripCollection::BeforeRemove(vint index, GuiControl* const& child)
|
|
{
|
|
GuiStackItemComposition* stackItem = stackComposition->GetStackItems().Get(index);
|
|
|
|
auto eventHandler = eventHandlers[index];
|
|
child->VisibleChanged.Detach(eventHandler);
|
|
eventHandlers.RemoveAt(index);
|
|
|
|
GuiToolstripCollectionBase::BeforeRemove(index, child);
|
|
}
|
|
|
|
void GuiToolstripCollection::AfterInsert(vint index, GuiControl* const& child)
|
|
{
|
|
{
|
|
GuiStackItemComposition* stackItem = new GuiStackItemComposition;
|
|
stackItem->AddChild(child->GetBoundsComposition());
|
|
stackComposition->InsertStackItem(index, stackItem);
|
|
}
|
|
{
|
|
auto eventHandler = child->VisibleChanged.AttachMethod(this, &GuiToolstripCollection::OnItemVisibleChanged);
|
|
eventHandlers.Insert(index, eventHandler);
|
|
}
|
|
UpdateItemVisibility(index, child);
|
|
GuiToolstripCollectionBase::AfterInsert(index, child);
|
|
}
|
|
|
|
void GuiToolstripCollection::AfterRemove(vint index, vint count)
|
|
{
|
|
for (vint i = 0; i < count; i++)
|
|
{
|
|
auto stackItem = stackComposition->GetStackItems().Get(index);
|
|
stackComposition->RemoveChild(stackItem);
|
|
SafeDeleteComposition(stackItem);
|
|
}
|
|
GuiToolstripCollectionBase::AfterRemove(index, count);
|
|
}
|
|
|
|
GuiToolstripCollection::GuiToolstripCollection(IToolstripUpdateLayout* _contentCallback, compositions::GuiStackComposition* _stackComposition)
|
|
:GuiToolstripCollectionBase(_contentCallback)
|
|
,stackComposition(_stackComposition)
|
|
{
|
|
}
|
|
|
|
GuiToolstripCollection::~GuiToolstripCollection()
|
|
{
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiToolstripMenu
|
|
***********************************************************************/
|
|
|
|
void GuiToolstripMenu::UpdateLayout()
|
|
{
|
|
sharedSizeRootComposition->ForceCalculateSizeImmediately();
|
|
}
|
|
|
|
GuiToolstripMenu::GuiToolstripMenu(theme::ThemeName themeName, GuiControl* _owner)
|
|
:GuiMenu(themeName, _owner)
|
|
{
|
|
sharedSizeRootComposition = new GuiSharedSizeRootComposition();
|
|
sharedSizeRootComposition->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
sharedSizeRootComposition->SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
containerComposition->AddChild(sharedSizeRootComposition);
|
|
|
|
stackComposition=new GuiStackComposition;
|
|
stackComposition->SetDirection(GuiStackComposition::Vertical);
|
|
stackComposition->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
stackComposition->SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
sharedSizeRootComposition->AddChild(stackComposition);
|
|
|
|
toolstripItems = new GuiToolstripCollection(this, stackComposition);
|
|
}
|
|
|
|
GuiToolstripMenu::~GuiToolstripMenu()
|
|
{
|
|
}
|
|
|
|
collections::ObservableListBase<GuiControl*>& GuiToolstripMenu::GetToolstripItems()
|
|
{
|
|
return *toolstripItems.Obj();
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiToolstripMenuBar
|
|
***********************************************************************/
|
|
|
|
GuiToolstripMenuBar::GuiToolstripMenuBar(theme::ThemeName themeName)
|
|
:GuiMenuBar(themeName)
|
|
{
|
|
stackComposition=new GuiStackComposition;
|
|
stackComposition->SetDirection(GuiStackComposition::Horizontal);
|
|
stackComposition->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
stackComposition->SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
containerComposition->AddChild(stackComposition);
|
|
|
|
toolstripItems=new GuiToolstripCollection(nullptr, stackComposition);
|
|
}
|
|
|
|
GuiToolstripMenuBar::~GuiToolstripMenuBar()
|
|
{
|
|
}
|
|
|
|
collections::ObservableListBase<GuiControl*>& GuiToolstripMenuBar::GetToolstripItems()
|
|
{
|
|
return *toolstripItems.Obj();
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiToolstripToolBar
|
|
***********************************************************************/
|
|
|
|
GuiToolstripToolBar::GuiToolstripToolBar(theme::ThemeName themeName)
|
|
:GuiControl(themeName)
|
|
{
|
|
stackComposition=new GuiStackComposition;
|
|
stackComposition->SetDirection(GuiStackComposition::Horizontal);
|
|
stackComposition->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
stackComposition->SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
containerComposition->AddChild(stackComposition);
|
|
|
|
toolstripItems=new GuiToolstripCollection(nullptr, stackComposition);
|
|
}
|
|
|
|
GuiToolstripToolBar::~GuiToolstripToolBar()
|
|
{
|
|
}
|
|
|
|
collections::ObservableListBase<GuiControl*>& GuiToolstripToolBar::GetToolstripItems()
|
|
{
|
|
return *toolstripItems.Obj();
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiToolstripButton
|
|
***********************************************************************/
|
|
|
|
void GuiToolstripButton::SetCallback(IToolstripUpdateLayout* _callback)
|
|
{
|
|
callback = _callback;
|
|
}
|
|
|
|
void GuiToolstripButton::OnActiveAlt()
|
|
{
|
|
auto host = GetSubMenuHost();
|
|
if (host == this)
|
|
{
|
|
GuiMenuButton::OnActiveAlt();
|
|
}
|
|
else
|
|
{
|
|
host->QueryTypedService<IGuiAltAction>()->OnActiveAlt();
|
|
}
|
|
}
|
|
|
|
void GuiToolstripButton::UpdateCommandContent()
|
|
{
|
|
if(command)
|
|
{
|
|
SetLargeImage(command->GetLargeImage());
|
|
SetImage(command->GetImage());
|
|
SetText(command->GetText());
|
|
SetEnabled(command->GetEnabled());
|
|
SetSelected(command->GetSelected());
|
|
if(command->GetShortcut())
|
|
{
|
|
SetShortcutText(command->GetShortcut()->GetName());
|
|
}
|
|
else
|
|
{
|
|
SetShortcutText(L"");
|
|
}
|
|
}
|
|
else
|
|
{
|
|
SetLargeImage(nullptr);
|
|
SetImage(nullptr);
|
|
SetText(L"");
|
|
SetEnabled(true);
|
|
SetSelected(false);
|
|
SetShortcutText(L"");
|
|
}
|
|
}
|
|
|
|
void GuiToolstripButton::OnLayoutAwaredPropertyChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
if (callback)
|
|
{
|
|
callback->UpdateLayout();
|
|
}
|
|
}
|
|
|
|
void GuiToolstripButton::OnClicked(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
if(command)
|
|
{
|
|
command->Executed.ExecuteWithNewSender(arguments, sender);
|
|
}
|
|
}
|
|
|
|
void GuiToolstripButton::OnCommandDescriptionChanged(compositions::GuiGraphicsComposition* sender, compositions::GuiEventArgs& arguments)
|
|
{
|
|
UpdateCommandContent();
|
|
}
|
|
|
|
GuiToolstripButton::GuiToolstripButton(theme::ThemeName themeName)
|
|
:GuiMenuButton(themeName)
|
|
,command(0)
|
|
{
|
|
SetAutoFocus(false);
|
|
Clicked.AttachMethod(this, &GuiToolstripButton::OnClicked);
|
|
TextChanged.AttachMethod(this, &GuiToolstripButton::OnLayoutAwaredPropertyChanged);
|
|
ShortcutTextChanged.AttachMethod(this, &GuiToolstripButton::OnLayoutAwaredPropertyChanged);
|
|
}
|
|
|
|
GuiToolstripButton::~GuiToolstripButton()
|
|
{
|
|
}
|
|
|
|
GuiToolstripCommand* GuiToolstripButton::GetCommand()
|
|
{
|
|
return command;
|
|
}
|
|
|
|
void GuiToolstripButton::SetCommand(GuiToolstripCommand* value)
|
|
{
|
|
if(command!=value)
|
|
{
|
|
if(command)
|
|
{
|
|
command->DescriptionChanged.Detach(descriptionChangedHandler);
|
|
}
|
|
command=0;
|
|
descriptionChangedHandler=0;
|
|
if(value)
|
|
{
|
|
command=value;
|
|
descriptionChangedHandler=command->DescriptionChanged.AttachMethod(this, &GuiToolstripButton::OnCommandDescriptionChanged);
|
|
}
|
|
UpdateCommandContent();
|
|
}
|
|
}
|
|
|
|
GuiToolstripMenu* GuiToolstripButton::GetToolstripSubMenu()
|
|
{
|
|
return dynamic_cast<GuiToolstripMenu*>(GetSubMenu());
|
|
}
|
|
|
|
GuiToolstripMenu* GuiToolstripButton::EnsureToolstripSubMenu()
|
|
{
|
|
if (!GetSubMenu())
|
|
{
|
|
CreateToolstripSubMenu({});
|
|
}
|
|
return dynamic_cast<GuiToolstripMenu*>(GetSubMenu());
|
|
}
|
|
|
|
void GuiToolstripButton::CreateToolstripSubMenu(TemplateProperty<templates::GuiMenuTemplate> subMenuTemplate)
|
|
{
|
|
if (!subMenu)
|
|
{
|
|
auto newSubMenu = new GuiToolstripMenu(theme::ThemeName::Menu, this);
|
|
if (subMenuTemplate)
|
|
{
|
|
newSubMenu->SetControlTemplate(subMenuTemplate);
|
|
}
|
|
else
|
|
{
|
|
newSubMenu->SetControlTemplate(GetControlTemplateObject(true)->GetSubMenuTemplate());
|
|
}
|
|
SetSubMenu(newSubMenu, true);
|
|
}
|
|
}
|
|
|
|
IDescriptable* GuiToolstripButton::QueryService(const WString& identifier)
|
|
{
|
|
if (identifier == IToolstripUpdateLayoutInvoker::Identifier)
|
|
{
|
|
return (IToolstripUpdateLayoutInvoker*)this;
|
|
}
|
|
else
|
|
{
|
|
return GuiMenuButton::QueryService(identifier);
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiToolstripNestedContainer
|
|
***********************************************************************/
|
|
|
|
void GuiToolstripNestedContainer::UpdateLayout()
|
|
{
|
|
if (callback)
|
|
{
|
|
callback->UpdateLayout();
|
|
}
|
|
}
|
|
|
|
void GuiToolstripNestedContainer::SetCallback(IToolstripUpdateLayout* _callback)
|
|
{
|
|
callback = _callback;
|
|
}
|
|
|
|
GuiToolstripNestedContainer::GuiToolstripNestedContainer(theme::ThemeName themeName)
|
|
:GuiControl(themeName)
|
|
{
|
|
}
|
|
|
|
GuiToolstripNestedContainer::~GuiToolstripNestedContainer()
|
|
{
|
|
}
|
|
|
|
IDescriptable* GuiToolstripNestedContainer::QueryService(const WString& identifier)
|
|
{
|
|
if (identifier == IToolstripUpdateLayoutInvoker::Identifier)
|
|
{
|
|
return (IToolstripUpdateLayoutInvoker*)this;
|
|
}
|
|
else
|
|
{
|
|
return GuiControl::QueryService(identifier);
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiToolstripGroupContainer::GroupCollection
|
|
***********************************************************************/
|
|
|
|
void GuiToolstripGroupContainer::GroupCollection::BeforeRemove(vint index, GuiControl* const& child)
|
|
{
|
|
auto controlStackItem = container->stackComposition->GetStackItems()[index * 2];
|
|
CHECK_ERROR(controlStackItem->RemoveChild(child->GetBoundsComposition()), L"GuiToolstripGroupContainer::GroupCollection::BeforeRemove(vint, GuiControl* const&)#Internal error");
|
|
}
|
|
|
|
void GuiToolstripGroupContainer::GroupCollection::AfterInsert(vint index, GuiControl* const& child)
|
|
{
|
|
auto controlStackItem = new GuiStackItemComposition;
|
|
child->GetBoundsComposition()->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
CHECK_ERROR(controlStackItem->AddChild(child->GetBoundsComposition()), L"GuiToolstripGroupContainer::GroupCollection::AfterInsert(vint, GuiControl* const&)#Internal error");
|
|
|
|
if (container->stackComposition->GetStackItems().Count() > 0)
|
|
{
|
|
auto splitterStackItem = new GuiStackItemComposition;
|
|
auto splitter = new GuiControl(container->splitterThemeName);
|
|
if (splitterTemplate)
|
|
{
|
|
splitter->SetControlTemplate(splitterTemplate);
|
|
}
|
|
splitter->GetBoundsComposition()->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
splitterStackItem->AddChild(splitter->GetBoundsComposition());
|
|
container->stackComposition->InsertStackItem(index == 0 ? 0 : index * 2 - 1, splitterStackItem);
|
|
}
|
|
|
|
container->stackComposition->InsertStackItem(index * 2, controlStackItem);
|
|
|
|
GuiToolstripCollectionBase::AfterInsert(index, child);
|
|
}
|
|
|
|
void GuiToolstripGroupContainer::GroupCollection::AfterRemove(vint index, vint count)
|
|
{
|
|
vint min = index * 2;
|
|
vint max = (index + count - 1) * 2;
|
|
for (vint i = min; i <= max; i++)
|
|
{
|
|
auto stackItem = container->stackComposition->GetStackItems()[min];
|
|
container->stackComposition->RemoveChild(stackItem);
|
|
SafeDeleteComposition(stackItem);
|
|
}
|
|
GuiToolstripCollectionBase::AfterRemove(index, count);
|
|
}
|
|
|
|
GuiToolstripGroupContainer::GroupCollection::GroupCollection(GuiToolstripGroupContainer* _container)
|
|
:GuiToolstripCollectionBase(_container)
|
|
, container(_container)
|
|
{
|
|
}
|
|
|
|
GuiToolstripGroupContainer::GroupCollection::~GroupCollection()
|
|
{
|
|
}
|
|
|
|
GuiToolstripGroupContainer::ControlTemplatePropertyType GuiToolstripGroupContainer::GroupCollection::GetSplitterTemplate()
|
|
{
|
|
return splitterTemplate;
|
|
}
|
|
|
|
void GuiToolstripGroupContainer::GroupCollection::SetSplitterTemplate(const ControlTemplatePropertyType& value)
|
|
{
|
|
splitterTemplate = value;
|
|
RebuildSplitters();
|
|
}
|
|
|
|
void GuiToolstripGroupContainer::GroupCollection::RebuildSplitters()
|
|
{
|
|
auto stack = container->stackComposition;
|
|
vint count = stack->GetStackItems().Count();
|
|
for (vint i = 1; i < count; i += 2)
|
|
{
|
|
auto stackItem = stack->GetStackItems()[i];
|
|
{
|
|
auto control = stackItem->Children()[0]->GetAssociatedControl();
|
|
CHECK_ERROR(control != nullptr, L"GuiToolstripGroupContainer::GroupCollection::RebuildSplitters()#Internal error");
|
|
stackItem->RemoveChild(control->GetBoundsComposition());
|
|
delete control;
|
|
}
|
|
{
|
|
auto control = new GuiControl(container->splitterThemeName);
|
|
if (splitterTemplate)
|
|
{
|
|
control->SetControlTemplate(splitterTemplate);
|
|
}
|
|
control->GetBoundsComposition()->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
stackItem->AddChild(control->GetBoundsComposition());
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiToolstripGroupContainer
|
|
***********************************************************************/
|
|
|
|
void GuiToolstripGroupContainer::OnParentLineChanged()
|
|
{
|
|
auto direction = GuiStackComposition::Horizontal;
|
|
if (auto service = QueryTypedService<IGuiMenuService>())
|
|
{
|
|
if (service->GetPreferredDirection() == IGuiMenuService::Vertical)
|
|
{
|
|
direction = GuiStackComposition::Vertical;
|
|
}
|
|
}
|
|
|
|
if (direction != stackComposition->GetDirection())
|
|
{
|
|
if (direction == GuiStackComposition::Vertical)
|
|
{
|
|
splitterThemeName = theme::ThemeName::MenuSplitter;
|
|
}
|
|
else
|
|
{
|
|
splitterThemeName = theme::ThemeName::ToolstripSplitter;
|
|
}
|
|
|
|
stackComposition->SetDirection(direction);
|
|
splitterThemeName = splitterThemeName;
|
|
groupCollection->RebuildSplitters();
|
|
UpdateLayout();
|
|
}
|
|
|
|
GuiControl::OnParentLineChanged();
|
|
}
|
|
|
|
GuiToolstripGroupContainer::GuiToolstripGroupContainer(theme::ThemeName themeName)
|
|
:GuiToolstripNestedContainer(themeName)
|
|
, splitterThemeName(theme::ThemeName::ToolstripSplitter)
|
|
{
|
|
stackComposition = new GuiStackComposition;
|
|
stackComposition->SetDirection(GuiStackComposition::Horizontal);
|
|
stackComposition->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
stackComposition->SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
containerComposition->AddChild(stackComposition);
|
|
|
|
groupCollection = new GroupCollection(this);
|
|
}
|
|
|
|
GuiToolstripGroupContainer::~GuiToolstripGroupContainer()
|
|
{
|
|
}
|
|
|
|
GuiToolstripGroupContainer::ControlTemplatePropertyType GuiToolstripGroupContainer::GetSplitterTemplate()
|
|
{
|
|
return groupCollection->GetSplitterTemplate();
|
|
}
|
|
|
|
void GuiToolstripGroupContainer::SetSplitterTemplate(const ControlTemplatePropertyType& value)
|
|
{
|
|
groupCollection->SetSplitterTemplate(value);
|
|
}
|
|
|
|
collections::ObservableListBase<GuiControl*>& GuiToolstripGroupContainer::GetToolstripItems()
|
|
{
|
|
return *groupCollection.Obj();
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiToolstripGroup
|
|
***********************************************************************/
|
|
|
|
void GuiToolstripGroup::OnParentLineChanged()
|
|
{
|
|
auto direction = GuiStackComposition::Horizontal;
|
|
if (auto service = QueryTypedService<IGuiMenuService>())
|
|
{
|
|
if (service->GetPreferredDirection() == IGuiMenuService::Vertical)
|
|
{
|
|
direction = GuiStackComposition::Vertical;
|
|
}
|
|
}
|
|
|
|
if (direction != stackComposition->GetDirection())
|
|
{
|
|
stackComposition->SetDirection(direction);
|
|
UpdateLayout();
|
|
}
|
|
|
|
GuiControl::OnParentLineChanged();
|
|
}
|
|
|
|
GuiToolstripGroup::GuiToolstripGroup(theme::ThemeName themeName)
|
|
:GuiToolstripNestedContainer(themeName)
|
|
{
|
|
stackComposition = new GuiStackComposition;
|
|
stackComposition->SetDirection(GuiStackComposition::Horizontal);
|
|
stackComposition->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
stackComposition->SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
containerComposition->AddChild(stackComposition);
|
|
|
|
toolstripItems = new GuiToolstripCollection(nullptr, stackComposition);
|
|
}
|
|
|
|
GuiToolstripGroup::~GuiToolstripGroup()
|
|
{
|
|
}
|
|
|
|
collections::ObservableListBase<GuiControl*>& GuiToolstripGroup::GetToolstripItems()
|
|
{
|
|
return *toolstripItems.Obj();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\GRAPHICSCOMPOSITION\GUIGRAPHICSAXIS.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace compositions
|
|
{
|
|
|
|
/***********************************************************************
|
|
GuiDefaultAxis
|
|
***********************************************************************/
|
|
|
|
GuiDefaultAxis::GuiDefaultAxis()
|
|
{
|
|
}
|
|
|
|
GuiDefaultAxis::~GuiDefaultAxis()
|
|
{
|
|
}
|
|
|
|
Size GuiDefaultAxis::RealSizeToVirtualSize(Size size)
|
|
{
|
|
return size;
|
|
}
|
|
|
|
Size GuiDefaultAxis::VirtualSizeToRealSize(Size size)
|
|
{
|
|
return size;
|
|
}
|
|
|
|
Point GuiDefaultAxis::RealPointToVirtualPoint(Size realFullSize, Point point)
|
|
{
|
|
return point;
|
|
}
|
|
|
|
Point GuiDefaultAxis::VirtualPointToRealPoint(Size realFullSize, Point point)
|
|
{
|
|
return point;
|
|
}
|
|
|
|
Rect GuiDefaultAxis::RealRectToVirtualRect(Size realFullSize, Rect rect)
|
|
{
|
|
return rect;
|
|
}
|
|
|
|
Rect GuiDefaultAxis::VirtualRectToRealRect(Size realFullSize, Rect rect)
|
|
{
|
|
return rect;
|
|
}
|
|
|
|
Margin GuiDefaultAxis::RealMarginToVirtualMargin(Margin margin)
|
|
{
|
|
return margin;
|
|
}
|
|
|
|
Margin GuiDefaultAxis::VirtualMarginToRealMargin(Margin margin)
|
|
{
|
|
return margin;
|
|
}
|
|
|
|
KeyDirection GuiDefaultAxis::RealKeyDirectionToVirtualKeyDirection(KeyDirection key)
|
|
{
|
|
return key;
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiAxis
|
|
***********************************************************************/
|
|
|
|
GuiAxis::GuiAxis(AxisDirection _axisDirection)
|
|
:axisDirection(_axisDirection)
|
|
{
|
|
}
|
|
|
|
GuiAxis::~GuiAxis()
|
|
{
|
|
}
|
|
|
|
AxisDirection GuiAxis::GetDirection()
|
|
{
|
|
return axisDirection;
|
|
}
|
|
|
|
Size GuiAxis::RealSizeToVirtualSize(Size size)
|
|
{
|
|
switch(axisDirection)
|
|
{
|
|
case AxisDirection::LeftDown:
|
|
case AxisDirection::RightDown:
|
|
case AxisDirection::LeftUp:
|
|
case AxisDirection::RightUp:
|
|
return Size(size.x, size.y);
|
|
case AxisDirection::DownLeft:
|
|
case AxisDirection::DownRight:
|
|
case AxisDirection::UpLeft:
|
|
case AxisDirection::UpRight:
|
|
return Size(size.y, size.x);
|
|
}
|
|
return size;
|
|
}
|
|
|
|
Size GuiAxis::VirtualSizeToRealSize(Size size)
|
|
{
|
|
return RealSizeToVirtualSize(size);
|
|
}
|
|
|
|
Point GuiAxis::RealPointToVirtualPoint(Size realFullSize, Point point)
|
|
{
|
|
Rect rect(point, Size(0, 0));
|
|
return RealRectToVirtualRect(realFullSize, rect).LeftTop();
|
|
}
|
|
|
|
Point GuiAxis::VirtualPointToRealPoint(Size realFullSize, Point point)
|
|
{
|
|
Rect rect(point, Size(0, 0));
|
|
return VirtualRectToRealRect(realFullSize, rect).LeftTop();
|
|
}
|
|
|
|
Rect GuiAxis::RealRectToVirtualRect(Size realFullSize, Rect rect)
|
|
{
|
|
vint x1=rect.x1;
|
|
vint x2=realFullSize.x-rect.x2;
|
|
vint y1=rect.y1;
|
|
vint y2=realFullSize.y-rect.y2;
|
|
vint w=rect.Width();
|
|
vint h=rect.Height();
|
|
switch(axisDirection)
|
|
{
|
|
case AxisDirection::LeftDown:
|
|
return Rect(Point(x2, y1), Size(w, h));
|
|
case AxisDirection::RightDown:
|
|
return Rect(Point(x1, y1), Size(w, h));
|
|
case AxisDirection::LeftUp:
|
|
return Rect(Point(x2, y2), Size(w, h));
|
|
case AxisDirection::RightUp:
|
|
return Rect(Point(x1, y2), Size(w, h));
|
|
case AxisDirection::DownLeft:
|
|
return Rect(Point(y1, x2), Size(h, w));
|
|
case AxisDirection::DownRight:
|
|
return Rect(Point(y1, x1), Size(h, w));
|
|
case AxisDirection::UpLeft:
|
|
return Rect(Point(y2, x2), Size(h, w));
|
|
case AxisDirection::UpRight:
|
|
return Rect(Point(y2, x1), Size(h, w));
|
|
}
|
|
return rect;
|
|
}
|
|
|
|
Rect GuiAxis::VirtualRectToRealRect(Size realFullSize, Rect rect)
|
|
{
|
|
realFullSize=RealSizeToVirtualSize(realFullSize);
|
|
vint x1=rect.x1;
|
|
vint x2=realFullSize.x-rect.x2;
|
|
vint y1=rect.y1;
|
|
vint y2=realFullSize.y-rect.y2;
|
|
vint w=rect.Width();
|
|
vint h=rect.Height();
|
|
switch(axisDirection)
|
|
{
|
|
case AxisDirection::LeftDown:
|
|
return Rect(Point(x2, y1), Size(w, h));
|
|
case AxisDirection::RightDown:
|
|
return Rect(Point(x1, y1), Size(w, h));
|
|
case AxisDirection::LeftUp:
|
|
return Rect(Point(x2, y2), Size(w, h));
|
|
case AxisDirection::RightUp:
|
|
return Rect(Point(x1, y2), Size(w, h));
|
|
case AxisDirection::DownLeft:
|
|
return Rect(Point(y2, x1), Size(h, w));
|
|
case AxisDirection::DownRight:
|
|
return Rect(Point(y1, x1), Size(h, w));
|
|
case AxisDirection::UpLeft:
|
|
return Rect(Point(y2, x2), Size(h, w));
|
|
case AxisDirection::UpRight:
|
|
return Rect(Point(y1, x2), Size(h, w));
|
|
}
|
|
return rect;
|
|
}
|
|
|
|
Margin GuiAxis::RealMarginToVirtualMargin(Margin margin)
|
|
{
|
|
vint x1=margin.left;
|
|
vint x2=margin.right;
|
|
vint y1=margin.top;
|
|
vint y2=margin.bottom;
|
|
switch(axisDirection)
|
|
{
|
|
case AxisDirection::LeftDown:
|
|
return Margin(x2, y1, x1, y2);
|
|
case AxisDirection::RightDown:
|
|
return Margin(x1, y1, x2, y2);
|
|
case AxisDirection::LeftUp:
|
|
return Margin(x2, y2, x1, y1);
|
|
case AxisDirection::RightUp:
|
|
return Margin(x1, y2, x2, y1);
|
|
case AxisDirection::DownLeft:
|
|
return Margin(y1, x2, y2, x1);
|
|
case AxisDirection::DownRight:
|
|
return Margin(y1, x1, y2, x2);
|
|
case AxisDirection::UpLeft:
|
|
return Margin(y2, x2, y1, x1);
|
|
case AxisDirection::UpRight:
|
|
return Margin(y2, x1, y1, x2);
|
|
}
|
|
return margin;
|
|
}
|
|
|
|
Margin GuiAxis::VirtualMarginToRealMargin(Margin margin)
|
|
{
|
|
vint x1=margin.left;
|
|
vint x2=margin.right;
|
|
vint y1=margin.top;
|
|
vint y2=margin.bottom;
|
|
switch(axisDirection)
|
|
{
|
|
case AxisDirection::LeftDown:
|
|
return Margin(x2, y1, x1, y2);
|
|
case AxisDirection::RightDown:
|
|
return Margin(x1, y1, x2, y2);
|
|
case AxisDirection::LeftUp:
|
|
return Margin(x2, y2, x1, y1);
|
|
case AxisDirection::RightUp:
|
|
return Margin(x1, y2, x2, y1);
|
|
case AxisDirection::DownLeft:
|
|
return Margin(y2, x1, y1, x2);
|
|
case AxisDirection::DownRight:
|
|
return Margin(y1, x1, y2, x2);
|
|
case AxisDirection::UpLeft:
|
|
return Margin(y2, x2, y1, x1);
|
|
case AxisDirection::UpRight:
|
|
return Margin(y1, x2, y2, x1);
|
|
default:;
|
|
}
|
|
return margin;
|
|
}
|
|
|
|
KeyDirection GuiAxis::RealKeyDirectionToVirtualKeyDirection(KeyDirection key)
|
|
{
|
|
bool pageKey=false;
|
|
switch(key)
|
|
{
|
|
case KeyDirection::PageUp:
|
|
pageKey=true;
|
|
key=KeyDirection::Up;
|
|
break;
|
|
case KeyDirection::PageDown:
|
|
pageKey=true;
|
|
key=KeyDirection::Down;
|
|
break;
|
|
case KeyDirection::PageLeft:
|
|
pageKey=true;
|
|
key=KeyDirection::Left;
|
|
break;
|
|
case KeyDirection::PageRight:
|
|
pageKey=true;
|
|
key=KeyDirection::Right;
|
|
break;
|
|
default:;
|
|
}
|
|
|
|
switch(key)
|
|
{
|
|
case KeyDirection::Up:
|
|
switch(axisDirection)
|
|
{
|
|
case AxisDirection::LeftDown: key=KeyDirection::Up; break;
|
|
case AxisDirection::RightDown: key=KeyDirection::Up; break;
|
|
case AxisDirection::LeftUp: key=KeyDirection::Down; break;
|
|
case AxisDirection::RightUp: key=KeyDirection::Down; break;
|
|
case AxisDirection::DownLeft: key=KeyDirection::Left; break;
|
|
case AxisDirection::DownRight: key=KeyDirection::Left; break;
|
|
case AxisDirection::UpLeft: key=KeyDirection::Right; break;
|
|
case AxisDirection::UpRight: key=KeyDirection::Right; break;
|
|
}
|
|
break;
|
|
case KeyDirection::Down:
|
|
switch(axisDirection)
|
|
{
|
|
case AxisDirection::LeftDown: key=KeyDirection::Down; break;
|
|
case AxisDirection::RightDown: key=KeyDirection::Down; break;
|
|
case AxisDirection::LeftUp: key=KeyDirection::Up; break;
|
|
case AxisDirection::RightUp: key=KeyDirection::Up; break;
|
|
case AxisDirection::DownLeft: key=KeyDirection::Right; break;
|
|
case AxisDirection::DownRight: key=KeyDirection::Right; break;
|
|
case AxisDirection::UpLeft: key=KeyDirection::Left; break;
|
|
case AxisDirection::UpRight: key=KeyDirection::Left; break;
|
|
}
|
|
break;
|
|
case KeyDirection::Left:
|
|
switch(axisDirection)
|
|
{
|
|
case AxisDirection::LeftDown: key=KeyDirection::Right; break;
|
|
case AxisDirection::RightDown: key=KeyDirection::Left; break;
|
|
case AxisDirection::LeftUp: key=KeyDirection::Right; break;
|
|
case AxisDirection::RightUp: key=KeyDirection::Left; break;
|
|
case AxisDirection::DownLeft: key=KeyDirection::Down; break;
|
|
case AxisDirection::DownRight: key=KeyDirection::Up; break;
|
|
case AxisDirection::UpLeft: key=KeyDirection::Down; break;
|
|
case AxisDirection::UpRight: key=KeyDirection::Up; break;
|
|
}
|
|
break;
|
|
case KeyDirection::Right:
|
|
switch(axisDirection)
|
|
{
|
|
case AxisDirection::LeftDown: key=KeyDirection::Left; break;
|
|
case AxisDirection::RightDown: key=KeyDirection::Right; break;
|
|
case AxisDirection::LeftUp: key=KeyDirection::Left; break;
|
|
case AxisDirection::RightUp: key=KeyDirection::Right; break;
|
|
case AxisDirection::DownLeft: key=KeyDirection::Up; break;
|
|
case AxisDirection::DownRight: key=KeyDirection::Down; break;
|
|
case AxisDirection::UpLeft: key=KeyDirection::Up; break;
|
|
case AxisDirection::UpRight: key=KeyDirection::Down; break;
|
|
}
|
|
break;
|
|
case KeyDirection::Home:
|
|
switch(axisDirection)
|
|
{
|
|
case AxisDirection::LeftDown: key=KeyDirection::Home; break;
|
|
case AxisDirection::RightDown: key=KeyDirection::Home; break;
|
|
case AxisDirection::LeftUp: key=KeyDirection::End; break;
|
|
case AxisDirection::RightUp: key=KeyDirection::End; break;
|
|
case AxisDirection::DownLeft: key=KeyDirection::Home; break;
|
|
case AxisDirection::DownRight: key=KeyDirection::Home; break;
|
|
case AxisDirection::UpLeft: key=KeyDirection::End; break;
|
|
case AxisDirection::UpRight: key=KeyDirection::End; break;
|
|
}
|
|
break;
|
|
case KeyDirection::End:
|
|
switch(axisDirection)
|
|
{
|
|
case AxisDirection::LeftDown: key=KeyDirection::End; break;
|
|
case AxisDirection::RightDown: key=KeyDirection::End; break;
|
|
case AxisDirection::LeftUp: key=KeyDirection::Home; break;
|
|
case AxisDirection::RightUp: key=KeyDirection::Home; break;
|
|
case AxisDirection::DownLeft: key=KeyDirection::End; break;
|
|
case AxisDirection::DownRight: key=KeyDirection::End; break;
|
|
case AxisDirection::UpLeft: key=KeyDirection::Home; break;
|
|
case AxisDirection::UpRight: key=KeyDirection::Home; break;
|
|
}
|
|
break;
|
|
default:;
|
|
}
|
|
|
|
if(pageKey)
|
|
{
|
|
switch(key)
|
|
{
|
|
case KeyDirection::Up:
|
|
key=KeyDirection::PageUp;
|
|
break;
|
|
case KeyDirection::Down:
|
|
key=KeyDirection::PageDown;
|
|
break;
|
|
case KeyDirection::Left:
|
|
key=KeyDirection::PageLeft;
|
|
break;
|
|
case KeyDirection::Right:
|
|
key=KeyDirection::PageRight;
|
|
break;
|
|
default:;
|
|
}
|
|
}
|
|
return key;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\GRAPHICSCOMPOSITION\GUIGRAPHICSBASICCOMPOSITION.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace compositions
|
|
{
|
|
using namespace collections;
|
|
using namespace controls;
|
|
using namespace elements;
|
|
|
|
/***********************************************************************
|
|
GuiWindowComposition
|
|
***********************************************************************/
|
|
|
|
GuiWindowComposition::GuiWindowComposition()
|
|
{
|
|
}
|
|
|
|
GuiWindowComposition::~GuiWindowComposition()
|
|
{
|
|
}
|
|
|
|
Rect GuiWindowComposition::GetBounds()
|
|
{
|
|
Rect bounds;
|
|
if (relatedHostRecord)
|
|
{
|
|
if (auto window = relatedHostRecord->host->GetNativeWindow())
|
|
{
|
|
bounds = Rect(Point(0, 0), window->Convert(window->GetClientSize()));
|
|
}
|
|
}
|
|
UpdatePreviousBounds(bounds);
|
|
return bounds;
|
|
}
|
|
|
|
void GuiWindowComposition::SetMargin(Margin value)
|
|
{
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiBoundsComposition
|
|
***********************************************************************/
|
|
|
|
GuiBoundsComposition::GuiBoundsComposition()
|
|
{
|
|
}
|
|
|
|
GuiBoundsComposition::~GuiBoundsComposition()
|
|
{
|
|
}
|
|
|
|
bool GuiBoundsComposition::GetSizeAffectParent()
|
|
{
|
|
return sizeAffectParent;
|
|
}
|
|
|
|
void GuiBoundsComposition::SetSizeAffectParent(bool value)
|
|
{
|
|
sizeAffectParent = value;
|
|
}
|
|
|
|
bool GuiBoundsComposition::IsSizeAffectParent()
|
|
{
|
|
return sizeAffectParent;
|
|
}
|
|
|
|
Rect GuiBoundsComposition::GetPreferredBounds()
|
|
{
|
|
Rect result = GetBoundsInternal(compositionBounds);
|
|
if (GetParent() && IsAlignedToParent())
|
|
{
|
|
if (alignmentToParent.left >= 0)
|
|
{
|
|
vint offset = alignmentToParent.left - result.x1;
|
|
result.x1 += offset;
|
|
result.x2 += offset;
|
|
}
|
|
if (alignmentToParent.top >= 0)
|
|
{
|
|
vint offset = alignmentToParent.top - result.y1;
|
|
result.y1 += offset;
|
|
result.y2 += offset;
|
|
}
|
|
if (alignmentToParent.right >= 0)
|
|
{
|
|
result.x2 += alignmentToParent.right;
|
|
}
|
|
if (alignmentToParent.bottom >= 0)
|
|
{
|
|
result.y2 += alignmentToParent.bottom;
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
Rect GuiBoundsComposition::GetBounds()
|
|
{
|
|
Rect result = GetPreferredBounds();
|
|
if (GetParent() && IsAlignedToParent())
|
|
{
|
|
Size clientSize = GetParent()->GetClientArea().GetSize();
|
|
if (alignmentToParent.left >= 0 && alignmentToParent.right >= 0)
|
|
{
|
|
result.x1 = alignmentToParent.left;
|
|
result.x2 = clientSize.x - alignmentToParent.right;
|
|
}
|
|
else if (alignmentToParent.left >= 0)
|
|
{
|
|
vint width = result.Width();
|
|
result.x1 = alignmentToParent.left;
|
|
result.x2 = result.x1 + width;
|
|
}
|
|
else if (alignmentToParent.right >= 0)
|
|
{
|
|
vint width = result.Width();
|
|
result.x2 = clientSize.x - alignmentToParent.right;
|
|
result.x1 = result.x2 - width;
|
|
}
|
|
|
|
if (alignmentToParent.top >= 0 && alignmentToParent.bottom >= 0)
|
|
{
|
|
result.y1 = alignmentToParent.top;
|
|
result.y2 = clientSize.y - alignmentToParent.bottom;
|
|
}
|
|
else if (alignmentToParent.top >= 0)
|
|
{
|
|
vint height = result.Height();
|
|
result.y1 = alignmentToParent.top;
|
|
result.y2 = result.y1 + height;
|
|
}
|
|
else if (alignmentToParent.bottom >= 0)
|
|
{
|
|
vint height = result.Height();
|
|
result.y2 = clientSize.y - alignmentToParent.bottom;
|
|
result.y1 = result.y2 - height;
|
|
}
|
|
}
|
|
UpdatePreviousBounds(result);
|
|
return result;
|
|
}
|
|
|
|
void GuiBoundsComposition::SetBounds(Rect value)
|
|
{
|
|
compositionBounds = value;
|
|
InvokeOnCompositionStateChanged();
|
|
}
|
|
|
|
Margin GuiBoundsComposition::GetAlignmentToParent()
|
|
{
|
|
return alignmentToParent;
|
|
}
|
|
|
|
void GuiBoundsComposition::SetAlignmentToParent(Margin value)
|
|
{
|
|
alignmentToParent = value;
|
|
InvokeOnCompositionStateChanged();
|
|
}
|
|
|
|
bool GuiBoundsComposition::IsAlignedToParent()
|
|
{
|
|
return alignmentToParent != Margin(-1, -1, -1, -1);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\GRAPHICSCOMPOSITION\GUIGRAPHICSCOMPOSITIONBASE.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace compositions
|
|
{
|
|
using namespace collections;
|
|
using namespace controls;
|
|
using namespace elements;
|
|
|
|
void InvokeOnCompositionStateChanged(compositions::GuiGraphicsComposition* composition)
|
|
{
|
|
composition->InvokeOnCompositionStateChanged();
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiGraphicsComposition
|
|
***********************************************************************/
|
|
|
|
void GuiGraphicsComposition::OnControlParentChanged(controls::GuiControl* control)
|
|
{
|
|
if(associatedControl && associatedControl!=control)
|
|
{
|
|
if(associatedControl->GetParent())
|
|
{
|
|
associatedControl->GetParent()->OnChildRemoved(associatedControl);
|
|
}
|
|
if(control)
|
|
{
|
|
control->OnChildInserted(associatedControl);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
for(vint i=0;i<children.Count();i++)
|
|
{
|
|
children[i]->OnControlParentChanged(control);
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiGraphicsComposition::OnChildInserted(GuiGraphicsComposition* child)
|
|
{
|
|
child->OnControlParentChanged(GetRelatedControl());
|
|
}
|
|
|
|
void GuiGraphicsComposition::OnChildRemoved(GuiGraphicsComposition* child)
|
|
{
|
|
child->OnControlParentChanged(0);
|
|
}
|
|
|
|
void GuiGraphicsComposition::OnParentChanged(GuiGraphicsComposition* oldParent, GuiGraphicsComposition* newParent)
|
|
{
|
|
OnParentLineChanged();
|
|
}
|
|
|
|
void GuiGraphicsComposition::OnParentLineChanged()
|
|
{
|
|
for (vint i = 0; i < children.Count(); i++)
|
|
{
|
|
children[i]->OnParentLineChanged();
|
|
}
|
|
}
|
|
|
|
void GuiGraphicsComposition::OnRenderContextChanged()
|
|
{
|
|
}
|
|
|
|
void GuiGraphicsComposition::UpdateRelatedHostRecord(GraphicsHostRecord* record)
|
|
{
|
|
relatedHostRecord = record;
|
|
auto renderTarget = GetRenderTarget();
|
|
|
|
if (ownedElement)
|
|
{
|
|
if (auto renderer = ownedElement->GetRenderer())
|
|
{
|
|
renderer->SetRenderTarget(renderTarget);
|
|
}
|
|
}
|
|
|
|
for (vint i = 0; i < children.Count(); i++)
|
|
{
|
|
children[i]->UpdateRelatedHostRecord(record);
|
|
}
|
|
|
|
if (HasEventReceiver())
|
|
{
|
|
GetEventReceiver()->renderTargetChanged.Execute(GuiEventArgs(this));
|
|
}
|
|
if (associatedControl)
|
|
{
|
|
associatedControl->OnRenderTargetChanged(renderTarget);
|
|
}
|
|
|
|
OnRenderContextChanged();
|
|
}
|
|
|
|
void GuiGraphicsComposition::SetAssociatedControl(controls::GuiControl* control)
|
|
{
|
|
if (associatedControl)
|
|
{
|
|
for (vint i = 0; i < children.Count(); i++)
|
|
{
|
|
children[i]->OnControlParentChanged(0);
|
|
}
|
|
}
|
|
associatedControl = control;
|
|
if (associatedControl)
|
|
{
|
|
for (vint i = 0; i < children.Count(); i++)
|
|
{
|
|
children[i]->OnControlParentChanged(associatedControl);
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiGraphicsComposition::InvokeOnCompositionStateChanged()
|
|
{
|
|
if (relatedHostRecord)
|
|
{
|
|
relatedHostRecord->host->RequestRender();
|
|
}
|
|
}
|
|
|
|
bool GuiGraphicsComposition::SharedPtrDestructorProc(DescriptableObject* obj, bool forceDisposing)
|
|
{
|
|
GuiGraphicsComposition* value=dynamic_cast<GuiGraphicsComposition*>(obj);
|
|
if(value->parent)
|
|
{
|
|
if (!forceDisposing) return false;
|
|
}
|
|
SafeDeleteComposition(value);
|
|
return true;
|
|
}
|
|
|
|
GuiGraphicsComposition::GuiGraphicsComposition()
|
|
{
|
|
sharedPtrDestructorProc = &GuiGraphicsComposition::SharedPtrDestructorProc;
|
|
}
|
|
|
|
GuiGraphicsComposition::~GuiGraphicsComposition()
|
|
{
|
|
for(vint i=0;i<children.Count();i++)
|
|
{
|
|
delete children[i];
|
|
}
|
|
}
|
|
|
|
bool GuiGraphicsComposition::IsRendering()
|
|
{
|
|
return isRendering;
|
|
}
|
|
|
|
GuiGraphicsComposition* GuiGraphicsComposition::GetParent()
|
|
{
|
|
return parent;
|
|
}
|
|
|
|
const GuiGraphicsComposition::CompositionList& GuiGraphicsComposition::Children()
|
|
{
|
|
return children;
|
|
}
|
|
|
|
bool GuiGraphicsComposition::AddChild(GuiGraphicsComposition* child)
|
|
{
|
|
return InsertChild(children.Count(), child);
|
|
}
|
|
|
|
bool GuiGraphicsComposition::InsertChild(vint index, GuiGraphicsComposition* child)
|
|
{
|
|
CHECK_ERROR(!isRendering, L"GuiGraphicsComposition::InsertChild(vint, GuiGraphicsComposition*)#Cannot modify composition tree during rendering.");
|
|
if (!child) return false;
|
|
if (child->GetParent()) return false;
|
|
children.Insert(index, child);
|
|
|
|
// composition parent changed -> control parent changed -> related host changed
|
|
child->parent = this;
|
|
child->OnParentChanged(nullptr, this);
|
|
OnChildInserted(child);
|
|
child->UpdateRelatedHostRecord(relatedHostRecord);
|
|
|
|
InvokeOnCompositionStateChanged();
|
|
return true;
|
|
}
|
|
|
|
bool GuiGraphicsComposition::RemoveChild(GuiGraphicsComposition* child)
|
|
{
|
|
CHECK_ERROR(!isRendering, L"GuiGraphicsComposition::InsertChild(vint, GuiGraphicsComposition*)#Cannot modify composition tree during rendering.");
|
|
if (!child) return false;
|
|
vint index = children.IndexOf(child);
|
|
if (index == -1) return false;
|
|
|
|
// composition parent changed -> control parent changed -> related host changed
|
|
child->parent = nullptr;
|
|
child->OnParentChanged(this, nullptr);
|
|
OnChildRemoved(child);
|
|
child->UpdateRelatedHostRecord(nullptr);
|
|
|
|
GuiGraphicsHost* host = GetRelatedGraphicsHost();
|
|
if (host)
|
|
{
|
|
host->DisconnectComposition(child);
|
|
}
|
|
children.RemoveAt(index);
|
|
InvokeOnCompositionStateChanged();
|
|
return true;
|
|
}
|
|
|
|
bool GuiGraphicsComposition::MoveChild(GuiGraphicsComposition* child, vint newIndex)
|
|
{
|
|
if(!child) return false;
|
|
vint index=children.IndexOf(child);
|
|
if(index==-1) return false;
|
|
children.RemoveAt(index);
|
|
children.Insert(newIndex, child);
|
|
InvokeOnCompositionStateChanged();
|
|
return true;
|
|
}
|
|
|
|
Ptr<IGuiGraphicsElement> GuiGraphicsComposition::GetOwnedElement()
|
|
{
|
|
return ownedElement;
|
|
}
|
|
|
|
void GuiGraphicsComposition::SetOwnedElement(Ptr<IGuiGraphicsElement> element)
|
|
{
|
|
if (ownedElement != element)
|
|
{
|
|
if (ownedElement)
|
|
{
|
|
if (auto renderer = ownedElement->GetRenderer())
|
|
{
|
|
renderer->SetRenderTarget(nullptr);
|
|
}
|
|
ownedElement->SetOwnerComposition(nullptr);
|
|
}
|
|
ownedElement = element;
|
|
if (ownedElement)
|
|
{
|
|
if (auto renderer = ownedElement->GetRenderer())
|
|
{
|
|
renderer->SetRenderTarget(GetRenderTarget());
|
|
}
|
|
ownedElement->SetOwnerComposition(this);
|
|
}
|
|
InvokeOnCompositionStateChanged();
|
|
}
|
|
}
|
|
|
|
bool GuiGraphicsComposition::GetVisible()
|
|
{
|
|
return visible;
|
|
}
|
|
|
|
void GuiGraphicsComposition::SetVisible(bool value)
|
|
{
|
|
visible = value;
|
|
InvokeOnCompositionStateChanged();
|
|
}
|
|
|
|
GuiGraphicsComposition::MinSizeLimitation GuiGraphicsComposition::GetMinSizeLimitation()
|
|
{
|
|
return minSizeLimitation;
|
|
}
|
|
|
|
void GuiGraphicsComposition::SetMinSizeLimitation(MinSizeLimitation value)
|
|
{
|
|
minSizeLimitation = value;
|
|
InvokeOnCompositionStateChanged();
|
|
}
|
|
|
|
elements::IGuiGraphicsRenderTarget* GuiGraphicsComposition::GetRenderTarget()
|
|
{
|
|
return relatedHostRecord ? relatedHostRecord->renderTarget : nullptr;
|
|
}
|
|
|
|
void GuiGraphicsComposition::Render(Size offset)
|
|
{
|
|
auto renderTarget = GetRenderTarget();
|
|
if (visible && renderTarget && !renderTarget->IsClipperCoverWholeTarget())
|
|
{
|
|
Rect bounds = GetBounds();
|
|
bounds.x1 += margin.left;
|
|
bounds.y1 += margin.top;
|
|
bounds.x2 -= margin.right;
|
|
bounds.y2 -= margin.bottom;
|
|
|
|
if (bounds.x1 <= bounds.x2 && bounds.y1 <= bounds.y2)
|
|
{
|
|
bounds.x1 += offset.x;
|
|
bounds.x2 += offset.x;
|
|
bounds.y1 += offset.y;
|
|
bounds.y2 += offset.y;
|
|
|
|
isRendering = true;
|
|
if (ownedElement)
|
|
{
|
|
IGuiGraphicsRenderer* renderer = ownedElement->GetRenderer();
|
|
if (renderer)
|
|
{
|
|
renderer->Render(bounds);
|
|
}
|
|
}
|
|
if (children.Count() > 0)
|
|
{
|
|
bounds.x1 += internalMargin.left;
|
|
bounds.y1 += internalMargin.top;
|
|
bounds.x2 -= internalMargin.right;
|
|
bounds.y2 -= internalMargin.bottom;
|
|
if (bounds.x1 <= bounds.x2 && bounds.y1 <= bounds.y2)
|
|
{
|
|
offset = bounds.GetSize();
|
|
renderTarget->PushClipper(bounds);
|
|
if (!renderTarget->IsClipperCoverWholeTarget())
|
|
{
|
|
for (vint i = 0; i < children.Count(); i++)
|
|
{
|
|
children[i]->Render(Size(bounds.x1, bounds.y1));
|
|
}
|
|
}
|
|
renderTarget->PopClipper();
|
|
}
|
|
}
|
|
isRendering = false;
|
|
}
|
|
}
|
|
}
|
|
|
|
GuiGraphicsEventReceiver* GuiGraphicsComposition::GetEventReceiver()
|
|
{
|
|
if(!eventReceiver)
|
|
{
|
|
eventReceiver=new GuiGraphicsEventReceiver(this);
|
|
}
|
|
return eventReceiver.Obj();
|
|
}
|
|
|
|
bool GuiGraphicsComposition::HasEventReceiver()
|
|
{
|
|
return eventReceiver;
|
|
}
|
|
|
|
GuiGraphicsComposition* GuiGraphicsComposition::FindComposition(Point location, bool forMouseEvent)
|
|
{
|
|
if (!visible) return 0;
|
|
Rect bounds = GetBounds();
|
|
Rect relativeBounds = Rect(Point(0, 0), bounds.GetSize());
|
|
if (relativeBounds.Contains(location))
|
|
{
|
|
Rect clientArea = GetClientArea();
|
|
for (vint i = children.Count() - 1; i >= 0; i--)
|
|
{
|
|
GuiGraphicsComposition* child = children[i];
|
|
Rect childBounds = child->GetBounds();
|
|
vint offsetX = childBounds.x1 + (clientArea.x1 - bounds.x1);
|
|
vint offsetY = childBounds.y1 + (clientArea.y1 - bounds.y1);
|
|
Point newLocation = location - Size(offsetX, offsetY);
|
|
GuiGraphicsComposition* childResult = child->FindComposition(newLocation, forMouseEvent);
|
|
if (childResult)
|
|
{
|
|
return childResult;
|
|
}
|
|
}
|
|
|
|
if (!forMouseEvent || !transparentToMouse)
|
|
{
|
|
return this;
|
|
}
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
bool GuiGraphicsComposition::GetTransparentToMouse()
|
|
{
|
|
return transparentToMouse;
|
|
}
|
|
|
|
void GuiGraphicsComposition::SetTransparentToMouse(bool value)
|
|
{
|
|
transparentToMouse = value;
|
|
}
|
|
|
|
Rect GuiGraphicsComposition::GetGlobalBounds()
|
|
{
|
|
Rect bounds = GetBounds();
|
|
GuiGraphicsComposition* composition = parent;
|
|
while (composition)
|
|
{
|
|
Rect clientArea = composition->GetClientArea();
|
|
Rect parentBounds = composition->GetBounds();
|
|
bounds.x1 += clientArea.x1;
|
|
bounds.x2 += clientArea.x1;
|
|
bounds.y1 += clientArea.y1;
|
|
bounds.y2 += clientArea.y1;
|
|
composition = composition->parent;
|
|
}
|
|
return bounds;
|
|
}
|
|
|
|
controls::GuiControl* GuiGraphicsComposition::GetAssociatedControl()
|
|
{
|
|
return associatedControl;
|
|
}
|
|
|
|
GuiGraphicsHost* GuiGraphicsComposition::GetAssociatedHost()
|
|
{
|
|
if (relatedHostRecord && relatedHostRecord->host->GetMainComposition() == this)
|
|
{
|
|
return relatedHostRecord->host;
|
|
}
|
|
else
|
|
{
|
|
return nullptr;
|
|
}
|
|
}
|
|
|
|
INativeCursor* GuiGraphicsComposition::GetAssociatedCursor()
|
|
{
|
|
return associatedCursor;
|
|
}
|
|
|
|
void GuiGraphicsComposition::SetAssociatedCursor(INativeCursor* cursor)
|
|
{
|
|
associatedCursor = cursor;
|
|
}
|
|
|
|
INativeWindowListener::HitTestResult GuiGraphicsComposition::GetAssociatedHitTestResult()
|
|
{
|
|
return associatedHitTestResult;
|
|
}
|
|
|
|
void GuiGraphicsComposition::SetAssociatedHitTestResult(INativeWindowListener::HitTestResult value)
|
|
{
|
|
associatedHitTestResult = value;
|
|
}
|
|
|
|
controls::GuiControl* GuiGraphicsComposition::GetRelatedControl()
|
|
{
|
|
GuiGraphicsComposition* composition = this;
|
|
while (composition)
|
|
{
|
|
if (composition->GetAssociatedControl())
|
|
{
|
|
return composition->GetAssociatedControl();
|
|
}
|
|
else
|
|
{
|
|
composition = composition->GetParent();
|
|
}
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
GuiGraphicsHost* GuiGraphicsComposition::GetRelatedGraphicsHost()
|
|
{
|
|
return relatedHostRecord ? relatedHostRecord->host : nullptr;
|
|
}
|
|
|
|
controls::GuiControlHost* GuiGraphicsComposition::GetRelatedControlHost()
|
|
{
|
|
if (auto control = GetRelatedControl())
|
|
{
|
|
return control->GetRelatedControlHost();
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
INativeCursor* GuiGraphicsComposition::GetRelatedCursor()
|
|
{
|
|
GuiGraphicsComposition* composition = this;
|
|
while (composition)
|
|
{
|
|
if (composition->GetAssociatedCursor())
|
|
{
|
|
return composition->GetAssociatedCursor();
|
|
}
|
|
else
|
|
{
|
|
composition = composition->GetParent();
|
|
}
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
Margin GuiGraphicsComposition::GetMargin()
|
|
{
|
|
return margin;
|
|
}
|
|
|
|
void GuiGraphicsComposition::SetMargin(Margin value)
|
|
{
|
|
margin = value;
|
|
InvokeOnCompositionStateChanged();
|
|
}
|
|
|
|
Margin GuiGraphicsComposition::GetInternalMargin()
|
|
{
|
|
return internalMargin;
|
|
}
|
|
|
|
void GuiGraphicsComposition::SetInternalMargin(Margin value)
|
|
{
|
|
internalMargin = value;
|
|
InvokeOnCompositionStateChanged();
|
|
}
|
|
|
|
Size GuiGraphicsComposition::GetPreferredMinSize()
|
|
{
|
|
return preferredMinSize;
|
|
}
|
|
|
|
void GuiGraphicsComposition::SetPreferredMinSize(Size value)
|
|
{
|
|
preferredMinSize = value;
|
|
InvokeOnCompositionStateChanged();
|
|
}
|
|
|
|
Rect GuiGraphicsComposition::GetClientArea()
|
|
{
|
|
Rect bounds=GetBounds();
|
|
bounds.x1+=margin.left+internalMargin.left;
|
|
bounds.y1+=margin.top+internalMargin.top;
|
|
bounds.x2-=margin.right+internalMargin.right;
|
|
bounds.y2-=margin.bottom+internalMargin.bottom;
|
|
return bounds;
|
|
}
|
|
|
|
void GuiGraphicsComposition::ForceCalculateSizeImmediately()
|
|
{
|
|
isRendering = true;
|
|
for (vint i = 0; i < children.Count(); i++)
|
|
{
|
|
children[i]->ForceCalculateSizeImmediately();
|
|
}
|
|
isRendering = false;
|
|
InvokeOnCompositionStateChanged();
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiGraphicsSite
|
|
***********************************************************************/
|
|
|
|
Rect GuiGraphicsSite::GetBoundsInternal(Rect expectedBounds)
|
|
{
|
|
Size minSize = GetMinPreferredClientSize();
|
|
if (minSize.x < preferredMinSize.x) minSize.x = preferredMinSize.x;
|
|
if (minSize.y < preferredMinSize.y) minSize.y = preferredMinSize.y;
|
|
|
|
minSize.x += margin.left + margin.right + internalMargin.left + internalMargin.right;
|
|
minSize.y += margin.top + margin.bottom + internalMargin.top + internalMargin.bottom;
|
|
vint w = expectedBounds.Width();
|
|
vint h = expectedBounds.Height();
|
|
if (minSize.x < w) minSize.x = w;
|
|
if (minSize.y < h) minSize.y = h;
|
|
return Rect(expectedBounds.LeftTop(), minSize);
|
|
}
|
|
|
|
void GuiGraphicsSite::UpdatePreviousBounds(Rect bounds)
|
|
{
|
|
if (previousBounds != bounds)
|
|
{
|
|
previousBounds = bounds;
|
|
BoundsChanged.Execute(GuiEventArgs(this));
|
|
InvokeOnCompositionStateChanged();
|
|
}
|
|
}
|
|
|
|
GuiGraphicsSite::GuiGraphicsSite()
|
|
{
|
|
BoundsChanged.SetAssociatedComposition(this);
|
|
}
|
|
|
|
GuiGraphicsSite::~GuiGraphicsSite()
|
|
{
|
|
}
|
|
|
|
bool GuiGraphicsSite::IsSizeAffectParent()
|
|
{
|
|
return true;
|
|
}
|
|
|
|
Size GuiGraphicsSite::GetMinPreferredClientSize()
|
|
{
|
|
Size minSize;
|
|
if (minSizeLimitation != GuiGraphicsComposition::NoLimit)
|
|
{
|
|
if (ownedElement)
|
|
{
|
|
IGuiGraphicsRenderer* renderer = ownedElement->GetRenderer();
|
|
if (renderer)
|
|
{
|
|
minSize = renderer->GetMinSize();
|
|
}
|
|
}
|
|
}
|
|
if (minSizeLimitation == GuiGraphicsComposition::LimitToElementAndChildren)
|
|
{
|
|
vint childCount = Children().Count();
|
|
for (vint i = 0; i < childCount; i++)
|
|
{
|
|
GuiGraphicsComposition* child = children[i];
|
|
if (child->IsSizeAffectParent())
|
|
{
|
|
Rect childBounds = child->GetPreferredBounds();
|
|
if (minSize.x < childBounds.x2) minSize.x = childBounds.x2;
|
|
if (minSize.y < childBounds.y2) minSize.y = childBounds.y2;
|
|
}
|
|
}
|
|
}
|
|
return minSize;
|
|
}
|
|
|
|
Rect GuiGraphicsSite::GetPreferredBounds()
|
|
{
|
|
return GetBoundsInternal(Rect(Point(0, 0), GetMinPreferredClientSize()));
|
|
}
|
|
|
|
/***********************************************************************
|
|
Helper Functions
|
|
***********************************************************************/
|
|
|
|
void NotifyFinalizeInstance(controls::GuiControl* value)
|
|
{
|
|
if (value)
|
|
{
|
|
NotifyFinalizeInstance(value->GetBoundsComposition());
|
|
}
|
|
}
|
|
|
|
void NotifyFinalizeInstance(GuiGraphicsComposition* value)
|
|
{
|
|
if (value)
|
|
{
|
|
bool finalized = false;
|
|
if (auto root = dynamic_cast<GuiInstanceRootObject*>(value))
|
|
{
|
|
if (root->IsFinalized())
|
|
{
|
|
finalized = true;
|
|
}
|
|
else
|
|
{
|
|
root->FinalizeInstance();
|
|
}
|
|
}
|
|
|
|
if (auto control = value->GetAssociatedControl())
|
|
{
|
|
if (auto root = dynamic_cast<GuiInstanceRootObject*>(control))
|
|
{
|
|
if (root->IsFinalized())
|
|
{
|
|
finalized = true;
|
|
}
|
|
else
|
|
{
|
|
root->FinalizeInstance();
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!finalized)
|
|
{
|
|
vint count = value->Children().Count();
|
|
for (vint i = 0; i < count; i++)
|
|
{
|
|
NotifyFinalizeInstance(value->Children()[i]);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void SafeDeleteControlInternal(controls::GuiControl* value)
|
|
{
|
|
if(value)
|
|
{
|
|
if (value->GetRelatedControlHost() != value)
|
|
{
|
|
GuiGraphicsComposition* bounds = value->GetBoundsComposition();
|
|
if (bounds->GetParent())
|
|
{
|
|
bounds->GetParent()->RemoveChild(bounds);
|
|
}
|
|
}
|
|
delete value;
|
|
}
|
|
}
|
|
|
|
void SafeDeleteCompositionInternal(GuiGraphicsComposition* value)
|
|
{
|
|
if (value)
|
|
{
|
|
if (value->GetParent())
|
|
{
|
|
value->GetParent()->RemoveChild(value);
|
|
}
|
|
|
|
if (value->GetAssociatedControl())
|
|
{
|
|
SafeDeleteControlInternal(value->GetAssociatedControl());
|
|
}
|
|
else
|
|
{
|
|
for (vint i = value->Children().Count() - 1; i >= 0; i--)
|
|
{
|
|
SafeDeleteCompositionInternal(value->Children().Get(i));
|
|
}
|
|
delete value;
|
|
}
|
|
}
|
|
}
|
|
|
|
void SafeDeleteControl(controls::GuiControl* value)
|
|
{
|
|
if (auto controlHost = dynamic_cast<controls::GuiControlHost*>(value))
|
|
{
|
|
controlHost->DeleteAfterProcessingAllEvents();
|
|
}
|
|
else
|
|
{
|
|
NotifyFinalizeInstance(value);
|
|
SafeDeleteControlInternal(value);
|
|
}
|
|
}
|
|
|
|
void SafeDeleteComposition(GuiGraphicsComposition* value)
|
|
{
|
|
NotifyFinalizeInstance(value);
|
|
SafeDeleteCompositionInternal(value);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\GRAPHICSCOMPOSITION\GUIGRAPHICSEVENTRECEIVER.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace compositions
|
|
{
|
|
|
|
/***********************************************************************
|
|
Event Receiver
|
|
***********************************************************************/
|
|
|
|
GuiGraphicsEventReceiver::GuiGraphicsEventReceiver(GuiGraphicsComposition* _sender)
|
|
:sender(_sender)
|
|
,leftButtonDown(_sender)
|
|
,leftButtonUp(_sender)
|
|
,leftButtonDoubleClick(_sender)
|
|
,middleButtonDown(_sender)
|
|
,middleButtonUp(_sender)
|
|
,middleButtonDoubleClick(_sender)
|
|
,rightButtonDown(_sender)
|
|
,rightButtonUp(_sender)
|
|
,rightButtonDoubleClick(_sender)
|
|
,horizontalWheel(_sender)
|
|
,verticalWheel(_sender)
|
|
,mouseMove(_sender)
|
|
,mouseEnter(_sender)
|
|
,mouseLeave(_sender)
|
|
,previewKey(_sender)
|
|
,keyDown(_sender)
|
|
,keyUp(_sender)
|
|
,systemKeyDown(_sender)
|
|
,systemKeyUp(_sender)
|
|
,previewCharInput(_sender)
|
|
,charInput(_sender)
|
|
,gotFocus(_sender)
|
|
,lostFocus(_sender)
|
|
,caretNotify(_sender)
|
|
,clipboardNotify(_sender)
|
|
{
|
|
}
|
|
|
|
GuiGraphicsEventReceiver::~GuiGraphicsEventReceiver()
|
|
{
|
|
}
|
|
|
|
GuiGraphicsComposition* GuiGraphicsEventReceiver::GetAssociatedComposition()
|
|
{
|
|
return sender;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\GRAPHICSCOMPOSITION\GUIGRAPHICSFLOWCOMPOSITION.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace compositions
|
|
{
|
|
using namespace collections;
|
|
|
|
/***********************************************************************
|
|
GuiFlowComposition
|
|
***********************************************************************/
|
|
|
|
void GuiFlowComposition::UpdateFlowItemBounds(bool forceUpdate)
|
|
{
|
|
if (forceUpdate || needUpdate)
|
|
{
|
|
needUpdate = false;
|
|
InvokeOnCompositionStateChanged();
|
|
|
|
auto clientMargin = axis->RealMarginToVirtualMargin(extraMargin);
|
|
if (clientMargin.left < 0) clientMargin.left = 0;
|
|
if (clientMargin.top < 0) clientMargin.top = 0;
|
|
if (clientMargin.right < 0) clientMargin.right = 0;
|
|
if (clientMargin.bottom < 0) clientMargin.bottom = 0;
|
|
|
|
auto realFullSize = previousBounds.GetSize();
|
|
auto clientSize = axis->RealSizeToVirtualSize(realFullSize);
|
|
clientSize.x -= (clientMargin.left + clientMargin.right);
|
|
clientSize.y -= (clientMargin.top + clientMargin.bottom);
|
|
|
|
flowItemBounds.Resize(flowItems.Count());
|
|
for (vint i = 0; i < flowItems.Count(); i++)
|
|
{
|
|
flowItemBounds[i] = Rect(Point(0, 0), flowItems[i]->GetMinSize());
|
|
}
|
|
|
|
vint currentIndex = 0;
|
|
vint rowTop = 0;
|
|
|
|
while (currentIndex < flowItems.Count())
|
|
{
|
|
auto itemSize = axis->RealSizeToVirtualSize(flowItemBounds[currentIndex].GetSize());
|
|
vint rowWidth = itemSize.x;
|
|
vint rowHeight = itemSize.y;
|
|
vint rowItemCount = 1;
|
|
|
|
for (vint i = currentIndex + 1; i < flowItems.Count(); i++)
|
|
{
|
|
itemSize = axis->RealSizeToVirtualSize(flowItemBounds[i].GetSize());
|
|
vint itemWidth = itemSize.x + columnPadding;
|
|
if (rowWidth + itemWidth > clientSize.x)
|
|
{
|
|
break;
|
|
}
|
|
rowWidth += itemWidth;
|
|
if (rowHeight < itemSize.y)
|
|
{
|
|
rowHeight = itemSize.y;
|
|
}
|
|
rowItemCount++;
|
|
}
|
|
|
|
vint baseLine = 0;
|
|
Array<vint> itemBaseLines(rowItemCount);
|
|
for (vint i = 0; i < rowItemCount; i++)
|
|
{
|
|
vint index = currentIndex + i;
|
|
vint itemBaseLine = 0;
|
|
itemSize = axis->RealSizeToVirtualSize(flowItemBounds[index].GetSize());
|
|
|
|
auto option = flowItems[index]->GetFlowOption();
|
|
switch (option.baseline)
|
|
{
|
|
case GuiFlowOption::FromTop:
|
|
itemBaseLine = option.distance;
|
|
break;
|
|
case GuiFlowOption::FromBottom:
|
|
itemBaseLine = itemSize.y - option.distance;
|
|
break;
|
|
case GuiFlowOption::Percentage:
|
|
itemBaseLine = (vint)(itemSize.y*option.percentage);
|
|
break;
|
|
}
|
|
|
|
itemBaseLines[i] = itemBaseLine;
|
|
if (baseLine < itemBaseLine)
|
|
{
|
|
baseLine = itemBaseLine;
|
|
}
|
|
}
|
|
|
|
vint rowUsedWidth = 0;
|
|
for (vint i = 0; i < rowItemCount; i++)
|
|
{
|
|
vint index = currentIndex + i;
|
|
itemSize = axis->RealSizeToVirtualSize(flowItemBounds[index].GetSize());
|
|
|
|
vint itemLeft = 0;
|
|
vint itemTop = rowTop + baseLine - itemBaseLines[i];
|
|
|
|
switch (alignment)
|
|
{
|
|
case FlowAlignment::Left:
|
|
itemLeft = rowUsedWidth + i * columnPadding;
|
|
break;
|
|
case FlowAlignment::Center:
|
|
itemLeft = rowUsedWidth + i * columnPadding + (clientSize.x - rowWidth) / 2;
|
|
break;
|
|
case FlowAlignment::Extend:
|
|
if (i == 0)
|
|
{
|
|
itemLeft = rowUsedWidth;
|
|
}
|
|
else
|
|
{
|
|
itemLeft = rowUsedWidth + (vint)((double)(clientSize.x - rowWidth) * i / (rowItemCount - 1)) + i * columnPadding;
|
|
}
|
|
break;
|
|
}
|
|
|
|
flowItemBounds[index] = axis->VirtualRectToRealRect(
|
|
realFullSize,
|
|
Rect(
|
|
Point(
|
|
itemLeft + clientMargin.left,
|
|
itemTop + clientMargin.top
|
|
),
|
|
itemSize
|
|
)
|
|
);
|
|
rowUsedWidth += itemSize.x;
|
|
}
|
|
|
|
rowTop += rowHeight + rowPadding;
|
|
currentIndex += rowItemCount;
|
|
}
|
|
|
|
minHeight = rowTop == 0 ? 0 : rowTop - rowPadding;
|
|
}
|
|
}
|
|
|
|
void GuiFlowComposition::OnBoundsChanged(GuiGraphicsComposition* sender, GuiEventArgs& arguments)
|
|
{
|
|
UpdateFlowItemBounds(true);
|
|
}
|
|
|
|
void GuiFlowComposition::OnChildInserted(GuiGraphicsComposition* child)
|
|
{
|
|
GuiBoundsComposition::OnChildInserted(child);
|
|
auto item = dynamic_cast<GuiFlowItemComposition*>(child);
|
|
if (item && !flowItems.Contains(item))
|
|
{
|
|
flowItems.Add(item);
|
|
needUpdate = true;
|
|
}
|
|
}
|
|
|
|
void GuiFlowComposition::OnChildRemoved(GuiGraphicsComposition* child)
|
|
{
|
|
GuiBoundsComposition::OnChildRemoved(child);
|
|
auto item = dynamic_cast<GuiFlowItemComposition*>(child);
|
|
if (item)
|
|
{
|
|
flowItems.Remove(item);
|
|
needUpdate = true;
|
|
}
|
|
}
|
|
|
|
GuiFlowComposition::GuiFlowComposition()
|
|
:axis(new GuiDefaultAxis)
|
|
{
|
|
BoundsChanged.AttachMethod(this, &GuiFlowComposition::OnBoundsChanged);
|
|
}
|
|
|
|
GuiFlowComposition::~GuiFlowComposition()
|
|
{
|
|
}
|
|
|
|
const GuiFlowComposition::ItemCompositionList& GuiFlowComposition::GetFlowItems()
|
|
{
|
|
return flowItems;
|
|
}
|
|
|
|
bool GuiFlowComposition::InsertFlowItem(vint index, GuiFlowItemComposition* item)
|
|
{
|
|
index = flowItems.Insert(index, item);
|
|
if (!AddChild(item))
|
|
{
|
|
flowItems.RemoveAt(index);
|
|
return false;
|
|
}
|
|
else
|
|
{
|
|
needUpdate = true;
|
|
return true;
|
|
}
|
|
}
|
|
|
|
|
|
Margin GuiFlowComposition::GetExtraMargin()
|
|
{
|
|
return extraMargin;
|
|
}
|
|
|
|
void GuiFlowComposition::SetExtraMargin(Margin value)
|
|
{
|
|
extraMargin = value;
|
|
needUpdate = true;
|
|
InvokeOnCompositionStateChanged();
|
|
}
|
|
|
|
vint GuiFlowComposition::GetRowPadding()
|
|
{
|
|
return rowPadding;
|
|
}
|
|
|
|
void GuiFlowComposition::SetRowPadding(vint value)
|
|
{
|
|
rowPadding = value;
|
|
needUpdate = true;
|
|
InvokeOnCompositionStateChanged();
|
|
}
|
|
|
|
vint GuiFlowComposition::GetColumnPadding()
|
|
{
|
|
return columnPadding;
|
|
}
|
|
|
|
void GuiFlowComposition::SetColumnPadding(vint value)
|
|
{
|
|
columnPadding = value;
|
|
needUpdate = true;
|
|
InvokeOnCompositionStateChanged();
|
|
}
|
|
|
|
Ptr<IGuiAxis> GuiFlowComposition::GetAxis()
|
|
{
|
|
return axis;
|
|
}
|
|
|
|
void GuiFlowComposition::SetAxis(Ptr<IGuiAxis> value)
|
|
{
|
|
if (value)
|
|
{
|
|
axis = value;
|
|
needUpdate = true;
|
|
InvokeOnCompositionStateChanged();
|
|
}
|
|
}
|
|
|
|
FlowAlignment GuiFlowComposition::GetAlignment()
|
|
{
|
|
return alignment;
|
|
}
|
|
|
|
void GuiFlowComposition::SetAlignment(FlowAlignment value)
|
|
{
|
|
alignment = value;
|
|
needUpdate = true;
|
|
InvokeOnCompositionStateChanged();
|
|
}
|
|
|
|
void GuiFlowComposition::ForceCalculateSizeImmediately()
|
|
{
|
|
GuiBoundsComposition::ForceCalculateSizeImmediately();
|
|
UpdateFlowItemBounds(true);
|
|
}
|
|
|
|
Size GuiFlowComposition::GetMinPreferredClientSize()
|
|
{
|
|
Size minSize = GuiBoundsComposition::GetMinPreferredClientSize();
|
|
if (GetMinSizeLimitation() == GuiGraphicsComposition::LimitToElementAndChildren)
|
|
{
|
|
auto clientSize = axis->VirtualSizeToRealSize(Size(0, minHeight));
|
|
FOREACH(GuiFlowItemComposition*, item, flowItems)
|
|
{
|
|
auto itemSize = item->GetPreferredBounds().GetSize();
|
|
if (clientSize.x < itemSize.x) clientSize.x = itemSize.x;
|
|
if (clientSize.y < itemSize.y) clientSize.y = itemSize.y;
|
|
}
|
|
if (minSize.x < clientSize.x) minSize.x = clientSize.x;
|
|
if (minSize.y < clientSize.y) minSize.y = clientSize.y;
|
|
}
|
|
|
|
vint x = 0;
|
|
vint y = 0;
|
|
if (extraMargin.left > 0) x += extraMargin.left;
|
|
if (extraMargin.right > 0) x += extraMargin.right;
|
|
if (extraMargin.top > 0) y += extraMargin.top;
|
|
if (extraMargin.bottom > 0) y += extraMargin.bottom;
|
|
return minSize + Size(x, y);
|
|
}
|
|
|
|
Rect GuiFlowComposition::GetBounds()
|
|
{
|
|
if (!needUpdate)
|
|
{
|
|
for (vint i = 0; i < flowItems.Count(); i++)
|
|
{
|
|
if (flowItemBounds[i].GetSize() != flowItems[i]->GetMinSize())
|
|
{
|
|
needUpdate = true;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (needUpdate)
|
|
{
|
|
UpdateFlowItemBounds(true);
|
|
}
|
|
|
|
bounds = GuiBoundsComposition::GetBounds();
|
|
return bounds;
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiFlowItemComposition
|
|
***********************************************************************/
|
|
|
|
void GuiFlowItemComposition::OnParentChanged(GuiGraphicsComposition* oldParent, GuiGraphicsComposition* newParent)
|
|
{
|
|
GuiGraphicsSite::OnParentChanged(oldParent, newParent);
|
|
flowParent = newParent == 0 ? 0 : dynamic_cast<GuiFlowComposition*>(newParent);
|
|
}
|
|
|
|
Size GuiFlowItemComposition::GetMinSize()
|
|
{
|
|
return GetBoundsInternal(bounds).GetSize();
|
|
}
|
|
|
|
GuiFlowItemComposition::GuiFlowItemComposition()
|
|
{
|
|
SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
}
|
|
|
|
GuiFlowItemComposition::~GuiFlowItemComposition()
|
|
{
|
|
}
|
|
|
|
bool GuiFlowItemComposition::IsSizeAffectParent()
|
|
{
|
|
return false;
|
|
}
|
|
|
|
Rect GuiFlowItemComposition::GetBounds()
|
|
{
|
|
Rect result = bounds;
|
|
if(flowParent)
|
|
{
|
|
flowParent->UpdateFlowItemBounds(false);
|
|
vint index = flowParent->flowItems.IndexOf(this);
|
|
if (index != -1)
|
|
{
|
|
result = flowParent->flowItemBounds[index];
|
|
}
|
|
|
|
result = Rect(
|
|
result.Left() - extraMargin.left,
|
|
result.Top() - extraMargin.top,
|
|
result.Right() + extraMargin.right,
|
|
result.Bottom() + extraMargin.bottom
|
|
);
|
|
}
|
|
UpdatePreviousBounds(result);
|
|
return result;
|
|
}
|
|
|
|
void GuiFlowItemComposition::SetBounds(Rect value)
|
|
{
|
|
bounds = value;
|
|
InvokeOnCompositionStateChanged();
|
|
}
|
|
|
|
Margin GuiFlowItemComposition::GetExtraMargin()
|
|
{
|
|
return extraMargin;
|
|
}
|
|
|
|
void GuiFlowItemComposition::SetExtraMargin(Margin value)
|
|
{
|
|
extraMargin = value;
|
|
InvokeOnCompositionStateChanged();
|
|
}
|
|
|
|
GuiFlowOption GuiFlowItemComposition::GetFlowOption()
|
|
{
|
|
return option;
|
|
}
|
|
|
|
void GuiFlowItemComposition::SetFlowOption(GuiFlowOption value)
|
|
{
|
|
option = value;
|
|
if (flowParent)
|
|
{
|
|
flowParent->needUpdate = true;
|
|
InvokeOnCompositionStateChanged();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\GRAPHICSCOMPOSITION\GUIGRAPHICSREPEATCOMPOSITION.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace compositions
|
|
{
|
|
using namespace reflection::description;
|
|
using namespace collections;
|
|
using namespace controls;
|
|
using namespace elements;
|
|
|
|
/***********************************************************************
|
|
GuiRepeatCompositionBase
|
|
***********************************************************************/
|
|
|
|
void GuiRepeatCompositionBase::OnItemChanged(vint index, vint oldCount, vint newCount)
|
|
{
|
|
if (itemTemplate && itemSource)
|
|
{
|
|
for (vint i = oldCount - 1; i >= 0; i--)
|
|
{
|
|
RemoveItem(index + i);
|
|
}
|
|
|
|
for (vint i = 0; i < newCount; i++)
|
|
{
|
|
InstallItem(index + i);
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiRepeatCompositionBase::RemoveItem(vint index)
|
|
{
|
|
GuiItemEventArgs arguments(dynamic_cast<GuiGraphicsComposition*>(this));
|
|
arguments.itemIndex = index;
|
|
ItemRemoved.Execute(arguments);
|
|
|
|
auto item = RemoveRepeatComposition(index);
|
|
SafeDeleteComposition(item);
|
|
}
|
|
|
|
void GuiRepeatCompositionBase::InstallItem(vint index)
|
|
{
|
|
auto source = itemSource->Get(index);
|
|
auto templateItem = itemTemplate(source);
|
|
auto item = InsertRepeatComposition(index);
|
|
|
|
templateItem->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
item->SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
item->AddChild(templateItem);
|
|
|
|
GuiItemEventArgs arguments(dynamic_cast<GuiGraphicsComposition*>(this));
|
|
arguments.itemIndex = index;
|
|
ItemInserted.Execute(arguments);
|
|
}
|
|
|
|
void GuiRepeatCompositionBase::ClearItems()
|
|
{
|
|
for (vint i = GetRepeatCompositionCount() - 1; i >= 0; i--)
|
|
{
|
|
RemoveItem(i);
|
|
}
|
|
}
|
|
|
|
void GuiRepeatCompositionBase::InstallItems()
|
|
{
|
|
if (itemTemplate && itemSource)
|
|
{
|
|
vint count = itemSource->GetCount();
|
|
for (vint i = 0; i < count; i++)
|
|
{
|
|
InstallItem(i);
|
|
}
|
|
}
|
|
}
|
|
|
|
GuiRepeatCompositionBase::GuiRepeatCompositionBase()
|
|
{
|
|
}
|
|
|
|
GuiRepeatCompositionBase::~GuiRepeatCompositionBase()
|
|
{
|
|
}
|
|
|
|
GuiRepeatCompositionBase::ItemStyleProperty GuiRepeatCompositionBase::GetItemTemplate()
|
|
{
|
|
return itemTemplate;
|
|
}
|
|
|
|
void GuiRepeatCompositionBase::SetItemTemplate(ItemStyleProperty value)
|
|
{
|
|
ClearItems();
|
|
itemTemplate = value;
|
|
if (itemTemplate && itemSource)
|
|
{
|
|
InstallItems();
|
|
}
|
|
}
|
|
|
|
Ptr<IValueEnumerable> GuiRepeatCompositionBase::GetItemSource()
|
|
{
|
|
return itemSource;
|
|
}
|
|
|
|
void GuiRepeatCompositionBase::SetItemSource(Ptr<IValueEnumerable> value)
|
|
{
|
|
if (value != itemSource)
|
|
{
|
|
if (itemChangedHandler)
|
|
{
|
|
itemSource.Cast<IValueObservableList>()->ItemChanged.Remove(itemChangedHandler);
|
|
}
|
|
|
|
ClearItems();
|
|
itemSource = value.Cast<IValueList>();
|
|
if (!itemSource && value)
|
|
{
|
|
itemSource = IValueList::Create(GetLazyList<Value>(value));
|
|
}
|
|
|
|
if (itemTemplate && itemSource)
|
|
{
|
|
InstallItems();
|
|
}
|
|
if (auto observable = itemSource.Cast<IValueObservableList>())
|
|
{
|
|
itemChangedHandler = observable->ItemChanged.Add(this, &GuiRepeatCompositionBase::OnItemChanged);
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiRepeatStackComposition
|
|
***********************************************************************/
|
|
|
|
vint GuiRepeatStackComposition::GetRepeatCompositionCount()
|
|
{
|
|
return stackItems.Count();
|
|
}
|
|
|
|
GuiGraphicsComposition* GuiRepeatStackComposition::GetRepeatComposition(vint index)
|
|
{
|
|
return stackItems[index];
|
|
}
|
|
|
|
GuiGraphicsComposition* GuiRepeatStackComposition::InsertRepeatComposition(vint index)
|
|
{
|
|
CHECK_ERROR(0 <= index && index <= stackItems.Count(), L"GuiRepeatStackComposition::InsertRepeatComposition(vint)#Index out of range.");
|
|
auto item = new GuiStackItemComposition;
|
|
InsertStackItem(index, item);
|
|
return item;
|
|
}
|
|
|
|
GuiGraphicsComposition* GuiRepeatStackComposition::RemoveRepeatComposition(vint index)
|
|
{
|
|
auto item = stackItems[index];
|
|
RemoveChild(item);
|
|
return item;
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiRepeatFlowComposition
|
|
***********************************************************************/
|
|
|
|
vint GuiRepeatFlowComposition::GetRepeatCompositionCount()
|
|
{
|
|
return flowItems.Count();
|
|
}
|
|
|
|
GuiGraphicsComposition* GuiRepeatFlowComposition::GetRepeatComposition(vint index)
|
|
{
|
|
return flowItems[index];
|
|
}
|
|
|
|
GuiGraphicsComposition* GuiRepeatFlowComposition::InsertRepeatComposition(vint index)
|
|
{
|
|
CHECK_ERROR(0 <= index && index <= flowItems.Count(), L"GuiRepeatStackComposition::InsertRepeatComposition(vint)#Index out of range.");
|
|
auto item = new GuiFlowItemComposition;
|
|
InsertFlowItem(index, item);
|
|
return item;
|
|
}
|
|
|
|
GuiGraphicsComposition* GuiRepeatFlowComposition::RemoveRepeatComposition(vint index)
|
|
{
|
|
auto item = flowItems[index];
|
|
RemoveChild(item);
|
|
return item;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\GRAPHICSCOMPOSITION\GUIGRAPHICSRESPONSIVECOMPOSITION.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace compositions
|
|
{
|
|
using namespace collections;
|
|
using namespace controls;
|
|
|
|
/***********************************************************************
|
|
GuiResponsiveCompositionBase
|
|
***********************************************************************/
|
|
|
|
void GuiResponsiveCompositionBase::OnParentLineChanged()
|
|
{
|
|
GuiBoundsComposition::OnParentLineChanged();
|
|
GuiResponsiveCompositionBase* responsive = nullptr;
|
|
{
|
|
auto parent = GetParent();
|
|
while (parent)
|
|
{
|
|
if ((responsive = dynamic_cast<GuiResponsiveCompositionBase*>(parent)))
|
|
{
|
|
break;
|
|
}
|
|
parent = parent->GetParent();
|
|
}
|
|
}
|
|
|
|
if (responsiveParent != responsive)
|
|
{
|
|
if (responsiveParent)
|
|
{
|
|
responsiveParent->OnResponsiveChildRemoved(this);
|
|
responsiveParent->OnResponsiveChildLevelUpdated();
|
|
}
|
|
responsiveParent = responsive;
|
|
if (responsiveParent)
|
|
{
|
|
responsiveParent->OnResponsiveChildInserted(this);
|
|
responsiveParent->OnResponsiveChildLevelUpdated();
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiResponsiveCompositionBase::OnResponsiveChildInserted(GuiResponsiveCompositionBase* child)
|
|
{
|
|
}
|
|
|
|
void GuiResponsiveCompositionBase::OnResponsiveChildRemoved(GuiResponsiveCompositionBase* child)
|
|
{
|
|
}
|
|
|
|
void GuiResponsiveCompositionBase::OnResponsiveChildLevelUpdated()
|
|
{
|
|
if (responsiveParent)
|
|
{
|
|
responsiveParent->OnResponsiveChildLevelUpdated();
|
|
}
|
|
else
|
|
{
|
|
InvokeOnCompositionStateChanged();
|
|
}
|
|
}
|
|
|
|
GuiResponsiveCompositionBase::GuiResponsiveCompositionBase()
|
|
{
|
|
SetMinSizeLimitation(LimitToElementAndChildren);
|
|
SetPreferredMinSize(Size(1, 1));
|
|
|
|
LevelCountChanged.SetAssociatedComposition(this);
|
|
CurrentLevelChanged.SetAssociatedComposition(this);
|
|
}
|
|
|
|
GuiResponsiveCompositionBase::~GuiResponsiveCompositionBase()
|
|
{
|
|
}
|
|
|
|
ResponsiveDirection GuiResponsiveCompositionBase::GetDirection()
|
|
{
|
|
return direction;
|
|
}
|
|
|
|
void GuiResponsiveCompositionBase::SetDirection(ResponsiveDirection value)
|
|
{
|
|
if (direction != value)
|
|
{
|
|
direction = value;
|
|
OnResponsiveChildLevelUpdated();
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiResponsiveSharedCollection
|
|
***********************************************************************/
|
|
|
|
void GuiResponsiveSharedCollection::BeforeInsert(vint index, controls::GuiControl* const& value)
|
|
{
|
|
CHECK_ERROR(!value->GetBoundsComposition()->GetParent(), L"GuiResponsiveSharedCollection::BeforeInsert(vint, GuiResponsiveSharedCollection* const&)#Cannot insert a shared control that is currently in use.");
|
|
}
|
|
|
|
void GuiResponsiveSharedCollection::AfterInsert(vint index, controls::GuiControl* const& value)
|
|
{
|
|
view->OnResponsiveChildLevelUpdated();
|
|
}
|
|
|
|
void GuiResponsiveSharedCollection::BeforeRemove(vint index, controls::GuiControl* const& value)
|
|
{
|
|
CHECK_ERROR(!value->GetBoundsComposition()->GetParent(), L"GuiResponsiveSharedCollection::BeforeRemove(vint, GuiResponsiveSharedCollection* const&)#Cannot remove a shared control that is currently in use.");
|
|
}
|
|
|
|
void GuiResponsiveSharedCollection::AfterRemove(vint index, vint count)
|
|
{
|
|
view->OnResponsiveChildLevelUpdated();
|
|
}
|
|
|
|
GuiResponsiveSharedCollection::GuiResponsiveSharedCollection(GuiResponsiveViewComposition* _view)
|
|
:view(_view)
|
|
{
|
|
}
|
|
|
|
GuiResponsiveSharedCollection::~GuiResponsiveSharedCollection()
|
|
{
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiResponsiveViewCollection
|
|
***********************************************************************/
|
|
|
|
void GuiResponsiveViewCollection::BeforeInsert(vint index, GuiResponsiveCompositionBase* const& value)
|
|
{
|
|
CHECK_ERROR(!value->GetParent(), L"GuiResponsiveViewCollection::BeforeRemove(vint, GuiResponsiveCompositionBase* const&)#Cannot insert a view that is currently in use.");
|
|
}
|
|
|
|
void GuiResponsiveViewCollection::AfterInsert(vint index, GuiResponsiveCompositionBase* const& value)
|
|
{
|
|
if (!view->currentView)
|
|
{
|
|
view->skipUpdatingLevels = true;
|
|
view->currentView = value;
|
|
view->currentView->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
view->AddChild(view->currentView);
|
|
view->skipUpdatingLevels = false;
|
|
}
|
|
view->OnResponsiveChildLevelUpdated();
|
|
}
|
|
|
|
void GuiResponsiveViewCollection::BeforeRemove(vint index, GuiResponsiveCompositionBase* const& value)
|
|
{
|
|
CHECK_ERROR(!value->GetParent(), L"GuiResponsiveViewCollection::BeforeRemove(vint, GuiResponsiveCompositionBase* const&)#Cannot remove a view that is currently in use.");
|
|
}
|
|
|
|
void GuiResponsiveViewCollection::AfterRemove(vint index, vint count)
|
|
{
|
|
view->OnResponsiveChildLevelUpdated();
|
|
}
|
|
|
|
GuiResponsiveViewCollection::GuiResponsiveViewCollection(GuiResponsiveViewComposition* _view)
|
|
:view(_view)
|
|
{
|
|
}
|
|
|
|
GuiResponsiveViewCollection::~GuiResponsiveViewCollection()
|
|
{
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiResponsiveSharedComposition
|
|
***********************************************************************/
|
|
|
|
void GuiResponsiveSharedComposition::SetSharedControl()
|
|
{
|
|
if (shared && view && !view->destructing)
|
|
{
|
|
auto sharedParent = shared->GetBoundsComposition()->GetParent();
|
|
CHECK_ERROR(view->sharedControls.Contains(shared), L"GuiResponsiveSharedComposition::SetSharedControl()#The specified shared control is not in GuiResponsiveViewComposition::GetSharedControls().");
|
|
CHECK_ERROR(!sharedParent || sharedParent == this, L"GuiResponsiveSharedComposition::SetSharedControl()#The specified shared control has not been released. This usually means this control is not in GuiResponsiveViewComposition::GetSharedControls().");
|
|
|
|
if (!sharedParent)
|
|
{
|
|
shared->GetBoundsComposition()->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
AddChild(shared->GetBoundsComposition());
|
|
view->usedSharedControls.Add(shared);
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiResponsiveSharedComposition::OnParentLineChanged()
|
|
{
|
|
GuiBoundsComposition::OnParentLineChanged();
|
|
if (view && view->destructing)
|
|
{
|
|
return;
|
|
}
|
|
|
|
GuiResponsiveViewComposition* currentView = nullptr;
|
|
{
|
|
auto parent = GetParent();
|
|
while (parent)
|
|
{
|
|
if ((currentView = dynamic_cast<GuiResponsiveViewComposition*>(parent)))
|
|
{
|
|
break;
|
|
}
|
|
parent = parent->GetParent();
|
|
}
|
|
}
|
|
|
|
if (currentView != view && view && shared)
|
|
{
|
|
RemoveChild(shared->GetBoundsComposition());
|
|
view->usedSharedControls.Remove(shared);
|
|
}
|
|
view = currentView;
|
|
|
|
SetSharedControl();
|
|
}
|
|
|
|
GuiResponsiveSharedComposition::GuiResponsiveSharedComposition()
|
|
{
|
|
SetMinSizeLimitation(LimitToElementAndChildren);
|
|
}
|
|
|
|
GuiResponsiveSharedComposition::~GuiResponsiveSharedComposition()
|
|
{
|
|
}
|
|
|
|
controls::GuiControl* GuiResponsiveSharedComposition::GetShared()
|
|
{
|
|
return shared;
|
|
}
|
|
|
|
void GuiResponsiveSharedComposition::SetShared(controls::GuiControl* value)
|
|
{
|
|
if (shared != value)
|
|
{
|
|
CHECK_ERROR(!shared || !shared->GetBoundsComposition()->GetParent(), L"GuiResponsiveSharedComposition::SetShared(GuiControl*)#Cannot replace a shared control that is currently in use.");
|
|
shared = value;
|
|
SetSharedControl();
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiResponsiveViewComposition
|
|
***********************************************************************/
|
|
|
|
bool GuiResponsiveViewComposition::CalculateLevelCount()
|
|
{
|
|
vint old = levelCount;
|
|
if (views.Count() == 0)
|
|
{
|
|
levelCount = 1;
|
|
}
|
|
else
|
|
{
|
|
levelCount = 0;
|
|
for (vint i = 0; i < views.Count(); i++)
|
|
{
|
|
auto view = views[i];
|
|
if (((vint)direction & (vint)view->GetDirection()) != 0)
|
|
{
|
|
levelCount += view->GetLevelCount();
|
|
}
|
|
else
|
|
{
|
|
levelCount += 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (old != levelCount)
|
|
{
|
|
LevelCountChanged.Execute(GuiEventArgs(this));
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool GuiResponsiveViewComposition::CalculateCurrentLevel()
|
|
{
|
|
vint old = currentLevel;
|
|
currentLevel = 0;
|
|
for (vint i = views.Count() - 1; i >= 0; i--)
|
|
{
|
|
auto view = views[i];
|
|
if (((vint)direction & (vint)view->GetDirection()) != 0)
|
|
{
|
|
if (currentView == view)
|
|
{
|
|
currentLevel += view->GetCurrentLevel() + 1;
|
|
break;
|
|
}
|
|
else
|
|
{
|
|
currentLevel += view->GetLevelCount();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
currentLevel++;
|
|
}
|
|
}
|
|
currentLevel--;
|
|
|
|
if (old != currentLevel)
|
|
{
|
|
CurrentLevelChanged.Execute(GuiEventArgs(this));
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void GuiResponsiveViewComposition::OnResponsiveChildLevelUpdated()
|
|
{
|
|
if (!skipUpdatingLevels)
|
|
{
|
|
CalculateLevelCount();
|
|
CalculateCurrentLevel();
|
|
GuiResponsiveCompositionBase::OnResponsiveChildLevelUpdated();
|
|
}
|
|
}
|
|
|
|
GuiResponsiveViewComposition::GuiResponsiveViewComposition()
|
|
:sharedControls(this)
|
|
, views(this)
|
|
{
|
|
BeforeSwitchingView.SetAssociatedComposition(this);
|
|
}
|
|
|
|
GuiResponsiveViewComposition::~GuiResponsiveViewComposition()
|
|
{
|
|
destructing = true;
|
|
|
|
FOREACH(GuiResponsiveCompositionBase*, view, views)
|
|
{
|
|
if (view != currentView)
|
|
{
|
|
SafeDeleteComposition(view);
|
|
}
|
|
}
|
|
|
|
FOREACH(GuiControl*, shared, From(sharedControls).Except(usedSharedControls))
|
|
{
|
|
SafeDeleteControl(shared);
|
|
}
|
|
}
|
|
|
|
vint GuiResponsiveViewComposition::GetLevelCount()
|
|
{
|
|
return levelCount;
|
|
}
|
|
|
|
vint GuiResponsiveViewComposition::GetCurrentLevel()
|
|
{
|
|
return currentLevel;
|
|
}
|
|
|
|
bool GuiResponsiveViewComposition::LevelDown()
|
|
{
|
|
skipUpdatingLevels = true;
|
|
if (((vint)direction & (vint)currentView->GetDirection()) != 0 && !currentView->LevelDown())
|
|
{
|
|
vint index = views.IndexOf(currentView);
|
|
if (index < views.Count() - 1)
|
|
{
|
|
RemoveChild(currentView);
|
|
currentView = views[index + 1];
|
|
currentView->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
{
|
|
GuiItemEventArgs arguments(this);
|
|
arguments.itemIndex = views.IndexOf(currentView);
|
|
BeforeSwitchingView.Execute(arguments);
|
|
}
|
|
AddChild(currentView);
|
|
}
|
|
}
|
|
skipUpdatingLevels = false;
|
|
|
|
auto x = CalculateLevelCount();
|
|
auto y = CalculateCurrentLevel();
|
|
if (!x && !y) return false;
|
|
InvokeOnCompositionStateChanged();
|
|
return true;
|
|
}
|
|
|
|
bool GuiResponsiveViewComposition::LevelUp()
|
|
{
|
|
skipUpdatingLevels = true;
|
|
if (((vint)direction & (vint)currentView->GetDirection()) != 0 && !currentView->LevelUp())
|
|
{
|
|
vint index = views.IndexOf(currentView);
|
|
if (index > 0)
|
|
{
|
|
RemoveChild(currentView);
|
|
currentView = views[index - 1];
|
|
currentView->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
{
|
|
GuiItemEventArgs arguments(this);
|
|
arguments.itemIndex = views.IndexOf(currentView);
|
|
BeforeSwitchingView.Execute(arguments);
|
|
}
|
|
AddChild(currentView);
|
|
}
|
|
}
|
|
skipUpdatingLevels = false;
|
|
|
|
auto x = CalculateLevelCount();
|
|
auto y = CalculateCurrentLevel();
|
|
if (!x && !y) return false;
|
|
InvokeOnCompositionStateChanged();
|
|
return true;
|
|
}
|
|
|
|
GuiResponsiveCompositionBase* GuiResponsiveViewComposition::GetCurrentView()
|
|
{
|
|
return currentView;
|
|
}
|
|
|
|
collections::ObservableListBase<controls::GuiControl*>& GuiResponsiveViewComposition::GetSharedControls()
|
|
{
|
|
return sharedControls;
|
|
}
|
|
|
|
collections::ObservableListBase<GuiResponsiveCompositionBase*>& GuiResponsiveViewComposition::GetViews()
|
|
{
|
|
return views;
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiResponsiveFixedComposition
|
|
***********************************************************************/
|
|
|
|
void GuiResponsiveFixedComposition::OnResponsiveChildLevelUpdated()
|
|
{
|
|
InvokeOnCompositionStateChanged();
|
|
}
|
|
|
|
GuiResponsiveFixedComposition::GuiResponsiveFixedComposition()
|
|
{
|
|
}
|
|
|
|
GuiResponsiveFixedComposition::~GuiResponsiveFixedComposition()
|
|
{
|
|
}
|
|
|
|
vint GuiResponsiveFixedComposition::GetLevelCount()
|
|
{
|
|
return 1;
|
|
}
|
|
|
|
vint GuiResponsiveFixedComposition::GetCurrentLevel()
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
bool GuiResponsiveFixedComposition::LevelDown()
|
|
{
|
|
return false;
|
|
}
|
|
|
|
bool GuiResponsiveFixedComposition::LevelUp()
|
|
{
|
|
return false;
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiResponsiveStackComposition
|
|
***********************************************************************/
|
|
|
|
#define DEFINE_AVAILABLE \
|
|
auto availables = From(responsiveChildren) \
|
|
.Where([=](GuiResponsiveCompositionBase* child) \
|
|
{ \
|
|
return ((vint)direction & (vint)child->GetDirection()) != 0; \
|
|
}) \
|
|
|
|
bool GuiResponsiveStackComposition::CalculateLevelCount()
|
|
{
|
|
vint old = levelCount;
|
|
DEFINE_AVAILABLE;
|
|
if (availables.IsEmpty())
|
|
{
|
|
levelCount = 1;
|
|
}
|
|
else
|
|
{
|
|
levelCount = availables
|
|
.Select([](GuiResponsiveCompositionBase* child)
|
|
{
|
|
return child->GetLevelCount() - 1;
|
|
})
|
|
.Aggregate([](vint a, vint b)
|
|
{
|
|
return a + b;
|
|
}) + 1;
|
|
}
|
|
|
|
if (old != levelCount)
|
|
{
|
|
LevelCountChanged.Execute(GuiEventArgs(this));
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool GuiResponsiveStackComposition::CalculateCurrentLevel()
|
|
{
|
|
vint old = currentLevel;
|
|
DEFINE_AVAILABLE;
|
|
if (availables.IsEmpty())
|
|
{
|
|
currentLevel = 0;
|
|
}
|
|
else
|
|
{
|
|
currentLevel = availables
|
|
.Select([](GuiResponsiveCompositionBase* child)
|
|
{
|
|
return child->GetCurrentLevel();
|
|
})
|
|
.Aggregate([](vint a, vint b)
|
|
{
|
|
return a + b;
|
|
});
|
|
}
|
|
|
|
if (old != currentLevel)
|
|
{
|
|
CurrentLevelChanged.Execute(GuiEventArgs(this));
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void GuiResponsiveStackComposition::OnResponsiveChildInserted(GuiResponsiveCompositionBase* child)
|
|
{
|
|
responsiveChildren.Add(child);
|
|
}
|
|
|
|
void GuiResponsiveStackComposition::OnResponsiveChildRemoved(GuiResponsiveCompositionBase* child)
|
|
{
|
|
responsiveChildren.Remove(child);
|
|
}
|
|
|
|
void GuiResponsiveStackComposition::OnResponsiveChildLevelUpdated()
|
|
{
|
|
CalculateLevelCount();
|
|
CalculateCurrentLevel();
|
|
GuiResponsiveCompositionBase::OnResponsiveChildLevelUpdated();
|
|
}
|
|
|
|
bool GuiResponsiveStackComposition::ChangeLevel(bool levelDown)
|
|
{
|
|
DEFINE_AVAILABLE;
|
|
|
|
SortedList<GuiResponsiveCompositionBase*> ignored;
|
|
while (true)
|
|
{
|
|
GuiResponsiveCompositionBase* selected = nullptr;
|
|
vint size = 0;
|
|
|
|
FOREACH(GuiResponsiveCompositionBase*, child, availables)
|
|
{
|
|
if (!ignored.Contains(child))
|
|
{
|
|
Size childSize = child->GetPreferredBounds().GetSize();
|
|
vint childSizeToCompare =
|
|
direction == ResponsiveDirection::Horizontal ? childSize.x :
|
|
direction == ResponsiveDirection::Vertical ? childSize.y :
|
|
childSize.x * childSize.y;
|
|
|
|
if (!selected || (levelDown ? size < childSizeToCompare : size > childSizeToCompare))
|
|
{
|
|
selected = child;
|
|
size = childSizeToCompare;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!selected)
|
|
{
|
|
break;
|
|
}
|
|
else if (levelDown ? selected->LevelDown() : selected->LevelUp())
|
|
{
|
|
break;
|
|
}
|
|
else
|
|
{
|
|
ignored.Add(selected);
|
|
}
|
|
}
|
|
|
|
if (!CalculateCurrentLevel()) return false;
|
|
InvokeOnCompositionStateChanged();
|
|
return true;
|
|
}
|
|
|
|
GuiResponsiveStackComposition::GuiResponsiveStackComposition()
|
|
{
|
|
}
|
|
|
|
GuiResponsiveStackComposition::~GuiResponsiveStackComposition()
|
|
{
|
|
}
|
|
|
|
vint GuiResponsiveStackComposition::GetLevelCount()
|
|
{
|
|
return levelCount;
|
|
}
|
|
|
|
vint GuiResponsiveStackComposition::GetCurrentLevel()
|
|
{
|
|
return currentLevel;
|
|
}
|
|
|
|
bool GuiResponsiveStackComposition::LevelDown()
|
|
{
|
|
return ChangeLevel(true);
|
|
}
|
|
|
|
bool GuiResponsiveStackComposition::LevelUp()
|
|
{
|
|
return ChangeLevel(false);
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiResponsiveGroupComposition
|
|
***********************************************************************/
|
|
|
|
bool GuiResponsiveGroupComposition::CalculateLevelCount()
|
|
{
|
|
vint old = levelCount;
|
|
DEFINE_AVAILABLE;
|
|
if (availables.IsEmpty())
|
|
{
|
|
levelCount = 1;
|
|
}
|
|
else
|
|
{
|
|
levelCount = availables
|
|
.Select([](GuiResponsiveCompositionBase* child)
|
|
{
|
|
return child->GetLevelCount();
|
|
})
|
|
.Max();
|
|
}
|
|
|
|
if (old != levelCount)
|
|
{
|
|
LevelCountChanged.Execute(GuiEventArgs(this));
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool GuiResponsiveGroupComposition::CalculateCurrentLevel()
|
|
{
|
|
vint old = currentLevel;
|
|
DEFINE_AVAILABLE;
|
|
if (availables.IsEmpty())
|
|
{
|
|
currentLevel = 0;
|
|
}
|
|
else
|
|
{
|
|
currentLevel = availables
|
|
.Select([](GuiResponsiveCompositionBase* child)
|
|
{
|
|
return child->GetCurrentLevel();
|
|
})
|
|
.Max();
|
|
}
|
|
|
|
if (old != currentLevel)
|
|
{
|
|
CurrentLevelChanged.Execute(GuiEventArgs(this));
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void GuiResponsiveGroupComposition::OnResponsiveChildInserted(GuiResponsiveCompositionBase* child)
|
|
{
|
|
responsiveChildren.Add(child);
|
|
}
|
|
|
|
void GuiResponsiveGroupComposition::OnResponsiveChildRemoved(GuiResponsiveCompositionBase* child)
|
|
{
|
|
responsiveChildren.Remove(child);
|
|
}
|
|
|
|
void GuiResponsiveGroupComposition::OnResponsiveChildLevelUpdated()
|
|
{
|
|
CalculateLevelCount();
|
|
CalculateCurrentLevel();
|
|
GuiResponsiveCompositionBase::OnResponsiveChildLevelUpdated();
|
|
}
|
|
|
|
GuiResponsiveGroupComposition::GuiResponsiveGroupComposition()
|
|
{
|
|
}
|
|
|
|
GuiResponsiveGroupComposition::~GuiResponsiveGroupComposition()
|
|
{
|
|
}
|
|
|
|
vint GuiResponsiveGroupComposition::GetLevelCount()
|
|
{
|
|
return levelCount;
|
|
}
|
|
|
|
vint GuiResponsiveGroupComposition::GetCurrentLevel()
|
|
{
|
|
return currentLevel;
|
|
}
|
|
|
|
bool GuiResponsiveGroupComposition::LevelDown()
|
|
{
|
|
DEFINE_AVAILABLE;
|
|
vint level = currentLevel;
|
|
FOREACH(GuiResponsiveCompositionBase*, child, availables)
|
|
{
|
|
if (child->GetCurrentLevel() >= level)
|
|
{
|
|
if (!child->LevelDown())
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!CalculateCurrentLevel()) return false;
|
|
InvokeOnCompositionStateChanged();
|
|
return true;
|
|
}
|
|
|
|
bool GuiResponsiveGroupComposition::LevelUp()
|
|
{
|
|
DEFINE_AVAILABLE;
|
|
vint level = currentLevel;
|
|
FOREACH(GuiResponsiveCompositionBase*, child, availables)
|
|
{
|
|
while (child->GetCurrentLevel() <= level)
|
|
{
|
|
if (!child->LevelUp())
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!CalculateCurrentLevel()) return false;
|
|
InvokeOnCompositionStateChanged();
|
|
return true;
|
|
}
|
|
|
|
#undef DEFINE_AVAILABLE
|
|
|
|
/***********************************************************************
|
|
GuiResponsiveContainerComposition
|
|
***********************************************************************/
|
|
|
|
#define RESPONSIVE_INVALID_SIZE Size(-1, -1)
|
|
|
|
void GuiResponsiveContainerComposition::AdjustLevel()
|
|
{
|
|
if (!responsiveTarget) return;
|
|
const Size containerSize = GetBounds().GetSize();
|
|
const Size responsiveOriginalSize = responsiveTarget->GetPreferredBounds().GetSize();
|
|
const bool testX = (vint)responsiveTarget->GetDirection() & (vint)ResponsiveDirection::Horizontal;
|
|
const bool testY = (vint)responsiveTarget->GetDirection() & (vint)ResponsiveDirection::Vertical;
|
|
|
|
#define RESPONSIVE_IF_CONTAINER(OP, SIZE) ((testX && (containerSize).x OP SIZE.x) || (testY && (containerSize).y OP SIZE.y))
|
|
|
|
if (upperLevelSize != RESPONSIVE_INVALID_SIZE && RESPONSIVE_IF_CONTAINER(>=, upperLevelSize))
|
|
{
|
|
upperLevelSize = RESPONSIVE_INVALID_SIZE;
|
|
}
|
|
|
|
if (upperLevelSize == RESPONSIVE_INVALID_SIZE && RESPONSIVE_IF_CONTAINER(>=, responsiveOriginalSize))
|
|
{
|
|
while (true)
|
|
{
|
|
if (responsiveTarget->GetCurrentLevel() == responsiveTarget->GetLevelCount() - 1)
|
|
{
|
|
break;
|
|
}
|
|
else if (responsiveTarget->LevelUp())
|
|
{
|
|
responsiveTarget->ForceCalculateSizeImmediately();
|
|
auto currentSize = responsiveTarget->GetPreferredBounds().GetSize();
|
|
if (RESPONSIVE_IF_CONTAINER(<, currentSize))
|
|
{
|
|
upperLevelSize = currentSize;
|
|
responsiveTarget->LevelDown();
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
while (true)
|
|
{
|
|
responsiveTarget->ForceCalculateSizeImmediately();
|
|
auto currentSize = responsiveTarget->GetPreferredBounds().GetSize();
|
|
if (RESPONSIVE_IF_CONTAINER(>=, currentSize))
|
|
{
|
|
break;
|
|
}
|
|
|
|
if (responsiveTarget->GetCurrentLevel() == 0)
|
|
{
|
|
break;
|
|
}
|
|
else if(responsiveTarget->LevelDown())
|
|
{
|
|
upperLevelSize = currentSize;
|
|
}
|
|
else
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
#undef RESPONSIVE_IF_CONTAINER
|
|
}
|
|
|
|
void GuiResponsiveContainerComposition::OnBoundsChanged(GuiGraphicsComposition* sender, GuiEventArgs& arguments)
|
|
{
|
|
auto control = GetRelatedControl();
|
|
if (control)
|
|
{
|
|
control->InvokeOrDelayIfRendering([=]()
|
|
{
|
|
AdjustLevel();
|
|
});
|
|
}
|
|
else
|
|
{
|
|
AdjustLevel();
|
|
}
|
|
}
|
|
|
|
GuiResponsiveContainerComposition::GuiResponsiveContainerComposition()
|
|
:upperLevelSize(RESPONSIVE_INVALID_SIZE)
|
|
{
|
|
BoundsChanged.AttachMethod(this, &GuiResponsiveContainerComposition::OnBoundsChanged);
|
|
}
|
|
|
|
GuiResponsiveContainerComposition::~GuiResponsiveContainerComposition()
|
|
{
|
|
}
|
|
|
|
GuiResponsiveCompositionBase* GuiResponsiveContainerComposition::GetResponsiveTarget()
|
|
{
|
|
return responsiveTarget;
|
|
}
|
|
|
|
void GuiResponsiveContainerComposition::SetResponsiveTarget(GuiResponsiveCompositionBase* value)
|
|
{
|
|
if (responsiveTarget != value)
|
|
{
|
|
if (responsiveTarget)
|
|
{
|
|
RemoveChild(responsiveTarget);
|
|
}
|
|
|
|
responsiveTarget = value;
|
|
upperLevelSize = RESPONSIVE_INVALID_SIZE;
|
|
|
|
if (responsiveTarget)
|
|
{
|
|
responsiveTarget->SetAlignmentToParent(Margin(0, 0, 0, 0));
|
|
while (responsiveTarget->LevelUp());
|
|
AddChild(responsiveTarget);
|
|
|
|
GuiEventArgs arguments(this);
|
|
OnBoundsChanged(this, arguments);
|
|
}
|
|
}
|
|
}
|
|
|
|
#undef RESPONSIVE_INVALID_SIZE
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
.\GRAPHICSCOMPOSITION\GUIGRAPHICSSHAREDSIZECOMPOSITION.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace compositions
|
|
{
|
|
using namespace reflection::description;
|
|
using namespace collections;
|
|
using namespace controls;
|
|
using namespace elements;
|
|
|
|
/***********************************************************************
|
|
GuiSharedSizeItemComposition
|
|
***********************************************************************/
|
|
|
|
void GuiSharedSizeItemComposition::Update()
|
|
{
|
|
if (parentRoot)
|
|
{
|
|
parentRoot->ForceCalculateSizeImmediately();
|
|
}
|
|
InvokeOnCompositionStateChanged();
|
|
}
|
|
|
|
void GuiSharedSizeItemComposition::OnParentLineChanged()
|
|
{
|
|
GuiBoundsComposition::OnParentLineChanged();
|
|
if (parentRoot)
|
|
{
|
|
parentRoot->childItems.Remove(this);
|
|
parentRoot = nullptr;
|
|
}
|
|
|
|
auto current = GetParent();
|
|
while (current)
|
|
{
|
|
if (auto item = dynamic_cast<GuiSharedSizeItemComposition*>(current))
|
|
{
|
|
break;
|
|
}
|
|
else if (auto root = dynamic_cast<GuiSharedSizeRootComposition*>(current))
|
|
{
|
|
parentRoot = root;
|
|
break;
|
|
}
|
|
current = current->GetParent();
|
|
}
|
|
|
|
if (parentRoot)
|
|
{
|
|
parentRoot->childItems.Add(this);
|
|
Size minSize;
|
|
if (sharedWidth)
|
|
{
|
|
vint index = parentRoot->itemWidths.Keys().IndexOf(group);
|
|
if (index != -1)
|
|
{
|
|
minSize.x = parentRoot->itemWidths.Values()[index];
|
|
}
|
|
}
|
|
if (sharedHeight)
|
|
{
|
|
vint index = parentRoot->itemHeights.Keys().IndexOf(group);
|
|
if (index != -1)
|
|
{
|
|
minSize.y = parentRoot->itemHeights.Values()[index];
|
|
}
|
|
}
|
|
SetPreferredMinSize(minSize);
|
|
}
|
|
}
|
|
|
|
GuiSharedSizeItemComposition::GuiSharedSizeItemComposition()
|
|
{
|
|
SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
}
|
|
|
|
GuiSharedSizeItemComposition::~GuiSharedSizeItemComposition()
|
|
{
|
|
}
|
|
|
|
const WString& GuiSharedSizeItemComposition::GetGroup()
|
|
{
|
|
return group;
|
|
}
|
|
|
|
void GuiSharedSizeItemComposition::SetGroup(const WString& value)
|
|
{
|
|
if (group != value)
|
|
{
|
|
group = value;
|
|
Update();
|
|
}
|
|
}
|
|
|
|
bool GuiSharedSizeItemComposition::GetSharedWidth()
|
|
{
|
|
return sharedWidth;
|
|
}
|
|
|
|
void GuiSharedSizeItemComposition::SetSharedWidth(bool value)
|
|
{
|
|
if (sharedWidth != value)
|
|
{
|
|
sharedWidth = value;
|
|
Update();
|
|
}
|
|
}
|
|
|
|
bool GuiSharedSizeItemComposition::GetSharedHeight()
|
|
{
|
|
return sharedHeight;
|
|
}
|
|
|
|
void GuiSharedSizeItemComposition::SetSharedHeight(bool value)
|
|
{
|
|
if (sharedHeight != value)
|
|
{
|
|
sharedHeight = value;
|
|
Update();
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiSharedSizeRootComposition
|
|
***********************************************************************/
|
|
|
|
void GuiSharedSizeRootComposition::AddSizeComponent(collections::Dictionary<WString, vint>& sizes, const WString& group, vint sizeComponent)
|
|
{
|
|
vint index = sizes.Keys().IndexOf(group);
|
|
if (index == -1)
|
|
{
|
|
sizes.Add(group, sizeComponent);
|
|
}
|
|
else if (sizes.Values().Get(index) < sizeComponent)
|
|
{
|
|
sizes.Set(group, sizeComponent);
|
|
}
|
|
}
|
|
|
|
void GuiSharedSizeRootComposition::CollectSizes(collections::Dictionary<WString, vint>& widths, collections::Dictionary<WString, vint>& heights)
|
|
{
|
|
FOREACH(GuiSharedSizeItemComposition*, item, childItems)
|
|
{
|
|
auto group = item->GetGroup();
|
|
auto minSize = item->GetPreferredMinSize();
|
|
item->SetPreferredMinSize(Size(0, 0));
|
|
auto size = item->GetPreferredBounds().GetSize();
|
|
|
|
if (item->GetSharedWidth())
|
|
{
|
|
AddSizeComponent(widths, group, size.x);
|
|
}
|
|
if (item->GetSharedHeight())
|
|
{
|
|
AddSizeComponent(heights, group, size.y);
|
|
}
|
|
|
|
item->SetPreferredMinSize(minSize);
|
|
}
|
|
}
|
|
|
|
void GuiSharedSizeRootComposition::AlignSizes(collections::Dictionary<WString, vint>& widths, collections::Dictionary<WString, vint>& heights)
|
|
{
|
|
FOREACH(GuiSharedSizeItemComposition*, item, childItems)
|
|
{
|
|
auto group = item->GetGroup();
|
|
auto size = item->GetPreferredMinSize();
|
|
|
|
if (item->GetSharedWidth())
|
|
{
|
|
size.x = widths[group];
|
|
}
|
|
if (item->GetSharedHeight())
|
|
{
|
|
size.y = heights[group];
|
|
}
|
|
|
|
item->SetPreferredMinSize(size);
|
|
}
|
|
}
|
|
|
|
GuiSharedSizeRootComposition::GuiSharedSizeRootComposition()
|
|
{
|
|
}
|
|
|
|
GuiSharedSizeRootComposition::~GuiSharedSizeRootComposition()
|
|
{
|
|
}
|
|
|
|
void GuiSharedSizeRootComposition::ForceCalculateSizeImmediately()
|
|
{
|
|
itemWidths.Clear();
|
|
itemHeights.Clear();
|
|
|
|
CollectSizes(itemWidths, itemHeights);
|
|
AlignSizes(itemWidths, itemHeights);
|
|
GuiBoundsComposition::ForceCalculateSizeImmediately();
|
|
}
|
|
|
|
Rect GuiSharedSizeRootComposition::GetBounds()
|
|
{
|
|
Dictionary<WString, vint> widths, heights;
|
|
CollectSizes(widths, heights);
|
|
bool minSizeModified = CompareEnumerable(itemWidths, widths) != 0 || CompareEnumerable(itemHeights, heights) != 0;
|
|
|
|
if (minSizeModified)
|
|
{
|
|
CopyFrom(itemWidths, widths);
|
|
CopyFrom(itemHeights, heights);
|
|
AlignSizes(itemWidths, itemHeights);
|
|
GuiBoundsComposition::ForceCalculateSizeImmediately();
|
|
}
|
|
return GuiBoundsComposition::GetBounds();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\GRAPHICSCOMPOSITION\GUIGRAPHICSSPECIALIZEDCOMPOSITION.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace compositions
|
|
{
|
|
|
|
/***********************************************************************
|
|
GuiSideAlignedComposition
|
|
***********************************************************************/
|
|
|
|
GuiSideAlignedComposition::GuiSideAlignedComposition()
|
|
:direction(Top)
|
|
,maxLength(10)
|
|
,maxRatio(1.0)
|
|
{
|
|
}
|
|
|
|
GuiSideAlignedComposition::~GuiSideAlignedComposition()
|
|
{
|
|
}
|
|
|
|
GuiSideAlignedComposition::Direction GuiSideAlignedComposition::GetDirection()
|
|
{
|
|
return direction;
|
|
}
|
|
|
|
void GuiSideAlignedComposition::SetDirection(Direction value)
|
|
{
|
|
direction = value;
|
|
InvokeOnCompositionStateChanged();
|
|
}
|
|
|
|
vint GuiSideAlignedComposition::GetMaxLength()
|
|
{
|
|
return maxLength;
|
|
}
|
|
|
|
void GuiSideAlignedComposition::SetMaxLength(vint value)
|
|
{
|
|
if (value < 0) value = 0;
|
|
maxLength = value;
|
|
InvokeOnCompositionStateChanged();
|
|
}
|
|
|
|
double GuiSideAlignedComposition::GetMaxRatio()
|
|
{
|
|
return maxRatio;
|
|
}
|
|
|
|
void GuiSideAlignedComposition::SetMaxRatio(double value)
|
|
{
|
|
maxRatio =
|
|
value < 0 ? 0 :
|
|
value>1 ? 1 :
|
|
value;
|
|
InvokeOnCompositionStateChanged();
|
|
}
|
|
|
|
bool GuiSideAlignedComposition::IsSizeAffectParent()
|
|
{
|
|
return false;
|
|
}
|
|
|
|
Rect GuiSideAlignedComposition::GetBounds()
|
|
{
|
|
Rect result;
|
|
GuiGraphicsComposition* parent = GetParent();
|
|
if (parent)
|
|
{
|
|
Rect bounds = parent->GetBounds();
|
|
vint w = (vint)(bounds.Width()*maxRatio);
|
|
vint h = (vint)(bounds.Height()*maxRatio);
|
|
if (w > maxLength) w = maxLength;
|
|
if (h > maxLength) h = maxLength;
|
|
switch (direction)
|
|
{
|
|
case Left:
|
|
{
|
|
bounds.x2 = bounds.x1 + w;
|
|
}
|
|
break;
|
|
case Top:
|
|
{
|
|
bounds.y2 = bounds.y1 + h;
|
|
}
|
|
break;
|
|
case Right:
|
|
{
|
|
bounds.x1 = bounds.x2 - w;
|
|
}
|
|
break;
|
|
case Bottom:
|
|
{
|
|
bounds.y1 = bounds.y2 - h;
|
|
}
|
|
break;
|
|
}
|
|
result = bounds;
|
|
}
|
|
UpdatePreviousBounds(result);
|
|
return result;
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiPartialViewComposition
|
|
***********************************************************************/
|
|
|
|
GuiPartialViewComposition::GuiPartialViewComposition()
|
|
:wRatio(0.0)
|
|
,wPageSize(1.0)
|
|
,hRatio(0.0)
|
|
,hPageSize(1.0)
|
|
{
|
|
}
|
|
|
|
GuiPartialViewComposition::~GuiPartialViewComposition()
|
|
{
|
|
}
|
|
|
|
double GuiPartialViewComposition::GetWidthRatio()
|
|
{
|
|
return wRatio;
|
|
}
|
|
|
|
double GuiPartialViewComposition::GetWidthPageSize()
|
|
{
|
|
return wPageSize;
|
|
}
|
|
|
|
double GuiPartialViewComposition::GetHeightRatio()
|
|
{
|
|
return hRatio;
|
|
}
|
|
|
|
double GuiPartialViewComposition::GetHeightPageSize()
|
|
{
|
|
return hPageSize;
|
|
}
|
|
|
|
void GuiPartialViewComposition::SetWidthRatio(double value)
|
|
{
|
|
wRatio = value;
|
|
InvokeOnCompositionStateChanged();
|
|
}
|
|
|
|
void GuiPartialViewComposition::SetWidthPageSize(double value)
|
|
{
|
|
wPageSize = value;
|
|
InvokeOnCompositionStateChanged();
|
|
}
|
|
|
|
void GuiPartialViewComposition::SetHeightRatio(double value)
|
|
{
|
|
hRatio = value;
|
|
InvokeOnCompositionStateChanged();
|
|
}
|
|
|
|
void GuiPartialViewComposition::SetHeightPageSize(double value)
|
|
{
|
|
hPageSize = value;
|
|
InvokeOnCompositionStateChanged();
|
|
}
|
|
|
|
bool GuiPartialViewComposition::IsSizeAffectParent()
|
|
{
|
|
return false;
|
|
}
|
|
|
|
Rect GuiPartialViewComposition::GetBounds()
|
|
{
|
|
Rect result;
|
|
GuiGraphicsComposition* parent = GetParent();
|
|
if (parent)
|
|
{
|
|
Rect bounds = parent->GetBounds();
|
|
vint w = bounds.Width();
|
|
vint h = bounds.Height();
|
|
vint pw = (vint)(wPageSize*w);
|
|
vint ph = (vint)(hPageSize*h);
|
|
|
|
vint ow = preferredMinSize.x - pw;
|
|
if (ow < 0) ow = 0;
|
|
vint oh = preferredMinSize.y - ph;
|
|
if (oh < 0) oh = 0;
|
|
|
|
w -= ow;
|
|
h -= oh;
|
|
pw += ow;
|
|
ph += oh;
|
|
|
|
result = Rect(Point((vint)(wRatio*w), (vint)(hRatio*h)), Size(pw, ph));
|
|
}
|
|
UpdatePreviousBounds(result);
|
|
return result;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
.\GRAPHICSCOMPOSITION\GUIGRAPHICSSTACKCOMPOSITION.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace compositions
|
|
{
|
|
|
|
/***********************************************************************
|
|
GuiStackComposition
|
|
***********************************************************************/
|
|
|
|
void GuiStackComposition::UpdateStackItemBounds()
|
|
{
|
|
if (stackItemBounds.Count() != stackItems.Count())
|
|
{
|
|
stackItemBounds.Resize(stackItems.Count());
|
|
}
|
|
|
|
stackItemTotalSize = Size(0, 0);
|
|
Point offset;
|
|
for (vint i = 0; i < stackItems.Count(); i++)
|
|
{
|
|
vint offsetX = 0;
|
|
vint offsetY = 0;
|
|
Size itemSize = stackItems[i]->GetMinSize();
|
|
stackItemBounds[i] = Rect(offset, itemSize);
|
|
|
|
#define ACCUMULATE(U, V) \
|
|
{ \
|
|
if (stackItemTotalSize.V < itemSize.V) \
|
|
{ \
|
|
stackItemTotalSize.V = itemSize.V; \
|
|
} \
|
|
if (i > 0) \
|
|
{ \
|
|
stackItemTotalSize.U += padding; \
|
|
} \
|
|
stackItemTotalSize.U += itemSize.U; \
|
|
} \
|
|
|
|
switch (direction)
|
|
{
|
|
case GuiStackComposition::Horizontal:
|
|
case GuiStackComposition::ReversedHorizontal:
|
|
ACCUMULATE(x, y)
|
|
break;
|
|
case GuiStackComposition::Vertical:
|
|
case GuiStackComposition::ReversedVertical:
|
|
ACCUMULATE(y, x)
|
|
break;
|
|
}
|
|
|
|
#undef ACCUMULATE
|
|
offset.x += itemSize.x + padding;
|
|
offset.y += itemSize.y + padding;
|
|
}
|
|
EnsureStackItemVisible();
|
|
}
|
|
|
|
void GuiStackComposition::EnsureStackItemVisible()
|
|
{
|
|
#define ADJUSTMENT(U, V) \
|
|
if (itemBounds.U() <= 0) \
|
|
{ \
|
|
adjustment -= itemBounds.U(); \
|
|
} \
|
|
else \
|
|
{ \
|
|
vint overflow = itemBounds.V() - previousBounds.V(); \
|
|
if (overflow > 0) \
|
|
{ \
|
|
adjustment -= overflow; \
|
|
} \
|
|
} \
|
|
|
|
if (ensuringVisibleStackItem)
|
|
{
|
|
Rect itemBounds = ensuringVisibleStackItem->GetBounds();
|
|
switch (direction)
|
|
{
|
|
case Horizontal:
|
|
case ReversedHorizontal:
|
|
ADJUSTMENT(Left, Right)
|
|
break;
|
|
case Vertical:
|
|
case ReversedVertical:
|
|
ADJUSTMENT(Top, Bottom)
|
|
break;
|
|
}
|
|
}
|
|
|
|
InvokeOnCompositionStateChanged();
|
|
#undef ADJUSTMENT
|
|
}
|
|
|
|
void GuiStackComposition::OnBoundsChanged(GuiGraphicsComposition* sender, GuiEventArgs& arguments)
|
|
{
|
|
EnsureStackItemVisible();
|
|
}
|
|
|
|
void GuiStackComposition::OnChildInserted(GuiGraphicsComposition* child)
|
|
{
|
|
GuiBoundsComposition::OnChildInserted(child);
|
|
GuiStackItemComposition* item = dynamic_cast<GuiStackItemComposition*>(child);
|
|
if (item)
|
|
{
|
|
if (!stackItems.Contains(item))
|
|
{
|
|
stackItems.Add(item);
|
|
}
|
|
UpdateStackItemBounds();
|
|
}
|
|
}
|
|
|
|
void GuiStackComposition::OnChildRemoved(GuiGraphicsComposition* child)
|
|
{
|
|
GuiBoundsComposition::OnChildRemoved(child);
|
|
GuiStackItemComposition* item = dynamic_cast<GuiStackItemComposition*>(child);
|
|
if (item)
|
|
{
|
|
stackItems.Remove(item);
|
|
if (item == ensuringVisibleStackItem)
|
|
{
|
|
ensuringVisibleStackItem = 0;
|
|
}
|
|
UpdateStackItemBounds();
|
|
}
|
|
}
|
|
|
|
GuiStackComposition::GuiStackComposition()
|
|
{
|
|
BoundsChanged.AttachMethod(this, &GuiStackComposition::OnBoundsChanged);
|
|
}
|
|
|
|
GuiStackComposition::~GuiStackComposition()
|
|
{
|
|
}
|
|
|
|
const GuiStackComposition::ItemCompositionList& GuiStackComposition::GetStackItems()
|
|
{
|
|
return stackItems;
|
|
}
|
|
|
|
bool GuiStackComposition::InsertStackItem(vint index, GuiStackItemComposition* item)
|
|
{
|
|
index = stackItems.Insert(index, item);
|
|
if (!AddChild(item))
|
|
{
|
|
stackItems.RemoveAt(index);
|
|
return false;
|
|
}
|
|
else
|
|
{
|
|
return true;
|
|
}
|
|
}
|
|
|
|
GuiStackComposition::Direction GuiStackComposition::GetDirection()
|
|
{
|
|
return direction;
|
|
}
|
|
|
|
void GuiStackComposition::SetDirection(Direction value)
|
|
{
|
|
direction = value;
|
|
EnsureStackItemVisible();
|
|
}
|
|
|
|
vint GuiStackComposition::GetPadding()
|
|
{
|
|
return padding;
|
|
}
|
|
|
|
void GuiStackComposition::SetPadding(vint value)
|
|
{
|
|
padding = value;
|
|
EnsureStackItemVisible();
|
|
}
|
|
|
|
void GuiStackComposition::ForceCalculateSizeImmediately()
|
|
{
|
|
GuiBoundsComposition::ForceCalculateSizeImmediately();
|
|
UpdateStackItemBounds();
|
|
}
|
|
|
|
Size GuiStackComposition::GetMinPreferredClientSize()
|
|
{
|
|
Size minSize = GuiBoundsComposition::GetMinPreferredClientSize();
|
|
if (GetMinSizeLimitation() == GuiGraphicsComposition::LimitToElementAndChildren)
|
|
{
|
|
if (!ensuringVisibleStackItem || direction == Vertical || direction == ReversedVertical)
|
|
{
|
|
if (minSize.x < stackItemTotalSize.x)
|
|
{
|
|
minSize.x = stackItemTotalSize.x;
|
|
}
|
|
}
|
|
if (!ensuringVisibleStackItem || direction == Horizontal || direction == ReversedHorizontal)
|
|
{
|
|
if (minSize.y < stackItemTotalSize.y)
|
|
{
|
|
minSize.y = stackItemTotalSize.y;
|
|
}
|
|
}
|
|
}
|
|
|
|
vint x = 0;
|
|
vint y = 0;
|
|
if (extraMargin.left > 0) x += extraMargin.left;
|
|
if (extraMargin.right > 0) x += extraMargin.right;
|
|
if (extraMargin.top > 0) y += extraMargin.top;
|
|
if (extraMargin.bottom > 0) y += extraMargin.bottom;
|
|
return minSize + Size(x, y);
|
|
}
|
|
|
|
Rect GuiStackComposition::GetBounds()
|
|
{
|
|
for (vint i = 0; i < stackItems.Count(); i++)
|
|
{
|
|
if (stackItemBounds[i].GetSize() != stackItems[i]->GetMinSize())
|
|
{
|
|
UpdateStackItemBounds();
|
|
break;
|
|
}
|
|
}
|
|
|
|
Rect bounds = GuiBoundsComposition::GetBounds();
|
|
previousBounds = bounds;
|
|
UpdatePreviousBounds(previousBounds);
|
|
return bounds;
|
|
}
|
|
|
|
Margin GuiStackComposition::GetExtraMargin()
|
|
{
|
|
return extraMargin;
|
|
}
|
|
|
|
void GuiStackComposition::SetExtraMargin(Margin value)
|
|
{
|
|
extraMargin=value;
|
|
EnsureStackItemVisible();
|
|
}
|
|
|
|
bool GuiStackComposition::IsStackItemClipped()
|
|
{
|
|
Rect clientArea = GetClientArea();
|
|
switch(direction)
|
|
{
|
|
case Horizontal:
|
|
case ReversedHorizontal:
|
|
{
|
|
vint width = stackItemTotalSize.x
|
|
+ (extraMargin.left > 0 ? extraMargin.left : 0)
|
|
+ (extraMargin.right > 0 ? extraMargin.right : 0)
|
|
;
|
|
return width > clientArea.Width();
|
|
}
|
|
break;
|
|
case Vertical:
|
|
case ReversedVertical:
|
|
{
|
|
vint height = stackItemTotalSize.y
|
|
+ (extraMargin.top > 0 ? extraMargin.top : 0)
|
|
+ (extraMargin.bottom > 0 ? extraMargin.bottom : 0)
|
|
;
|
|
return height > clientArea.Height();
|
|
}
|
|
break;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool GuiStackComposition::EnsureVisible(vint index)
|
|
{
|
|
if (0 <= index && index < stackItems.Count())
|
|
{
|
|
ensuringVisibleStackItem = stackItems[index];
|
|
}
|
|
else
|
|
{
|
|
ensuringVisibleStackItem = 0;
|
|
}
|
|
EnsureStackItemVisible();
|
|
return ensuringVisibleStackItem != 0;
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiStackItemComposition
|
|
***********************************************************************/
|
|
|
|
void GuiStackItemComposition::OnParentChanged(GuiGraphicsComposition* oldParent, GuiGraphicsComposition* newParent)
|
|
{
|
|
GuiGraphicsSite::OnParentChanged(oldParent, newParent);
|
|
stackParent = newParent == 0 ? 0 : dynamic_cast<GuiStackComposition*>(newParent);
|
|
}
|
|
|
|
Size GuiStackItemComposition::GetMinSize()
|
|
{
|
|
return GetBoundsInternal(bounds).GetSize();
|
|
}
|
|
|
|
GuiStackItemComposition::GuiStackItemComposition()
|
|
:stackParent(0)
|
|
{
|
|
SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
}
|
|
|
|
GuiStackItemComposition::~GuiStackItemComposition()
|
|
{
|
|
}
|
|
|
|
bool GuiStackItemComposition::IsSizeAffectParent()
|
|
{
|
|
return false;
|
|
}
|
|
|
|
Rect GuiStackItemComposition::GetBounds()
|
|
{
|
|
Rect result = bounds;
|
|
if(stackParent)
|
|
{
|
|
vint index = stackParent->stackItems.IndexOf(this);
|
|
if (index != -1)
|
|
{
|
|
result = stackParent->stackItemBounds[index];
|
|
}
|
|
|
|
Rect parentBounds = stackParent->previousBounds;
|
|
Margin margin = stackParent->extraMargin;
|
|
if (margin.left <= 0) margin.left = 0;
|
|
if (margin.top <= 0) margin.top = 0;
|
|
if (margin.right <= 0) margin.right = 0;
|
|
if (margin.bottom <= 0) margin.bottom = 0;
|
|
|
|
auto x = result.Left();
|
|
auto y = result.Top();
|
|
auto w = result.Width();
|
|
auto h = result.Height();
|
|
|
|
switch (stackParent->direction)
|
|
{
|
|
case GuiStackComposition::Horizontal:
|
|
x += margin.left + stackParent->adjustment;
|
|
y = margin.top;
|
|
h = parentBounds.Height() - margin.top - margin.bottom;
|
|
break;
|
|
case GuiStackComposition::ReversedHorizontal:
|
|
x = parentBounds.Width() - margin.right - x - w + stackParent->adjustment;
|
|
y = margin.top;
|
|
h = parentBounds.Height() - margin.top - margin.bottom;
|
|
break;
|
|
case GuiStackComposition::Vertical:
|
|
x = margin.left;
|
|
y += margin.top + stackParent->adjustment;
|
|
w = parentBounds.Width() - margin.left - margin.right;
|
|
break;
|
|
case GuiStackComposition::ReversedVertical:
|
|
x = margin.left;
|
|
y = parentBounds.Height() - margin.bottom - y - h + stackParent->adjustment;
|
|
w = parentBounds.Width() - margin.left - margin.right;
|
|
break;
|
|
}
|
|
|
|
result = Rect(
|
|
x - extraMargin.left,
|
|
y - extraMargin.top,
|
|
x + w + extraMargin.right,
|
|
y + h + extraMargin.bottom
|
|
);
|
|
}
|
|
UpdatePreviousBounds(result);
|
|
return result;
|
|
}
|
|
|
|
void GuiStackItemComposition::SetBounds(Rect value)
|
|
{
|
|
bounds = value;
|
|
InvokeOnCompositionStateChanged();
|
|
}
|
|
|
|
Margin GuiStackItemComposition::GetExtraMargin()
|
|
{
|
|
return extraMargin;
|
|
}
|
|
|
|
void GuiStackItemComposition::SetExtraMargin(Margin value)
|
|
{
|
|
extraMargin = value;
|
|
InvokeOnCompositionStateChanged();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
.\GRAPHICSCOMPOSITION\GUIGRAPHICSTABLECOMPOSITION.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace compositions
|
|
{
|
|
using namespace collections;
|
|
using namespace controls;
|
|
using namespace elements;
|
|
|
|
/***********************************************************************
|
|
GuiTableComposition
|
|
***********************************************************************/
|
|
|
|
namespace update_cell_bounds_helpers
|
|
{
|
|
vint First(vint a, vint b)
|
|
{
|
|
return a;
|
|
}
|
|
|
|
vint Second(vint a, vint b)
|
|
{
|
|
return b;
|
|
}
|
|
|
|
vint X(Size s)
|
|
{
|
|
return s.x;
|
|
}
|
|
|
|
vint Y(Size s)
|
|
{
|
|
return s.y;
|
|
}
|
|
|
|
vint RL(GuiCellComposition* cell)
|
|
{
|
|
return cell->GetRow();
|
|
}
|
|
|
|
vint CL(GuiCellComposition* cell)
|
|
{
|
|
return cell->GetColumn();
|
|
}
|
|
|
|
vint RS(GuiCellComposition* cell)
|
|
{
|
|
return cell->GetRowSpan();
|
|
}
|
|
|
|
vint CS(GuiCellComposition* cell)
|
|
{
|
|
return cell->GetColumnSpan();
|
|
}
|
|
}
|
|
using namespace update_cell_bounds_helpers;
|
|
|
|
vint GuiTableComposition::GetSiteIndex(vint _rows, vint _columns, vint _row, vint _column)
|
|
{
|
|
return _row*_columns + _column;
|
|
}
|
|
|
|
void GuiTableComposition::SetSitedCell(vint _row, vint _column, GuiCellComposition* cell)
|
|
{
|
|
cellCompositions[GetSiteIndex(rows, columns, _row, _column)] = cell;
|
|
}
|
|
|
|
void GuiTableComposition::UpdateCellBoundsInternal(
|
|
collections::Array<vint>& dimSizes,
|
|
vint& dimSize,
|
|
vint& dimSizeWithPercentage,
|
|
collections::Array<GuiCellOption>& dimOptions,
|
|
vint GuiTableComposition::* dim1,
|
|
vint GuiTableComposition::* dim2,
|
|
vint(*getSize)(Size),
|
|
vint(*getLocation)(GuiCellComposition*),
|
|
vint(*getSpan)(GuiCellComposition*),
|
|
vint(*getRow)(vint, vint),
|
|
vint(*getCol)(vint, vint),
|
|
vint maxPass
|
|
)
|
|
{
|
|
for (vint pass = 0; pass < maxPass; pass++)
|
|
{
|
|
for (vint i = 0; i < this->*dim1; i++)
|
|
{
|
|
GuiCellOption option = dimOptions[i];
|
|
if (pass == 0)
|
|
{
|
|
dimSizes[i] = 0;
|
|
}
|
|
switch (option.composeType)
|
|
{
|
|
case GuiCellOption::Absolute:
|
|
{
|
|
dimSizes[i] = option.absolute;
|
|
}
|
|
break;
|
|
case GuiCellOption::MinSize:
|
|
{
|
|
for (vint j = 0; j < this->*dim2; j++)
|
|
{
|
|
GuiCellComposition* cell = GetSitedCell(getRow(i, j), getCol(i, j));
|
|
if (cell)
|
|
{
|
|
bool accept = false;
|
|
if (pass == 0)
|
|
{
|
|
accept = getSpan(cell) == 1;
|
|
}
|
|
else
|
|
{
|
|
accept = getLocation(cell) + getSpan(cell) == i + 1;
|
|
}
|
|
if (accept)
|
|
{
|
|
vint size = getSize(cell->GetPreferredBounds().GetSize());
|
|
vint span = getSpan(cell);
|
|
for (vint k = 1; k < span; k++)
|
|
{
|
|
size -= dimSizes[i - k] + cellPadding;
|
|
}
|
|
if (dimSizes[i] < size)
|
|
{
|
|
dimSizes[i] = size;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
default:;
|
|
}
|
|
}
|
|
}
|
|
|
|
bool percentageExists = false;
|
|
for (vint i = 0; i < this->*dim1; i++)
|
|
{
|
|
GuiCellOption option = dimOptions[i];
|
|
if (option.composeType == GuiCellOption::Percentage)
|
|
{
|
|
if (0.001 < option.percentage)
|
|
{
|
|
percentageExists = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (percentageExists)
|
|
{
|
|
for (vint i = 0; i < this->*dim1; i++)
|
|
{
|
|
GuiCellOption option = dimOptions[i];
|
|
if (option.composeType == GuiCellOption::Percentage)
|
|
{
|
|
if (0.001 < option.percentage)
|
|
{
|
|
for (vint j = 0; j < this->*dim2; j++)
|
|
{
|
|
GuiCellComposition* cell = GetSitedCell(getRow(i, j), getCol(i, j));
|
|
if (cell)
|
|
{
|
|
vint size = getSize(cell->GetPreferredBounds().GetSize());
|
|
vint start = getLocation(cell);
|
|
vint span = getSpan(cell);
|
|
size -= (span - 1)*cellPadding;
|
|
double totalPercentage = 0;
|
|
|
|
for (vint k = start; k < start + span; k++)
|
|
{
|
|
if (dimOptions[k].composeType == GuiCellOption::Percentage)
|
|
{
|
|
if (0.001 < dimOptions[k].percentage)
|
|
{
|
|
totalPercentage += dimOptions[k].percentage;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
size -= dimSizes[k];
|
|
}
|
|
}
|
|
|
|
size = (vint)ceil(size*option.percentage / totalPercentage);
|
|
if (dimSizes[i] < size)
|
|
{
|
|
dimSizes[i] = size;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
vint percentageTotalSize = 0;
|
|
for (vint i = 0; i < this->*dim1; i++)
|
|
{
|
|
GuiCellOption option = dimOptions[i];
|
|
if (option.composeType == GuiCellOption::Percentage)
|
|
{
|
|
if (0.001 < option.percentage)
|
|
{
|
|
vint size = (vint)ceil(dimSizes[i] / option.percentage);
|
|
if (percentageTotalSize < size)
|
|
{
|
|
percentageTotalSize = size;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
double totalPercentage = 0;
|
|
for (vint i = 0; i < this->*dim1; i++)
|
|
{
|
|
GuiCellOption option = dimOptions[i];
|
|
if (option.composeType == GuiCellOption::Percentage)
|
|
{
|
|
if (0.001 < option.percentage)
|
|
{
|
|
totalPercentage += option.percentage;
|
|
}
|
|
}
|
|
}
|
|
|
|
for (vint i = 0; i < this->*dim1; i++)
|
|
{
|
|
GuiCellOption option = dimOptions[i];
|
|
if (option.composeType == GuiCellOption::Percentage)
|
|
{
|
|
if (0.001 < option.percentage)
|
|
{
|
|
vint size = (vint)ceil(percentageTotalSize*option.percentage / totalPercentage);
|
|
if (dimSizes[i] < size)
|
|
{
|
|
dimSizes[i] = size;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
for (vint i = 0; i < this->*dim1; i++)
|
|
{
|
|
if (dimOptions[i].composeType != GuiCellOption::Percentage)
|
|
{
|
|
dimSize += dimSizes[i];
|
|
}
|
|
dimSizeWithPercentage += dimSizes[i];
|
|
}
|
|
}
|
|
|
|
void GuiTableComposition::UpdateCellBoundsPercentages(
|
|
collections::Array<vint>& dimSizes,
|
|
vint dimSize,
|
|
vint maxDimSize,
|
|
collections::Array<GuiCellOption>& dimOptions
|
|
)
|
|
{
|
|
if (maxDimSize > dimSize)
|
|
{
|
|
double totalPercentage = 0;
|
|
vint percentageCount = 0;
|
|
for (vint i = 0; i < dimOptions.Count(); i++)
|
|
{
|
|
GuiCellOption option = dimOptions[i];
|
|
if (option.composeType == GuiCellOption::Percentage)
|
|
{
|
|
totalPercentage += option.percentage;
|
|
percentageCount++;
|
|
}
|
|
}
|
|
if (percentageCount > 0 && totalPercentage > 0.001)
|
|
{
|
|
for (vint i = 0; i < dimOptions.Count(); i++)
|
|
{
|
|
GuiCellOption option = dimOptions[i];
|
|
if (option.composeType == GuiCellOption::Percentage)
|
|
{
|
|
dimSizes[i] = (vint)((maxDimSize - dimSize)*option.percentage / totalPercentage);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
vint GuiTableComposition::UpdateCellBoundsOffsets(
|
|
collections::Array<vint>& offsets,
|
|
collections::Array<vint>& sizes,
|
|
vint max
|
|
)
|
|
{
|
|
offsets[0] = 0;
|
|
for (vint i = 1; i < offsets.Count(); i++)
|
|
{
|
|
offsets[i] = offsets[i - 1] + cellPadding + sizes[i - 1];
|
|
}
|
|
|
|
vint last = offsets.Count() - 1;
|
|
vint right = offsets[last] + sizes[last];
|
|
return max - right;
|
|
}
|
|
|
|
void GuiTableComposition::OnRenderContextChanged()
|
|
{
|
|
if(GetRenderTarget())
|
|
{
|
|
UpdateCellBounds();
|
|
}
|
|
}
|
|
|
|
GuiTableComposition::GuiTableComposition()
|
|
:rows(0)
|
|
, columns(0)
|
|
, cellPadding(0)
|
|
, borderVisible(true)
|
|
, rowExtending(0)
|
|
, columnExtending(0)
|
|
{
|
|
ConfigChanged.SetAssociatedComposition(this);
|
|
SetRowsAndColumns(1, 1);
|
|
}
|
|
|
|
GuiTableComposition::~GuiTableComposition()
|
|
{
|
|
}
|
|
|
|
vint GuiTableComposition::GetRows()
|
|
{
|
|
return rows;
|
|
}
|
|
|
|
vint GuiTableComposition::GetColumns()
|
|
{
|
|
return columns;
|
|
}
|
|
|
|
bool GuiTableComposition::SetRowsAndColumns(vint _rows, vint _columns)
|
|
{
|
|
if (_rows <= 0 || _columns <= 0) return false;
|
|
rowOptions.Resize(_rows);
|
|
columnOptions.Resize(_columns);
|
|
cellCompositions.Resize(_rows*_columns);
|
|
cellBounds.Resize(_rows*_columns);
|
|
for (vint i = 0; i < _rows*_columns; i++)
|
|
{
|
|
cellCompositions[i] = 0;
|
|
cellBounds[i] = Rect();
|
|
}
|
|
rows = _rows;
|
|
columns = _columns;
|
|
vint childCount = Children().Count();
|
|
for (vint i = 0; i < childCount; i++)
|
|
{
|
|
GuiCellComposition* cell = dynamic_cast<GuiCellComposition*>(Children().Get(i));
|
|
if (cell)
|
|
{
|
|
cell->OnTableRowsAndColumnsChanged();
|
|
}
|
|
}
|
|
ConfigChanged.Execute(GuiEventArgs(this));
|
|
UpdateCellBounds();
|
|
return true;
|
|
}
|
|
|
|
GuiCellComposition* GuiTableComposition::GetSitedCell(vint _row, vint _column)
|
|
{
|
|
return cellCompositions[GetSiteIndex(rows, columns, _row, _column)];
|
|
}
|
|
|
|
GuiCellOption GuiTableComposition::GetRowOption(vint _row)
|
|
{
|
|
return rowOptions[_row];
|
|
}
|
|
|
|
void GuiTableComposition::SetRowOption(vint _row, GuiCellOption option)
|
|
{
|
|
rowOptions[_row] = option;
|
|
UpdateCellBounds();
|
|
ConfigChanged.Execute(GuiEventArgs(this));
|
|
}
|
|
|
|
GuiCellOption GuiTableComposition::GetColumnOption(vint _column)
|
|
{
|
|
return columnOptions[_column];
|
|
}
|
|
|
|
void GuiTableComposition::SetColumnOption(vint _column, GuiCellOption option)
|
|
{
|
|
columnOptions[_column] = option;
|
|
UpdateCellBounds();
|
|
ConfigChanged.Execute(GuiEventArgs(this));
|
|
}
|
|
|
|
vint GuiTableComposition::GetCellPadding()
|
|
{
|
|
return cellPadding;
|
|
}
|
|
|
|
void GuiTableComposition::SetCellPadding(vint value)
|
|
{
|
|
if (value < 0) value = 0;
|
|
cellPadding = value;
|
|
UpdateCellBounds();
|
|
}
|
|
|
|
bool GuiTableComposition::GetBorderVisible()
|
|
{
|
|
return borderVisible;
|
|
}
|
|
|
|
void GuiTableComposition::SetBorderVisible(bool value)
|
|
{
|
|
if (borderVisible != value)
|
|
{
|
|
borderVisible = value;
|
|
UpdateCellBounds();
|
|
}
|
|
}
|
|
|
|
Rect GuiTableComposition::GetCellArea()
|
|
{
|
|
Rect bounds(Point(0, 0), GuiBoundsComposition::GetBounds().GetSize());
|
|
vint borderThickness = borderVisible ? cellPadding : 0;
|
|
bounds.x1 += margin.left + internalMargin.left + borderThickness;
|
|
bounds.y1 += margin.top + internalMargin.top + borderThickness;
|
|
bounds.x2 -= margin.right + internalMargin.right + borderThickness;
|
|
bounds.y2 -= margin.bottom + internalMargin.bottom + borderThickness;
|
|
if (bounds.x2 < bounds.x1) bounds.x2 = bounds.x1;
|
|
if (bounds.y2 < bounds.y1) bounds.y2 = bounds.y1;
|
|
return bounds;
|
|
}
|
|
|
|
void GuiTableComposition::UpdateCellBounds()
|
|
{
|
|
rowOffsets.Resize(rows);
|
|
rowSizes.Resize(rows);
|
|
columnOffsets.Resize(columns);
|
|
columnSizes.Resize(columns);
|
|
|
|
vint rowTotal = (rows - 1) * cellPadding;
|
|
vint columnTotal = (columns - 1) * cellPadding;
|
|
vint rowTotalWithPercentage = rowTotal;
|
|
vint columnTotalWithPercentage = columnTotal;
|
|
|
|
UpdateCellBoundsInternal(
|
|
rowSizes,
|
|
rowTotal,
|
|
rowTotalWithPercentage,
|
|
rowOptions,
|
|
&GuiTableComposition::rows,
|
|
&GuiTableComposition::columns,
|
|
&Y,
|
|
&RL,
|
|
&RS,
|
|
&First,
|
|
&Second,
|
|
1
|
|
);
|
|
UpdateCellBoundsInternal(
|
|
columnSizes,
|
|
columnTotal,
|
|
columnTotalWithPercentage,
|
|
columnOptions,
|
|
&GuiTableComposition::columns,
|
|
&GuiTableComposition::rows,
|
|
&X,
|
|
&CL,
|
|
&CS,
|
|
&Second,
|
|
&First,
|
|
1
|
|
);
|
|
|
|
Rect area = GetCellArea();
|
|
UpdateCellBoundsPercentages(rowSizes, rowTotal, area.Height(), rowOptions);
|
|
UpdateCellBoundsPercentages(columnSizes, columnTotal, area.Width(), columnOptions);
|
|
rowExtending = UpdateCellBoundsOffsets(rowOffsets, rowSizes, area.Height());
|
|
columnExtending = UpdateCellBoundsOffsets(columnOffsets, columnSizes, area.Width());
|
|
|
|
for (vint i = 0; i < rows; i++)
|
|
{
|
|
for (vint j = 0; j < columns; j++)
|
|
{
|
|
vint index = GetSiteIndex(rows, columns, i, j);
|
|
cellBounds[index] = Rect(Point(columnOffsets[j], rowOffsets[i]), Size(columnSizes[j], rowSizes[i]));
|
|
}
|
|
}
|
|
|
|
tableContentMinSize = Size(columnTotalWithPercentage, rowTotalWithPercentage);
|
|
InvokeOnCompositionStateChanged();
|
|
}
|
|
|
|
void GuiTableComposition::ForceCalculateSizeImmediately()
|
|
{
|
|
GuiBoundsComposition::ForceCalculateSizeImmediately();
|
|
UpdateCellBounds();
|
|
UpdateCellBounds();
|
|
}
|
|
|
|
Size GuiTableComposition::GetMinPreferredClientSize()
|
|
{
|
|
vint offset = (borderVisible ? 2 * cellPadding : 0);
|
|
return Size(tableContentMinSize.x + offset, tableContentMinSize.y + offset);
|
|
}
|
|
|
|
Rect GuiTableComposition::GetBounds()
|
|
{
|
|
Rect cached = previousBounds;
|
|
Rect result = GuiBoundsComposition::GetBounds();
|
|
|
|
bool cellMinSizeModified = false;
|
|
SortedList<GuiCellComposition*> cells;
|
|
FOREACH(GuiCellComposition*, cell, cellCompositions)
|
|
{
|
|
if (cell && !cells.Contains(cell))
|
|
{
|
|
cells.Add(cell);
|
|
Size newSize = cell->GetPreferredBounds().GetSize();
|
|
if (cell->lastPreferredSize != newSize)
|
|
{
|
|
cell->lastPreferredSize = newSize;
|
|
cellMinSizeModified = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (cached != result || cellMinSizeModified)
|
|
{
|
|
UpdateCellBounds();
|
|
}
|
|
return result;
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiCellComposition
|
|
***********************************************************************/
|
|
|
|
void GuiCellComposition::ClearSitedCells(GuiTableComposition* table)
|
|
{
|
|
if (row != -1 && column != -1)
|
|
{
|
|
for (vint r = 0; r < rowSpan; r++)
|
|
{
|
|
for (vint c = 0; c < columnSpan; c++)
|
|
{
|
|
table->SetSitedCell(row + r, column + c, 0);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiCellComposition::SetSitedCells(GuiTableComposition* table)
|
|
{
|
|
for (vint r = 0; r < rowSpan; r++)
|
|
{
|
|
for (vint c = 0; c < columnSpan; c++)
|
|
{
|
|
table->SetSitedCell(row + r, column + c, this);
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiCellComposition::ResetSiteInternal()
|
|
{
|
|
row = -1;
|
|
column = -1;
|
|
rowSpan = 1;
|
|
columnSpan = 1;
|
|
}
|
|
|
|
bool GuiCellComposition::SetSiteInternal(vint _row, vint _column, vint _rowSpan, vint _columnSpan)
|
|
{
|
|
if (tableParent)
|
|
{
|
|
if (_row < 0 || _row >= tableParent->rows || _column < 0 || _column >= tableParent->columns) return false;
|
|
if (_rowSpan<1 || _row + _rowSpan>tableParent->rows || _columnSpan<1 || _column + _columnSpan>tableParent->columns) return false;
|
|
|
|
for (vint r = 0; r < _rowSpan; r++)
|
|
{
|
|
for (vint c = 0; c < _columnSpan; c++)
|
|
{
|
|
GuiCellComposition* cell = tableParent->GetSitedCell(_row + r, _column + c);
|
|
if (cell && cell != this)
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
ClearSitedCells(tableParent);
|
|
}
|
|
|
|
row = _row;
|
|
column = _column;
|
|
rowSpan = _rowSpan;
|
|
columnSpan = _columnSpan;
|
|
|
|
if (tableParent)
|
|
{
|
|
SetSitedCells(tableParent);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
void GuiCellComposition::OnParentChanged(GuiGraphicsComposition* oldParent, GuiGraphicsComposition* newParent)
|
|
{
|
|
GuiGraphicsSite::OnParentChanged(oldParent, newParent);
|
|
if (tableParent)
|
|
{
|
|
ClearSitedCells(tableParent);
|
|
}
|
|
tableParent = dynamic_cast<GuiTableComposition*>(newParent);
|
|
if (!tableParent || !SetSiteInternal(row, column, rowSpan, columnSpan))
|
|
{
|
|
ResetSiteInternal();
|
|
}
|
|
if (tableParent)
|
|
{
|
|
if (row != -1 && column != -1)
|
|
{
|
|
SetSiteInternal(row, column, rowSpan, columnSpan);
|
|
}
|
|
tableParent->UpdateCellBounds();
|
|
}
|
|
}
|
|
|
|
void GuiCellComposition::OnTableRowsAndColumnsChanged()
|
|
{
|
|
if(!SetSiteInternal(row, column, rowSpan, columnSpan))
|
|
{
|
|
ResetSiteInternal();
|
|
}
|
|
}
|
|
|
|
GuiCellComposition::GuiCellComposition()
|
|
:row(-1)
|
|
,column(-1)
|
|
,rowSpan(1)
|
|
,columnSpan(1)
|
|
,tableParent(0)
|
|
{
|
|
SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
}
|
|
|
|
GuiCellComposition::~GuiCellComposition()
|
|
{
|
|
}
|
|
|
|
GuiTableComposition* GuiCellComposition::GetTableParent()
|
|
{
|
|
return tableParent;
|
|
}
|
|
|
|
vint GuiCellComposition::GetRow()
|
|
{
|
|
return row;
|
|
}
|
|
|
|
vint GuiCellComposition::GetRowSpan()
|
|
{
|
|
return rowSpan;
|
|
}
|
|
|
|
vint GuiCellComposition::GetColumn()
|
|
{
|
|
return column;
|
|
}
|
|
|
|
vint GuiCellComposition::GetColumnSpan()
|
|
{
|
|
return columnSpan;
|
|
}
|
|
|
|
bool GuiCellComposition::SetSite(vint _row, vint _column, vint _rowSpan, vint _columnSpan)
|
|
{
|
|
if (!SetSiteInternal(_row, _column, _rowSpan, _columnSpan))
|
|
{
|
|
return false;
|
|
}
|
|
|
|
if (tableParent)
|
|
{
|
|
tableParent->UpdateCellBounds();
|
|
}
|
|
return true;
|
|
}
|
|
|
|
Rect GuiCellComposition::GetBounds()
|
|
{
|
|
Rect result;
|
|
if(tableParent && row!=-1 && column!=-1)
|
|
{
|
|
Rect bounds1, bounds2;
|
|
{
|
|
vint index=tableParent->GetSiteIndex(tableParent->rows, tableParent->columns, row, column);
|
|
bounds1=tableParent->cellBounds[index];
|
|
}
|
|
{
|
|
vint index=tableParent->GetSiteIndex(tableParent->rows, tableParent->columns, row+rowSpan-1, column+columnSpan-1);
|
|
bounds2=tableParent->cellBounds[index];
|
|
if(tableParent->GetMinSizeLimitation()==GuiGraphicsComposition::NoLimit)
|
|
{
|
|
if(row+rowSpan==tableParent->rows)
|
|
{
|
|
bounds2.y2+=tableParent->rowExtending;
|
|
}
|
|
if(column+columnSpan==tableParent->columns)
|
|
{
|
|
bounds2.x2+=tableParent->columnExtending;
|
|
}
|
|
}
|
|
}
|
|
vint offset = tableParent->borderVisible ? tableParent->cellPadding : 0;
|
|
result = Rect(bounds1.x1 + offset, bounds1.y1 + offset, bounds2.x2 + offset, bounds2.y2 + offset);
|
|
}
|
|
else
|
|
{
|
|
result = Rect();
|
|
}
|
|
UpdatePreviousBounds(result);
|
|
return result;
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiTableSplitterCompositionBase
|
|
***********************************************************************/
|
|
|
|
void GuiTableSplitterCompositionBase::OnParentChanged(GuiGraphicsComposition* oldParent, GuiGraphicsComposition* newParent)
|
|
{
|
|
GuiGraphicsSite::OnParentChanged(oldParent, newParent);
|
|
tableParent = dynamic_cast<GuiTableComposition*>(newParent);
|
|
}
|
|
|
|
void GuiTableSplitterCompositionBase::OnLeftButtonDown(GuiGraphicsComposition* sender, GuiMouseEventArgs& arguments)
|
|
{
|
|
dragging = true;
|
|
draggingPoint = Point(arguments.x, arguments.y);
|
|
}
|
|
|
|
void GuiTableSplitterCompositionBase::OnLeftButtonUp(GuiGraphicsComposition* sender, GuiMouseEventArgs& arguments)
|
|
{
|
|
dragging = false;
|
|
}
|
|
|
|
void GuiTableSplitterCompositionBase::OnMouseMoveHelper(
|
|
vint cellsBefore,
|
|
vint GuiTableComposition::* cells,
|
|
collections::Array<vint>& cellSizes,
|
|
vint offset,
|
|
GuiCellOption(GuiTableComposition::*getOption)(vint),
|
|
void(GuiTableComposition::*setOption)(vint, GuiCellOption)
|
|
)
|
|
{
|
|
if (dragging)
|
|
{
|
|
if (tableParent)
|
|
{
|
|
if (0 < cellsBefore && cellsBefore < tableParent->*cells)
|
|
{
|
|
auto o1 = (tableParent->*getOption)(cellsBefore - 1);
|
|
auto o2 = (tableParent->*getOption)(cellsBefore);
|
|
|
|
vint indexStart = -1;
|
|
vint indexEnd = -1;
|
|
vint indexStep = -1;
|
|
vint max = 0;
|
|
|
|
if (offset < 0)
|
|
{
|
|
indexStart = cellsBefore - 1;
|
|
indexEnd = -1;
|
|
indexStep = -1;
|
|
}
|
|
else if (offset > 0)
|
|
{
|
|
indexStart = cellsBefore;
|
|
indexEnd = tableParent->*cells;
|
|
indexStep = 1;
|
|
}
|
|
else
|
|
{
|
|
return;
|
|
}
|
|
|
|
{
|
|
auto o = (tableParent->*getOption)(indexStart);
|
|
if (o.composeType == GuiCellOption::Absolute)
|
|
{
|
|
max = o.absolute - 1;
|
|
}
|
|
else
|
|
{
|
|
for (vint i = indexStart; i != indexEnd; i += indexStep)
|
|
{
|
|
o = (tableParent->*getOption)(i);
|
|
if (o.composeType == GuiCellOption::Absolute)
|
|
{
|
|
break;
|
|
}
|
|
else if (o.composeType == GuiCellOption::Percentage)
|
|
{
|
|
max += cellSizes[i] - 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (max <= 0)
|
|
{
|
|
return;
|
|
}
|
|
}
|
|
|
|
if (offset < 0)
|
|
{
|
|
if (max < -offset)
|
|
{
|
|
offset = -max;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (max < offset)
|
|
{
|
|
offset = max;
|
|
}
|
|
}
|
|
|
|
if (o1.composeType == GuiCellOption::Absolute)
|
|
{
|
|
o1.absolute += offset;
|
|
(tableParent->*setOption)(cellsBefore - 1, o1);
|
|
}
|
|
if (o2.composeType == GuiCellOption::Absolute)
|
|
{
|
|
o2.absolute -= offset;
|
|
(tableParent->*setOption)(cellsBefore, o2);
|
|
}
|
|
tableParent->ForceCalculateSizeImmediately();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
Rect GuiTableSplitterCompositionBase::GetBoundsHelper(
|
|
vint cellsBefore,
|
|
vint GuiTableComposition::* cells,
|
|
vint(Rect::* dimSize)()const,
|
|
collections::Array<vint>& cellOffsets,
|
|
vint Rect::* dimU1,
|
|
vint Rect::* dimU2,
|
|
vint Rect::* dimV1,
|
|
vint Rect::* dimV2
|
|
)
|
|
{
|
|
Rect result(0, 0, 0, 0);
|
|
if (tableParent)
|
|
{
|
|
if (0 < cellsBefore && cellsBefore < tableParent->*cells)
|
|
{
|
|
vint offset = tableParent->borderVisible ? tableParent->cellPadding : 0;
|
|
result.*dimU1 = offset;
|
|
result.*dimU2 = offset + (tableParent->GetCellArea().*dimSize)();
|
|
result.*dimV1 = offset + cellOffsets[cellsBefore] - tableParent->cellPadding;
|
|
result.*dimV2 = (result.*dimV1) + tableParent->cellPadding;
|
|
}
|
|
}
|
|
UpdatePreviousBounds(result);
|
|
return result;
|
|
}
|
|
|
|
GuiTableSplitterCompositionBase::GuiTableSplitterCompositionBase()
|
|
:tableParent(0)
|
|
, dragging(false)
|
|
{
|
|
SetAssociatedCursor(GetCurrentController()->ResourceService()->GetSystemCursor(INativeCursor::SizeNS));
|
|
GetEventReceiver()->leftButtonDown.AttachMethod(this, &GuiTableSplitterCompositionBase::OnLeftButtonDown);
|
|
GetEventReceiver()->leftButtonUp.AttachMethod(this, &GuiTableSplitterCompositionBase::OnLeftButtonUp);
|
|
}
|
|
|
|
GuiTableSplitterCompositionBase::~GuiTableSplitterCompositionBase()
|
|
{
|
|
}
|
|
|
|
GuiTableComposition* GuiTableSplitterCompositionBase::GetTableParent()
|
|
{
|
|
return tableParent;
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiRowSplitterComposition
|
|
***********************************************************************/
|
|
|
|
void GuiRowSplitterComposition::OnMouseMove(GuiGraphicsComposition* sender, GuiMouseEventArgs& arguments)
|
|
{
|
|
OnMouseMoveHelper(
|
|
rowsToTheTop,
|
|
&GuiTableComposition::rows,
|
|
tableParent->rowSizes,
|
|
arguments.y - draggingPoint.y,
|
|
&GuiTableComposition::GetRowOption,
|
|
&GuiTableComposition::SetRowOption
|
|
);
|
|
}
|
|
|
|
GuiRowSplitterComposition::GuiRowSplitterComposition()
|
|
:rowsToTheTop(0)
|
|
{
|
|
SetAssociatedCursor(GetCurrentController()->ResourceService()->GetSystemCursor(INativeCursor::SizeNS));
|
|
GetEventReceiver()->mouseMove.AttachMethod(this, &GuiRowSplitterComposition::OnMouseMove);
|
|
}
|
|
|
|
GuiRowSplitterComposition::~GuiRowSplitterComposition()
|
|
{
|
|
}
|
|
|
|
vint GuiRowSplitterComposition::GetRowsToTheTop()
|
|
{
|
|
return rowsToTheTop;
|
|
}
|
|
|
|
void GuiRowSplitterComposition::SetRowsToTheTop(vint value)
|
|
{
|
|
rowsToTheTop = value;
|
|
InvokeOnCompositionStateChanged();
|
|
}
|
|
|
|
Rect GuiRowSplitterComposition::GetBounds()
|
|
{
|
|
return GetBoundsHelper(
|
|
rowsToTheTop,
|
|
&GuiTableComposition::rows,
|
|
&Rect::Width,
|
|
tableParent->rowOffsets,
|
|
&Rect::x1,
|
|
&Rect::x2,
|
|
&Rect::y1,
|
|
&Rect::y2
|
|
);
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiColumnSplitterComposition
|
|
***********************************************************************/
|
|
|
|
void GuiColumnSplitterComposition::OnMouseMove(GuiGraphicsComposition* sender, GuiMouseEventArgs& arguments)
|
|
{
|
|
OnMouseMoveHelper(
|
|
columnsToTheLeft,
|
|
&GuiTableComposition::columns,
|
|
tableParent->columnSizes,
|
|
arguments.x - draggingPoint.x,
|
|
&GuiTableComposition::GetColumnOption,
|
|
&GuiTableComposition::SetColumnOption
|
|
);
|
|
}
|
|
|
|
GuiColumnSplitterComposition::GuiColumnSplitterComposition()
|
|
:columnsToTheLeft(0)
|
|
{
|
|
SetAssociatedCursor(GetCurrentController()->ResourceService()->GetSystemCursor(INativeCursor::SizeWE));
|
|
GetEventReceiver()->mouseMove.AttachMethod(this, &GuiColumnSplitterComposition::OnMouseMove);
|
|
}
|
|
|
|
GuiColumnSplitterComposition::~GuiColumnSplitterComposition()
|
|
{
|
|
}
|
|
|
|
vint GuiColumnSplitterComposition::GetColumnsToTheLeft()
|
|
{
|
|
return columnsToTheLeft;
|
|
}
|
|
|
|
void GuiColumnSplitterComposition::SetColumnsToTheLeft(vint value)
|
|
{
|
|
columnsToTheLeft = value;
|
|
InvokeOnCompositionStateChanged();
|
|
}
|
|
|
|
Rect GuiColumnSplitterComposition::GetBounds()
|
|
{
|
|
return GetBoundsHelper(
|
|
columnsToTheLeft,
|
|
&GuiTableComposition::columns,
|
|
&Rect::Height,
|
|
tableParent->columnOffsets,
|
|
&Rect::y1,
|
|
&Rect::y2,
|
|
&Rect::x1,
|
|
&Rect::x2
|
|
);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\GRAPHICSELEMENT\GUIGRAPHICSDOCUMENTELEMENT.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
using namespace collections;
|
|
|
|
namespace presentation
|
|
{
|
|
namespace elements
|
|
{
|
|
|
|
/***********************************************************************
|
|
SetPropertiesVisitor
|
|
***********************************************************************/
|
|
|
|
namespace visitors
|
|
{
|
|
class SetPropertiesVisitor : public Object, public DocumentRun::IVisitor
|
|
{
|
|
typedef GuiDocumentElement::GuiDocumentElementRenderer Renderer;
|
|
typedef DocumentModel::ResolvedStyle ResolvedStyle;
|
|
public:
|
|
vint start;
|
|
vint length;
|
|
vint selectionBegin;
|
|
vint selectionEnd;
|
|
List<ResolvedStyle> styles;
|
|
|
|
DocumentModel* model;
|
|
Renderer* renderer;
|
|
Ptr<Renderer::ParagraphCache> cache;
|
|
IGuiGraphicsParagraph* paragraph;
|
|
|
|
SetPropertiesVisitor(DocumentModel* _model, Renderer* _renderer, Ptr<Renderer::ParagraphCache> _cache, vint _selectionBegin, vint _selectionEnd)
|
|
:start(0)
|
|
,length(0)
|
|
,model(_model)
|
|
,renderer(_renderer)
|
|
,cache(_cache)
|
|
,paragraph(_cache->graphicsParagraph.Obj())
|
|
,selectionBegin(_selectionBegin)
|
|
,selectionEnd(_selectionEnd)
|
|
{
|
|
ResolvedStyle style;
|
|
style=model->GetStyle(DocumentModel::DefaultStyleName, style);
|
|
styles.Add(style);
|
|
}
|
|
|
|
void VisitContainer(DocumentContainerRun* run)
|
|
{
|
|
FOREACH(Ptr<DocumentRun>, subRun, run->runs)
|
|
{
|
|
subRun->Accept(this);
|
|
}
|
|
}
|
|
|
|
void ApplyStyle(vint start, vint length, const ResolvedStyle& style)
|
|
{
|
|
paragraph->SetFont(start, length, style.style.fontFamily);
|
|
paragraph->SetSize(start, length, style.style.size);
|
|
paragraph->SetStyle(start, length,
|
|
(IGuiGraphicsParagraph::TextStyle)
|
|
( (style.style.bold?IGuiGraphicsParagraph::Bold:0)
|
|
| (style.style.italic?IGuiGraphicsParagraph::Italic:0)
|
|
| (style.style.underline?IGuiGraphicsParagraph::Underline:0)
|
|
| (style.style.strikeline?IGuiGraphicsParagraph::Strikeline:0)
|
|
));
|
|
}
|
|
|
|
void ApplyColor(vint start, vint length, const ResolvedStyle& style)
|
|
{
|
|
paragraph->SetColor(start, length, style.color);
|
|
paragraph->SetBackgroundColor(start, length, style.backgroundColor);
|
|
}
|
|
|
|
void Visit(DocumentTextRun* run)override
|
|
{
|
|
length=run->GetRepresentationText().Length();
|
|
if(length>0)
|
|
{
|
|
ResolvedStyle style=styles[styles.Count()-1];
|
|
ApplyStyle(start, length, style);
|
|
ApplyColor(start, length, style);
|
|
|
|
vint styleStart=start;
|
|
vint styleEnd=styleStart+length;
|
|
if(styleStart<selectionEnd && selectionBegin<styleEnd)
|
|
{
|
|
vint s2=styleStart>selectionBegin?styleStart:selectionBegin;
|
|
vint s3=selectionEnd<styleEnd?selectionEnd:styleEnd;
|
|
|
|
if(s2<s3)
|
|
{
|
|
ResolvedStyle selectionStyle=model->GetStyle(DocumentModel::SelectionStyleName, style);
|
|
ApplyColor(s2, s3-s2, selectionStyle);
|
|
}
|
|
}
|
|
}
|
|
start+=length;
|
|
}
|
|
|
|
void Visit(DocumentStylePropertiesRun* run)override
|
|
{
|
|
ResolvedStyle style=styles[styles.Count()-1];
|
|
style=model->GetStyle(run->style, style);
|
|
styles.Add(style);
|
|
VisitContainer(run);
|
|
styles.RemoveAt(styles.Count()-1);
|
|
}
|
|
|
|
void Visit(DocumentStyleApplicationRun* run)override
|
|
{
|
|
ResolvedStyle style=styles[styles.Count()-1];
|
|
style=model->GetStyle(run->styleName, style);
|
|
styles.Add(style);
|
|
VisitContainer(run);
|
|
styles.RemoveAt(styles.Count()-1);
|
|
}
|
|
|
|
void Visit(DocumentHyperlinkRun* run)override
|
|
{
|
|
ResolvedStyle style=styles[styles.Count()-1];
|
|
style=model->GetStyle(run->styleName, style);
|
|
styles.Add(style);
|
|
VisitContainer(run);
|
|
styles.RemoveAt(styles.Count()-1);
|
|
}
|
|
|
|
void Visit(DocumentImageRun* run)override
|
|
{
|
|
length=run->GetRepresentationText().Length();
|
|
|
|
Ptr<GuiImageFrameElement> element=GuiImageFrameElement::Create();
|
|
element->SetImage(run->image, run->frameIndex);
|
|
element->SetStretch(true);
|
|
|
|
IGuiGraphicsParagraph::InlineObjectProperties properties;
|
|
properties.size=run->size;
|
|
properties.baseline=run->baseline;
|
|
properties.breakCondition=IGuiGraphicsParagraph::Alone;
|
|
properties.backgroundImage = element;
|
|
|
|
paragraph->SetInlineObject(start, length, properties);
|
|
|
|
if(start<selectionEnd && selectionBegin<start+length)
|
|
{
|
|
ResolvedStyle style=styles[styles.Count()-1];
|
|
ResolvedStyle selectionStyle=model->GetStyle(DocumentModel::SelectionStyleName, style);
|
|
ApplyColor(start, length, selectionStyle);
|
|
}
|
|
start+=length;
|
|
}
|
|
|
|
void Visit(DocumentEmbeddedObjectRun* run)override
|
|
{
|
|
length=run->GetRepresentationText().Length();
|
|
|
|
IGuiGraphicsParagraph::InlineObjectProperties properties;
|
|
properties.breakCondition=IGuiGraphicsParagraph::Alone;
|
|
|
|
if (run->name != L"")
|
|
{
|
|
vint index = renderer->nameCallbackIdMap.Keys().IndexOf(run->name);
|
|
if (index != -1)
|
|
{
|
|
auto id = renderer->nameCallbackIdMap.Values()[index];
|
|
index = cache->embeddedObjects.Keys().IndexOf(id);
|
|
if (index != -1)
|
|
{
|
|
auto eo = cache->embeddedObjects.Values()[index];
|
|
if (eo->start == start)
|
|
{
|
|
properties.size = eo->size;
|
|
properties.callbackId = id;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
auto eo = MakePtr<Renderer::EmbeddedObject>();
|
|
eo->name = run->name;
|
|
eo->size = Size(0, 0);
|
|
eo->start = start;
|
|
|
|
vint id = -1;
|
|
vint count = renderer->freeCallbackIds.Count();
|
|
if (count > 0)
|
|
{
|
|
id = renderer->freeCallbackIds[count - 1];
|
|
renderer->freeCallbackIds.RemoveAt(count - 1);
|
|
}
|
|
else
|
|
{
|
|
id = renderer->usedCallbackIds++;
|
|
}
|
|
|
|
renderer->nameCallbackIdMap.Add(eo->name, id);
|
|
cache->embeddedObjects.Add(id, eo);
|
|
properties.callbackId = id;
|
|
}
|
|
}
|
|
|
|
paragraph->SetInlineObject(start, length, properties);
|
|
|
|
if(start<selectionEnd && selectionBegin<start+length)
|
|
{
|
|
ResolvedStyle style=styles[styles.Count()-1];
|
|
ResolvedStyle selectionStyle=model->GetStyle(DocumentModel::SelectionStyleName, style);
|
|
ApplyColor(start, length, selectionStyle);
|
|
}
|
|
start+=length;
|
|
}
|
|
|
|
void Visit(DocumentParagraphRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
|
|
static vint SetProperty(DocumentModel* model, Renderer* renderer, Ptr<Renderer::ParagraphCache> cache, Ptr<DocumentParagraphRun> run, vint selectionBegin, vint selectionEnd)
|
|
{
|
|
SetPropertiesVisitor visitor(model, renderer, cache, selectionBegin, selectionEnd);
|
|
run->Accept(&visitor);
|
|
return visitor.length;
|
|
}
|
|
};
|
|
}
|
|
using namespace visitors;
|
|
|
|
/***********************************************************************
|
|
GuiDocumentElement::GuiDocumentElementRenderer
|
|
***********************************************************************/
|
|
|
|
Size GuiDocumentElement::GuiDocumentElementRenderer::OnRenderInlineObject(vint callbackId, Rect location)
|
|
{
|
|
if (element->callback)
|
|
{
|
|
auto cache = paragraphCaches[renderingParagraph];
|
|
auto relativeLocation = Rect(Point(location.x1 + renderingParagraphOffset.x, location.y1 + renderingParagraphOffset.y), location.GetSize());
|
|
auto eo = cache->embeddedObjects[callbackId];
|
|
auto size = element->callback->OnRenderEmbeddedObject(eo->name, relativeLocation);
|
|
eo->resized = eo->size != size;
|
|
eo->size = size;
|
|
return eo->size;
|
|
}
|
|
else
|
|
{
|
|
return Size();
|
|
}
|
|
}
|
|
|
|
void GuiDocumentElement::GuiDocumentElementRenderer::InitializeInternal()
|
|
{
|
|
}
|
|
|
|
void GuiDocumentElement::GuiDocumentElementRenderer::FinalizeInternal()
|
|
{
|
|
}
|
|
|
|
void GuiDocumentElement::GuiDocumentElementRenderer::RenderTargetChangedInternal(IGuiGraphicsRenderTarget* oldRenderTarget, IGuiGraphicsRenderTarget* newRenderTarget)
|
|
{
|
|
for(vint i=0;i<paragraphCaches.Count();i++)
|
|
{
|
|
ParagraphCache* cache=paragraphCaches[i].Obj();
|
|
if(cache)
|
|
{
|
|
cache->graphicsParagraph=0;
|
|
}
|
|
}
|
|
}
|
|
|
|
Ptr<GuiDocumentElement::GuiDocumentElementRenderer::ParagraphCache> GuiDocumentElement::GuiDocumentElementRenderer::EnsureAndGetCache(vint paragraphIndex, bool createParagraph)
|
|
{
|
|
if(paragraphIndex<0 || paragraphIndex>=paragraphCaches.Count()) return 0;
|
|
Ptr<DocumentParagraphRun> paragraph=element->document->paragraphs[paragraphIndex];
|
|
Ptr<ParagraphCache> cache=paragraphCaches[paragraphIndex];
|
|
if(!cache)
|
|
{
|
|
cache=new ParagraphCache;
|
|
cache->fullText=paragraph->GetText(false);
|
|
paragraphCaches[paragraphIndex]=cache;
|
|
}
|
|
|
|
if(createParagraph)
|
|
{
|
|
if(!cache->graphicsParagraph)
|
|
{
|
|
cache->graphicsParagraph=layoutProvider->CreateParagraph(cache->fullText, renderTarget, this);
|
|
cache->graphicsParagraph->SetParagraphAlignment(paragraph->alignment ? paragraph->alignment.Value() : Alignment::Left);
|
|
SetPropertiesVisitor::SetProperty(element->document.Obj(), this, cache, paragraph, cache->selectionBegin, cache->selectionEnd);
|
|
}
|
|
if(cache->graphicsParagraph->GetMaxWidth()!=lastMaxWidth)
|
|
{
|
|
cache->graphicsParagraph->SetMaxWidth(lastMaxWidth);
|
|
}
|
|
|
|
vint paragraphHeight=paragraphHeights[paragraphIndex];
|
|
vint height=cache->graphicsParagraph->GetHeight();
|
|
if(paragraphHeight!=height)
|
|
{
|
|
cachedTotalHeight+=height-paragraphHeight;
|
|
paragraphHeight=height;
|
|
paragraphHeights[paragraphIndex]=paragraphHeight;
|
|
minSize=Size(0, cachedTotalHeight);
|
|
}
|
|
}
|
|
|
|
return cache;
|
|
}
|
|
|
|
bool GuiDocumentElement::GuiDocumentElementRenderer::GetParagraphIndexFromPoint(Point point, vint& top, vint& index)
|
|
{
|
|
vint y=0;
|
|
for(vint i=0;i<paragraphHeights.Count();i++)
|
|
{
|
|
vint paragraphHeight=paragraphHeights[i];
|
|
vint nextY=y+paragraphHeight+paragraphDistance;
|
|
top=y;
|
|
index=i;
|
|
|
|
if(nextY<=point.y)
|
|
{
|
|
y=nextY;
|
|
continue;
|
|
}
|
|
else
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
GuiDocumentElement::GuiDocumentElementRenderer::GuiDocumentElementRenderer()
|
|
:paragraphDistance(0)
|
|
,lastMaxWidth(-1)
|
|
,cachedTotalHeight(0)
|
|
,layoutProvider(GetGuiGraphicsResourceManager()->GetLayoutProvider())
|
|
,lastCaret(-1, -1)
|
|
,lastCaretFrontSide(false)
|
|
{
|
|
}
|
|
|
|
void GuiDocumentElement::GuiDocumentElementRenderer::Render(Rect bounds)
|
|
{
|
|
if (element->callback)
|
|
{
|
|
element->callback->OnStartRender();
|
|
}
|
|
renderTarget->PushClipper(bounds);
|
|
if(!renderTarget->IsClipperCoverWholeTarget())
|
|
{
|
|
vint maxWidth=bounds.Width();
|
|
Rect clipper=renderTarget->GetClipper();
|
|
vint cx=bounds.Left();
|
|
vint cy=bounds.Top();
|
|
vint y1=clipper.Top()-bounds.Top();
|
|
vint y2=y1+clipper.Height();
|
|
vint y=0;
|
|
|
|
lastMaxWidth=maxWidth;
|
|
|
|
for(vint i=0;i<paragraphHeights.Count();i++)
|
|
{
|
|
vint paragraphHeight=paragraphHeights[i];
|
|
if(y+paragraphHeight<=y1)
|
|
{
|
|
y+=paragraphHeight+paragraphDistance;
|
|
continue;
|
|
}
|
|
else if(y>=y2)
|
|
{
|
|
break;
|
|
}
|
|
else
|
|
{
|
|
Ptr<DocumentParagraphRun> paragraph=element->document->paragraphs[i];
|
|
Ptr<ParagraphCache> cache=paragraphCaches[i];
|
|
bool created=cache && cache->graphicsParagraph;
|
|
cache=EnsureAndGetCache(i, true);
|
|
if(!created && i==lastCaret.row && element->caretVisible)
|
|
{
|
|
cache->graphicsParagraph->OpenCaret(lastCaret.column, lastCaretColor, lastCaretFrontSide);
|
|
}
|
|
|
|
paragraphHeight=cache->graphicsParagraph->GetHeight();
|
|
|
|
renderingParagraph = i;
|
|
renderingParagraphOffset = Point(cx - bounds.x1, cy + y - bounds.y1);
|
|
cache->graphicsParagraph->Render(Rect(Point(cx, cy+y), Size(maxWidth, paragraphHeight)));
|
|
renderingParagraph = -1;
|
|
|
|
bool resized = false;
|
|
for (vint j = 0; j < cache->embeddedObjects.Count(); j++)
|
|
{
|
|
auto eo = cache->embeddedObjects.Values()[j];
|
|
if (eo->resized)
|
|
{
|
|
eo->resized = false;
|
|
resized = true;
|
|
}
|
|
}
|
|
|
|
if (resized)
|
|
{
|
|
cache->graphicsParagraph = 0;
|
|
}
|
|
}
|
|
|
|
y+=paragraphHeight+paragraphDistance;
|
|
}
|
|
}
|
|
renderTarget->PopClipper();
|
|
if (element->callback)
|
|
{
|
|
element->callback->OnFinishRender();
|
|
}
|
|
}
|
|
|
|
void GuiDocumentElement::GuiDocumentElementRenderer::OnElementStateChanged()
|
|
{
|
|
if (element->document && element->document->paragraphs.Count() > 0)
|
|
{
|
|
vint defaultSize = GetCurrentController()->ResourceService()->GetDefaultFont().size;
|
|
paragraphDistance = defaultSize;
|
|
vint defaultHeight = defaultSize;
|
|
|
|
paragraphCaches.Resize(element->document->paragraphs.Count());
|
|
paragraphHeights.Resize(element->document->paragraphs.Count());
|
|
|
|
for (vint i = 0; i < paragraphCaches.Count(); i++)
|
|
{
|
|
paragraphCaches[i] = 0;
|
|
}
|
|
for (vint i = 0; i < paragraphHeights.Count(); i++)
|
|
{
|
|
paragraphHeights[i] = defaultHeight;
|
|
}
|
|
|
|
cachedTotalHeight = paragraphHeights.Count() * (defaultHeight + paragraphDistance);
|
|
if (paragraphHeights.Count()>0)
|
|
{
|
|
cachedTotalHeight -= paragraphDistance;
|
|
}
|
|
minSize = Size(0, cachedTotalHeight);
|
|
}
|
|
else
|
|
{
|
|
paragraphCaches.Resize(0);
|
|
paragraphHeights.Resize(0);
|
|
cachedTotalHeight = 0;
|
|
minSize = Size(0, 0);
|
|
}
|
|
|
|
nameCallbackIdMap.Clear();
|
|
freeCallbackIds.Clear();
|
|
usedCallbackIds = 0;
|
|
}
|
|
|
|
void GuiDocumentElement::GuiDocumentElementRenderer::NotifyParagraphUpdated(vint index, vint oldCount, vint newCount, bool updatedText)
|
|
{
|
|
if (0 <= index && index < paragraphCaches.Count() && 0 <= oldCount && index + oldCount <= paragraphCaches.Count() && 0 <= newCount)
|
|
{
|
|
vint paragraphCount = element->document->paragraphs.Count();
|
|
CHECK_ERROR(updatedText || oldCount == newCount, L"GuiDocumentlement::GuiDocumentElementRenderer::NotifyParagraphUpdated(vint, vint, vint, bool)#Illegal values of oldCount and newCount.");
|
|
CHECK_ERROR(paragraphCount - paragraphCaches.Count() == newCount - oldCount, L"GuiDocumentElement::GuiDocumentElementRenderer::NotifyParagraphUpdated(vint, vint, vint, bool)#Illegal values of oldCount and newCount.");
|
|
|
|
ParagraphCacheArray oldCaches;
|
|
CopyFrom(oldCaches, paragraphCaches);
|
|
paragraphCaches.Resize(paragraphCount);
|
|
|
|
ParagraphHeightArray oldHeights;
|
|
CopyFrom(oldHeights, paragraphHeights);
|
|
paragraphHeights.Resize(paragraphCount);
|
|
|
|
vint defaultHeight = GetCurrentController()->ResourceService()->GetDefaultFont().size;
|
|
cachedTotalHeight = 0;
|
|
|
|
for (vint i = 0; i < paragraphCount; i++)
|
|
{
|
|
if (i < index)
|
|
{
|
|
paragraphCaches[i] = oldCaches[i];
|
|
paragraphHeights[i] = oldHeights[i];
|
|
}
|
|
else if (i < index + newCount)
|
|
{
|
|
paragraphCaches[i] = 0;
|
|
paragraphHeights[i] = defaultHeight;
|
|
if (!updatedText && i < index + oldCount)
|
|
{
|
|
auto cache = oldCaches[i];
|
|
if(cache)
|
|
{
|
|
cache->graphicsParagraph = 0;
|
|
}
|
|
paragraphCaches[i] = cache;
|
|
paragraphHeights[i] = oldHeights[i];
|
|
}
|
|
}
|
|
else
|
|
{
|
|
paragraphCaches[i] = oldCaches[i - (newCount - oldCount)];
|
|
paragraphHeights[i] = oldHeights[i - (newCount - oldCount)];
|
|
}
|
|
cachedTotalHeight += paragraphHeights[i] + paragraphDistance;
|
|
}
|
|
if (paragraphCount > 0)
|
|
{
|
|
cachedTotalHeight -= paragraphDistance;
|
|
}
|
|
|
|
if (updatedText)
|
|
{
|
|
vint count = oldCount < newCount ? oldCount : newCount;
|
|
for (vint i = 0; i < count; i++)
|
|
{
|
|
if (auto cache = oldCaches[index + i])
|
|
{
|
|
for (vint j = 0; j < cache->embeddedObjects.Count(); j++)
|
|
{
|
|
auto id = cache->embeddedObjects.Keys()[j];
|
|
auto name = cache->embeddedObjects.Values()[j]->name;
|
|
nameCallbackIdMap.Remove(name);
|
|
freeCallbackIds.Add(id);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
Ptr<DocumentHyperlinkRun::Package> GuiDocumentElement::GuiDocumentElementRenderer::GetHyperlinkFromPoint(Point point)
|
|
{
|
|
vint top=0;
|
|
vint index=-1;
|
|
if(GetParagraphIndexFromPoint(point, top, index))
|
|
{
|
|
Ptr<ParagraphCache> cache=EnsureAndGetCache(index, true);
|
|
Point paragraphPoint(point.x, point.y-top);
|
|
|
|
vint start=-1;
|
|
vint length=0;
|
|
if(cache->graphicsParagraph->GetInlineObjectFromPoint(paragraphPoint, start, length))
|
|
{
|
|
return element->document->GetHyperlink(index, start, start+length);
|
|
}
|
|
|
|
vint caret=cache->graphicsParagraph->GetCaretFromPoint(paragraphPoint);
|
|
return element->document->GetHyperlink(index, caret, caret);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
void GuiDocumentElement::GuiDocumentElementRenderer::OpenCaret(TextPos caret, Color color, bool frontSide)
|
|
{
|
|
CloseCaret(caret);
|
|
lastCaret=caret;
|
|
lastCaretColor=color;
|
|
lastCaretFrontSide=frontSide;
|
|
|
|
Ptr<ParagraphCache> cache=paragraphCaches[lastCaret.row];
|
|
if(cache && cache->graphicsParagraph)
|
|
{
|
|
cache->graphicsParagraph->OpenCaret(lastCaret.column, lastCaretColor, lastCaretFrontSide);
|
|
}
|
|
}
|
|
|
|
void GuiDocumentElement::GuiDocumentElementRenderer::CloseCaret(TextPos caret)
|
|
{
|
|
if(lastCaret!=TextPos(-1, -1))
|
|
{
|
|
if(0<=lastCaret.row && lastCaret.row<paragraphCaches.Count())
|
|
{
|
|
Ptr<ParagraphCache> cache=paragraphCaches[lastCaret.row];
|
|
if(cache && cache->graphicsParagraph)
|
|
{
|
|
cache->graphicsParagraph->CloseCaret();
|
|
}
|
|
}
|
|
}
|
|
lastCaret=caret;
|
|
}
|
|
|
|
void GuiDocumentElement::GuiDocumentElementRenderer::SetSelection(TextPos begin, TextPos end)
|
|
{
|
|
if(begin>end)
|
|
{
|
|
TextPos t=begin;
|
|
begin=end;
|
|
end=t;
|
|
}
|
|
if(begin==end)
|
|
{
|
|
begin=TextPos(-1, -1);
|
|
end=TextPos(-1, -1);
|
|
}
|
|
|
|
for(vint i=0;i<paragraphCaches.Count();i++)
|
|
{
|
|
if(begin.row<=i && i<=end.row)
|
|
{
|
|
Ptr<ParagraphCache> cache=EnsureAndGetCache(i, false);
|
|
vint newBegin=i==begin.row?begin.column:0;
|
|
vint newEnd=i==end.row?end.column:cache->fullText.Length();
|
|
|
|
if(cache->selectionBegin!=newBegin || cache->selectionEnd!=newEnd)
|
|
{
|
|
cache->selectionBegin=newBegin;
|
|
cache->selectionEnd=newEnd;
|
|
NotifyParagraphUpdated(i, 1, 1, false);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
Ptr<ParagraphCache> cache=paragraphCaches[i];
|
|
if(cache)
|
|
{
|
|
if(cache->selectionBegin!=-1 || cache->selectionEnd!=-1)
|
|
{
|
|
cache->selectionBegin=-1;
|
|
cache->selectionEnd=-1;
|
|
NotifyParagraphUpdated(i, 1, 1, false);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
TextPos GuiDocumentElement::GuiDocumentElementRenderer::CalculateCaret(TextPos comparingCaret, IGuiGraphicsParagraph::CaretRelativePosition position, bool& preferFrontSide)
|
|
{
|
|
Ptr<ParagraphCache> cache=EnsureAndGetCache(comparingCaret.row, true);
|
|
if(cache)
|
|
{
|
|
switch(position)
|
|
{
|
|
case IGuiGraphicsParagraph::CaretFirst:
|
|
{
|
|
preferFrontSide=false;
|
|
vint caret=cache->graphicsParagraph->GetCaret(0, IGuiGraphicsParagraph::CaretFirst, preferFrontSide);
|
|
return TextPos(comparingCaret.row, caret);
|
|
}
|
|
case IGuiGraphicsParagraph::CaretLast:
|
|
{
|
|
preferFrontSide=true;
|
|
vint caret=cache->graphicsParagraph->GetCaret(0, IGuiGraphicsParagraph::CaretLast, preferFrontSide);
|
|
return TextPos(comparingCaret.row, caret);
|
|
}
|
|
case IGuiGraphicsParagraph::CaretLineFirst:
|
|
{
|
|
preferFrontSide=false;
|
|
vint caret=cache->graphicsParagraph->GetCaret(comparingCaret.column, IGuiGraphicsParagraph::CaretLineFirst, preferFrontSide);
|
|
return TextPos(comparingCaret.row, caret);
|
|
}
|
|
case IGuiGraphicsParagraph::CaretLineLast:
|
|
{
|
|
preferFrontSide=true;
|
|
vint caret=cache->graphicsParagraph->GetCaret(comparingCaret.column, IGuiGraphicsParagraph::CaretLineLast, preferFrontSide);
|
|
return TextPos(comparingCaret.row, caret);
|
|
}
|
|
case IGuiGraphicsParagraph::CaretMoveUp:
|
|
{
|
|
vint caret=cache->graphicsParagraph->GetCaret(comparingCaret.column, IGuiGraphicsParagraph::CaretMoveUp, preferFrontSide);
|
|
if(caret==comparingCaret.column && comparingCaret.row>0)
|
|
{
|
|
Rect caretBounds=cache->graphicsParagraph->GetCaretBounds(comparingCaret.column, preferFrontSide);
|
|
Ptr<ParagraphCache> anotherCache=EnsureAndGetCache(comparingCaret.row-1, true);
|
|
vint height=anotherCache->graphicsParagraph->GetHeight();
|
|
caret=anotherCache->graphicsParagraph->GetCaretFromPoint(Point(caretBounds.x1, height));
|
|
return TextPos(comparingCaret.row-1, caret);
|
|
}
|
|
else
|
|
{
|
|
return TextPos(comparingCaret.row, caret);
|
|
}
|
|
}
|
|
case IGuiGraphicsParagraph::CaretMoveDown:
|
|
{
|
|
vint caret=cache->graphicsParagraph->GetCaret(comparingCaret.column, IGuiGraphicsParagraph::CaretMoveDown, preferFrontSide);
|
|
if(caret==comparingCaret.column && comparingCaret.row<paragraphCaches.Count()-1)
|
|
{
|
|
Rect caretBounds=cache->graphicsParagraph->GetCaretBounds(comparingCaret.column, preferFrontSide);
|
|
Ptr<ParagraphCache> anotherCache=EnsureAndGetCache(comparingCaret.row+1, true);
|
|
caret=anotherCache->graphicsParagraph->GetCaretFromPoint(Point(caretBounds.x1, 0));
|
|
return TextPos(comparingCaret.row+1, caret);
|
|
}
|
|
else
|
|
{
|
|
return TextPos(comparingCaret.row, caret);
|
|
}
|
|
}
|
|
case IGuiGraphicsParagraph::CaretMoveLeft:
|
|
{
|
|
preferFrontSide=false;
|
|
vint caret=cache->graphicsParagraph->GetCaret(comparingCaret.column, IGuiGraphicsParagraph::CaretMoveLeft, preferFrontSide);
|
|
if(caret==comparingCaret.column && comparingCaret.row>0)
|
|
{
|
|
Ptr<ParagraphCache> anotherCache=EnsureAndGetCache(comparingCaret.row-1, true);
|
|
caret=anotherCache->graphicsParagraph->GetCaret(0, IGuiGraphicsParagraph::CaretLast, preferFrontSide);
|
|
return TextPos(comparingCaret.row-1, caret);
|
|
}
|
|
else
|
|
{
|
|
return TextPos(comparingCaret.row, caret);
|
|
}
|
|
}
|
|
case IGuiGraphicsParagraph::CaretMoveRight:
|
|
{
|
|
preferFrontSide=true;
|
|
vint caret=cache->graphicsParagraph->GetCaret(comparingCaret.column, IGuiGraphicsParagraph::CaretMoveRight, preferFrontSide);
|
|
if(caret==comparingCaret.column && comparingCaret.row<paragraphCaches.Count()-1)
|
|
{
|
|
Ptr<ParagraphCache> anotherCache=EnsureAndGetCache(comparingCaret.row+1, true);
|
|
caret=anotherCache->graphicsParagraph->GetCaret(0, IGuiGraphicsParagraph::CaretFirst, preferFrontSide);
|
|
return TextPos(comparingCaret.row+1, caret);
|
|
}
|
|
else
|
|
{
|
|
return TextPos(comparingCaret.row, caret);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return comparingCaret;
|
|
}
|
|
|
|
TextPos GuiDocumentElement::GuiDocumentElementRenderer::CalculateCaretFromPoint(Point point)
|
|
{
|
|
vint top=0;
|
|
vint index=-1;
|
|
if(GetParagraphIndexFromPoint(point, top, index))
|
|
{
|
|
Ptr<ParagraphCache> cache=EnsureAndGetCache(index, true);
|
|
Point paragraphPoint(point.x, point.y-top);
|
|
vint caret=cache->graphicsParagraph->GetCaretFromPoint(paragraphPoint);
|
|
return TextPos(index, caret);
|
|
}
|
|
return TextPos(-1, -1);
|
|
}
|
|
|
|
Rect GuiDocumentElement::GuiDocumentElementRenderer::GetCaretBounds(TextPos caret, bool frontSide)
|
|
{
|
|
Ptr<ParagraphCache> cache=EnsureAndGetCache(caret.row, true);
|
|
if(cache)
|
|
{
|
|
Rect bounds=cache->graphicsParagraph->GetCaretBounds(caret.column, frontSide);
|
|
if(bounds!=Rect())
|
|
{
|
|
vint y=0;
|
|
for(vint i=0;i<caret.row;i++)
|
|
{
|
|
EnsureAndGetCache(i, true);
|
|
y+=paragraphHeights[i]+paragraphDistance;
|
|
}
|
|
|
|
bounds.y1+=y;
|
|
bounds.y2+=y;
|
|
return bounds;
|
|
}
|
|
}
|
|
return Rect();
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiDocumentElement
|
|
***********************************************************************/
|
|
|
|
void GuiDocumentElement::UpdateCaret()
|
|
{
|
|
auto elementRenderer = renderer.Cast<GuiDocumentElementRenderer>();
|
|
if (elementRenderer)
|
|
{
|
|
elementRenderer->SetSelection(caretBegin, caretEnd);
|
|
if (caretVisible)
|
|
{
|
|
elementRenderer->OpenCaret(caretEnd, caretColor, caretFrontSide);
|
|
}
|
|
else
|
|
{
|
|
elementRenderer->CloseCaret(caretEnd);
|
|
}
|
|
InvokeOnCompositionStateChanged();
|
|
}
|
|
}
|
|
|
|
GuiDocumentElement::GuiDocumentElement()
|
|
:caretVisible(false)
|
|
,caretFrontSide(false)
|
|
{
|
|
}
|
|
|
|
GuiDocumentElement::ICallback* GuiDocumentElement::GetCallback()
|
|
{
|
|
return callback;
|
|
}
|
|
|
|
void GuiDocumentElement::SetCallback(ICallback* value)
|
|
{
|
|
callback = value;
|
|
}
|
|
|
|
Ptr<DocumentModel> GuiDocumentElement::GetDocument()
|
|
{
|
|
return document;
|
|
}
|
|
|
|
void GuiDocumentElement::SetDocument(Ptr<DocumentModel> value)
|
|
{
|
|
document=value;
|
|
InvokeOnElementStateChanged();
|
|
SetCaret(TextPos(), TextPos(), false);
|
|
}
|
|
|
|
TextPos GuiDocumentElement::GetCaretBegin()
|
|
{
|
|
return caretBegin;
|
|
}
|
|
|
|
TextPos GuiDocumentElement::GetCaretEnd()
|
|
{
|
|
return caretEnd;
|
|
}
|
|
|
|
bool GuiDocumentElement::IsCaretEndPreferFrontSide()
|
|
{
|
|
return caretFrontSide;
|
|
}
|
|
|
|
void GuiDocumentElement::SetCaret(TextPos begin, TextPos end, bool frontSide)
|
|
{
|
|
caretBegin=begin;
|
|
caretEnd=end;
|
|
if(caretBegin<caretEnd)
|
|
{
|
|
caretFrontSide=true;
|
|
}
|
|
else if(caretBegin>caretEnd)
|
|
{
|
|
caretFrontSide=false;
|
|
}
|
|
else
|
|
{
|
|
caretFrontSide=frontSide;
|
|
}
|
|
UpdateCaret();
|
|
}
|
|
|
|
bool GuiDocumentElement::GetCaretVisible()
|
|
{
|
|
return caretVisible;
|
|
}
|
|
|
|
void GuiDocumentElement::SetCaretVisible(bool value)
|
|
{
|
|
caretVisible=value;
|
|
UpdateCaret();
|
|
}
|
|
|
|
Color GuiDocumentElement::GetCaretColor()
|
|
{
|
|
return caretColor;
|
|
}
|
|
|
|
void GuiDocumentElement::SetCaretColor(Color value)
|
|
{
|
|
caretColor=value;
|
|
UpdateCaret();
|
|
}
|
|
|
|
TextPos GuiDocumentElement::CalculateCaret(TextPos comparingCaret, IGuiGraphicsParagraph::CaretRelativePosition position, bool& preferFrontSide)
|
|
{
|
|
if (auto elementRenderer = renderer.Cast<GuiDocumentElementRenderer>())
|
|
{
|
|
TextPos caret=elementRenderer->CalculateCaret(comparingCaret, position, preferFrontSide);
|
|
return caret.column==-1?comparingCaret:caret;
|
|
}
|
|
else
|
|
{
|
|
return comparingCaret;
|
|
}
|
|
}
|
|
|
|
TextPos GuiDocumentElement::CalculateCaretFromPoint(Point point)
|
|
{
|
|
if (auto elementRenderer = renderer.Cast<GuiDocumentElementRenderer>())
|
|
{
|
|
return elementRenderer->CalculateCaretFromPoint(point);
|
|
}
|
|
else
|
|
{
|
|
return TextPos(0, 0);
|
|
}
|
|
}
|
|
|
|
Rect GuiDocumentElement::GetCaretBounds(TextPos caret, bool frontSide)
|
|
{
|
|
if (auto elementRenderer = renderer.Cast<GuiDocumentElementRenderer>())
|
|
{
|
|
return elementRenderer->GetCaretBounds(caret, frontSide);
|
|
}
|
|
else
|
|
{
|
|
return Rect();
|
|
}
|
|
}
|
|
|
|
void GuiDocumentElement::NotifyParagraphUpdated(vint index, vint oldCount, vint newCount, bool updatedText)
|
|
{
|
|
if (auto elementRenderer = renderer.Cast<GuiDocumentElementRenderer>())
|
|
{
|
|
elementRenderer->NotifyParagraphUpdated(index, oldCount, newCount, updatedText);
|
|
InvokeOnCompositionStateChanged();
|
|
}
|
|
}
|
|
|
|
void GuiDocumentElement::EditRun(TextPos begin, TextPos end, Ptr<DocumentModel> model, bool copy)
|
|
{
|
|
if (auto elementRenderer = renderer.Cast<GuiDocumentElementRenderer>())
|
|
{
|
|
if (begin > end)
|
|
{
|
|
TextPos temp = begin;
|
|
begin = end;
|
|
end = temp;
|
|
}
|
|
|
|
vint newRows = document->EditRun(begin, end, model, copy);
|
|
if (newRows != -1)
|
|
{
|
|
elementRenderer->NotifyParagraphUpdated(begin.row, end.row - begin.row + 1, newRows, true);
|
|
}
|
|
InvokeOnCompositionStateChanged();
|
|
}
|
|
}
|
|
|
|
void GuiDocumentElement::EditText(TextPos begin, TextPos end, bool frontSide, const collections::Array<WString>& text)
|
|
{
|
|
if (auto elementRenderer = renderer.Cast<GuiDocumentElementRenderer>())
|
|
{
|
|
if (begin > end)
|
|
{
|
|
TextPos temp = begin;
|
|
begin = end;
|
|
end = temp;
|
|
}
|
|
|
|
vint newRows = document->EditText(begin, end, frontSide, text);
|
|
if (newRows != -1)
|
|
{
|
|
elementRenderer->NotifyParagraphUpdated(begin.row, end.row - begin.row + 1, newRows, true);
|
|
}
|
|
InvokeOnCompositionStateChanged();
|
|
}
|
|
}
|
|
|
|
void GuiDocumentElement::EditStyle(TextPos begin, TextPos end, Ptr<DocumentStyleProperties> style)
|
|
{
|
|
if (auto elementRenderer = renderer.Cast<GuiDocumentElementRenderer>())
|
|
{
|
|
if (begin > end)
|
|
{
|
|
TextPos temp = begin;
|
|
begin = end;
|
|
end = temp;
|
|
}
|
|
|
|
if (document->EditStyle(begin, end, style))
|
|
{
|
|
elementRenderer->NotifyParagraphUpdated(begin.row, end.row - begin.row + 1, end.row - begin.row + 1, false);
|
|
}
|
|
InvokeOnCompositionStateChanged();
|
|
}
|
|
}
|
|
|
|
void GuiDocumentElement::EditImage(TextPos begin, TextPos end, Ptr<GuiImageData> image)
|
|
{
|
|
if (auto elementRenderer = renderer.Cast<GuiDocumentElementRenderer>())
|
|
{
|
|
if (begin > end)
|
|
{
|
|
TextPos temp = begin;
|
|
begin = end;
|
|
end = temp;
|
|
}
|
|
|
|
if (document->EditImage(begin, end, image))
|
|
{
|
|
elementRenderer->NotifyParagraphUpdated(begin.row, end.row - begin.row + 1, 1, true);
|
|
}
|
|
InvokeOnCompositionStateChanged();
|
|
}
|
|
}
|
|
|
|
void GuiDocumentElement::EditHyperlink(vint paragraphIndex, vint begin, vint end, const WString& reference, const WString& normalStyleName, const WString& activeStyleName)
|
|
{
|
|
if (auto elementRenderer = renderer.Cast<GuiDocumentElementRenderer>())
|
|
{
|
|
if (begin > end)
|
|
{
|
|
vint temp = begin;
|
|
begin = end;
|
|
end = temp;
|
|
}
|
|
|
|
if (document->EditHyperlink(paragraphIndex, begin, end, reference, normalStyleName, activeStyleName))
|
|
{
|
|
elementRenderer->NotifyParagraphUpdated(paragraphIndex, 1, 1, false);
|
|
}
|
|
InvokeOnCompositionStateChanged();
|
|
}
|
|
}
|
|
|
|
void GuiDocumentElement::RemoveHyperlink(vint paragraphIndex, vint begin, vint end)
|
|
{
|
|
if (auto elementRenderer = renderer.Cast<GuiDocumentElementRenderer>())
|
|
{
|
|
if (begin > end)
|
|
{
|
|
vint temp = begin;
|
|
begin = end;
|
|
end = temp;
|
|
}
|
|
|
|
if (document->RemoveHyperlink(paragraphIndex, begin, end))
|
|
{
|
|
elementRenderer->NotifyParagraphUpdated(paragraphIndex, 1, 1, false);
|
|
}
|
|
InvokeOnCompositionStateChanged();
|
|
}
|
|
}
|
|
|
|
void GuiDocumentElement::EditStyleName(TextPos begin, TextPos end, const WString& styleName)
|
|
{
|
|
if (auto elementRenderer = renderer.Cast<GuiDocumentElementRenderer>())
|
|
{
|
|
if (begin > end)
|
|
{
|
|
TextPos temp = begin;
|
|
begin = end;
|
|
end = temp;
|
|
}
|
|
|
|
if (document->EditStyleName(begin, end, styleName))
|
|
{
|
|
elementRenderer->NotifyParagraphUpdated(begin.row, end.row - begin.row + 1, end.row - begin.row + 1, false);
|
|
}
|
|
InvokeOnCompositionStateChanged();
|
|
}
|
|
}
|
|
|
|
void GuiDocumentElement::RemoveStyleName(TextPos begin, TextPos end)
|
|
{
|
|
if (auto elementRenderer = renderer.Cast<GuiDocumentElementRenderer>())
|
|
{
|
|
if (begin > end)
|
|
{
|
|
TextPos temp = begin;
|
|
begin = end;
|
|
end = temp;
|
|
}
|
|
|
|
if (document->RemoveStyleName(begin, end))
|
|
{
|
|
elementRenderer->NotifyParagraphUpdated(begin.row, end.row - begin.row + 1, end.row - begin.row + 1, false);
|
|
}
|
|
InvokeOnCompositionStateChanged();
|
|
}
|
|
}
|
|
|
|
void GuiDocumentElement::RenameStyle(const WString& oldStyleName, const WString& newStyleName)
|
|
{
|
|
if (auto elementRenderer = renderer.Cast<GuiDocumentElementRenderer>())
|
|
{
|
|
document->RenameStyle(oldStyleName, newStyleName);
|
|
}
|
|
}
|
|
|
|
void GuiDocumentElement::ClearStyle(TextPos begin, TextPos end)
|
|
{
|
|
if (auto elementRenderer = renderer.Cast<GuiDocumentElementRenderer>())
|
|
{
|
|
if (begin > end)
|
|
{
|
|
TextPos temp = begin;
|
|
begin = end;
|
|
end = temp;
|
|
}
|
|
|
|
if (document->ClearStyle(begin, end))
|
|
{
|
|
elementRenderer->NotifyParagraphUpdated(begin.row, end.row - begin.row + 1, end.row - begin.row + 1, false);
|
|
}
|
|
InvokeOnCompositionStateChanged();
|
|
}
|
|
}
|
|
|
|
Ptr<DocumentStyleProperties> GuiDocumentElement::SummarizeStyle(TextPos begin, TextPos end)
|
|
{
|
|
if (auto elementRenderer = renderer.Cast<GuiDocumentElementRenderer>())
|
|
{
|
|
if (begin > end)
|
|
{
|
|
TextPos temp = begin;
|
|
begin = end;
|
|
end = temp;
|
|
}
|
|
|
|
return document->SummarizeStyle(begin, end);
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
Nullable<WString> GuiDocumentElement::SummarizeStyleName(TextPos begin, TextPos end)
|
|
{
|
|
if (auto elementRenderer = renderer.Cast<GuiDocumentElementRenderer>())
|
|
{
|
|
if (begin > end)
|
|
{
|
|
TextPos temp = begin;
|
|
begin = end;
|
|
end = temp;
|
|
}
|
|
|
|
return document->SummarizeStyleName(begin, end);
|
|
}
|
|
return {};
|
|
}
|
|
|
|
void GuiDocumentElement::SetParagraphAlignment(TextPos begin, TextPos end, const collections::Array<Nullable<Alignment>>& alignments)
|
|
{
|
|
if (auto elementRenderer = renderer.Cast<GuiDocumentElementRenderer>())
|
|
{
|
|
vint first = begin.row;
|
|
vint last = end.row;
|
|
if (first > last)
|
|
{
|
|
vint temp = first;
|
|
first = last;
|
|
last = temp;
|
|
}
|
|
|
|
if (0 <= first && first < document->paragraphs.Count() && 0 <= last && last < document->paragraphs.Count() && last - first + 1 == alignments.Count())
|
|
{
|
|
for (vint i = first; i <= last; i++)
|
|
{
|
|
document->paragraphs[i]->alignment = alignments[i - first];
|
|
}
|
|
elementRenderer->NotifyParagraphUpdated(first, alignments.Count(), alignments.Count(), false);
|
|
}
|
|
InvokeOnCompositionStateChanged();
|
|
}
|
|
}
|
|
|
|
Nullable<Alignment> GuiDocumentElement::SummarizeParagraphAlignment(TextPos begin, TextPos end)
|
|
{
|
|
if (auto elementRenderer = renderer.Cast<GuiDocumentElementRenderer>())
|
|
{
|
|
if (begin > end)
|
|
{
|
|
TextPos temp = begin;
|
|
begin = end;
|
|
end = temp;
|
|
}
|
|
|
|
return document->SummarizeParagraphAlignment(begin, end);
|
|
}
|
|
return {};
|
|
}
|
|
|
|
Ptr<DocumentHyperlinkRun::Package> GuiDocumentElement::GetHyperlinkFromPoint(Point point)
|
|
{
|
|
if (auto elementRenderer = renderer.Cast<GuiDocumentElementRenderer>())
|
|
{
|
|
return elementRenderer->GetHyperlinkFromPoint(point);
|
|
}
|
|
return nullptr;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\GRAPHICSELEMENT\GUIGRAPHICSELEMENT.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace elements
|
|
{
|
|
using namespace collections;
|
|
|
|
/***********************************************************************
|
|
GuiFocusRectangleElement
|
|
***********************************************************************/
|
|
|
|
GuiFocusRectangleElement::GuiFocusRectangleElement()
|
|
{
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiSolidBorderElement
|
|
***********************************************************************/
|
|
|
|
GuiSolidBorderElement::GuiSolidBorderElement()
|
|
:color(0, 0, 0)
|
|
{
|
|
}
|
|
|
|
Color GuiSolidBorderElement::GetColor()
|
|
{
|
|
return color;
|
|
}
|
|
|
|
void GuiSolidBorderElement::SetColor(Color value)
|
|
{
|
|
if(color!=value)
|
|
{
|
|
color=value;
|
|
InvokeOnElementStateChanged();
|
|
}
|
|
}
|
|
|
|
ElementShape GuiSolidBorderElement::GetShape()
|
|
{
|
|
return shape;
|
|
}
|
|
|
|
void GuiSolidBorderElement::SetShape(ElementShape value)
|
|
{
|
|
shape=value;
|
|
}
|
|
|
|
/***********************************************************************
|
|
Gui3DBorderElement
|
|
***********************************************************************/
|
|
|
|
Gui3DBorderElement::Gui3DBorderElement()
|
|
{
|
|
}
|
|
|
|
Color Gui3DBorderElement::GetColor1()
|
|
{
|
|
return color1;
|
|
}
|
|
|
|
void Gui3DBorderElement::SetColor1(Color value)
|
|
{
|
|
SetColors(value, color2);
|
|
}
|
|
|
|
Color Gui3DBorderElement::GetColor2()
|
|
{
|
|
return color2;
|
|
}
|
|
|
|
void Gui3DBorderElement::SetColor2(Color value)
|
|
{
|
|
SetColors(color1, value);
|
|
}
|
|
|
|
void Gui3DBorderElement::SetColors(Color value1, Color value2)
|
|
{
|
|
if(color1!=value1 || color2!=value2)
|
|
{
|
|
color1=value1;
|
|
color2=value2;
|
|
InvokeOnElementStateChanged();
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
Gui3DSplitterElement
|
|
***********************************************************************/
|
|
|
|
Gui3DSplitterElement::Gui3DSplitterElement()
|
|
:direction(Horizontal)
|
|
{
|
|
}
|
|
|
|
Color Gui3DSplitterElement::GetColor1()
|
|
{
|
|
return color1;
|
|
}
|
|
|
|
void Gui3DSplitterElement::SetColor1(Color value)
|
|
{
|
|
SetColors(value, color2);
|
|
}
|
|
|
|
Color Gui3DSplitterElement::GetColor2()
|
|
{
|
|
return color2;
|
|
}
|
|
|
|
void Gui3DSplitterElement::SetColor2(Color value)
|
|
{
|
|
SetColors(color1, value);
|
|
}
|
|
|
|
void Gui3DSplitterElement::SetColors(Color value1, Color value2)
|
|
{
|
|
if(color1!=value1 || color2!=value2)
|
|
{
|
|
color1=value1;
|
|
color2=value2;
|
|
InvokeOnElementStateChanged();
|
|
}
|
|
}
|
|
|
|
Gui3DSplitterElement::Direction Gui3DSplitterElement::GetDirection()
|
|
{
|
|
return direction;
|
|
}
|
|
|
|
void Gui3DSplitterElement::SetDirection(Direction value)
|
|
{
|
|
if(direction!=value)
|
|
{
|
|
direction=value;
|
|
InvokeOnElementStateChanged();
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiSolidBackgroundElement
|
|
***********************************************************************/
|
|
|
|
GuiSolidBackgroundElement::GuiSolidBackgroundElement()
|
|
:color(255, 255, 255)
|
|
{
|
|
}
|
|
|
|
Color GuiSolidBackgroundElement::GetColor()
|
|
{
|
|
return color;
|
|
}
|
|
|
|
void GuiSolidBackgroundElement::SetColor(Color value)
|
|
{
|
|
if(color!=value)
|
|
{
|
|
color=value;
|
|
InvokeOnElementStateChanged();
|
|
}
|
|
}
|
|
|
|
ElementShape GuiSolidBackgroundElement::GetShape()
|
|
{
|
|
return shape;
|
|
}
|
|
|
|
void GuiSolidBackgroundElement::SetShape(ElementShape value)
|
|
{
|
|
shape=value;
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiGradientBackgroundElement
|
|
***********************************************************************/
|
|
|
|
GuiGradientBackgroundElement::GuiGradientBackgroundElement()
|
|
:direction(Horizontal)
|
|
{
|
|
}
|
|
|
|
Color GuiGradientBackgroundElement::GetColor1()
|
|
{
|
|
return color1;
|
|
}
|
|
|
|
void GuiGradientBackgroundElement::SetColor1(Color value)
|
|
{
|
|
SetColors(value, color2);
|
|
}
|
|
|
|
Color GuiGradientBackgroundElement::GetColor2()
|
|
{
|
|
return color2;
|
|
}
|
|
|
|
void GuiGradientBackgroundElement::SetColor2(Color value)
|
|
{
|
|
SetColors(color1, value);
|
|
}
|
|
|
|
void GuiGradientBackgroundElement::SetColors(Color value1, Color value2)
|
|
{
|
|
if(color1!=value1 || color2!=value2)
|
|
{
|
|
color1=value1;
|
|
color2=value2;
|
|
InvokeOnElementStateChanged();
|
|
}
|
|
}
|
|
|
|
GuiGradientBackgroundElement::Direction GuiGradientBackgroundElement::GetDirection()
|
|
{
|
|
return direction;
|
|
}
|
|
|
|
void GuiGradientBackgroundElement::SetDirection(Direction value)
|
|
{
|
|
if(direction!=value)
|
|
{
|
|
direction=value;
|
|
InvokeOnElementStateChanged();
|
|
}
|
|
}
|
|
|
|
ElementShape GuiGradientBackgroundElement::GetShape()
|
|
{
|
|
return shape;
|
|
}
|
|
|
|
void GuiGradientBackgroundElement::SetShape(ElementShape value)
|
|
{
|
|
shape=value;
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiInnerShadowElement
|
|
***********************************************************************/
|
|
|
|
GuiInnerShadowElement::GuiInnerShadowElement()
|
|
{
|
|
}
|
|
|
|
Color GuiInnerShadowElement::GetColor()
|
|
{
|
|
return color;
|
|
}
|
|
|
|
void GuiInnerShadowElement::SetColor(Color value)
|
|
{
|
|
if (color != value)
|
|
{
|
|
color = value;
|
|
InvokeOnElementStateChanged();
|
|
}
|
|
}
|
|
|
|
vint GuiInnerShadowElement::GetThickness()
|
|
{
|
|
return thickness;
|
|
}
|
|
|
|
void GuiInnerShadowElement::SetThickness(vint value)
|
|
{
|
|
if (thickness != value)
|
|
{
|
|
thickness = value;
|
|
InvokeOnElementStateChanged();
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiSolidLabelElement
|
|
***********************************************************************/
|
|
|
|
GuiSolidLabelElement::GuiSolidLabelElement()
|
|
:color(0, 0, 0)
|
|
,hAlignment(Alignment::Left)
|
|
,vAlignment(Alignment::Top)
|
|
,wrapLine(false)
|
|
,ellipse(false)
|
|
,multiline(false)
|
|
,wrapLineHeightCalculation(false)
|
|
{
|
|
fontProperties.fontFamily=L"Lucida Console";
|
|
fontProperties.size=12;
|
|
}
|
|
|
|
Color GuiSolidLabelElement::GetColor()
|
|
{
|
|
return color;
|
|
}
|
|
|
|
void GuiSolidLabelElement::SetColor(Color value)
|
|
{
|
|
if(color!=value)
|
|
{
|
|
color=value;
|
|
InvokeOnElementStateChanged();
|
|
}
|
|
}
|
|
|
|
const FontProperties& GuiSolidLabelElement::GetFont()
|
|
{
|
|
return fontProperties;
|
|
}
|
|
|
|
void GuiSolidLabelElement::SetFont(const FontProperties& value)
|
|
{
|
|
if(fontProperties!=value)
|
|
{
|
|
fontProperties=value;
|
|
InvokeOnElementStateChanged();
|
|
}
|
|
}
|
|
|
|
const WString& GuiSolidLabelElement::GetText()
|
|
{
|
|
return text;
|
|
}
|
|
|
|
void GuiSolidLabelElement::SetText(const WString& value)
|
|
{
|
|
if(text!=value)
|
|
{
|
|
text=value;
|
|
InvokeOnElementStateChanged();
|
|
}
|
|
}
|
|
|
|
Alignment GuiSolidLabelElement::GetHorizontalAlignment()
|
|
{
|
|
return hAlignment;
|
|
}
|
|
|
|
Alignment GuiSolidLabelElement::GetVerticalAlignment()
|
|
{
|
|
return vAlignment;
|
|
}
|
|
|
|
void GuiSolidLabelElement::SetHorizontalAlignment(Alignment value)
|
|
{
|
|
SetAlignments(value, vAlignment);
|
|
}
|
|
|
|
void GuiSolidLabelElement::SetVerticalAlignment(Alignment value)
|
|
{
|
|
SetAlignments(hAlignment, value);
|
|
}
|
|
|
|
void GuiSolidLabelElement::SetAlignments(Alignment horizontal, Alignment vertical)
|
|
{
|
|
if(hAlignment!=horizontal || vAlignment!=vertical)
|
|
{
|
|
hAlignment=horizontal;
|
|
vAlignment=vertical;
|
|
InvokeOnElementStateChanged();
|
|
}
|
|
}
|
|
|
|
bool GuiSolidLabelElement::GetWrapLine()
|
|
{
|
|
return wrapLine;
|
|
}
|
|
|
|
void GuiSolidLabelElement::SetWrapLine(bool value)
|
|
{
|
|
if(wrapLine!=value)
|
|
{
|
|
wrapLine=value;
|
|
InvokeOnElementStateChanged();
|
|
}
|
|
}
|
|
|
|
bool GuiSolidLabelElement::GetEllipse()
|
|
{
|
|
return ellipse;
|
|
}
|
|
|
|
void GuiSolidLabelElement::SetEllipse(bool value)
|
|
{
|
|
if(ellipse!=value)
|
|
{
|
|
ellipse=value;
|
|
InvokeOnElementStateChanged();
|
|
}
|
|
}
|
|
|
|
bool GuiSolidLabelElement::GetMultiline()
|
|
{
|
|
return multiline;
|
|
}
|
|
|
|
void GuiSolidLabelElement::SetMultiline(bool value)
|
|
{
|
|
if(multiline!=value)
|
|
{
|
|
multiline=value;
|
|
InvokeOnElementStateChanged();
|
|
}
|
|
}
|
|
|
|
bool GuiSolidLabelElement::GetWrapLineHeightCalculation()
|
|
{
|
|
return wrapLineHeightCalculation;
|
|
}
|
|
|
|
void GuiSolidLabelElement::SetWrapLineHeightCalculation(bool value)
|
|
{
|
|
if(wrapLineHeightCalculation!=value)
|
|
{
|
|
wrapLineHeightCalculation=value;
|
|
InvokeOnElementStateChanged();
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiImageFrameElement
|
|
***********************************************************************/
|
|
|
|
GuiImageFrameElement::GuiImageFrameElement()
|
|
:frameIndex(0)
|
|
,hAlignment(Alignment::Left)
|
|
,vAlignment(Alignment::Top)
|
|
,stretch(false)
|
|
,enabled(true)
|
|
{
|
|
}
|
|
|
|
Ptr<INativeImage> GuiImageFrameElement::GetImage()
|
|
{
|
|
return image;
|
|
}
|
|
|
|
vint GuiImageFrameElement::GetFrameIndex()
|
|
{
|
|
return frameIndex;
|
|
}
|
|
|
|
void GuiImageFrameElement::SetImage(Ptr<INativeImage> value)
|
|
{
|
|
SetImage(value, frameIndex);
|
|
}
|
|
|
|
void GuiImageFrameElement::SetFrameIndex(vint value)
|
|
{
|
|
SetImage(image, value);
|
|
}
|
|
|
|
void GuiImageFrameElement::SetImage(Ptr<INativeImage> _image, vint _frameIndex)
|
|
{
|
|
if(image!=_image || frameIndex!=_frameIndex)
|
|
{
|
|
if(!_image)
|
|
{
|
|
image=0;
|
|
frameIndex=0;
|
|
}
|
|
else if(0<=_frameIndex && _frameIndex<_image->GetFrameCount())
|
|
{
|
|
image=_image;
|
|
frameIndex=_frameIndex;
|
|
}
|
|
InvokeOnElementStateChanged();
|
|
}
|
|
}
|
|
|
|
Alignment GuiImageFrameElement::GetHorizontalAlignment()
|
|
{
|
|
return hAlignment;
|
|
}
|
|
|
|
Alignment GuiImageFrameElement::GetVerticalAlignment()
|
|
{
|
|
return vAlignment;
|
|
}
|
|
|
|
void GuiImageFrameElement::SetHorizontalAlignment(Alignment value)
|
|
{
|
|
SetAlignments(value, vAlignment);
|
|
}
|
|
|
|
void GuiImageFrameElement::SetVerticalAlignment(Alignment value)
|
|
{
|
|
SetAlignments(hAlignment, value);
|
|
}
|
|
|
|
void GuiImageFrameElement::SetAlignments(Alignment horizontal, Alignment vertical)
|
|
{
|
|
if(hAlignment!=horizontal || vAlignment!=vertical)
|
|
{
|
|
hAlignment=horizontal;
|
|
vAlignment=vertical;
|
|
InvokeOnElementStateChanged();
|
|
}
|
|
}
|
|
|
|
bool GuiImageFrameElement::GetStretch()
|
|
{
|
|
return stretch;
|
|
}
|
|
|
|
void GuiImageFrameElement::SetStretch(bool value)
|
|
{
|
|
if(stretch!=value)
|
|
{
|
|
stretch=value;
|
|
InvokeOnElementStateChanged();
|
|
}
|
|
}
|
|
|
|
bool GuiImageFrameElement::GetEnabled()
|
|
{
|
|
return enabled;
|
|
}
|
|
|
|
void GuiImageFrameElement::SetEnabled(bool value)
|
|
{
|
|
if(enabled!=value)
|
|
{
|
|
enabled=value;
|
|
InvokeOnElementStateChanged();
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiPolygonElement
|
|
***********************************************************************/
|
|
|
|
GuiPolygonElement::GuiPolygonElement()
|
|
{
|
|
}
|
|
|
|
Size GuiPolygonElement::GetSize()
|
|
{
|
|
return size;
|
|
}
|
|
|
|
void GuiPolygonElement::SetSize(Size value)
|
|
{
|
|
if(size!=value)
|
|
{
|
|
size=value;
|
|
InvokeOnElementStateChanged();
|
|
}
|
|
}
|
|
|
|
const Point& GuiPolygonElement::GetPoint(vint index)
|
|
{
|
|
return points[index];
|
|
}
|
|
|
|
vint GuiPolygonElement::GetPointCount()
|
|
{
|
|
return points.Count();
|
|
}
|
|
|
|
void GuiPolygonElement::SetPoints(const Point* p, vint count)
|
|
{
|
|
points.Resize(count);
|
|
if(count>0)
|
|
{
|
|
memcpy(&points[0], p, sizeof(*p)*count);
|
|
}
|
|
InvokeOnElementStateChanged();
|
|
}
|
|
|
|
const GuiPolygonElement::PointArray& GuiPolygonElement::GetPointsArray()
|
|
{
|
|
return points;
|
|
}
|
|
|
|
void GuiPolygonElement::SetPointsArray(const PointArray& value)
|
|
{
|
|
CopyFrom(points, value);
|
|
InvokeOnElementStateChanged();
|
|
}
|
|
|
|
Color GuiPolygonElement::GetBorderColor()
|
|
{
|
|
return borderColor;
|
|
}
|
|
|
|
void GuiPolygonElement::SetBorderColor(Color value)
|
|
{
|
|
if(borderColor!=value)
|
|
{
|
|
borderColor=value;
|
|
InvokeOnElementStateChanged();
|
|
}
|
|
}
|
|
|
|
Color GuiPolygonElement::GetBackgroundColor()
|
|
{
|
|
return backgroundColor;
|
|
}
|
|
|
|
void GuiPolygonElement::SetBackgroundColor(Color value)
|
|
{
|
|
if(backgroundColor!=value)
|
|
{
|
|
backgroundColor=value;
|
|
InvokeOnElementStateChanged();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\GRAPHICSELEMENT\GUIGRAPHICSRESOURCEMANAGER.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace elements
|
|
{
|
|
using namespace collections;
|
|
|
|
/***********************************************************************
|
|
GuiGraphicsResourceManager
|
|
***********************************************************************/
|
|
|
|
GuiGraphicsResourceManager::GuiGraphicsResourceManager()
|
|
{
|
|
}
|
|
|
|
GuiGraphicsResourceManager::~GuiGraphicsResourceManager()
|
|
{
|
|
}
|
|
|
|
bool GuiGraphicsResourceManager::RegisterElementFactory(IGuiGraphicsElementFactory* factory)
|
|
{
|
|
if(elementFactories.Keys().Contains(factory->GetElementTypeName()))
|
|
{
|
|
return false;
|
|
}
|
|
else
|
|
{
|
|
elementFactories.Add(factory->GetElementTypeName(), factory);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
bool GuiGraphicsResourceManager::RegisterRendererFactory(const WString& elementTypeName, IGuiGraphicsRendererFactory* factory)
|
|
{
|
|
if(rendererFactories.Keys().Contains(elementTypeName))
|
|
{
|
|
return false;
|
|
}
|
|
else
|
|
{
|
|
rendererFactories.Add(elementTypeName, factory);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
IGuiGraphicsElementFactory* GuiGraphicsResourceManager::GetElementFactory(const WString& elementTypeName)
|
|
{
|
|
vint index=elementFactories.Keys().IndexOf(elementTypeName);
|
|
return index==-1?0:elementFactories.Values().Get(index).Obj();
|
|
}
|
|
|
|
IGuiGraphicsRendererFactory* GuiGraphicsResourceManager::GetRendererFactory(const WString& elementTypeName)
|
|
{
|
|
vint index=rendererFactories.Keys().IndexOf(elementTypeName);
|
|
return index==-1?nullptr:rendererFactories.Values().Get(index).Obj();
|
|
}
|
|
|
|
GuiGraphicsResourceManager* guiGraphicsResourceManager=0;
|
|
|
|
GuiGraphicsResourceManager* GetGuiGraphicsResourceManager()
|
|
{
|
|
return guiGraphicsResourceManager;
|
|
}
|
|
|
|
void SetGuiGraphicsResourceManager(GuiGraphicsResourceManager* resourceManager)
|
|
{
|
|
guiGraphicsResourceManager=resourceManager;
|
|
}
|
|
|
|
bool RegisterFactories(IGuiGraphicsElementFactory* elementFactory, IGuiGraphicsRendererFactory* rendererFactory)
|
|
{
|
|
if(guiGraphicsResourceManager && elementFactory && rendererFactory)
|
|
{
|
|
if(guiGraphicsResourceManager->RegisterElementFactory(elementFactory))
|
|
{
|
|
if(guiGraphicsResourceManager->RegisterRendererFactory(elementFactory->GetElementTypeName(), rendererFactory))
|
|
{
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\GRAPHICSELEMENT\GUIGRAPHICSTEXTELEMENT.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
using namespace collections;
|
|
|
|
namespace presentation
|
|
{
|
|
namespace elements
|
|
{
|
|
namespace text
|
|
{
|
|
|
|
/***********************************************************************
|
|
text::TextLine
|
|
***********************************************************************/
|
|
|
|
TextLine::TextLine()
|
|
:text(0)
|
|
,att(0)
|
|
,availableOffsetCount(0)
|
|
,bufferLength(0)
|
|
,dataLength(0)
|
|
,lexerFinalState(-1)
|
|
,contextFinalState(-1)
|
|
{
|
|
}
|
|
|
|
TextLine::~TextLine()
|
|
{
|
|
}
|
|
|
|
vint TextLine::CalculateBufferLength(vint dataLength)
|
|
{
|
|
if(dataLength<1)dataLength=1;
|
|
vint bufferLength=dataLength-dataLength%BlockSize;
|
|
if(bufferLength<dataLength)
|
|
{
|
|
bufferLength+=BlockSize;
|
|
}
|
|
return bufferLength;
|
|
}
|
|
|
|
void TextLine::Initialize()
|
|
{
|
|
Finalize();
|
|
text=new wchar_t[BlockSize];
|
|
att=new CharAtt[BlockSize];
|
|
bufferLength=BlockSize;
|
|
|
|
memset(text, 0, sizeof(wchar_t)*bufferLength);
|
|
memset(att, 0, sizeof(CharAtt)*bufferLength);
|
|
}
|
|
|
|
void TextLine::Finalize()
|
|
{
|
|
if(text)
|
|
{
|
|
delete[] text;
|
|
text=0;
|
|
}
|
|
if(att)
|
|
{
|
|
delete[] att;
|
|
att=0;
|
|
}
|
|
availableOffsetCount=0;
|
|
bufferLength=0;
|
|
dataLength=0;
|
|
}
|
|
|
|
bool TextLine::IsReady()
|
|
{
|
|
return text && att;
|
|
}
|
|
|
|
bool TextLine::Modify(vint start, vint count, const wchar_t* input, vint inputCount)
|
|
{
|
|
if(!text || !att || start<0 || count<0 || start+count>dataLength || inputCount<0) return false;
|
|
|
|
vint newDataLength=dataLength-count+inputCount;
|
|
vint newBufferLength=CalculateBufferLength(newDataLength);
|
|
if(newBufferLength!=bufferLength)
|
|
{
|
|
wchar_t* newText=new wchar_t[newBufferLength];
|
|
memcpy(newText, text, start*sizeof(wchar_t));
|
|
memcpy(newText+start, input, inputCount*sizeof(wchar_t));
|
|
memcpy(newText+start+inputCount, text+start+count, (dataLength-start-count)*sizeof(wchar_t));
|
|
|
|
CharAtt* newAtt=new CharAtt[newBufferLength];
|
|
memcpy(newAtt, att, start*sizeof(CharAtt));
|
|
memset(newAtt+start, 0, inputCount*sizeof(CharAtt));
|
|
memcpy(newAtt+start+inputCount, att+start+count, (dataLength-start-count)*sizeof(CharAtt));
|
|
|
|
delete[] text;
|
|
delete[] att;
|
|
text=newText;
|
|
att=newAtt;
|
|
}
|
|
else
|
|
{
|
|
memmove(text+start+inputCount, text+start+count, (dataLength-start-count)*sizeof(wchar_t));
|
|
memmove(att+start+inputCount, att+start+count, (dataLength-start-count)*sizeof(CharAtt));
|
|
memcpy(text+start, input, inputCount*sizeof(wchar_t));
|
|
memset(att+start, 0, inputCount*sizeof(CharAtt));
|
|
}
|
|
dataLength=newDataLength;
|
|
bufferLength=newBufferLength;
|
|
if(availableOffsetCount>start)
|
|
{
|
|
availableOffsetCount=start;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
TextLine TextLine::Split(vint index)
|
|
{
|
|
if(index<0 || index>dataLength) return TextLine();
|
|
vint count=dataLength-index;
|
|
TextLine line;
|
|
line.Initialize();
|
|
line.Modify(0, 0, text+index, count);
|
|
memcpy(line.att, att+index, count*sizeof(CharAtt));
|
|
Modify(index, count, L"", 0);
|
|
return line;
|
|
}
|
|
|
|
void TextLine::AppendAndFinalize(TextLine& line)
|
|
{
|
|
vint oldDataLength=dataLength;
|
|
Modify(oldDataLength, 0, line.text, line.dataLength);
|
|
memcpy(att+oldDataLength, line.att, line.dataLength*sizeof(CharAtt));
|
|
line.Finalize();
|
|
}
|
|
|
|
/***********************************************************************
|
|
text::CharMeasurer
|
|
***********************************************************************/
|
|
|
|
CharMeasurer::CharMeasurer(vint _rowHeight)
|
|
:rowHeight(_rowHeight)
|
|
{
|
|
memset(widths, 0, sizeof(widths));
|
|
}
|
|
|
|
CharMeasurer::~CharMeasurer()
|
|
{
|
|
}
|
|
|
|
void CharMeasurer::SetRenderTarget(IGuiGraphicsRenderTarget* value)
|
|
{
|
|
if(oldRenderTarget!=value)
|
|
{
|
|
oldRenderTarget=value;
|
|
rowHeight=GetRowHeightInternal(oldRenderTarget);
|
|
memset(widths, 0, sizeof(widths));
|
|
}
|
|
}
|
|
|
|
vint CharMeasurer::MeasureWidth(UnicodeCodePoint codePoint)
|
|
{
|
|
vuint32_t index = codePoint.GetCodePoint();
|
|
if (0 <= index && index < 65536)
|
|
{
|
|
vint w = widths[index];
|
|
if (w == 0)
|
|
{
|
|
widths[index] = w = MeasureWidthInternal(codePoint, oldRenderTarget);
|
|
}
|
|
return w;
|
|
}
|
|
else if (index < 0x110000)
|
|
{
|
|
return MeasureWidthInternal(codePoint, oldRenderTarget);
|
|
}
|
|
else
|
|
{
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
vint CharMeasurer::GetRowHeight()
|
|
{
|
|
return rowHeight;
|
|
}
|
|
|
|
/***********************************************************************
|
|
text::TextLines
|
|
***********************************************************************/
|
|
|
|
TextLines::TextLines(GuiColorizedTextElement* _ownerElement)
|
|
:ownerElement(_ownerElement)
|
|
,charMeasurer(0)
|
|
,renderTarget(0)
|
|
,tabWidth(1)
|
|
,tabSpaceCount(4)
|
|
,passwordChar(L'\0')
|
|
{
|
|
TextLine line;
|
|
line.Initialize();
|
|
lines.Add(line);
|
|
}
|
|
|
|
TextLines::~TextLines()
|
|
{
|
|
RemoveLines(0, lines.Count());
|
|
}
|
|
|
|
//--------------------------------------------------------
|
|
|
|
vint TextLines::GetCount()
|
|
{
|
|
return lines.Count();
|
|
}
|
|
|
|
TextLine& TextLines::GetLine(vint row)
|
|
{
|
|
return lines[row];
|
|
}
|
|
|
|
CharMeasurer* TextLines::GetCharMeasurer()
|
|
{
|
|
return charMeasurer;
|
|
}
|
|
|
|
void TextLines::SetCharMeasurer(CharMeasurer* value)
|
|
{
|
|
charMeasurer=value;
|
|
if(charMeasurer) charMeasurer->SetRenderTarget(renderTarget);
|
|
ClearMeasurement();
|
|
}
|
|
|
|
IGuiGraphicsRenderTarget* TextLines::GetRenderTarget()
|
|
{
|
|
return renderTarget;
|
|
}
|
|
|
|
void TextLines::SetRenderTarget(IGuiGraphicsRenderTarget* value)
|
|
{
|
|
renderTarget=value;
|
|
if(charMeasurer) charMeasurer->SetRenderTarget(renderTarget);
|
|
ClearMeasurement();
|
|
}
|
|
|
|
WString TextLines::GetText(TextPos start, TextPos end)
|
|
{
|
|
if(!IsAvailable(start) || !IsAvailable(end) || start>end) return L"";
|
|
|
|
if(start.row==end.row)
|
|
{
|
|
return WString(lines[start.row].text+start.column, end.column-start.column);
|
|
}
|
|
|
|
vint count=0;
|
|
for(vint i=start.row+1;i<end.row;i++)
|
|
{
|
|
count+=lines[i].dataLength;
|
|
}
|
|
count+=lines[start.row].dataLength-start.column;
|
|
count+=end.column;
|
|
|
|
Array<wchar_t> buffer;
|
|
buffer.Resize(count+(end.row-start.row)*2);
|
|
wchar_t* writing=&buffer[0];
|
|
|
|
for(vint i=start.row;i<=end.row;i++)
|
|
{
|
|
wchar_t* text=lines[i].text;
|
|
vint chars=0;
|
|
if(i==start.row)
|
|
{
|
|
text+=start.column;
|
|
chars=lines[i].dataLength-start.column;
|
|
}
|
|
else if(i==end.row)
|
|
{
|
|
chars=end.column;
|
|
}
|
|
else
|
|
{
|
|
chars=lines[i].dataLength;
|
|
}
|
|
|
|
if(i!=start.row)
|
|
{
|
|
*writing++=L'\r';
|
|
*writing++=L'\n';
|
|
}
|
|
memcpy(writing, text, chars*sizeof(wchar_t));
|
|
writing+=chars;
|
|
}
|
|
return WString(&buffer[0], buffer.Count());
|
|
}
|
|
|
|
WString TextLines::GetText()
|
|
{
|
|
return GetText(TextPos(0, 0), TextPos(lines.Count()-1, lines[lines.Count()-1].dataLength));
|
|
}
|
|
|
|
void TextLines::SetText(const WString& value)
|
|
{
|
|
Modify(TextPos(0, 0), TextPos(lines.Count()-1, lines[lines.Count()-1].dataLength), value);
|
|
}
|
|
|
|
//--------------------------------------------------------
|
|
|
|
bool TextLines::RemoveLines(vint start, vint count)
|
|
{
|
|
if(start<0 || count<0 || start+count>lines.Count()) return false;
|
|
for(vint i=start+count-1;i>=start;i--)
|
|
{
|
|
lines[i].Finalize();
|
|
}
|
|
lines.RemoveRange(start, count);
|
|
return true;
|
|
}
|
|
|
|
bool TextLines::IsAvailable(TextPos pos)
|
|
{
|
|
return 0<=pos.row && pos.row<lines.Count() && 0<=pos.column && pos.column<=lines[pos.row].dataLength;
|
|
}
|
|
|
|
TextPos TextLines::Normalize(TextPos pos)
|
|
{
|
|
if(pos.row<0)
|
|
{
|
|
return TextPos(0, 0);
|
|
}
|
|
else if(pos.row>=lines.Count())
|
|
{
|
|
return TextPos(lines.Count()-1, lines[lines.Count()-1].dataLength);
|
|
}
|
|
else
|
|
{
|
|
TextLine& line=lines[pos.row];
|
|
if(pos.column<0)
|
|
{
|
|
return TextPos(pos.row, 0);
|
|
}
|
|
else if(pos.column>line.dataLength)
|
|
{
|
|
return TextPos(pos.row, line.dataLength);
|
|
}
|
|
else
|
|
{
|
|
return pos;
|
|
}
|
|
}
|
|
}
|
|
|
|
TextPos TextLines::Modify(TextPos start, TextPos end, const wchar_t** inputs, vint* inputCounts, vint rows)
|
|
{
|
|
if(!IsAvailable(start) || !IsAvailable(end) || start>end) return TextPos(-1, -1);
|
|
if (ownerElement)
|
|
{
|
|
ownerElement->InvokeOnElementStateChanged();
|
|
}
|
|
|
|
if(rows==1)
|
|
{
|
|
if(start.row==end.row)
|
|
{
|
|
lines[start.row].Modify(start.column, end.column-start.column, inputs[0], inputCounts[0]);
|
|
}
|
|
else
|
|
{
|
|
if(end.row-start.row>1)
|
|
{
|
|
RemoveLines(start.row+1, end.row-start.row-1);
|
|
}
|
|
vint modifyCount=lines[start.row].dataLength-start.column+end.column;
|
|
lines[start.row].AppendAndFinalize(lines[start.row+1]);
|
|
lines.RemoveAt(start.row+1);
|
|
lines[start.row].Modify(start.column, modifyCount, inputs[0], inputCounts[0]);
|
|
}
|
|
return TextPos(start.row, start.column+inputCounts[0]);
|
|
}
|
|
|
|
if(start.row==end.row)
|
|
{
|
|
TextLine newLine=lines[start.row].Split(end.column);
|
|
lines.Insert(start.row+1, newLine);
|
|
end=TextPos(start.row+1, 0);
|
|
}
|
|
|
|
vint oldMiddleLines=end.row-start.row-1;
|
|
vint newMiddleLines=rows-2;
|
|
if(oldMiddleLines<newMiddleLines)
|
|
{
|
|
for(vint i=oldMiddleLines;i<newMiddleLines;i++)
|
|
{
|
|
TextLine line;
|
|
line.Initialize();
|
|
lines.Insert(end.row, line);
|
|
}
|
|
}
|
|
else if(oldMiddleLines>newMiddleLines)
|
|
{
|
|
RemoveLines(start.row+newMiddleLines+1, oldMiddleLines-newMiddleLines);
|
|
}
|
|
end.row+=newMiddleLines-oldMiddleLines;
|
|
|
|
lines[start.row].Modify(start.column, lines[start.row].dataLength-start.column, inputs[0], inputCounts[0]);
|
|
lines[end.row].Modify(0, end.column, inputs[rows-1], inputCounts[rows-1]);
|
|
for(vint i=1;i<rows-1;i++)
|
|
{
|
|
lines[start.row+i].Modify(0, lines[start.row+i].dataLength, inputs[i], inputCounts[i]);
|
|
}
|
|
return TextPos(end.row, inputCounts[rows-1]);
|
|
}
|
|
|
|
TextPos TextLines::Modify(TextPos start, TextPos end, const wchar_t* input, vint inputCount)
|
|
{
|
|
List<const wchar_t*> inputs;
|
|
List<vint> inputCounts;
|
|
const wchar_t* previous=input;
|
|
const wchar_t* current=input;
|
|
|
|
while(true)
|
|
{
|
|
if(current==input+inputCount)
|
|
{
|
|
inputs.Add(previous);
|
|
inputCounts.Add(current-previous);
|
|
break;
|
|
}
|
|
else if(*current==L'\r' || *current==L'\n')
|
|
{
|
|
inputs.Add(previous);
|
|
inputCounts.Add(current-previous);
|
|
previous=current+(current[1]==L'\n'?2:1);
|
|
current=previous;
|
|
}
|
|
else
|
|
{
|
|
current++;
|
|
}
|
|
}
|
|
|
|
return Modify(start, end, &inputs[0], &inputCounts[0], inputs.Count());
|
|
}
|
|
|
|
TextPos TextLines::Modify(TextPos start, TextPos end, const wchar_t* input)
|
|
{
|
|
return Modify(start, end, input, wcslen(input));
|
|
}
|
|
|
|
TextPos TextLines::Modify(TextPos start, TextPos end, const WString& input)
|
|
{
|
|
return Modify(start, end, input.Buffer(), input.Length());
|
|
}
|
|
|
|
void TextLines::Clear()
|
|
{
|
|
RemoveLines(0, lines.Count());
|
|
TextLine line;
|
|
line.Initialize();
|
|
lines.Add(line);
|
|
if (ownerElement)
|
|
{
|
|
ownerElement->InvokeOnElementStateChanged();
|
|
}
|
|
}
|
|
|
|
//--------------------------------------------------------
|
|
|
|
void TextLines::ClearMeasurement()
|
|
{
|
|
for (vint i = 0; i < lines.Count(); i++)
|
|
{
|
|
lines[i].availableOffsetCount = 0;
|
|
}
|
|
|
|
tabWidth = tabSpaceCount * (charMeasurer ? charMeasurer->MeasureWidth({ L' ' }) : 1);
|
|
if (tabWidth == 0)
|
|
{
|
|
tabWidth = 1;
|
|
}
|
|
|
|
if (ownerElement)
|
|
{
|
|
ownerElement->InvokeOnElementStateChanged();
|
|
}
|
|
}
|
|
|
|
vint TextLines::GetTabSpaceCount()
|
|
{
|
|
return tabSpaceCount;
|
|
}
|
|
|
|
void TextLines::SetTabSpaceCount(vint value)
|
|
{
|
|
if(value<1) value=1;
|
|
if(tabSpaceCount!=value)
|
|
{
|
|
tabSpaceCount=value;
|
|
ClearMeasurement();
|
|
}
|
|
}
|
|
|
|
void TextLines::MeasureRow(vint row)
|
|
{
|
|
TextLine& line = lines[row];
|
|
vint offset = 0;
|
|
if (line.availableOffsetCount)
|
|
{
|
|
offset = line.att[line.availableOffsetCount - 1].rightOffset;
|
|
}
|
|
for (vint i = line.availableOffsetCount; i < line.dataLength; i++)
|
|
{
|
|
CharAtt& att = line.att[i];
|
|
wchar_t c = line.text[i];
|
|
vint width = 0;
|
|
vint passwordWidth = 0;
|
|
if (passwordChar)
|
|
{
|
|
passwordWidth = charMeasurer ? charMeasurer->MeasureWidth({ passwordChar }) : 1;
|
|
}
|
|
|
|
if (c == L'\t')
|
|
{
|
|
width = tabWidth - offset % tabWidth;
|
|
}
|
|
#if defined VCZH_MSVC
|
|
else if (UTF16SPFirst(c) && (i + 1 < line.dataLength) && UTF16SPSecond(line.text[i + 1]))
|
|
{
|
|
width = passwordChar ? passwordWidth : (charMeasurer ? charMeasurer->MeasureWidth({ c, line.text[i + 1] }) : 1);
|
|
offset += width;
|
|
att.rightOffset = (int)offset;
|
|
line.att[i + 1].rightOffset = (int)offset;
|
|
i++;
|
|
continue;
|
|
}
|
|
#endif
|
|
else
|
|
{
|
|
width = passwordChar ? passwordWidth : (charMeasurer ? charMeasurer->MeasureWidth({ c }) : 1);
|
|
}
|
|
offset += width;
|
|
att.rightOffset = (int)offset;
|
|
}
|
|
line.availableOffsetCount = line.dataLength;
|
|
}
|
|
|
|
vint TextLines::GetRowWidth(vint row)
|
|
{
|
|
if(row<0 || row>=lines.Count()) return -1;
|
|
TextLine& line=lines[row];
|
|
if(line.dataLength==0)
|
|
{
|
|
return 0;
|
|
}
|
|
else
|
|
{
|
|
MeasureRow(row);
|
|
return line.att[line.dataLength-1].rightOffset;
|
|
}
|
|
}
|
|
|
|
vint TextLines::GetRowHeight()
|
|
{
|
|
return charMeasurer ? charMeasurer->GetRowHeight() : 1;
|
|
}
|
|
|
|
vint TextLines::GetMaxWidth()
|
|
{
|
|
vint width=0;
|
|
for(vint i=0;i<lines.Count();i++)
|
|
{
|
|
vint rowWidth=GetRowWidth(i);
|
|
if(width<rowWidth)
|
|
{
|
|
width=rowWidth;
|
|
}
|
|
}
|
|
return width;
|
|
}
|
|
|
|
vint TextLines::GetMaxHeight()
|
|
{
|
|
return lines.Count() * GetRowHeight();
|
|
}
|
|
|
|
TextPos TextLines::GetTextPosFromPoint(Point point)
|
|
{
|
|
vint h = GetRowHeight();
|
|
if(point.y<0)
|
|
{
|
|
point.y=0;
|
|
}
|
|
else if(point.y>=h*lines.Count())
|
|
{
|
|
point.y=h*lines.Count()-1;
|
|
}
|
|
|
|
vint row=point.y/h;
|
|
if(point.x<0)
|
|
{
|
|
return TextPos(row, 0);
|
|
}
|
|
else if(point.x>=GetRowWidth(row))
|
|
{
|
|
return TextPos(row, lines[row].dataLength);
|
|
}
|
|
TextLine& line=lines[row];
|
|
|
|
vint i1=0, i2=line.dataLength;
|
|
vint p1=0, p2=line.att[line.dataLength-1].rightOffset;
|
|
while(i2-i1>1)
|
|
{
|
|
vint i=(i1+i2)/2;
|
|
vint p=i==0?0:line.att[i-1].rightOffset;
|
|
if(point.x<p)
|
|
{
|
|
i2=i;
|
|
p2=p;
|
|
}
|
|
else
|
|
{
|
|
i1=i;
|
|
p1=p;
|
|
}
|
|
}
|
|
#if defined VCZH_MSVC
|
|
if (UTF16SPSecond(line.text[i1]) && i1 > 0 && UTF16SPFirst(line.text[i1 - 1]))
|
|
{
|
|
i1--;
|
|
}
|
|
#endif
|
|
return TextPos(row, i1);
|
|
}
|
|
|
|
Point TextLines::GetPointFromTextPos(TextPos pos)
|
|
{
|
|
if(IsAvailable(pos))
|
|
{
|
|
vint y = pos.row * GetRowHeight();
|
|
if(pos.column==0)
|
|
{
|
|
return Point(0, y);
|
|
}
|
|
else
|
|
{
|
|
MeasureRow(pos.row);
|
|
TextLine& line=lines[pos.row];
|
|
return Point(line.att[pos.column-1].rightOffset, y);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
return Point(-1, -1);
|
|
}
|
|
}
|
|
|
|
Rect TextLines::GetRectFromTextPos(TextPos pos)
|
|
{
|
|
Point point=GetPointFromTextPos(pos);
|
|
if(point==Point(-1, -1))
|
|
{
|
|
return Rect(-1, -1, -1, -1);
|
|
}
|
|
else
|
|
{
|
|
vint h = GetRowHeight();
|
|
TextLine& line=lines[pos.row];
|
|
if(pos.column==line.dataLength)
|
|
{
|
|
return Rect(point, Size(h/2, h));
|
|
}
|
|
else
|
|
{
|
|
return Rect(point, Size(line.att[pos.column].rightOffset-point.x, h));
|
|
}
|
|
}
|
|
}
|
|
|
|
//--------------------------------------------------------
|
|
|
|
wchar_t TextLines::GetPasswordChar()
|
|
{
|
|
return passwordChar;
|
|
}
|
|
|
|
void TextLines::SetPasswordChar(wchar_t value)
|
|
{
|
|
passwordChar=value;
|
|
ClearMeasurement();
|
|
}
|
|
}
|
|
|
|
using namespace text;
|
|
|
|
/***********************************************************************
|
|
GuiColorizedTextElement
|
|
***********************************************************************/
|
|
|
|
GuiColorizedTextElement::GuiColorizedTextElement()
|
|
:callback(0)
|
|
,isVisuallyEnabled(true)
|
|
,isFocused(false)
|
|
,caretVisible(false)
|
|
,lines(this)
|
|
{
|
|
}
|
|
|
|
text::TextLines& GuiColorizedTextElement::GetLines()
|
|
{
|
|
return lines;
|
|
}
|
|
|
|
GuiColorizedTextElement::ICallback* GuiColorizedTextElement::GetCallback()
|
|
{
|
|
return callback;
|
|
}
|
|
|
|
void GuiColorizedTextElement::SetCallback(ICallback* value)
|
|
{
|
|
callback=value;
|
|
if(!callback)
|
|
{
|
|
lines.SetCharMeasurer(0);
|
|
}
|
|
}
|
|
|
|
const GuiColorizedTextElement::ColorArray& GuiColorizedTextElement::GetColors()
|
|
{
|
|
return colors;
|
|
}
|
|
|
|
void GuiColorizedTextElement::SetColors(const ColorArray& value)
|
|
{
|
|
CopyFrom(colors, value);
|
|
if(callback) callback->ColorChanged();
|
|
InvokeOnElementStateChanged();
|
|
}
|
|
|
|
void GuiColorizedTextElement::ResetTextColorIndex(vint index)
|
|
{
|
|
vint lineCount = lines.GetCount();
|
|
for (vint i = 0; i < lineCount; i++)
|
|
{
|
|
auto& line = lines.GetLine(i);
|
|
line.lexerFinalState = -1;
|
|
line.contextFinalState = -1;
|
|
for (vint j = 0; j < line.dataLength; j++)
|
|
{
|
|
line.att[j].colorIndex = (vuint32_t)index;
|
|
}
|
|
}
|
|
}
|
|
|
|
const FontProperties& GuiColorizedTextElement::GetFont()
|
|
{
|
|
return font;
|
|
}
|
|
|
|
void GuiColorizedTextElement::SetFont(const FontProperties& value)
|
|
{
|
|
if(font!=value)
|
|
{
|
|
font=value;
|
|
if(callback)
|
|
{
|
|
callback->FontChanged();
|
|
}
|
|
InvokeOnElementStateChanged();
|
|
}
|
|
}
|
|
|
|
wchar_t GuiColorizedTextElement::GetPasswordChar()
|
|
{
|
|
return lines.GetPasswordChar();
|
|
}
|
|
|
|
void GuiColorizedTextElement::SetPasswordChar(wchar_t value)
|
|
{
|
|
if(lines.GetPasswordChar()!=value)
|
|
{
|
|
lines.SetPasswordChar(value);
|
|
InvokeOnElementStateChanged();
|
|
}
|
|
}
|
|
|
|
Point GuiColorizedTextElement::GetViewPosition()
|
|
{
|
|
return viewPosition;
|
|
}
|
|
|
|
void GuiColorizedTextElement::SetViewPosition(Point value)
|
|
{
|
|
if(viewPosition!=value)
|
|
{
|
|
viewPosition=value;
|
|
InvokeOnElementStateChanged();
|
|
}
|
|
}
|
|
|
|
bool GuiColorizedTextElement::GetVisuallyEnabled()
|
|
{
|
|
return isVisuallyEnabled;
|
|
}
|
|
|
|
void GuiColorizedTextElement::SetVisuallyEnabled(bool value)
|
|
{
|
|
if(isVisuallyEnabled!=value)
|
|
{
|
|
isVisuallyEnabled=value;
|
|
InvokeOnElementStateChanged();
|
|
}
|
|
}
|
|
|
|
bool GuiColorizedTextElement::GetFocused()
|
|
{
|
|
return isFocused;
|
|
}
|
|
|
|
void GuiColorizedTextElement::SetFocused(bool value)
|
|
{
|
|
if(isFocused!=value)
|
|
{
|
|
isFocused=value;
|
|
InvokeOnElementStateChanged();
|
|
}
|
|
}
|
|
|
|
TextPos GuiColorizedTextElement::GetCaretBegin()
|
|
{
|
|
return caretBegin;
|
|
}
|
|
|
|
void GuiColorizedTextElement::SetCaretBegin(TextPos value)
|
|
{
|
|
caretBegin=value;
|
|
InvokeOnElementStateChanged();
|
|
}
|
|
|
|
TextPos GuiColorizedTextElement::GetCaretEnd()
|
|
{
|
|
return caretEnd;
|
|
}
|
|
|
|
void GuiColorizedTextElement::SetCaretEnd(TextPos value)
|
|
{
|
|
caretEnd=value;
|
|
InvokeOnElementStateChanged();
|
|
}
|
|
|
|
bool GuiColorizedTextElement::GetCaretVisible()
|
|
{
|
|
return caretVisible;
|
|
}
|
|
|
|
void GuiColorizedTextElement::SetCaretVisible(bool value)
|
|
{
|
|
caretVisible=value;
|
|
InvokeOnElementStateChanged();
|
|
}
|
|
|
|
Color GuiColorizedTextElement::GetCaretColor()
|
|
{
|
|
return caretColor;
|
|
}
|
|
|
|
void GuiColorizedTextElement::SetCaretColor(Color value)
|
|
{
|
|
if(caretColor!=value)
|
|
{
|
|
caretColor=value;
|
|
InvokeOnElementStateChanged();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
.\GRAPHICSHOST\GUIGRAPHICSHOST.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace compositions
|
|
{
|
|
using namespace collections;
|
|
using namespace controls;
|
|
using namespace elements;
|
|
using namespace theme;
|
|
|
|
/***********************************************************************
|
|
GuiGraphicsTimerManager
|
|
***********************************************************************/
|
|
|
|
GuiGraphicsTimerManager::GuiGraphicsTimerManager()
|
|
{
|
|
}
|
|
|
|
GuiGraphicsTimerManager::~GuiGraphicsTimerManager()
|
|
{
|
|
}
|
|
|
|
void GuiGraphicsTimerManager::AddCallback(Ptr<IGuiGraphicsTimerCallback> callback)
|
|
{
|
|
callbacks.Add(callback);
|
|
}
|
|
|
|
void GuiGraphicsTimerManager::Play()
|
|
{
|
|
for (vint i = callbacks.Count() - 1; i >= 0; i--)
|
|
{
|
|
auto callback = callbacks[i];
|
|
if (!callback->Play())
|
|
{
|
|
callbacks.RemoveAt(i);
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiGraphicsHost
|
|
***********************************************************************/
|
|
|
|
void GuiGraphicsHost::RefreshRelatedHostRecord(INativeWindow* nativeWindow)
|
|
{
|
|
hostRecord.nativeWindow = nativeWindow;
|
|
hostRecord.renderTarget = nativeWindow ? GetGuiGraphicsResourceManager()->GetRenderTarget(nativeWindow) : nullptr;
|
|
windowComposition->UpdateRelatedHostRecord(&hostRecord);
|
|
}
|
|
|
|
void GuiGraphicsHost::DisconnectCompositionInternal(GuiGraphicsComposition* composition)
|
|
{
|
|
for(vint i=0;i<composition->Children().Count();i++)
|
|
{
|
|
DisconnectCompositionInternal(composition->Children().Get(i));
|
|
}
|
|
if(mouseCaptureComposition==composition)
|
|
{
|
|
if(hostRecord.nativeWindow)
|
|
{
|
|
hostRecord.nativeWindow->ReleaseCapture();
|
|
}
|
|
mouseCaptureComposition=0;
|
|
}
|
|
if(focusedComposition==composition)
|
|
{
|
|
focusedComposition=0;
|
|
}
|
|
mouseEnterCompositions.Remove(composition);
|
|
}
|
|
|
|
void GuiGraphicsHost::MouseCapture(const NativeWindowMouseInfo& info)
|
|
{
|
|
if (hostRecord.nativeWindow && (info.left || info.middle || info.right))
|
|
{
|
|
if (!hostRecord.nativeWindow->IsCapturing() && !info.nonClient)
|
|
{
|
|
hostRecord.nativeWindow->RequireCapture();
|
|
auto point = hostRecord.nativeWindow->Convert(NativePoint(info.x, info.y));
|
|
mouseCaptureComposition = windowComposition->FindComposition(point, true);
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiGraphicsHost::MouseUncapture(const NativeWindowMouseInfo& info)
|
|
{
|
|
if(hostRecord.nativeWindow && !(info.left || info.middle || info.right))
|
|
{
|
|
hostRecord.nativeWindow->ReleaseCapture();
|
|
mouseCaptureComposition=0;
|
|
}
|
|
}
|
|
|
|
void GuiGraphicsHost::OnCharInput(const NativeWindowCharInfo& info, GuiGraphicsComposition* composition, GuiCharEvent GuiGraphicsEventReceiver::* eventReceiverEvent)
|
|
{
|
|
List<GuiGraphicsComposition*> compositions;
|
|
while(composition)
|
|
{
|
|
if(composition->HasEventReceiver())
|
|
{
|
|
compositions.Add(composition);
|
|
}
|
|
composition=composition->GetParent();
|
|
}
|
|
|
|
GuiCharEventArgs arguments(composition);
|
|
(NativeWindowCharInfo&)arguments=info;
|
|
|
|
for(vint i=compositions.Count()-1;i>=0;i--)
|
|
{
|
|
compositions[i]->GetEventReceiver()->previewCharInput.Execute(arguments);
|
|
if(arguments.handled)
|
|
{
|
|
return;
|
|
}
|
|
}
|
|
|
|
for(vint i=0;i<compositions.Count();i++)
|
|
{
|
|
(compositions[i]->GetEventReceiver()->*eventReceiverEvent).Execute(arguments);
|
|
if(arguments.handled)
|
|
{
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiGraphicsHost::OnKeyInput(const NativeWindowKeyInfo& info, GuiGraphicsComposition* composition, GuiKeyEvent GuiGraphicsEventReceiver::* eventReceiverEvent)
|
|
{
|
|
List<GuiGraphicsComposition*> compositions;
|
|
{
|
|
auto current = composition;
|
|
while (current)
|
|
{
|
|
if (current->HasEventReceiver())
|
|
{
|
|
compositions.Add(current);
|
|
}
|
|
current = current->GetParent();
|
|
}
|
|
}
|
|
|
|
GuiKeyEventArgs arguments(composition);
|
|
(NativeWindowKeyInfo&)arguments = info;
|
|
|
|
for (vint i = compositions.Count() - 1; i >= 0; i--)
|
|
{
|
|
compositions[i]->GetEventReceiver()->previewKey.Execute(arguments);
|
|
if (arguments.handled)
|
|
{
|
|
return;
|
|
}
|
|
}
|
|
|
|
for (vint i = 0; i < compositions.Count(); i++)
|
|
{
|
|
(compositions[i]->GetEventReceiver()->*eventReceiverEvent).Execute(arguments);
|
|
if (arguments.handled)
|
|
{
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiGraphicsHost::RaiseMouseEvent(GuiMouseEventArgs& arguments, GuiGraphicsComposition* composition, GuiMouseEvent GuiGraphicsEventReceiver::* eventReceiverEvent)
|
|
{
|
|
arguments.compositionSource=composition;
|
|
arguments.eventSource=0;
|
|
vint x=arguments.x;
|
|
vint y=arguments.y;
|
|
|
|
while(composition)
|
|
{
|
|
if(composition->HasEventReceiver())
|
|
{
|
|
if(!arguments.eventSource)
|
|
{
|
|
arguments.eventSource=composition;
|
|
}
|
|
GuiGraphicsEventReceiver* eventReceiver=composition->GetEventReceiver();
|
|
(eventReceiver->*eventReceiverEvent).Execute(arguments);
|
|
if(arguments.handled)
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
GuiGraphicsComposition* parent=composition->GetParent();
|
|
if(parent)
|
|
{
|
|
Rect parentBounds=parent->GetBounds();
|
|
Rect clientArea=parent->GetClientArea();
|
|
Rect childBounds=composition->GetBounds();
|
|
|
|
x+=childBounds.x1+(clientArea.x1-parentBounds.x1);
|
|
y+=childBounds.y1+(clientArea.y1-parentBounds.y1);
|
|
arguments.x=x;
|
|
arguments.y=y;
|
|
}
|
|
composition=parent;
|
|
}
|
|
}
|
|
|
|
void GuiGraphicsHost::OnMouseInput(const NativeWindowMouseInfo& info, GuiMouseEvent GuiGraphicsEventReceiver::* eventReceiverEvent)
|
|
{
|
|
GuiGraphicsComposition* composition = 0;
|
|
if (mouseCaptureComposition)
|
|
{
|
|
composition = mouseCaptureComposition;
|
|
}
|
|
else
|
|
{
|
|
auto point = hostRecord.nativeWindow->Convert(NativePoint(info.x, info.y));
|
|
composition = windowComposition->FindComposition(point, true);
|
|
}
|
|
if (composition)
|
|
{
|
|
Rect bounds = composition->GetGlobalBounds();
|
|
Point point = hostRecord.nativeWindow->Convert(NativePoint(info.x, info.y));
|
|
GuiMouseEventArgs arguments;
|
|
arguments.ctrl = info.ctrl;
|
|
arguments.shift = info.shift;
|
|
arguments.left = info.left;
|
|
arguments.middle = info.middle;
|
|
arguments.right = info.right;
|
|
arguments.wheel = info.wheel;
|
|
arguments.nonClient = info.nonClient;
|
|
arguments.x = point.x - bounds.x1;
|
|
arguments.y = point.y - bounds.y1;
|
|
RaiseMouseEvent(arguments, composition, eventReceiverEvent);
|
|
}
|
|
}
|
|
|
|
void GuiGraphicsHost::RecreateRenderTarget()
|
|
{
|
|
windowComposition->UpdateRelatedHostRecord(nullptr);
|
|
GetGuiGraphicsResourceManager()->RecreateRenderTarget(hostRecord.nativeWindow);
|
|
RefreshRelatedHostRecord(hostRecord.nativeWindow);
|
|
}
|
|
|
|
INativeWindowListener::HitTestResult GuiGraphicsHost::HitTest(NativePoint location)
|
|
{
|
|
NativeRect bounds = hostRecord.nativeWindow->GetBounds();
|
|
NativeRect clientBounds = hostRecord.nativeWindow->GetClientBoundsInScreen();
|
|
NativePoint clientLocation(location.x + bounds.x1 - clientBounds.x1, location.y + bounds.y1 - clientBounds.y1);
|
|
auto point = hostRecord.nativeWindow->Convert(clientLocation);
|
|
GuiGraphicsComposition* hitComposition = windowComposition->FindComposition(point, false);
|
|
while (hitComposition)
|
|
{
|
|
INativeWindowListener::HitTestResult result = hitComposition->GetAssociatedHitTestResult();
|
|
if (result == INativeWindowListener::NoDecision)
|
|
{
|
|
hitComposition = hitComposition->GetParent();
|
|
}
|
|
else
|
|
{
|
|
return result;
|
|
}
|
|
}
|
|
return INativeWindowListener::NoDecision;
|
|
}
|
|
|
|
void GuiGraphicsHost::Moving(NativeRect& bounds, bool fixSizeOnly)
|
|
{
|
|
NativeRect oldBounds = hostRecord.nativeWindow->GetBounds();
|
|
minSize = windowComposition->GetPreferredBounds().GetSize();
|
|
NativeSize minWindowSize = hostRecord.nativeWindow->Convert(minSize) + (oldBounds.GetSize() - hostRecord.nativeWindow->GetClientSize());
|
|
if (bounds.Width() < minWindowSize.x)
|
|
{
|
|
if (fixSizeOnly)
|
|
{
|
|
if (bounds.Width() < minWindowSize.x)
|
|
{
|
|
bounds.x2 = bounds.x1 + minWindowSize.x;
|
|
}
|
|
}
|
|
else if (oldBounds.x1 != bounds.x1)
|
|
{
|
|
bounds.x1 = oldBounds.x2 - minWindowSize.x;
|
|
}
|
|
else if (oldBounds.x2 != bounds.x2)
|
|
{
|
|
bounds.x2 = oldBounds.x1 + minWindowSize.x;
|
|
}
|
|
}
|
|
if (bounds.Height() < minWindowSize.y)
|
|
{
|
|
if (fixSizeOnly)
|
|
{
|
|
if (bounds.Height() < minWindowSize.y)
|
|
{
|
|
bounds.y2 = bounds.y1 + minWindowSize.y;
|
|
}
|
|
}
|
|
else if (oldBounds.y1 != bounds.y1)
|
|
{
|
|
bounds.y1 = oldBounds.y2 - minWindowSize.y;
|
|
}
|
|
else if (oldBounds.y2 != bounds.y2)
|
|
{
|
|
bounds.y2 = oldBounds.y1 + minWindowSize.y;
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiGraphicsHost::Moved()
|
|
{
|
|
NativeSize size = hostRecord.nativeWindow->GetClientSize();
|
|
if (previousClientSize != size)
|
|
{
|
|
previousClientSize = size;
|
|
minSize = windowComposition->GetPreferredBounds().GetSize();
|
|
needRender = true;
|
|
}
|
|
}
|
|
|
|
void GuiGraphicsHost::DpiChanged()
|
|
{
|
|
RecreateRenderTarget();
|
|
needRender = true;
|
|
}
|
|
|
|
void GuiGraphicsHost::Paint()
|
|
{
|
|
if (!supressPaint)
|
|
{
|
|
needRender = true;
|
|
}
|
|
}
|
|
|
|
void GuiGraphicsHost::LeftButtonDown(const NativeWindowMouseInfo& info)
|
|
{
|
|
altActionManager->CloseAltHost();
|
|
MouseCapture(info);
|
|
OnMouseInput(info, &GuiGraphicsEventReceiver::leftButtonDown);
|
|
}
|
|
|
|
void GuiGraphicsHost::LeftButtonUp(const NativeWindowMouseInfo& info)
|
|
{
|
|
OnMouseInput(info, &GuiGraphicsEventReceiver::leftButtonUp);
|
|
MouseUncapture(info);
|
|
}
|
|
|
|
void GuiGraphicsHost::LeftButtonDoubleClick(const NativeWindowMouseInfo& info)
|
|
{
|
|
LeftButtonDown(info);
|
|
OnMouseInput(info, &GuiGraphicsEventReceiver::leftButtonDoubleClick);
|
|
}
|
|
|
|
void GuiGraphicsHost::RightButtonDown(const NativeWindowMouseInfo& info)
|
|
{
|
|
altActionManager->CloseAltHost();
|
|
MouseCapture(info);
|
|
OnMouseInput(info, &GuiGraphicsEventReceiver::rightButtonDown);
|
|
}
|
|
|
|
void GuiGraphicsHost::RightButtonUp(const NativeWindowMouseInfo& info)
|
|
{
|
|
OnMouseInput(info, &GuiGraphicsEventReceiver::rightButtonUp);
|
|
MouseUncapture(info);
|
|
}
|
|
|
|
void GuiGraphicsHost::RightButtonDoubleClick(const NativeWindowMouseInfo& info)
|
|
{
|
|
RightButtonDown(info);
|
|
OnMouseInput(info, &GuiGraphicsEventReceiver::rightButtonDoubleClick);
|
|
}
|
|
|
|
void GuiGraphicsHost::MiddleButtonDown(const NativeWindowMouseInfo& info)
|
|
{
|
|
altActionManager->CloseAltHost();
|
|
MouseCapture(info);
|
|
OnMouseInput(info, &GuiGraphicsEventReceiver::middleButtonDown);
|
|
}
|
|
|
|
void GuiGraphicsHost::MiddleButtonUp(const NativeWindowMouseInfo& info)
|
|
{
|
|
OnMouseInput(info, &GuiGraphicsEventReceiver::middleButtonUp);
|
|
MouseUncapture(info);
|
|
}
|
|
|
|
void GuiGraphicsHost::MiddleButtonDoubleClick(const NativeWindowMouseInfo& info)
|
|
{
|
|
MiddleButtonDown(info);
|
|
OnMouseInput(info, &GuiGraphicsEventReceiver::middleButtonDoubleClick);
|
|
}
|
|
|
|
void GuiGraphicsHost::HorizontalWheel(const NativeWindowMouseInfo& info)
|
|
{
|
|
OnMouseInput(info, &GuiGraphicsEventReceiver::horizontalWheel);
|
|
}
|
|
|
|
void GuiGraphicsHost::VerticalWheel(const NativeWindowMouseInfo& info)
|
|
{
|
|
OnMouseInput(info, &GuiGraphicsEventReceiver::verticalWheel);
|
|
}
|
|
|
|
void GuiGraphicsHost::MouseMoving(const NativeWindowMouseInfo& info)
|
|
{
|
|
CompositionList newCompositions;
|
|
{
|
|
auto point = hostRecord.nativeWindow->Convert(NativePoint(info.x, info.y));
|
|
GuiGraphicsComposition* composition = windowComposition->FindComposition(point, true);
|
|
while (composition)
|
|
{
|
|
newCompositions.Insert(0, composition);
|
|
composition = composition->GetParent();
|
|
}
|
|
}
|
|
|
|
vint firstDifferentIndex = mouseEnterCompositions.Count();
|
|
for (vint i = 0; i < mouseEnterCompositions.Count(); i++)
|
|
{
|
|
if (i == newCompositions.Count())
|
|
{
|
|
firstDifferentIndex = newCompositions.Count();
|
|
break;
|
|
}
|
|
if (mouseEnterCompositions[i] != newCompositions[i])
|
|
{
|
|
firstDifferentIndex = i;
|
|
break;
|
|
}
|
|
}
|
|
|
|
for (vint i = mouseEnterCompositions.Count() - 1; i >= firstDifferentIndex; i--)
|
|
{
|
|
GuiGraphicsComposition* composition = mouseEnterCompositions[i];
|
|
if (composition->HasEventReceiver())
|
|
{
|
|
composition->GetEventReceiver()->mouseLeave.Execute(GuiEventArgs(composition));
|
|
}
|
|
}
|
|
|
|
CopyFrom(mouseEnterCompositions, newCompositions);
|
|
for (vint i = firstDifferentIndex; i < mouseEnterCompositions.Count(); i++)
|
|
{
|
|
GuiGraphicsComposition* composition = mouseEnterCompositions[i];
|
|
if (composition->HasEventReceiver())
|
|
{
|
|
composition->GetEventReceiver()->mouseEnter.Execute(GuiEventArgs(composition));
|
|
}
|
|
}
|
|
|
|
INativeCursor* cursor = 0;
|
|
if (newCompositions.Count() > 0)
|
|
{
|
|
cursor = newCompositions[newCompositions.Count() - 1]->GetRelatedCursor();
|
|
}
|
|
if (cursor)
|
|
{
|
|
hostRecord.nativeWindow->SetWindowCursor(cursor);
|
|
}
|
|
else
|
|
{
|
|
hostRecord.nativeWindow->SetWindowCursor(GetCurrentController()->ResourceService()->GetDefaultSystemCursor());
|
|
}
|
|
|
|
OnMouseInput(info, &GuiGraphicsEventReceiver::mouseMove);
|
|
}
|
|
|
|
void GuiGraphicsHost::MouseEntered()
|
|
{
|
|
}
|
|
|
|
void GuiGraphicsHost::MouseLeaved()
|
|
{
|
|
for(vint i=mouseEnterCompositions.Count()-1;i>=0;i--)
|
|
{
|
|
GuiGraphicsComposition* composition=mouseEnterCompositions[i];
|
|
if(composition->HasEventReceiver())
|
|
{
|
|
composition->GetEventReceiver()->mouseLeave.Execute(GuiEventArgs(composition));
|
|
}
|
|
}
|
|
mouseEnterCompositions.Clear();
|
|
}
|
|
|
|
void GuiGraphicsHost::KeyDown(const NativeWindowKeyInfo& info)
|
|
{
|
|
if (altActionManager->KeyDown(info)) { return; }
|
|
if (tabActionManager->KeyDown(info, focusedComposition)) { return; }
|
|
if(shortcutKeyManager && shortcutKeyManager->Execute(info)) { return; }
|
|
|
|
if (focusedComposition && focusedComposition->HasEventReceiver())
|
|
{
|
|
OnKeyInput(info, focusedComposition, &GuiGraphicsEventReceiver::keyDown);
|
|
}
|
|
}
|
|
|
|
void GuiGraphicsHost::KeyUp(const NativeWindowKeyInfo& info)
|
|
{
|
|
if (altActionManager->KeyUp(info)) { return; }
|
|
|
|
if(focusedComposition && focusedComposition->HasEventReceiver())
|
|
{
|
|
OnKeyInput(info, focusedComposition, &GuiGraphicsEventReceiver::keyUp);
|
|
}
|
|
}
|
|
|
|
void GuiGraphicsHost::SysKeyDown(const NativeWindowKeyInfo& info)
|
|
{
|
|
if (altActionManager->SysKeyDown(info)) { return; }
|
|
|
|
if(focusedComposition && focusedComposition->HasEventReceiver())
|
|
{
|
|
OnKeyInput(info, focusedComposition, &GuiGraphicsEventReceiver::systemKeyDown);
|
|
}
|
|
}
|
|
|
|
void GuiGraphicsHost::SysKeyUp(const NativeWindowKeyInfo& info)
|
|
{
|
|
if (altActionManager->SysKeyUp(info)) { return; }
|
|
|
|
if (!info.ctrl && !info.shift && info.code == VKEY::_MENU && hostRecord.nativeWindow)
|
|
{
|
|
if (hostRecord.nativeWindow)
|
|
{
|
|
hostRecord.nativeWindow->SupressAlt();
|
|
}
|
|
}
|
|
|
|
if (focusedComposition && focusedComposition->HasEventReceiver())
|
|
{
|
|
OnKeyInput(info, focusedComposition, &GuiGraphicsEventReceiver::systemKeyUp);
|
|
}
|
|
}
|
|
|
|
void GuiGraphicsHost::Char(const NativeWindowCharInfo& info)
|
|
{
|
|
if (altActionManager->Char(info)) { return; }
|
|
if (tabActionManager->Char(info)) { return; }
|
|
|
|
if(focusedComposition && focusedComposition->HasEventReceiver())
|
|
{
|
|
OnCharInput(info, focusedComposition, &GuiGraphicsEventReceiver::charInput);
|
|
}
|
|
}
|
|
|
|
void GuiGraphicsHost::GlobalTimer()
|
|
{
|
|
timerManager.Play();
|
|
|
|
DateTime now=DateTime::UtcTime();
|
|
if(now.totalMilliseconds-lastCaretTime>=CaretInterval)
|
|
{
|
|
lastCaretTime=now.totalMilliseconds;
|
|
if(focusedComposition && focusedComposition->HasEventReceiver())
|
|
{
|
|
focusedComposition->GetEventReceiver()->caretNotify.Execute(GuiEventArgs(focusedComposition));
|
|
}
|
|
}
|
|
|
|
Render(false);
|
|
}
|
|
|
|
GuiGraphicsHost::GuiGraphicsHost(controls::GuiControlHost* _controlHost, GuiGraphicsComposition* boundsComposition)
|
|
:controlHost(_controlHost)
|
|
{
|
|
altActionManager = new GuiAltActionManager(controlHost);
|
|
tabActionManager = new GuiTabActionManager(controlHost);
|
|
hostRecord.host = this;
|
|
windowComposition=new GuiWindowComposition;
|
|
windowComposition->SetMinSizeLimitation(GuiGraphicsComposition::LimitToElementAndChildren);
|
|
windowComposition->AddChild(boundsComposition);
|
|
RefreshRelatedHostRecord(nullptr);
|
|
}
|
|
|
|
GuiGraphicsHost::~GuiGraphicsHost()
|
|
{
|
|
windowComposition->RemoveChild(windowComposition->Children()[0]);
|
|
NotifyFinalizeInstance(windowComposition);
|
|
|
|
delete altActionManager;
|
|
delete tabActionManager;
|
|
if (shortcutKeyManager)
|
|
{
|
|
delete shortcutKeyManager;
|
|
shortcutKeyManager = nullptr;
|
|
}
|
|
|
|
delete windowComposition;
|
|
}
|
|
|
|
INativeWindow* GuiGraphicsHost::GetNativeWindow()
|
|
{
|
|
return hostRecord.nativeWindow;
|
|
}
|
|
|
|
void GuiGraphicsHost::SetNativeWindow(INativeWindow* _nativeWindow)
|
|
{
|
|
if (hostRecord.nativeWindow != _nativeWindow)
|
|
{
|
|
if (hostRecord.nativeWindow)
|
|
{
|
|
GetCurrentController()->CallbackService()->UninstallListener(this);
|
|
hostRecord.nativeWindow->UninstallListener(this);
|
|
}
|
|
|
|
if (_nativeWindow)
|
|
{
|
|
_nativeWindow->InstallListener(this);
|
|
GetCurrentController()->CallbackService()->InstallListener(this);
|
|
previousClientSize = _nativeWindow->GetClientSize();
|
|
minSize = windowComposition->GetPreferredBounds().GetSize();
|
|
_nativeWindow->SetCaretPoint(_nativeWindow->Convert(caretPoint));
|
|
needRender = true;
|
|
}
|
|
|
|
RefreshRelatedHostRecord(_nativeWindow);
|
|
}
|
|
}
|
|
|
|
GuiGraphicsComposition* GuiGraphicsHost::GetMainComposition()
|
|
{
|
|
return windowComposition;
|
|
}
|
|
|
|
void GuiGraphicsHost::Render(bool forceUpdate)
|
|
{
|
|
if (!forceUpdate && !needRender)
|
|
{
|
|
return;
|
|
}
|
|
needRender = false;
|
|
|
|
if(hostRecord.nativeWindow && hostRecord.nativeWindow->IsVisible())
|
|
{
|
|
supressPaint = true;
|
|
hostRecord.renderTarget->StartRendering();
|
|
windowComposition->Render(Size());
|
|
{
|
|
auto bounds = windowComposition->GetBounds();
|
|
auto preferred = windowComposition->GetPreferredBounds();
|
|
auto width = bounds.Width() > preferred.Width() ? bounds.Width() : preferred.Width();
|
|
auto height = bounds.Height() > preferred.Height() ? bounds.Height() : preferred.Height();
|
|
if (width != bounds.Width() || height != bounds.Height())
|
|
{
|
|
controlHost->UpdateClientSizeAfterRendering(Size(width, height));
|
|
}
|
|
}
|
|
auto result = hostRecord.renderTarget->StopRendering();
|
|
hostRecord.nativeWindow->RedrawContent();
|
|
supressPaint = false;
|
|
|
|
switch (result)
|
|
{
|
|
case RenderTargetFailure::ResizeWhileRendering:
|
|
{
|
|
GetGuiGraphicsResourceManager()->ResizeRenderTarget(hostRecord.nativeWindow);
|
|
needRender = true;
|
|
}
|
|
break;
|
|
case RenderTargetFailure::LostDevice:
|
|
{
|
|
RecreateRenderTarget();
|
|
needRender = true;
|
|
}
|
|
break;
|
|
default:;
|
|
}
|
|
}
|
|
|
|
if (!needRender)
|
|
{
|
|
{
|
|
ProcList procs;
|
|
CopyFrom(procs, afterRenderProcs);
|
|
afterRenderProcs.Clear();
|
|
for (vint i = 0; i < procs.Count(); i++)
|
|
{
|
|
procs[i]();
|
|
}
|
|
}
|
|
{
|
|
ProcMap procs;
|
|
CopyFrom(procs, afterRenderKeyedProcs);
|
|
afterRenderKeyedProcs.Clear();
|
|
for (vint i = 0; i < procs.Count(); i++)
|
|
{
|
|
procs.Values()[i]();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiGraphicsHost::RequestRender()
|
|
{
|
|
needRender = true;
|
|
}
|
|
|
|
void GuiGraphicsHost::InvokeAfterRendering(const Func<void()>& proc, ProcKey key)
|
|
{
|
|
if (key.key == nullptr)
|
|
{
|
|
afterRenderProcs.Add(proc);
|
|
}
|
|
else
|
|
{
|
|
afterRenderKeyedProcs.Set(key, proc);
|
|
}
|
|
}
|
|
|
|
void GuiGraphicsHost::InvalidateTabOrderCache()
|
|
{
|
|
tabActionManager->InvalidateTabOrderCache();
|
|
}
|
|
|
|
IGuiShortcutKeyManager* GuiGraphicsHost::GetShortcutKeyManager()
|
|
{
|
|
return shortcutKeyManager;
|
|
}
|
|
|
|
void GuiGraphicsHost::SetShortcutKeyManager(IGuiShortcutKeyManager* value)
|
|
{
|
|
shortcutKeyManager=value;
|
|
}
|
|
|
|
bool GuiGraphicsHost::SetFocus(GuiGraphicsComposition* composition)
|
|
{
|
|
if(!composition || composition->GetRelatedGraphicsHost()!=this)
|
|
{
|
|
return false;
|
|
}
|
|
if(focusedComposition && focusedComposition->HasEventReceiver())
|
|
{
|
|
GuiEventArgs arguments;
|
|
arguments.compositionSource=focusedComposition;
|
|
arguments.eventSource=focusedComposition;
|
|
focusedComposition->GetEventReceiver()->lostFocus.Execute(arguments);
|
|
}
|
|
focusedComposition=composition;
|
|
SetCaretPoint(Point(0, 0));
|
|
if(focusedComposition && focusedComposition->HasEventReceiver())
|
|
{
|
|
GuiEventArgs arguments;
|
|
arguments.compositionSource=focusedComposition;
|
|
arguments.eventSource=focusedComposition;
|
|
focusedComposition->GetEventReceiver()->gotFocus.Execute(arguments);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
GuiGraphicsComposition* GuiGraphicsHost::GetFocusedComposition()
|
|
{
|
|
return focusedComposition;
|
|
}
|
|
|
|
Point GuiGraphicsHost::GetCaretPoint()
|
|
{
|
|
return caretPoint;
|
|
}
|
|
|
|
void GuiGraphicsHost::SetCaretPoint(Point value, GuiGraphicsComposition* referenceComposition)
|
|
{
|
|
if (referenceComposition)
|
|
{
|
|
Rect bounds = referenceComposition->GetGlobalBounds();
|
|
value.x += bounds.x1;
|
|
value.y += bounds.y1;
|
|
}
|
|
caretPoint = value;
|
|
if (hostRecord.nativeWindow)
|
|
{
|
|
hostRecord.nativeWindow->SetCaretPoint(hostRecord.nativeWindow->Convert(caretPoint));
|
|
}
|
|
}
|
|
|
|
GuiGraphicsTimerManager* GuiGraphicsHost::GetTimerManager()
|
|
{
|
|
return &timerManager;
|
|
}
|
|
|
|
void GuiGraphicsHost::DisconnectComposition(GuiGraphicsComposition* composition)
|
|
{
|
|
DisconnectCompositionInternal(composition);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\GRAPHICSHOST\GUIGRAPHICSHOST_ALT.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace compositions
|
|
{
|
|
using namespace collections;
|
|
using namespace controls;
|
|
using namespace theme;
|
|
|
|
const wchar_t* const IGuiAltAction::Identifier = L"vl::presentation::compositions::IGuiAltAction";
|
|
const wchar_t* const IGuiAltActionContainer::Identifier = L"vl::presentation::compositions::IGuiAltAction";
|
|
const wchar_t* const IGuiAltActionHost::Identifier = L"vl::presentation::compositions::IGuiAltAction";
|
|
|
|
/***********************************************************************
|
|
IGuiAltAction
|
|
***********************************************************************/
|
|
|
|
bool IGuiAltAction::IsLegalAlt(const WString& alt)
|
|
{
|
|
for (vint i = 0; i < alt.Length(); i++)
|
|
{
|
|
auto c = alt[i];
|
|
if (('A' <= c && c <= 'Z') || ('0' <= c && c <= '9'))
|
|
{
|
|
continue;
|
|
}
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
/***********************************************************************
|
|
IGuiAltActionHost
|
|
***********************************************************************/
|
|
|
|
void IGuiAltActionHost::CollectAltActionsFromControl(controls::GuiControl* control, bool includeThisControl, collections::Group<WString, IGuiAltAction*>& actions)
|
|
{
|
|
List<GuiControl*> controls;
|
|
controls.Add(control);
|
|
vint index = 0;
|
|
|
|
while (index < controls.Count())
|
|
{
|
|
auto current = controls[index++];
|
|
|
|
if (current != control || includeThisControl)
|
|
{
|
|
if (auto container = current->QueryTypedService<IGuiAltActionContainer>())
|
|
{
|
|
vint count = container->GetAltActionCount();
|
|
for (vint i = 0; i < count; i++)
|
|
{
|
|
auto action = container->GetAltAction(i);
|
|
actions.Add(action->GetAlt(), action);
|
|
}
|
|
continue;
|
|
}
|
|
else if (auto action = current->QueryTypedService<IGuiAltAction>())
|
|
{
|
|
if (action->IsAltAvailable())
|
|
{
|
|
if (action->IsAltEnabled())
|
|
{
|
|
actions.Add(action->GetAlt(), action);
|
|
continue;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
vint count = current->GetChildrenCount();
|
|
for (vint i = 0; i < count; i++)
|
|
{
|
|
controls.Add(current->GetChild(i));
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiAltActionHostBase
|
|
***********************************************************************/
|
|
|
|
void GuiAltActionHostBase::SetAltComposition(GuiGraphicsComposition* _composition)
|
|
{
|
|
composition = _composition;
|
|
}
|
|
|
|
void GuiAltActionHostBase::SetAltControl(controls::GuiControl* _control, bool _includeControl)
|
|
{
|
|
control = _control;
|
|
includeControl = _includeControl;
|
|
}
|
|
|
|
GuiGraphicsComposition* GuiAltActionHostBase::GetAltComposition()
|
|
{
|
|
CHECK_ERROR(composition, L"GuiAltActionHostBase::GetAltComposition()#Need to call SetAltComposition.");
|
|
return composition;
|
|
}
|
|
|
|
IGuiAltActionHost* GuiAltActionHostBase::GetPreviousAltHost()
|
|
{
|
|
return previousHost;
|
|
}
|
|
|
|
void GuiAltActionHostBase::OnActivatedAltHost(IGuiAltActionHost* _previousHost)
|
|
{
|
|
previousHost = _previousHost;
|
|
}
|
|
|
|
void GuiAltActionHostBase::OnDeactivatedAltHost()
|
|
{
|
|
previousHost = nullptr;
|
|
}
|
|
|
|
void GuiAltActionHostBase::CollectAltActions(collections::Group<WString, IGuiAltAction*>& actions)
|
|
{
|
|
CHECK_ERROR(control, L"GuiAltActionHostBase::CollectAltActions(Group<WString, IGuiAltAction*>&)#Need to call SetAltControl.");
|
|
CollectAltActionsFromControl(control, includeControl, actions);
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiAltActionManager
|
|
***********************************************************************/
|
|
|
|
void GuiAltActionManager::EnterAltHost(IGuiAltActionHost* host)
|
|
{
|
|
ClearAltHost();
|
|
|
|
Group<WString, IGuiAltAction*> actions;
|
|
host->CollectAltActions(actions);
|
|
if (actions.Count() == 0)
|
|
{
|
|
CloseAltHost();
|
|
return;
|
|
}
|
|
|
|
host->OnActivatedAltHost(currentAltHost);
|
|
currentAltHost = host;
|
|
CreateAltTitles(actions);
|
|
}
|
|
|
|
void GuiAltActionManager::LeaveAltHost()
|
|
{
|
|
if (currentAltHost)
|
|
{
|
|
ClearAltHost();
|
|
auto previousHost = currentAltHost->GetPreviousAltHost();
|
|
currentAltHost->OnDeactivatedAltHost();
|
|
currentAltHost = previousHost;
|
|
|
|
if (currentAltHost)
|
|
{
|
|
Group<WString, IGuiAltAction*> actions;
|
|
currentAltHost->CollectAltActions(actions);
|
|
CreateAltTitles(actions);
|
|
}
|
|
}
|
|
}
|
|
|
|
bool GuiAltActionManager::EnterAltKey(wchar_t key)
|
|
{
|
|
currentAltPrefix += key;
|
|
vint index = currentActiveAltActions.Keys().IndexOf(currentAltPrefix);
|
|
if (index == -1)
|
|
{
|
|
if (FilterTitles() == 0)
|
|
{
|
|
currentAltPrefix = currentAltPrefix.Left(currentAltPrefix.Length() - 1);
|
|
FilterTitles();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
auto action = currentActiveAltActions.Values()[index];
|
|
if (action->GetActivatingAltHost())
|
|
{
|
|
EnterAltHost(action->GetActivatingAltHost());
|
|
}
|
|
else
|
|
{
|
|
CloseAltHost();
|
|
}
|
|
action->OnActiveAlt();
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void GuiAltActionManager::LeaveAltKey()
|
|
{
|
|
if (currentAltPrefix.Length() >= 1)
|
|
{
|
|
currentAltPrefix = currentAltPrefix.Left(currentAltPrefix.Length() - 1);
|
|
}
|
|
FilterTitles();
|
|
}
|
|
|
|
void GuiAltActionManager::CreateAltTitles(const collections::Group<WString, IGuiAltAction*>& actions)
|
|
{
|
|
if (currentAltHost)
|
|
{
|
|
vint count = actions.Count();
|
|
for (vint i = 0; i < count; i++)
|
|
{
|
|
WString key = actions.Keys()[i];
|
|
const auto& values = actions.GetByIndex(i);
|
|
vint numberLength = 0;
|
|
if (values.Count() == 1 && key.Length() > 0)
|
|
{
|
|
numberLength = 0;
|
|
}
|
|
else if (values.Count() <= 10)
|
|
{
|
|
numberLength = 1;
|
|
}
|
|
else if (values.Count() <= 100)
|
|
{
|
|
numberLength = 2;
|
|
}
|
|
else if (values.Count() <= 1000)
|
|
{
|
|
numberLength = 3;
|
|
}
|
|
else
|
|
{
|
|
continue;
|
|
}
|
|
|
|
FOREACH_INDEXER(IGuiAltAction*, action, index, values)
|
|
{
|
|
WString key = actions.Keys()[i];
|
|
if (numberLength > 0)
|
|
{
|
|
WString number = itow(index);
|
|
while (number.Length() < numberLength)
|
|
{
|
|
number = L"0" + number;
|
|
}
|
|
key += number;
|
|
}
|
|
currentActiveAltActions.Add(key, action);
|
|
}
|
|
}
|
|
|
|
count = currentActiveAltActions.Count();
|
|
auto window = dynamic_cast<GuiWindow*>(currentAltHost->GetAltComposition()->GetRelatedControlHost());
|
|
for (vint i = 0; i < count; i++)
|
|
{
|
|
auto key = currentActiveAltActions.Keys()[i];
|
|
auto composition = currentActiveAltActions.Values()[i]->GetAltComposition();
|
|
|
|
auto label = new GuiLabel(theme::ThemeName::ShortcutKey);
|
|
if (auto labelStyle = window->GetControlTemplateObject(true)->GetShortcutKeyTemplate())
|
|
{
|
|
label->SetControlTemplate(labelStyle);
|
|
}
|
|
label->SetText(key);
|
|
composition->AddChild(label->GetBoundsComposition());
|
|
currentActiveAltTitles.Add(key, label);
|
|
}
|
|
|
|
FilterTitles();
|
|
}
|
|
}
|
|
|
|
vint GuiAltActionManager::FilterTitles()
|
|
{
|
|
vint count = currentActiveAltTitles.Count();
|
|
vint visibles = 0;
|
|
for (vint i = 0; i < count; i++)
|
|
{
|
|
auto key = currentActiveAltTitles.Keys()[i];
|
|
auto value = currentActiveAltTitles.Values()[i];
|
|
if (key.Length() >= currentAltPrefix.Length() && key.Left(currentAltPrefix.Length()) == currentAltPrefix)
|
|
{
|
|
value->SetVisible(true);
|
|
if (currentAltPrefix.Length() <= key.Length())
|
|
{
|
|
value->SetText(
|
|
key
|
|
.Insert(currentAltPrefix.Length(), L"[")
|
|
.Insert(currentAltPrefix.Length() + 2, L"]")
|
|
);
|
|
}
|
|
else
|
|
{
|
|
value->SetText(key);
|
|
}
|
|
visibles++;
|
|
}
|
|
else
|
|
{
|
|
value->SetVisible(false);
|
|
}
|
|
}
|
|
return visibles;
|
|
}
|
|
|
|
void GuiAltActionManager::ClearAltHost()
|
|
{
|
|
FOREACH(GuiControl*, title, currentActiveAltTitles.Values())
|
|
{
|
|
SafeDeleteControl(title);
|
|
}
|
|
currentActiveAltActions.Clear();
|
|
currentActiveAltTitles.Clear();
|
|
currentAltPrefix = L"";
|
|
}
|
|
|
|
void GuiAltActionManager::CloseAltHost()
|
|
{
|
|
ClearAltHost();
|
|
while (currentAltHost)
|
|
{
|
|
currentAltHost->OnDeactivatedAltHost();
|
|
currentAltHost = currentAltHost->GetPreviousAltHost();
|
|
}
|
|
}
|
|
|
|
GuiAltActionManager::GuiAltActionManager(controls::GuiControlHost* _controlHost)
|
|
:controlHost(_controlHost)
|
|
{
|
|
}
|
|
|
|
GuiAltActionManager::~GuiAltActionManager()
|
|
{
|
|
}
|
|
|
|
bool GuiAltActionManager::KeyDown(const NativeWindowKeyInfo& info)
|
|
{
|
|
if (!info.ctrl && !info.shift && currentAltHost)
|
|
{
|
|
if (info.code == VKEY::_ESCAPE)
|
|
{
|
|
LeaveAltHost();
|
|
return true;
|
|
}
|
|
else if (info.code == VKEY::_BACK)
|
|
{
|
|
LeaveAltKey();
|
|
}
|
|
else if (VKEY::_NUMPAD0 <= info.code && info.code <= VKEY::_NUMPAD9)
|
|
{
|
|
if (EnterAltKey((wchar_t)(L'0' + ((vint)info.code - (vint)VKEY::_NUMPAD0))))
|
|
{
|
|
supressAltKey = info.code;
|
|
return true;
|
|
}
|
|
}
|
|
else if ((VKEY::_0 <= info.code && info.code <= VKEY::_9) || (VKEY::_A <= info.code && info.code <= VKEY::_Z))
|
|
{
|
|
if (EnterAltKey((wchar_t)info.code))
|
|
{
|
|
supressAltKey = info.code;
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (currentAltHost)
|
|
{
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool GuiAltActionManager::KeyUp(const NativeWindowKeyInfo& info)
|
|
{
|
|
if (!info.ctrl && !info.shift && info.code == supressAltKey)
|
|
{
|
|
supressAltKey = VKEY::_UNKNOWN;
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool GuiAltActionManager::SysKeyDown(const NativeWindowKeyInfo& info)
|
|
{
|
|
if (!info.ctrl && !info.shift && info.code == VKEY::_MENU && !currentAltHost)
|
|
{
|
|
if (auto altHost = controlHost->QueryTypedService<IGuiAltActionHost>())
|
|
{
|
|
if (!altHost->GetPreviousAltHost())
|
|
{
|
|
EnterAltHost(altHost);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (currentAltHost)
|
|
{
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool GuiAltActionManager::SysKeyUp(const NativeWindowKeyInfo& info)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
bool GuiAltActionManager::Char(const NativeWindowCharInfo& info)
|
|
{
|
|
if (currentAltHost || supressAltKey != VKEY::_UNKNOWN)
|
|
{
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\GRAPHICSHOST\GUIGRAPHICSHOST_SHORTCUTKEY.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace compositions
|
|
{
|
|
|
|
/***********************************************************************
|
|
GuiShortcutKeyItem
|
|
***********************************************************************/
|
|
|
|
GuiShortcutKeyItem::GuiShortcutKeyItem(GuiShortcutKeyManager* _shortcutKeyManager, bool _ctrl, bool _shift, bool _alt, VKEY _key)
|
|
:shortcutKeyManager(_shortcutKeyManager)
|
|
,ctrl(_ctrl)
|
|
,shift(_shift)
|
|
,alt(_alt)
|
|
,key(_key)
|
|
{
|
|
}
|
|
|
|
GuiShortcutKeyItem::~GuiShortcutKeyItem()
|
|
{
|
|
}
|
|
|
|
IGuiShortcutKeyManager* GuiShortcutKeyItem::GetManager()
|
|
{
|
|
return shortcutKeyManager;
|
|
}
|
|
|
|
WString GuiShortcutKeyItem::GetName()
|
|
{
|
|
WString name;
|
|
if(ctrl) name+=L"Ctrl+";
|
|
if(shift) name+=L"Shift+";
|
|
if(alt) name+=L"Alt+";
|
|
name+=GetCurrentController()->InputService()->GetKeyName(key);
|
|
return name;
|
|
}
|
|
|
|
bool GuiShortcutKeyItem::CanActivate(const NativeWindowKeyInfo& info)
|
|
{
|
|
return
|
|
info.ctrl==ctrl &&
|
|
info.shift==shift &&
|
|
info.alt==alt &&
|
|
info.code==key;
|
|
}
|
|
|
|
bool GuiShortcutKeyItem::CanActivate(bool _ctrl, bool _shift, bool _alt, VKEY _key)
|
|
{
|
|
return
|
|
_ctrl==ctrl &&
|
|
_shift==shift &&
|
|
_alt==alt &&
|
|
_key==key;
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiShortcutKeyManager
|
|
***********************************************************************/
|
|
|
|
GuiShortcutKeyManager::GuiShortcutKeyManager()
|
|
{
|
|
}
|
|
|
|
GuiShortcutKeyManager::~GuiShortcutKeyManager()
|
|
{
|
|
}
|
|
|
|
vint GuiShortcutKeyManager::GetItemCount()
|
|
{
|
|
return shortcutKeyItems.Count();
|
|
}
|
|
|
|
IGuiShortcutKeyItem* GuiShortcutKeyManager::GetItem(vint index)
|
|
{
|
|
return shortcutKeyItems[index].Obj();
|
|
}
|
|
|
|
bool GuiShortcutKeyManager::Execute(const NativeWindowKeyInfo& info)
|
|
{
|
|
bool executed=false;
|
|
FOREACH(Ptr<GuiShortcutKeyItem>, item, shortcutKeyItems)
|
|
{
|
|
if(item->CanActivate(info))
|
|
{
|
|
GuiEventArgs arguments;
|
|
item->Executed.Execute(arguments);
|
|
executed=true;
|
|
}
|
|
}
|
|
return executed;
|
|
}
|
|
|
|
IGuiShortcutKeyItem* GuiShortcutKeyManager::CreateShortcut(bool ctrl, bool shift, bool alt, VKEY key)
|
|
{
|
|
FOREACH(Ptr<GuiShortcutKeyItem>, item, shortcutKeyItems)
|
|
{
|
|
if(item->CanActivate(ctrl, shift, alt, key))
|
|
{
|
|
return item.Obj();
|
|
}
|
|
}
|
|
Ptr<GuiShortcutKeyItem> item=new GuiShortcutKeyItem(this, ctrl, shift, alt, key);
|
|
shortcutKeyItems.Add(item);
|
|
return item.Obj();
|
|
}
|
|
|
|
bool GuiShortcutKeyManager::DestroyShortcut(bool ctrl, bool shift, bool alt, VKEY key)
|
|
{
|
|
FOREACH(Ptr<GuiShortcutKeyItem>, item, shortcutKeyItems)
|
|
{
|
|
if(item->CanActivate(ctrl, shift, alt, key))
|
|
{
|
|
shortcutKeyItems.Remove(item.Obj());
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
IGuiShortcutKeyItem* GuiShortcutKeyManager::TryGetShortcut(bool ctrl, bool shift, bool alt, VKEY key)
|
|
{
|
|
FOREACH(Ptr<GuiShortcutKeyItem>, item, shortcutKeyItems)
|
|
{
|
|
if(item->CanActivate(ctrl, shift, alt, key))
|
|
{
|
|
return item.Obj();
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\GRAPHICSHOST\GUIGRAPHICSHOST_TAB.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
namespace compositions
|
|
{
|
|
using namespace collections;
|
|
using namespace controls;
|
|
|
|
const wchar_t* const IGuiTabAction::Identifier = L"vl::presentation::compositions::IGuiTabAction";
|
|
|
|
/***********************************************************************
|
|
GuiTabActionManager
|
|
***********************************************************************/
|
|
|
|
namespace tab_focus
|
|
{
|
|
void CollectControls(GuiControl* current, bool includeCurrent, Group<vuint64_t, GuiControl*>& prioritized)
|
|
{
|
|
if (includeCurrent)
|
|
{
|
|
auto tabAction = current->QueryTypedService<IGuiTabAction>();
|
|
if (tabAction && (tabAction->IsTabAvailable() || tabAction->GetTabPriority() != -1))
|
|
{
|
|
vint priority = tabAction->GetTabPriority();
|
|
vuint64_t normalized = priority < 0 ? ~(vuint64_t)0 : (vuint64_t)priority;
|
|
prioritized.Add(normalized, current);
|
|
return;
|
|
}
|
|
}
|
|
|
|
vint count = current->GetChildrenCount();
|
|
for (vint i = 0; i < count; i++)
|
|
{
|
|
CollectControls(current->GetChild(i), true, prioritized);
|
|
}
|
|
}
|
|
|
|
void InsertPrioritized(List<GuiControl*>& controls, vint index, Group<vuint64_t, GuiControl*>& prioritized)
|
|
{
|
|
vint count = prioritized.Count();
|
|
for (vint i = 0; i < count; i++)
|
|
{
|
|
auto& values = prioritized.GetByIndex(i);
|
|
for (vint j = 0; j < values.Count(); j++)
|
|
{
|
|
controls.Insert(index++, values[j]);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
using namespace tab_focus;
|
|
|
|
void GuiTabActionManager::BuildControlList()
|
|
{
|
|
controlsInOrder.Clear();
|
|
{
|
|
Group<vuint64_t, GuiControl*> prioritized;
|
|
CollectControls(controlHost, false, prioritized);
|
|
InsertPrioritized(controlsInOrder, 0, prioritized);
|
|
}
|
|
|
|
for (vint i = 0; i < controlsInOrder.Count(); i++)
|
|
{
|
|
Group<vuint64_t, GuiControl*> prioritized;
|
|
CollectControls(controlsInOrder[i], false, prioritized);
|
|
InsertPrioritized(controlsInOrder, i + 1, prioritized);
|
|
}
|
|
}
|
|
|
|
controls::GuiControl* GuiTabActionManager::GetNextFocusControl(controls::GuiControl* focusedControl, vint offset)
|
|
{
|
|
if (!available)
|
|
{
|
|
BuildControlList();
|
|
available = true;
|
|
}
|
|
#define STEP_AND_NORMALIZE(INDEX) (((INDEX) + offset + controlsInOrder.Count()) % controlsInOrder.Count())
|
|
|
|
if (controlsInOrder.Count() == 0) return nullptr;
|
|
vint startIndex = controlsInOrder.IndexOf(focusedControl);
|
|
startIndex =
|
|
startIndex == -1 ? 0 :
|
|
STEP_AND_NORMALIZE(startIndex);
|
|
|
|
vint index = startIndex;
|
|
do
|
|
{
|
|
auto control = controlsInOrder[index];
|
|
if (auto tabAction = control->QueryTypedService<IGuiTabAction>())
|
|
{
|
|
if (tabAction->IsTabAvailable() && tabAction->IsTabEnabled())
|
|
{
|
|
return control;
|
|
}
|
|
}
|
|
|
|
index = STEP_AND_NORMALIZE(index);
|
|
} while (index != startIndex);
|
|
|
|
#undef STEP_AND_NORMALIZE
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
GuiTabActionManager::GuiTabActionManager(controls::GuiControlHost* _controlHost)
|
|
:controlHost(_controlHost)
|
|
{
|
|
}
|
|
|
|
GuiTabActionManager::~GuiTabActionManager()
|
|
{
|
|
}
|
|
|
|
void GuiTabActionManager::InvalidateTabOrderCache()
|
|
{
|
|
available = false;
|
|
controlsInOrder.Clear();
|
|
}
|
|
|
|
bool GuiTabActionManager::KeyDown(const NativeWindowKeyInfo& info, GuiGraphicsComposition* focusedComposition)
|
|
{
|
|
if (!info.ctrl && !info.alt && info.code == VKEY::_TAB)
|
|
{
|
|
GuiControl* focusedControl = nullptr;
|
|
if (focusedComposition)
|
|
{
|
|
focusedControl = focusedComposition->GetRelatedControl();
|
|
if (focusedControl && focusedControl->GetAcceptTabInput())
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
if (auto next = GetNextFocusControl(focusedControl, (info.shift ? -1 : 1)))
|
|
{
|
|
next->SetFocus();
|
|
supressTabOnce = true;
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool GuiTabActionManager::Char(const NativeWindowCharInfo& info)
|
|
{
|
|
bool supress = supressTabOnce;
|
|
supressTabOnce = false;
|
|
return supress && info.code == L'\t';
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\NATIVEWINDOW\GUINATIVEWINDOW.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
|
|
/***********************************************************************
|
|
INativeWindowListener
|
|
***********************************************************************/
|
|
|
|
INativeWindowListener::HitTestResult INativeWindowListener::HitTest(NativePoint location)
|
|
{
|
|
return INativeWindowListener::NoDecision;
|
|
}
|
|
|
|
void INativeWindowListener::Moving(NativeRect& bounds, bool fixSizeOnly)
|
|
{
|
|
}
|
|
|
|
void INativeWindowListener::Moved()
|
|
{
|
|
}
|
|
|
|
void INativeWindowListener::DpiChanged()
|
|
{
|
|
}
|
|
|
|
void INativeWindowListener::Enabled()
|
|
{
|
|
}
|
|
|
|
void INativeWindowListener::Disabled()
|
|
{
|
|
}
|
|
|
|
void INativeWindowListener::GotFocus()
|
|
{
|
|
}
|
|
|
|
void INativeWindowListener::LostFocus()
|
|
{
|
|
}
|
|
|
|
void INativeWindowListener::Activated()
|
|
{
|
|
}
|
|
|
|
void INativeWindowListener::Deactivated()
|
|
{
|
|
}
|
|
|
|
void INativeWindowListener::Opened()
|
|
{
|
|
}
|
|
|
|
void INativeWindowListener::Closing(bool& cancel)
|
|
{
|
|
}
|
|
|
|
void INativeWindowListener::Closed()
|
|
{
|
|
}
|
|
|
|
void INativeWindowListener::Paint()
|
|
{
|
|
}
|
|
|
|
void INativeWindowListener::Destroying()
|
|
{
|
|
}
|
|
|
|
void INativeWindowListener::Destroyed()
|
|
{
|
|
}
|
|
|
|
void INativeWindowListener::LeftButtonDown(const NativeWindowMouseInfo& info)
|
|
{
|
|
}
|
|
|
|
void INativeWindowListener::LeftButtonUp(const NativeWindowMouseInfo& info)
|
|
{
|
|
}
|
|
|
|
void INativeWindowListener::LeftButtonDoubleClick(const NativeWindowMouseInfo& info)
|
|
{
|
|
}
|
|
|
|
void INativeWindowListener::RightButtonDown(const NativeWindowMouseInfo& info)
|
|
{
|
|
}
|
|
|
|
void INativeWindowListener::RightButtonUp(const NativeWindowMouseInfo& info)
|
|
{
|
|
}
|
|
|
|
void INativeWindowListener::RightButtonDoubleClick(const NativeWindowMouseInfo& info)
|
|
{
|
|
}
|
|
|
|
void INativeWindowListener::MiddleButtonDown(const NativeWindowMouseInfo& info)
|
|
{
|
|
}
|
|
|
|
void INativeWindowListener::MiddleButtonUp(const NativeWindowMouseInfo& info)
|
|
{
|
|
}
|
|
|
|
void INativeWindowListener::MiddleButtonDoubleClick(const NativeWindowMouseInfo& info)
|
|
{
|
|
}
|
|
|
|
void INativeWindowListener::HorizontalWheel(const NativeWindowMouseInfo& info)
|
|
{
|
|
}
|
|
|
|
void INativeWindowListener::VerticalWheel(const NativeWindowMouseInfo& info)
|
|
{
|
|
}
|
|
|
|
void INativeWindowListener::MouseMoving(const NativeWindowMouseInfo& info)
|
|
{
|
|
}
|
|
|
|
void INativeWindowListener::MouseEntered()
|
|
{
|
|
}
|
|
|
|
void INativeWindowListener::MouseLeaved()
|
|
{
|
|
}
|
|
|
|
void INativeWindowListener::KeyDown(const NativeWindowKeyInfo& info)
|
|
{
|
|
}
|
|
|
|
void INativeWindowListener::KeyUp(const NativeWindowKeyInfo& info)
|
|
{
|
|
}
|
|
|
|
void INativeWindowListener::SysKeyDown(const NativeWindowKeyInfo& info)
|
|
{
|
|
}
|
|
|
|
void INativeWindowListener::SysKeyUp(const NativeWindowKeyInfo& info)
|
|
{
|
|
}
|
|
|
|
void INativeWindowListener::Char(const NativeWindowCharInfo& info)
|
|
{
|
|
}
|
|
|
|
/***********************************************************************
|
|
INativeControllerListener
|
|
***********************************************************************/
|
|
|
|
void INativeControllerListener::LeftButtonDown(NativePoint position)
|
|
{
|
|
}
|
|
|
|
void INativeControllerListener::LeftButtonUp(NativePoint position)
|
|
{
|
|
}
|
|
|
|
void INativeControllerListener::RightButtonDown(NativePoint position)
|
|
{
|
|
}
|
|
|
|
void INativeControllerListener::RightButtonUp(NativePoint position)
|
|
{
|
|
}
|
|
|
|
void INativeControllerListener::MouseMoving(NativePoint position)
|
|
{
|
|
}
|
|
|
|
void INativeControllerListener::GlobalTimer()
|
|
{
|
|
}
|
|
|
|
void INativeControllerListener::ClipboardUpdated()
|
|
{
|
|
}
|
|
|
|
void INativeControllerListener::NativeWindowCreated(INativeWindow* window)
|
|
{
|
|
}
|
|
|
|
void INativeControllerListener::NativeWindowDestroying(INativeWindow* window)
|
|
{
|
|
}
|
|
|
|
/***********************************************************************
|
|
Native Window Provider
|
|
***********************************************************************/
|
|
|
|
INativeController* currentController=0;
|
|
|
|
INativeController* GetCurrentController()
|
|
{
|
|
return currentController;
|
|
}
|
|
|
|
void SetCurrentController(INativeController* controller)
|
|
{
|
|
currentController=controller;
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\RESOURCES\GUIDOCUMENT.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
using namespace collections;
|
|
using namespace parsing::tabling;
|
|
using namespace parsing::xml;
|
|
using namespace regex;
|
|
using namespace stream;
|
|
|
|
/***********************************************************************
|
|
DocumentFontSize
|
|
***********************************************************************/
|
|
|
|
DocumentFontSize DocumentFontSize::Parse(const WString& value)
|
|
{
|
|
if (value.Length() > 0 && value[value.Length() - 1] == L'x')
|
|
{
|
|
return DocumentFontSize(wtof(value.Left(value.Length() - 1)), true);
|
|
}
|
|
else
|
|
{
|
|
return DocumentFontSize(wtof(value), false);
|
|
}
|
|
}
|
|
|
|
WString DocumentFontSize::ToString()const
|
|
{
|
|
return ftow(size) + (relative ? L"x" : L"");
|
|
}
|
|
|
|
/***********************************************************************
|
|
DocumentImageRun
|
|
***********************************************************************/
|
|
|
|
const wchar_t* DocumentImageRun::RepresentationText=L"[Image]";
|
|
const wchar_t* DocumentEmbeddedObjectRun::RepresentationText=L"[EmbeddedObject]";
|
|
|
|
/***********************************************************************
|
|
ExtractTextVisitor
|
|
***********************************************************************/
|
|
|
|
namespace document_operation_visitors
|
|
{
|
|
class ExtractTextVisitor : public Object, public DocumentRun::IVisitor
|
|
{
|
|
public:
|
|
stream::TextWriter& writer;
|
|
bool skipNonTextContent;
|
|
|
|
ExtractTextVisitor(stream::TextWriter& _writer, bool _skipNonTextContent)
|
|
:writer(_writer)
|
|
,skipNonTextContent(_skipNonTextContent)
|
|
{
|
|
}
|
|
|
|
void VisitContainer(DocumentContainerRun* run)
|
|
{
|
|
FOREACH(Ptr<DocumentRun>, subRun, run->runs)
|
|
{
|
|
subRun->Accept(this);
|
|
}
|
|
}
|
|
|
|
void VisitContent(DocumentContentRun* run)
|
|
{
|
|
writer.WriteString(run->GetRepresentationText());
|
|
}
|
|
|
|
void Visit(DocumentTextRun* run)override
|
|
{
|
|
VisitContent(run);
|
|
}
|
|
|
|
void Visit(DocumentStylePropertiesRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
|
|
void Visit(DocumentStyleApplicationRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
|
|
void Visit(DocumentHyperlinkRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
|
|
void Visit(DocumentImageRun* run)override
|
|
{
|
|
if(!skipNonTextContent)
|
|
{
|
|
VisitContent(run);
|
|
}
|
|
}
|
|
|
|
void Visit(DocumentEmbeddedObjectRun* run)override
|
|
{
|
|
if(!skipNonTextContent)
|
|
{
|
|
VisitContent(run);
|
|
}
|
|
}
|
|
|
|
void Visit(DocumentParagraphRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
};
|
|
}
|
|
using namespace document_operation_visitors;
|
|
|
|
/***********************************************************************
|
|
DocumentParagraphRun
|
|
***********************************************************************/
|
|
|
|
WString DocumentParagraphRun::GetText(bool skipNonTextContent)
|
|
{
|
|
return GenerateToStream([&](StreamWriter& writer)
|
|
{
|
|
GetText(writer, skipNonTextContent);
|
|
});
|
|
}
|
|
|
|
void DocumentParagraphRun::GetText(stream::TextWriter& writer, bool skipNonTextContent)
|
|
{
|
|
ExtractTextVisitor visitor(writer, skipNonTextContent);
|
|
Accept(&visitor);
|
|
}
|
|
|
|
/***********************************************************************
|
|
DocumentModel
|
|
***********************************************************************/
|
|
|
|
const wchar_t* DocumentModel::DefaultStyleName = L"#Default";
|
|
const wchar_t* DocumentModel::SelectionStyleName = L"#Selection";
|
|
const wchar_t* DocumentModel::ContextStyleName = L"#Context";
|
|
const wchar_t* DocumentModel::NormalLinkStyleName = L"#NormalLink";
|
|
const wchar_t* DocumentModel::ActiveLinkStyleName = L"#ActiveLink";
|
|
|
|
DocumentModel::DocumentModel()
|
|
{
|
|
{
|
|
FontProperties font=GetCurrentController()->ResourceService()->GetDefaultFont();
|
|
Ptr<DocumentStyleProperties> sp=new DocumentStyleProperties;
|
|
sp->face=font.fontFamily;
|
|
sp->size=DocumentFontSize((double)font.size, false);
|
|
sp->color=Color();
|
|
sp->backgroundColor=Color(0, 0, 0, 0);
|
|
sp->bold=font.bold;
|
|
sp->italic=font.italic;
|
|
sp->underline=font.underline;
|
|
sp->strikeline=font.strikeline;
|
|
sp->antialias=font.antialias;
|
|
sp->verticalAntialias=font.verticalAntialias;
|
|
|
|
Ptr<DocumentStyle> style=new DocumentStyle;
|
|
style->styles=sp;
|
|
styles.Add(L"#Default", style);
|
|
}
|
|
{
|
|
Ptr<DocumentStyleProperties> sp=new DocumentStyleProperties;
|
|
sp->color=Color(255, 255, 255);
|
|
sp->backgroundColor=Color(51, 153, 255);
|
|
|
|
Ptr<DocumentStyle> style=new DocumentStyle;
|
|
style->styles=sp;
|
|
styles.Add(L"#Selection", style);
|
|
}
|
|
{
|
|
Ptr<DocumentStyleProperties> sp=new DocumentStyleProperties;
|
|
|
|
Ptr<DocumentStyle> style=new DocumentStyle;
|
|
style->styles=sp;
|
|
styles.Add(L"#Context", style);
|
|
}
|
|
{
|
|
Ptr<DocumentStyleProperties> sp=new DocumentStyleProperties;
|
|
sp->color=Color(0, 0, 255);
|
|
sp->underline=true;
|
|
|
|
Ptr<DocumentStyle> style=new DocumentStyle;
|
|
style->parentStyleName=L"#Context";
|
|
style->styles=sp;
|
|
styles.Add(L"#NormalLink", style);
|
|
}
|
|
{
|
|
Ptr<DocumentStyleProperties> sp=new DocumentStyleProperties;
|
|
sp->color=Color(255, 128, 0);
|
|
sp->underline=true;
|
|
|
|
Ptr<DocumentStyle> style=new DocumentStyle;
|
|
style->parentStyleName=L"#Context";
|
|
style->styles=sp;
|
|
styles.Add(L"#ActiveLink", style);
|
|
}
|
|
}
|
|
|
|
void DocumentModel::MergeStyle(Ptr<DocumentStyleProperties> style, Ptr<DocumentStyleProperties> parent)
|
|
{
|
|
if(!style->face && parent->face) style->face =parent->face;
|
|
if(!style->size && parent->size) style->size =parent->size;
|
|
if(!style->color && parent->color) style->color =parent->color;
|
|
if(!style->backgroundColor && parent->backgroundColor) style->backgroundColor =parent->backgroundColor;
|
|
if(!style->bold && parent->bold) style->bold =parent->bold;
|
|
if(!style->italic && parent->italic) style->italic =parent->italic;
|
|
if(!style->underline && parent->underline) style->underline =parent->underline;
|
|
if(!style->strikeline && parent->strikeline) style->strikeline =parent->strikeline;
|
|
if(!style->antialias && parent->antialias) style->antialias =parent->antialias;
|
|
if(!style->verticalAntialias && parent->verticalAntialias) style->verticalAntialias =parent->verticalAntialias;
|
|
}
|
|
|
|
void DocumentModel::MergeBaselineStyle(Ptr<DocumentStyleProperties> style, const WString& styleName)
|
|
{
|
|
auto indexDst = styles.Keys().IndexOf(styleName);
|
|
Ptr<DocumentStyleProperties> sp = new DocumentStyleProperties;
|
|
MergeStyle(sp, style);
|
|
if (indexDst != -1)
|
|
{
|
|
MergeStyle(sp, styles.Values()[indexDst]->styles);
|
|
}
|
|
|
|
if (indexDst == -1)
|
|
{
|
|
auto style = new DocumentStyle;
|
|
style->styles = sp;
|
|
styles.Add(styleName, style);
|
|
}
|
|
else
|
|
{
|
|
styles.Values()[indexDst]->styles = sp;
|
|
}
|
|
|
|
FOREACH(Ptr<DocumentStyle>, style, styles.Values())
|
|
{
|
|
style->resolvedStyles = nullptr;
|
|
}
|
|
}
|
|
|
|
void DocumentModel::MergeBaselineStyle(Ptr<DocumentModel> baselineDocument, const WString& styleName)
|
|
{
|
|
auto indexSrc = baselineDocument->styles.Keys().IndexOf(styleName + L"-Override");
|
|
if (indexSrc == -1)
|
|
{
|
|
return;
|
|
}
|
|
|
|
auto csp = baselineDocument->styles.Values()[indexSrc]->styles;
|
|
MergeBaselineStyle(csp, styleName);
|
|
}
|
|
|
|
void DocumentModel::MergeBaselineStyles(Ptr<DocumentModel> baselineDocument)
|
|
{
|
|
MergeBaselineStyle(baselineDocument, DefaultStyleName);
|
|
MergeBaselineStyle(baselineDocument, SelectionStyleName);
|
|
MergeBaselineStyle(baselineDocument, ContextStyleName);
|
|
MergeBaselineStyle(baselineDocument, NormalLinkStyleName);
|
|
MergeBaselineStyle(baselineDocument, ActiveLinkStyleName);
|
|
}
|
|
|
|
void DocumentModel::MergeDefaultFont(const FontProperties& defaultFont)
|
|
{
|
|
Ptr<DocumentStyleProperties> style = new DocumentStyleProperties;
|
|
|
|
style->face =defaultFont.fontFamily;
|
|
style->size =DocumentFontSize((double)defaultFont.size, false);
|
|
style->bold =defaultFont.bold;
|
|
style->italic =defaultFont.italic;
|
|
style->underline =defaultFont.underline;
|
|
style->strikeline =defaultFont.strikeline;
|
|
style->antialias =defaultFont.antialias;
|
|
style->verticalAntialias =defaultFont.verticalAntialias;
|
|
|
|
MergeBaselineStyle(style, DefaultStyleName);
|
|
}
|
|
|
|
DocumentModel::ResolvedStyle DocumentModel::GetStyle(Ptr<DocumentStyleProperties> sp, const ResolvedStyle& context)
|
|
{
|
|
FontProperties font;
|
|
font.fontFamily =sp->face ?sp->face.Value() :context.style.fontFamily;
|
|
font.bold =sp->bold ?sp->bold.Value() :context.style.bold;
|
|
font.italic =sp->italic ?sp->italic.Value() :context.style.italic;
|
|
font.underline =sp->underline ?sp->underline.Value() :context.style.underline;
|
|
font.strikeline =sp->strikeline ?sp->strikeline.Value() :context.style.strikeline;
|
|
font.antialias =sp->antialias ?sp->antialias.Value() :context.style.antialias;
|
|
font.verticalAntialias =sp->verticalAntialias ?sp->verticalAntialias.Value() :context.style.verticalAntialias;
|
|
Color color =sp->color ?sp->color.Value() :context.color;
|
|
Color backgroundColor =sp->backgroundColor ?sp->backgroundColor.Value() :context.backgroundColor;
|
|
|
|
if (sp->size)
|
|
{
|
|
font.size = (vint)(sp->size.Value().relative ? context.style.size * sp->size.Value().size : sp->size.Value().size);
|
|
}
|
|
else
|
|
{
|
|
font.size = context.style.size;
|
|
}
|
|
return ResolvedStyle(font, color, backgroundColor);
|
|
}
|
|
|
|
DocumentModel::ResolvedStyle DocumentModel::GetStyle(const WString& styleName, const ResolvedStyle& context)
|
|
{
|
|
Ptr<DocumentStyle> selectedStyle;
|
|
{
|
|
vint index=styles.Keys().IndexOf(styleName);
|
|
if(index!=-1)
|
|
{
|
|
selectedStyle=styles.Values()[index];
|
|
}
|
|
else
|
|
{
|
|
selectedStyle=styles[L"#Default"];
|
|
}
|
|
}
|
|
|
|
if(!selectedStyle->resolvedStyles)
|
|
{
|
|
Ptr<DocumentStyleProperties> sp = new DocumentStyleProperties;
|
|
selectedStyle->resolvedStyles = sp;
|
|
|
|
Ptr<DocumentStyle> currentStyle;
|
|
WString currentName = styleName;
|
|
while(true)
|
|
{
|
|
vint index = styles.Keys().IndexOf(currentName);
|
|
if (index == -1) break;
|
|
currentStyle = styles.Values().Get(index);
|
|
currentName = currentStyle->parentStyleName;
|
|
MergeStyle(sp, currentStyle->styles);
|
|
}
|
|
}
|
|
|
|
Ptr<DocumentStyleProperties> sp=selectedStyle->resolvedStyles;
|
|
return GetStyle(sp, context);
|
|
}
|
|
|
|
WString DocumentModel::GetText(bool skipNonTextContent)
|
|
{
|
|
return GenerateToStream([&](StreamWriter& writer)
|
|
{
|
|
GetText(writer, skipNonTextContent);
|
|
});
|
|
}
|
|
|
|
void DocumentModel::GetText(stream::TextWriter& writer, bool skipNonTextContent)
|
|
{
|
|
for(vint i=0;i<paragraphs.Count();i++)
|
|
{
|
|
Ptr<DocumentParagraphRun> paragraph=paragraphs[i];
|
|
paragraph->GetText(writer, skipNonTextContent);
|
|
if(i<paragraphs.Count()-1)
|
|
{
|
|
writer.WriteString(L"\r\n\r\n");
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\RESOURCES\GUIDOCUMENTCLIPBOARD_DOCUMENT.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
using namespace collections;
|
|
using namespace parsing::xml;
|
|
using namespace stream;
|
|
|
|
namespace document_clipboard_visitors
|
|
{
|
|
class TraverseDocumentVisitor : public Object, public DocumentRun::IVisitor
|
|
{
|
|
public:
|
|
TraverseDocumentVisitor()
|
|
{
|
|
}
|
|
|
|
virtual void VisitContainer(DocumentContainerRun* run)
|
|
{
|
|
FOREACH(Ptr<DocumentRun>, childRun, run->runs)
|
|
{
|
|
childRun->Accept(this);
|
|
}
|
|
}
|
|
|
|
void Visit(DocumentTextRun* run)override
|
|
{
|
|
}
|
|
|
|
void Visit(DocumentStylePropertiesRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
|
|
void Visit(DocumentStyleApplicationRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
|
|
void Visit(DocumentHyperlinkRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
|
|
void Visit(DocumentImageRun* run)override
|
|
{
|
|
}
|
|
|
|
void Visit(DocumentEmbeddedObjectRun* run)override
|
|
{
|
|
}
|
|
|
|
void Visit(DocumentParagraphRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
};
|
|
|
|
class ModifyDocumentForClipboardVisitor : public TraverseDocumentVisitor
|
|
{
|
|
public:
|
|
ModifyDocumentForClipboardVisitor()
|
|
{
|
|
}
|
|
|
|
void VisitContainer(DocumentContainerRun* run)override
|
|
{
|
|
for (vint i = run->runs.Count() - 1; i >= 0; i--)
|
|
{
|
|
auto childRun = run->runs[i];
|
|
if (childRun.Cast<DocumentEmbeddedObjectRun>())
|
|
{
|
|
run->runs.RemoveAt(i);
|
|
}
|
|
}
|
|
TraverseDocumentVisitor::VisitContainer(run);
|
|
}
|
|
};
|
|
|
|
class CollectImageRunsVisitor : public TraverseDocumentVisitor
|
|
{
|
|
public:
|
|
List<Ptr<DocumentImageRun>> imageRuns;
|
|
|
|
CollectImageRunsVisitor()
|
|
{
|
|
}
|
|
|
|
void Visit(DocumentImageRun* run)override
|
|
{
|
|
run->source = L"res://Image_" + itow(imageRuns.Count());
|
|
imageRuns.Add(run);
|
|
}
|
|
};
|
|
}
|
|
using namespace document_clipboard_visitors;
|
|
|
|
void ModifyDocumentForClipboard(Ptr<DocumentModel> model)
|
|
{
|
|
ModifyDocumentForClipboardVisitor visitor;
|
|
FOREACH(Ptr<DocumentParagraphRun>, paragraph, model->paragraphs)
|
|
{
|
|
paragraph->Accept(&visitor);
|
|
}
|
|
}
|
|
|
|
Ptr<DocumentModel> LoadDocumentFromClipboardStream(stream::IStream& clipboardStream)
|
|
{
|
|
auto tempResource = MakePtr<GuiResource>();
|
|
auto tempResourceItem = MakePtr<GuiResourceItem>();
|
|
tempResource->AddItem(L"Document", tempResourceItem);
|
|
auto tempResolver = MakePtr<GuiResourcePathResolver>(tempResource, L"");
|
|
|
|
internal::ContextFreeReader reader(clipboardStream);
|
|
{
|
|
WString title;
|
|
vint32_t version = 0;
|
|
reader << title << version;
|
|
|
|
if (title != L"WCF_Document" || version < 1)
|
|
{
|
|
return nullptr;
|
|
}
|
|
}
|
|
|
|
WString xmlText;
|
|
reader << xmlText;
|
|
List<GuiResourceError> errors;
|
|
auto parser = GetParserManager()->GetParser<XmlDocument>(L"XML");
|
|
auto xml = parser->Parse({}, xmlText, errors);
|
|
if (errors.Count() > 0) return nullptr;
|
|
|
|
{
|
|
vint32_t count = 0;
|
|
reader << count;
|
|
for (vint i = 0; i < count; i++)
|
|
{
|
|
MemoryStream memoryStream;
|
|
reader << (IStream&)memoryStream;
|
|
if (auto image = GetCurrentController()->ImageService()->CreateImageFromStream(memoryStream))
|
|
{
|
|
auto imageItem = MakePtr<GuiResourceItem>();
|
|
imageItem->SetContent(L"Image", MakePtr<GuiImageData>(image, 0));
|
|
tempResource->AddItem(L"Image_" + itow(i), imageItem);
|
|
}
|
|
}
|
|
}
|
|
|
|
auto document = DocumentModel::LoadFromXml(tempResourceItem, xml, tempResolver, errors);
|
|
return document;
|
|
}
|
|
|
|
void SaveDocumentToClipboardStream(Ptr<DocumentModel> model, stream::IStream& clipboardStream)
|
|
{
|
|
CollectImageRunsVisitor visitor;
|
|
FOREACH(Ptr<DocumentParagraphRun>, paragraph, model->paragraphs)
|
|
{
|
|
paragraph->Accept(&visitor);
|
|
}
|
|
|
|
internal::ContextFreeWriter writer(clipboardStream);
|
|
{
|
|
WString title = L"WCF_Document";
|
|
vint32_t version = 1;
|
|
writer << title << version;
|
|
}
|
|
{
|
|
auto xmlText = GenerateToStream([&](StreamWriter& streamWriter)
|
|
{
|
|
auto xml = model->SaveToXml();
|
|
XmlPrint(xml, streamWriter);
|
|
});
|
|
writer << xmlText;
|
|
}
|
|
{
|
|
vint32_t count = (vint32_t)visitor.imageRuns.Count();
|
|
writer << count;
|
|
|
|
FOREACH(Ptr<DocumentImageRun>, imageRun, visitor.imageRuns)
|
|
{
|
|
MemoryStream memoryStream;
|
|
if (imageRun->image)
|
|
{
|
|
auto format = imageRun->image->GetFormat();
|
|
if (format == INativeImage::Gif)
|
|
{
|
|
format = INativeImage::Png;
|
|
}
|
|
|
|
imageRun->image->SaveToStream(memoryStream, format);
|
|
}
|
|
|
|
writer << (stream::IStream&)memoryStream;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\RESOURCES\GUIDOCUMENTCLIPBOARD_HTMLFORMAT.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
using namespace collections;
|
|
using namespace stream;
|
|
|
|
namespace document_clipboard_visitors
|
|
{
|
|
class GenerateHtmlVisitor : public Object, public DocumentRun::IVisitor
|
|
{
|
|
typedef DocumentModel::ResolvedStyle ResolvedStyle;
|
|
public:
|
|
List<ResolvedStyle> styles;
|
|
DocumentModel* model;
|
|
StreamWriter& writer;
|
|
|
|
GenerateHtmlVisitor(DocumentModel* _model, StreamWriter& _writer)
|
|
:model(_model)
|
|
, writer(_writer)
|
|
{
|
|
ResolvedStyle style;
|
|
style.color = Color(0, 0, 0, 0);
|
|
style.backgroundColor = Color(0, 0, 0, 0);
|
|
style = model->GetStyle(DocumentModel::DefaultStyleName, style);
|
|
styles.Add(style);
|
|
}
|
|
|
|
void VisitContainer(DocumentContainerRun* run)
|
|
{
|
|
FOREACH(Ptr<DocumentRun>, subRun, run->runs)
|
|
{
|
|
subRun->Accept(this);
|
|
}
|
|
}
|
|
|
|
WString ColorToString(Color c)
|
|
{
|
|
auto result = c.ToString();
|
|
if (result.Length() == 9) result = result.Left(7);
|
|
return result;
|
|
}
|
|
|
|
void Visit(DocumentTextRun* run)override
|
|
{
|
|
WString text = run->GetRepresentationText();
|
|
if (text.Length() > 0)
|
|
{
|
|
ResolvedStyle style = styles[styles.Count() - 1];
|
|
|
|
writer.WriteString(L"<span style=\"");
|
|
if (style.style.bold) writer.WriteString(L"font-weight:bold; ");
|
|
if (style.style.italic) writer.WriteString(L"font-style:italic; ");
|
|
if (style.style.underline && style.style.strikeline) writer.WriteString(L"text-decoration:underline line-through; ");
|
|
else if (style.style.underline) writer.WriteString(L"text-decoration:underline; ");
|
|
else if (style.style.strikeline) writer.WriteString(L"text-decoration:line-through; ");
|
|
if (style.style.fontFamily != L"") writer.WriteString(L"font-family:" + style.style.fontFamily + L"; ");
|
|
if (style.style.size != 0) writer.WriteString(L"font-size:" + itow(style.style.size) + L"px; ");
|
|
if (style.color.a != 0) writer.WriteString(L"color:" + ColorToString(style.color) + L"; ");
|
|
if (style.backgroundColor.a != 0)writer.WriteString(L"background-color:" + ColorToString(style.backgroundColor) + L"; ");
|
|
writer.WriteString(L"\">");
|
|
|
|
for (vint i = 0; i < text.Length(); i++)
|
|
{
|
|
switch (wchar_t c = text[i])
|
|
{
|
|
case L'&': writer.WriteString(L"&"); break;
|
|
case L'<': writer.WriteString(L"<"); break;
|
|
case L'>': writer.WriteString(L">"); break;
|
|
case L'\r': break;
|
|
case L'\n': writer.WriteString(L"<br>"); break;
|
|
case L' ': writer.WriteString(L" "); break;
|
|
case L'\t': writer.WriteString(L"<pre>\t</pre>"); break;
|
|
default: writer.WriteChar(c); break;
|
|
}
|
|
}
|
|
|
|
writer.WriteString(L"</span>");
|
|
}
|
|
}
|
|
|
|
void Visit(DocumentStylePropertiesRun* run)override
|
|
{
|
|
ResolvedStyle style = styles[styles.Count() - 1];
|
|
style = model->GetStyle(run->style, style);
|
|
styles.Add(style);
|
|
VisitContainer(run);
|
|
styles.RemoveAt(styles.Count() - 1);
|
|
}
|
|
|
|
void Visit(DocumentStyleApplicationRun* run)override
|
|
{
|
|
ResolvedStyle style = styles[styles.Count() - 1];
|
|
style = model->GetStyle(run->styleName, style);
|
|
styles.Add(style);
|
|
VisitContainer(run);
|
|
styles.RemoveAt(styles.Count() - 1);
|
|
}
|
|
|
|
void Visit(DocumentHyperlinkRun* run)override
|
|
{
|
|
writer.WriteString(L"<a href=\"");
|
|
for (vint i = 0; i < run->reference.Length(); i++)
|
|
{
|
|
switch (wchar_t c = run->reference[i])
|
|
{
|
|
case L'&': writer.WriteString(L"&"); break;
|
|
case L'<': writer.WriteString(L"<"); break;
|
|
case L'>': writer.WriteString(L">"); break;
|
|
case L'"': writer.WriteString(L"""); break;
|
|
case L'\'': writer.WriteString(L"'"); break;
|
|
default: writer.WriteChar(c); break;
|
|
}
|
|
}
|
|
writer.WriteString(L"\">");
|
|
|
|
ResolvedStyle style = styles[styles.Count() - 1];
|
|
style = model->GetStyle((run->normalStyleName == L"" ? DocumentModel::NormalLinkStyleName : run->normalStyleName), style);
|
|
styles.Add(style);
|
|
VisitContainer(run);
|
|
styles.RemoveAt(styles.Count() - 1);
|
|
|
|
writer.WriteString(L"</a>");
|
|
}
|
|
|
|
void Visit(DocumentImageRun* run)override
|
|
{
|
|
if (run->image)
|
|
{
|
|
writer.WriteString(L"<img width=\"" + itow(run->size.x) + L"\" height=\"" + itow(run->size.y) + L"\" src=\"data:image/");
|
|
auto format = run->image->GetFormat();
|
|
if (format == INativeImage::Gif)
|
|
{
|
|
format = INativeImage::Png;
|
|
}
|
|
|
|
switch (format)
|
|
{
|
|
case INativeImage::Bmp: writer.WriteString(L"bmp;base64,"); break;
|
|
case INativeImage::Gif: writer.WriteString(L"gif;base64,"); break;
|
|
case INativeImage::Icon: writer.WriteString(L"icon;base64,"); break;
|
|
case INativeImage::Jpeg: writer.WriteString(L"jpeg;base64,"); break;
|
|
case INativeImage::Png: writer.WriteString(L"png;base64,"); break;
|
|
case INativeImage::Tiff: writer.WriteString(L"tiff;base64,"); break;
|
|
case INativeImage::Wmp: writer.WriteString(L"wmp;base64,"); break;
|
|
default: writer.WriteString(L"unsupported;base64,\"/>"); return;
|
|
}
|
|
|
|
MemoryStream memoryStream;
|
|
run->image->SaveToStream(memoryStream, format);
|
|
memoryStream.SeekFromBegin(0);
|
|
while (true)
|
|
{
|
|
vuint8_t bytes[3] = { 0,0,0 };
|
|
vint read = memoryStream.Read(&bytes, sizeof(bytes));
|
|
if (read == 0) break;
|
|
|
|
vuint8_t b1 = bytes[0] / (1 << 2);
|
|
vuint8_t b2 = ((bytes[0] % (1 << 2)) << 4) + bytes[1] / (1 << 4);
|
|
vuint8_t b3 = ((bytes[1] % (1 << 4)) << 2) + bytes[2] / (1 << 6);
|
|
vuint8_t b4 = bytes[2] % (1 << 6);
|
|
|
|
const wchar_t* BASE64 =
|
|
L"ABCDEFG"
|
|
L"HIJKLMN"
|
|
L"OPQRST"
|
|
L"UVWXYZ"
|
|
L"abcdefg"
|
|
L"hijklmn"
|
|
L"opqrst"
|
|
L"uvwxyz"
|
|
L"0123456789"
|
|
L"+/";
|
|
#define BASE64_CHAR(b) BASE64[b]
|
|
switch (read)
|
|
{
|
|
case 1:
|
|
writer.WriteChar(BASE64_CHAR(b1));
|
|
writer.WriteChar(BASE64_CHAR(b2));
|
|
writer.WriteChar(L'=');
|
|
writer.WriteChar(L'=');
|
|
break;
|
|
case 2:
|
|
writer.WriteChar(BASE64_CHAR(b1));
|
|
writer.WriteChar(BASE64_CHAR(b2));
|
|
writer.WriteChar(BASE64_CHAR(b3));
|
|
writer.WriteChar(L'=');
|
|
break;
|
|
case 3:
|
|
writer.WriteChar(BASE64_CHAR(b1));
|
|
writer.WriteChar(BASE64_CHAR(b2));
|
|
writer.WriteChar(BASE64_CHAR(b3));
|
|
writer.WriteChar(BASE64_CHAR(b4));
|
|
break;
|
|
}
|
|
#undef BASE64_CHAR
|
|
}
|
|
|
|
writer.WriteString(L"\"/>");
|
|
}
|
|
}
|
|
|
|
void Visit(DocumentEmbeddedObjectRun* run)override
|
|
{
|
|
}
|
|
|
|
void Visit(DocumentParagraphRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
};
|
|
}
|
|
using namespace document_clipboard_visitors;
|
|
|
|
#define HTML_LINE(LINE) LINE "\r\n"
|
|
|
|
void SaveDocumentToHtmlUtf8(Ptr<DocumentModel> model, AString& header, AString& content, AString& footer)
|
|
{
|
|
header =
|
|
HTML_LINE("<!DOCTYPE html>")
|
|
HTML_LINE("<html>")
|
|
HTML_LINE("<header>")
|
|
HTML_LINE("<title>GacUI Document 1.0</title>")
|
|
HTML_LINE("<meta charset=\"utf-8\"/>")
|
|
HTML_LINE("</header>")
|
|
HTML_LINE("<body>")
|
|
;
|
|
|
|
MemoryStream memoryStream;
|
|
{
|
|
Utf8Encoder encoder;
|
|
EncoderStream encoderStream(memoryStream, encoder);
|
|
StreamWriter writer(encoderStream);
|
|
GenerateHtmlVisitor visitor(model.Obj(), writer);
|
|
|
|
FOREACH(Ptr<DocumentParagraphRun>, paragraph, model->paragraphs)
|
|
{
|
|
writer.WriteString(L"<p style=\"text-align:");
|
|
if (paragraph->alignment)
|
|
{
|
|
switch (paragraph->alignment.Value())
|
|
{
|
|
case Alignment::Left: writer.WriteString(L"left;"); break;
|
|
case Alignment::Center: writer.WriteString(L"center;"); break;
|
|
case Alignment::Right: writer.WriteString(L"right;"); break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
writer.WriteString(L"left;");
|
|
}
|
|
writer.WriteString(L"\">");
|
|
paragraph->Accept(&visitor);
|
|
writer.WriteString(L"</p>\r\n");
|
|
}
|
|
}
|
|
char zero = 0;
|
|
memoryStream.Write(&zero, sizeof(zero));
|
|
content = (const char*)memoryStream.GetInternalBuffer();
|
|
|
|
footer =
|
|
HTML_LINE("</body>")
|
|
HTML_LINE("</html>")
|
|
;
|
|
}
|
|
|
|
void SaveDocumentToHtmlClipboardStream(Ptr<DocumentModel> model, stream::IStream& clipboardStream)
|
|
{
|
|
AString header, content, footer;
|
|
SaveDocumentToHtmlUtf8(model, header, content, footer);
|
|
|
|
char clipboardHeader[] =
|
|
HTML_LINE("StartHTML:-1")
|
|
HTML_LINE("EndHTML:-1")
|
|
HTML_LINE("StartFragment:0000000000")
|
|
HTML_LINE("EndFragment:0000000000")
|
|
;
|
|
char commentStart[] = "<!--StartFragment-->";
|
|
char commentEnd[] = "<!--EndFragment-->";
|
|
vint offsetStart = sizeof(clipboardHeader) - 1 + header.Length() + sizeof(commentStart) - 1;
|
|
vint offsetEnd = offsetStart + content.Length();
|
|
|
|
AString offsetStartString = itoa(offsetStart);
|
|
AString offsetEndString = itoa(offsetEnd);
|
|
memcpy(strstr(clipboardHeader, "EndFragment:") - offsetStartString.Length() - 2, offsetStartString.Buffer(), offsetStartString.Length());
|
|
memcpy(clipboardHeader + sizeof(clipboardHeader) - 1 - offsetEndString.Length() - 2, offsetEndString.Buffer(), offsetEndString.Length());
|
|
|
|
clipboardStream.Write(clipboardHeader, sizeof(clipboardHeader) - 1);
|
|
if (header.Length() > 0) clipboardStream.Write((void*)header.Buffer(), header.Length());
|
|
clipboardStream.Write(commentStart, sizeof(commentStart) - 1);
|
|
if (content.Length() > 0) clipboardStream.Write((void*)content.Buffer(), content.Length());
|
|
clipboardStream.Write(commentEnd, sizeof(commentEnd) - 1);
|
|
if (footer.Length() > 0) clipboardStream.Write((void*)footer.Buffer(), footer.Length());
|
|
}
|
|
|
|
#undef HTML_LINE
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\RESOURCES\GUIDOCUMENTCLIPBOARD_RICHTEXTFORMAT.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
using namespace collections;
|
|
using namespace stream;
|
|
|
|
namespace document_clipboard_visitors
|
|
{
|
|
class GenerateRtfVisitor : public Object, public DocumentRun::IVisitor
|
|
{
|
|
typedef DocumentModel::ResolvedStyle ResolvedStyle;
|
|
public:
|
|
List<ResolvedStyle> styles;
|
|
DocumentModel* model;
|
|
StreamWriter& writer;
|
|
|
|
List<WString>& fontTable;
|
|
List<Color>& colorTable;
|
|
Dictionary<WString, vint> fontIndex;
|
|
Dictionary<Color, vint> colorIndex;
|
|
|
|
vint GetFont(const WString& fontName)
|
|
{
|
|
vint index = fontIndex.Keys().IndexOf(fontName);
|
|
if (index == -1)
|
|
{
|
|
index = fontTable.Add(fontName);
|
|
fontIndex.Add(fontName, index);
|
|
return index;
|
|
}
|
|
return fontIndex.Values()[index];
|
|
}
|
|
|
|
vint GetColor(Color color)
|
|
{
|
|
if (color.a == 0) return 0;
|
|
vint index = colorIndex.Keys().IndexOf(color);
|
|
if (index == -1)
|
|
{
|
|
index = colorTable.Add(color) + 1;
|
|
colorIndex.Add(color, index);
|
|
return index;
|
|
}
|
|
return colorIndex.Values()[index];
|
|
}
|
|
|
|
GenerateRtfVisitor(DocumentModel* _model, List<WString>& _fontTable, List<Color>& _colorTable, StreamWriter& _writer)
|
|
:model(_model)
|
|
, writer(_writer)
|
|
, fontTable(_fontTable)
|
|
, colorTable(_colorTable)
|
|
{
|
|
ResolvedStyle style;
|
|
style.color = Color(0, 0, 0, 0);
|
|
style.backgroundColor = Color(0, 0, 0, 0);
|
|
style = model->GetStyle(DocumentModel::DefaultStyleName, style);
|
|
styles.Add(style);
|
|
}
|
|
|
|
void VisitContainer(DocumentContainerRun* run)
|
|
{
|
|
FOREACH(Ptr<DocumentRun>, subRun, run->runs)
|
|
{
|
|
subRun->Accept(this);
|
|
}
|
|
}
|
|
|
|
void Visit(DocumentTextRun* run)override
|
|
{
|
|
WString text = run->GetRepresentationText();
|
|
if (text.Length() > 0)
|
|
{
|
|
ResolvedStyle style = styles[styles.Count() - 1];
|
|
|
|
writer.WriteString(L"{\\f" + itow(GetFont(style.style.fontFamily)));
|
|
writer.WriteString(L"\\fs" + itow((vint)(style.style.size * 1.5)));
|
|
writer.WriteString(L"\\cf" + itow(GetColor(style.color)));
|
|
writer.WriteString(L"\\cb" + itow(GetColor(style.backgroundColor)));
|
|
writer.WriteString(L"\\chshdng" + itow(GetColor(style.backgroundColor)));
|
|
writer.WriteString(L"\\chcbpat" + itow(GetColor(style.backgroundColor)));
|
|
|
|
if (style.style.bold) writer.WriteString(L"\\b");
|
|
if (style.style.italic) writer.WriteString(L"\\i");
|
|
if (style.style.underline) writer.WriteString(L"\\ul");
|
|
if (style.style.strikeline) writer.WriteString(L"\\strike");
|
|
|
|
for (vint i = 0; i < text.Length(); i++)
|
|
{
|
|
writer.WriteString(L"\\u" + itow(text[i]));
|
|
}
|
|
|
|
writer.WriteString(L"}");
|
|
}
|
|
}
|
|
|
|
void Visit(DocumentStylePropertiesRun* run)override
|
|
{
|
|
ResolvedStyle style = styles[styles.Count() - 1];
|
|
style = model->GetStyle(run->style, style);
|
|
styles.Add(style);
|
|
VisitContainer(run);
|
|
styles.RemoveAt(styles.Count() - 1);
|
|
}
|
|
|
|
void Visit(DocumentStyleApplicationRun* run)override
|
|
{
|
|
ResolvedStyle style = styles[styles.Count() - 1];
|
|
style = model->GetStyle(run->styleName, style);
|
|
styles.Add(style);
|
|
VisitContainer(run);
|
|
styles.RemoveAt(styles.Count() - 1);
|
|
}
|
|
|
|
void Visit(DocumentHyperlinkRun* run)override
|
|
{
|
|
ResolvedStyle style = styles[styles.Count() - 1];
|
|
style = model->GetStyle((run->normalStyleName == L"" ? DocumentModel::NormalLinkStyleName : run->normalStyleName), style);
|
|
styles.Add(style);
|
|
VisitContainer(run);
|
|
styles.RemoveAt(styles.Count() - 1);
|
|
}
|
|
|
|
void Visit(DocumentImageRun* run)override
|
|
{
|
|
if (run->image)
|
|
{
|
|
writer.WriteString(L"{\\pict\\pngblip");
|
|
writer.WriteString(L"\\picw" + itow(run->size.x) + L"\\pich" + itow(run->size.y));
|
|
writer.WriteString(L"\\picwgoal" + itow(run->size.x * 15) + L"\\pichgoal" + itow(run->size.y * 15) + L" ");
|
|
|
|
MemoryStream memoryStream;
|
|
run->image->SaveToStream(memoryStream, INativeImage::Png);
|
|
vint count = (vint)memoryStream.Size();
|
|
vuint8_t* buffer = (vuint8_t*)memoryStream.GetInternalBuffer();
|
|
for (vint i = 0; i < count; i++)
|
|
{
|
|
writer.WriteChar(L"0123456789abcdef"[buffer[i] / 16]);
|
|
writer.WriteChar(L"0123456789abcdef"[buffer[i] % 16]);
|
|
}
|
|
|
|
writer.WriteString(L"}");
|
|
}
|
|
}
|
|
|
|
void Visit(DocumentEmbeddedObjectRun* run)override
|
|
{
|
|
}
|
|
|
|
void Visit(DocumentParagraphRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
};
|
|
}
|
|
using namespace document_clipboard_visitors;
|
|
|
|
void SaveDocumentToRtf(Ptr<DocumentModel> model, AString& rtf)
|
|
{
|
|
List<WString> fontTable;
|
|
List<Color> colorTable;
|
|
MemoryStream bodyStream;
|
|
{
|
|
StreamWriter writer(bodyStream);
|
|
GenerateRtfVisitor visitor(model.Obj(), fontTable, colorTable, writer);
|
|
|
|
FOREACH(Ptr<DocumentParagraphRun>, paragraph, model->paragraphs)
|
|
{
|
|
if (paragraph->alignment)
|
|
{
|
|
switch (paragraph->alignment.Value())
|
|
{
|
|
case Alignment::Left: writer.WriteString(L"\\ql{"); break;
|
|
case Alignment::Center: writer.WriteString(L"\\qc{"); break;
|
|
case Alignment::Right: writer.WriteString(L"\\qr{"); break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
writer.WriteString(L"\\ql{");
|
|
}
|
|
paragraph->Accept(&visitor);
|
|
writer.WriteString(L"}\\par");
|
|
}
|
|
}
|
|
|
|
MemoryStream rtfStream;
|
|
{
|
|
Utf8Encoder encoder;
|
|
EncoderStream encoderStream(rtfStream, encoder);
|
|
StreamWriter writer(encoderStream);
|
|
|
|
writer.WriteString(L"{\\rtf1\\ansi\\deff0{\\fonttbl");
|
|
FOREACH_INDEXER(WString, fontName, index, fontTable)
|
|
{
|
|
writer.WriteString(L"{\\f");
|
|
writer.WriteString(itow(index));
|
|
writer.WriteString(L" ");
|
|
writer.WriteString(fontName);
|
|
writer.WriteString(L";}");
|
|
}
|
|
|
|
writer.WriteString(L"}{\\colortbl");
|
|
FOREACH_INDEXER(Color, color, index, colorTable)
|
|
{
|
|
writer.WriteString(L";\\red");
|
|
writer.WriteString(itow(color.r));
|
|
writer.WriteString(L"\\green");
|
|
writer.WriteString(itow(color.g));
|
|
writer.WriteString(L"\\blue");
|
|
writer.WriteString(itow(color.b));
|
|
}
|
|
|
|
writer.WriteString(L";}{\\*\\generator GacUI Document 1.0}\\uc0");
|
|
{
|
|
bodyStream.SeekFromBegin(0);
|
|
StreamReader reader(bodyStream);
|
|
writer.WriteString(reader.ReadToEnd());
|
|
}
|
|
writer.WriteString(L"}");
|
|
}
|
|
char zero = 0;
|
|
rtfStream.Write(&zero, sizeof(zero));
|
|
rtf = (const char*)rtfStream.GetInternalBuffer();
|
|
}
|
|
|
|
void SaveDocumentToRtfStream(Ptr<DocumentModel> model, stream::IStream& rtfStream)
|
|
{
|
|
AString rtf;
|
|
SaveDocumentToRtf(model, rtf);
|
|
rtfStream.Write((void*)rtf.Buffer(), rtf.Length());
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\RESOURCES\GUIDOCUMENTEDITOR_ADDCONTAINER.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
using namespace collections;
|
|
using namespace document_editor;
|
|
|
|
/***********************************************************************
|
|
Insert container runs on top of all text ranges that intersect with the specified range
|
|
AddStyleVisitor : Apply a style on the specified range
|
|
AddHyperlinkVisitor : Apply a hyperlink on the specified range
|
|
AddStyleNameVisitor : Apply a style name on the specified range
|
|
***********************************************************************/
|
|
|
|
namespace document_operation_visitors
|
|
{
|
|
class AddContainerVisitor : public Object, public DocumentRun::IVisitor
|
|
{
|
|
public:
|
|
RunRangeMap& runRanges;
|
|
vint start;
|
|
vint end;
|
|
bool insertStyle;
|
|
|
|
virtual Ptr<DocumentContainerRun> CreateContainer() = 0;
|
|
|
|
AddContainerVisitor(RunRangeMap& _runRanges, vint _start, vint _end)
|
|
:runRanges(_runRanges)
|
|
, start(_start)
|
|
, end(_end)
|
|
, insertStyle(false)
|
|
{
|
|
}
|
|
|
|
void VisitContainer(DocumentContainerRun* run)
|
|
{
|
|
for (vint i = run->runs.Count() - 1; i >= 0; i--)
|
|
{
|
|
Ptr<DocumentRun> subRun = run->runs[i];
|
|
RunRange range = runRanges[subRun.Obj()];
|
|
if (range.start<end && start<range.end)
|
|
{
|
|
insertStyle = false;
|
|
subRun->Accept(this);
|
|
if (insertStyle)
|
|
{
|
|
Ptr<DocumentContainerRun> containerRun = CreateContainer();
|
|
run->runs.RemoveAt(i);
|
|
containerRun->runs.Add(subRun);
|
|
run->runs.Insert(i, containerRun);
|
|
}
|
|
}
|
|
}
|
|
insertStyle = false;
|
|
}
|
|
|
|
void Visit(DocumentTextRun* run)override
|
|
{
|
|
insertStyle = true;
|
|
}
|
|
|
|
void Visit(DocumentStylePropertiesRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
|
|
void Visit(DocumentStyleApplicationRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
|
|
void Visit(DocumentHyperlinkRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
|
|
void Visit(DocumentImageRun* run)override
|
|
{
|
|
insertStyle = false;
|
|
}
|
|
|
|
void Visit(DocumentEmbeddedObjectRun* run)override
|
|
{
|
|
insertStyle = false;
|
|
}
|
|
|
|
void Visit(DocumentParagraphRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
};
|
|
|
|
class AddStyleVisitor : public AddContainerVisitor
|
|
{
|
|
public:
|
|
Ptr<DocumentStyleProperties> style;
|
|
|
|
Ptr<DocumentContainerRun> CreateContainer()override
|
|
{
|
|
Ptr<DocumentStylePropertiesRun> containerRun = new DocumentStylePropertiesRun;
|
|
containerRun->style = CopyStyle(style);
|
|
return containerRun;
|
|
}
|
|
|
|
AddStyleVisitor(RunRangeMap& _runRanges, vint _start, vint _end, Ptr<DocumentStyleProperties> _style)
|
|
:AddContainerVisitor(_runRanges, _start, _end)
|
|
, style(_style)
|
|
{
|
|
}
|
|
};
|
|
|
|
class AddHyperlinkVisitor : public AddContainerVisitor
|
|
{
|
|
public:
|
|
WString reference;
|
|
WString normalStyleName;
|
|
WString activeStyleName;
|
|
|
|
Ptr<DocumentContainerRun> CreateContainer()override
|
|
{
|
|
Ptr<DocumentHyperlinkRun> containerRun = new DocumentHyperlinkRun;
|
|
containerRun->reference = reference;
|
|
containerRun->normalStyleName = normalStyleName;
|
|
containerRun->activeStyleName = activeStyleName;
|
|
containerRun->styleName = normalStyleName;
|
|
return containerRun;
|
|
}
|
|
|
|
AddHyperlinkVisitor(RunRangeMap& _runRanges, vint _start, vint _end, const WString& _reference, const WString& _normalStyleName, const WString& _activeStyleName)
|
|
:AddContainerVisitor(_runRanges, _start, _end)
|
|
, reference(_reference)
|
|
, normalStyleName(_normalStyleName)
|
|
, activeStyleName(_activeStyleName)
|
|
{
|
|
}
|
|
};
|
|
|
|
class AddStyleNameVisitor : public AddContainerVisitor
|
|
{
|
|
public:
|
|
WString styleName;
|
|
|
|
Ptr<DocumentContainerRun> CreateContainer()override
|
|
{
|
|
Ptr<DocumentStyleApplicationRun> containerRun = new DocumentStyleApplicationRun;
|
|
containerRun->styleName = styleName;
|
|
return containerRun;
|
|
}
|
|
|
|
AddStyleNameVisitor(RunRangeMap& _runRanges, vint _start, vint _end, const WString& _styleName)
|
|
:AddContainerVisitor(_runRanges, _start, _end)
|
|
, styleName(_styleName)
|
|
{
|
|
}
|
|
};
|
|
}
|
|
using namespace document_operation_visitors;
|
|
|
|
namespace document_editor
|
|
{
|
|
void AddStyle(DocumentParagraphRun* run, RunRangeMap& runRanges, vint start, vint end, Ptr<DocumentStyleProperties> style)
|
|
{
|
|
AddStyleVisitor visitor(runRanges, start, end, style);
|
|
run->Accept(&visitor);
|
|
}
|
|
|
|
void AddHyperlink(DocumentParagraphRun* run, RunRangeMap& runRanges, vint start, vint end, const WString& reference, const WString& normalStyleName, const WString& activeStyleName)
|
|
{
|
|
AddHyperlinkVisitor visitor(runRanges, start, end, reference, normalStyleName, activeStyleName);
|
|
run->Accept(&visitor);
|
|
}
|
|
|
|
void AddStyleName(DocumentParagraphRun* run, RunRangeMap& runRanges, vint start, vint end, const WString& styleName)
|
|
{
|
|
AddStyleNameVisitor visitor(runRanges, start, end, styleName);
|
|
run->Accept(&visitor);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\RESOURCES\GUIDOCUMENTEDITOR_CLEARUNNECESSARYRUN.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
using namespace collections;
|
|
using namespace document_editor;
|
|
|
|
/***********************************************************************
|
|
Clear all runs that have an empty length
|
|
***********************************************************************/
|
|
|
|
namespace document_operation_visitors
|
|
{
|
|
class ClearRunVisitor : public Object, public DocumentRun::IVisitor
|
|
{
|
|
public:
|
|
vint start;
|
|
|
|
ClearRunVisitor()
|
|
:start(0)
|
|
{
|
|
}
|
|
|
|
void VisitContainer(DocumentContainerRun* run)
|
|
{
|
|
for (vint i = run->runs.Count() - 1; i >= 0; i--)
|
|
{
|
|
vint oldStart = start;
|
|
run->runs[i]->Accept(this);
|
|
if (oldStart == start)
|
|
{
|
|
run->runs.RemoveAt(i);
|
|
}
|
|
}
|
|
}
|
|
|
|
void VisitContent(DocumentContentRun* run)
|
|
{
|
|
start += run->GetRepresentationText().Length();
|
|
}
|
|
|
|
void Visit(DocumentTextRun* run)override
|
|
{
|
|
VisitContent(run);
|
|
}
|
|
|
|
void Visit(DocumentStylePropertiesRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
|
|
void Visit(DocumentStyleApplicationRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
|
|
void Visit(DocumentHyperlinkRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
|
|
void Visit(DocumentImageRun* run)override
|
|
{
|
|
VisitContent(run);
|
|
}
|
|
|
|
void Visit(DocumentEmbeddedObjectRun* run)override
|
|
{
|
|
VisitContent(run);
|
|
}
|
|
|
|
void Visit(DocumentParagraphRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
};
|
|
}
|
|
using namespace document_operation_visitors;
|
|
|
|
/***********************************************************************
|
|
Ensure DocumentStylePropertiesRun doesn't have a child which is another DocumentStylePropertiesRun
|
|
Remove DocumentStylePropertiesRun's property if it set a value but doesn't change the output
|
|
Remove DocumentStylePropertiesRun if it is empty or contains no text run
|
|
***********************************************************************/
|
|
|
|
namespace document_operation_visitors
|
|
{
|
|
class CompressStyleRunVisitor : public Object, public DocumentRun::IVisitor
|
|
{
|
|
public:
|
|
DocumentModel* model;
|
|
List<DocumentModel::ResolvedStyle> resolvedStyles;
|
|
List<Ptr<DocumentRun>> replacedRuns;
|
|
|
|
CompressStyleRunVisitor(DocumentModel* _model)
|
|
:model(_model)
|
|
{
|
|
DocumentModel::ResolvedStyle resolvedStyle;
|
|
resolvedStyle = model->GetStyle(DocumentModel::DefaultStyleName, resolvedStyle);
|
|
resolvedStyles.Add(resolvedStyle);
|
|
}
|
|
|
|
const DocumentModel::ResolvedStyle& GetCurrentResolvedStyle()
|
|
{
|
|
return resolvedStyles[resolvedStyles.Count() - 1];
|
|
}
|
|
|
|
void VisitContainer(DocumentContainerRun* run)
|
|
{
|
|
for (vint i = 0; i < run->runs.Count(); i++)
|
|
{
|
|
Ptr<DocumentRun> subRun = run->runs[i];
|
|
replacedRuns.Clear();
|
|
subRun->Accept(this);
|
|
if (replacedRuns.Count() > 0)
|
|
{
|
|
run->runs.RemoveAt(i);
|
|
for (vint j = 0; j < replacedRuns.Count(); j++)
|
|
{
|
|
run->runs.Insert(i + j, replacedRuns[j]);
|
|
}
|
|
i--;
|
|
}
|
|
}
|
|
}
|
|
|
|
void Visit(DocumentTextRun* run)override
|
|
{
|
|
}
|
|
|
|
bool OnlyImageOrObject(DocumentContainerRun* run)
|
|
{
|
|
bool onlyImageOrObject = true;
|
|
FOREACH(Ptr<DocumentRun>, subRun, run->runs)
|
|
{
|
|
if (!subRun.Cast<DocumentImageRun>() && !subRun.Cast<DocumentEmbeddedObjectRun>())
|
|
{
|
|
onlyImageOrObject = false;
|
|
break;
|
|
}
|
|
}
|
|
return onlyImageOrObject;
|
|
}
|
|
|
|
void Visit(DocumentStylePropertiesRun* run)override
|
|
{
|
|
if (OnlyImageOrObject(run))
|
|
{
|
|
CopyFrom(replacedRuns, run->runs);
|
|
return;
|
|
}
|
|
|
|
const DocumentModel::ResolvedStyle& currentResolvedStyle = GetCurrentResolvedStyle();
|
|
DocumentModel::ResolvedStyle resolvedStyle = model->GetStyle(run->style, currentResolvedStyle);
|
|
|
|
if (currentResolvedStyle.style.fontFamily == resolvedStyle.style.fontFamily) run->style->face = Nullable<WString>();
|
|
if (currentResolvedStyle.style.size == resolvedStyle.style.size) run->style->size = Nullable<DocumentFontSize>();
|
|
if (currentResolvedStyle.color == resolvedStyle.color) run->style->color = Nullable<Color>();
|
|
if (currentResolvedStyle.backgroundColor == resolvedStyle.backgroundColor) run->style->backgroundColor = Nullable<Color>();
|
|
if (currentResolvedStyle.style.bold == resolvedStyle.style.bold) run->style->bold = Nullable<bool>();
|
|
if (currentResolvedStyle.style.italic == resolvedStyle.style.italic) run->style->italic = Nullable<bool>();
|
|
if (currentResolvedStyle.style.underline == resolvedStyle.style.underline) run->style->underline = Nullable<bool>();
|
|
if (currentResolvedStyle.style.strikeline == resolvedStyle.style.strikeline) run->style->strikeline = Nullable<bool>();
|
|
if (currentResolvedStyle.style.antialias == resolvedStyle.style.antialias) run->style->antialias = Nullable<bool>();
|
|
if (currentResolvedStyle.style.verticalAntialias == resolvedStyle.style.verticalAntialias) run->style->verticalAntialias = Nullable<bool>();
|
|
|
|
if (run->style->face) goto CONTINUE_PROCESSING;
|
|
if (run->style->size) goto CONTINUE_PROCESSING;
|
|
if (run->style->color) goto CONTINUE_PROCESSING;
|
|
if (run->style->backgroundColor) goto CONTINUE_PROCESSING;
|
|
if (run->style->bold) goto CONTINUE_PROCESSING;
|
|
if (run->style->italic) goto CONTINUE_PROCESSING;
|
|
if (run->style->underline) goto CONTINUE_PROCESSING;
|
|
if (run->style->strikeline) goto CONTINUE_PROCESSING;
|
|
if (run->style->antialias) goto CONTINUE_PROCESSING;
|
|
if (run->style->verticalAntialias) goto CONTINUE_PROCESSING;
|
|
CopyFrom(replacedRuns, run->runs);
|
|
return;
|
|
|
|
CONTINUE_PROCESSING:
|
|
if (From(run->runs).Cast<DocumentStylePropertiesRun>().First(nullptr) != nullptr)
|
|
{
|
|
FOREACH(Ptr<DocumentRun>, subRun, run->runs)
|
|
{
|
|
if (auto styleRun = subRun.Cast<DocumentStylePropertiesRun>())
|
|
{
|
|
DocumentModel::MergeStyle(styleRun->style, run->style);
|
|
replacedRuns.Add(styleRun);
|
|
}
|
|
else
|
|
{
|
|
auto parentRun = CopyRun(run).Cast<DocumentStylePropertiesRun>();
|
|
parentRun->runs.Add(subRun);
|
|
replacedRuns.Add(parentRun);
|
|
}
|
|
}
|
|
return;
|
|
}
|
|
|
|
resolvedStyles.Add(resolvedStyle);
|
|
VisitContainer(run);
|
|
resolvedStyles.RemoveAt(resolvedStyles.Count() - 1);
|
|
}
|
|
|
|
void Visit(DocumentStyleApplicationRun* run)override
|
|
{
|
|
if (OnlyImageOrObject(run))
|
|
{
|
|
CopyFrom(replacedRuns, run->runs);
|
|
return;
|
|
}
|
|
|
|
const DocumentModel::ResolvedStyle& currentResolvedStyle = GetCurrentResolvedStyle();
|
|
DocumentModel::ResolvedStyle resolvedStyle = model->GetStyle(run->styleName, currentResolvedStyle);
|
|
resolvedStyles.Add(resolvedStyle);
|
|
VisitContainer(run);
|
|
resolvedStyles.RemoveAt(resolvedStyles.Count() - 1);
|
|
}
|
|
|
|
void Visit(DocumentHyperlinkRun* run)override
|
|
{
|
|
const DocumentModel::ResolvedStyle& currentResolvedStyle = GetCurrentResolvedStyle();
|
|
DocumentModel::ResolvedStyle resolvedStyle = model->GetStyle(run->styleName, currentResolvedStyle);
|
|
resolvedStyles.Add(resolvedStyle);
|
|
VisitContainer(run);
|
|
resolvedStyles.RemoveAt(resolvedStyles.Count() - 1);
|
|
}
|
|
|
|
void Visit(DocumentImageRun* run)override
|
|
{
|
|
}
|
|
|
|
void Visit(DocumentEmbeddedObjectRun* run)override
|
|
{
|
|
}
|
|
|
|
void Visit(DocumentParagraphRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
};
|
|
}
|
|
using namespace document_operation_visitors;
|
|
|
|
/***********************************************************************
|
|
Merge sibling runs if they are exactly the same
|
|
***********************************************************************/
|
|
|
|
namespace document_operation_visitors
|
|
{
|
|
class MergeSiblingRunVisitor : public Object, public DocumentRun::IVisitor
|
|
{
|
|
public:
|
|
Ptr<DocumentRun> nextRun;
|
|
Ptr<DocumentRun> replacedRun;
|
|
|
|
void Visit(DocumentTextRun* run)override
|
|
{
|
|
if (auto sibilingRun = nextRun.Cast<DocumentTextRun>())
|
|
{
|
|
run->text += sibilingRun->text;
|
|
replacedRun = run;
|
|
}
|
|
}
|
|
|
|
void Visit(DocumentStylePropertiesRun* run)override
|
|
{
|
|
if (auto sibilingRun = nextRun.Cast<DocumentStylePropertiesRun>())
|
|
{
|
|
if (run->style->face != sibilingRun->style->face) return;
|
|
if (run->style->size != sibilingRun->style->size) return;
|
|
if (run->style->color != sibilingRun->style->color) return;
|
|
if (run->style->backgroundColor != sibilingRun->style->backgroundColor) return;
|
|
if (run->style->bold != sibilingRun->style->bold) return;
|
|
if (run->style->italic != sibilingRun->style->italic) return;
|
|
if (run->style->underline != sibilingRun->style->underline) return;
|
|
if (run->style->strikeline != sibilingRun->style->strikeline) return;
|
|
if (run->style->antialias != sibilingRun->style->antialias) return;
|
|
if (run->style->verticalAntialias != sibilingRun->style->verticalAntialias) return;
|
|
|
|
CopyFrom(run->runs, sibilingRun->runs, true);
|
|
replacedRun = run;
|
|
}
|
|
}
|
|
|
|
void Visit(DocumentStyleApplicationRun* run)override
|
|
{
|
|
if (auto sibilingRun = nextRun.Cast<DocumentStyleApplicationRun>())
|
|
{
|
|
if (run->styleName == sibilingRun->styleName)
|
|
{
|
|
CopyFrom(run->runs, sibilingRun->runs, true);
|
|
replacedRun = run;
|
|
}
|
|
}
|
|
}
|
|
|
|
void Visit(DocumentHyperlinkRun* run)override
|
|
{
|
|
if (auto sibilingRun = nextRun.Cast<DocumentHyperlinkRun>())
|
|
{
|
|
if (run->styleName == sibilingRun->styleName &&
|
|
run->normalStyleName == sibilingRun->normalStyleName &&
|
|
run->activeStyleName == sibilingRun->activeStyleName &&
|
|
run->reference == sibilingRun->reference)
|
|
{
|
|
CopyFrom(run->runs, sibilingRun->runs, true);
|
|
replacedRun = run;
|
|
}
|
|
}
|
|
}
|
|
|
|
void Visit(DocumentImageRun* run)override
|
|
{
|
|
}
|
|
|
|
void Visit(DocumentEmbeddedObjectRun* run)override
|
|
{
|
|
}
|
|
|
|
void Visit(DocumentParagraphRun* run)override
|
|
{
|
|
}
|
|
};
|
|
|
|
class MergeSiblingRunRecursivelyVisitor : public Object, public DocumentRun::IVisitor
|
|
{
|
|
public:
|
|
Ptr<DocumentRun> replacedRun;
|
|
Ptr<DocumentRun> nextRun;
|
|
|
|
void VisitContainer(DocumentContainerRun* run)
|
|
{
|
|
for (vint i = 0; i < run->runs.Count() - 1; i++)
|
|
{
|
|
auto currentRun = run->runs[i];
|
|
auto nextRun = run->runs[i + 1];
|
|
|
|
MergeSiblingRunVisitor visitor;
|
|
visitor.nextRun = nextRun;
|
|
currentRun->Accept(&visitor);
|
|
|
|
if (visitor.replacedRun)
|
|
{
|
|
run->runs.RemoveAt(i + 1);
|
|
run->runs[i] = visitor.replacedRun;
|
|
i--;
|
|
}
|
|
}
|
|
|
|
for (vint i = 0; i < run->runs.Count() - 1; i++)
|
|
{
|
|
run->runs[i]->Accept(this);
|
|
}
|
|
}
|
|
|
|
void Visit(DocumentTextRun* run)override
|
|
{
|
|
}
|
|
|
|
void Visit(DocumentStylePropertiesRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
|
|
void Visit(DocumentStyleApplicationRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
|
|
void Visit(DocumentHyperlinkRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
|
|
void Visit(DocumentImageRun* run)override
|
|
{
|
|
}
|
|
|
|
void Visit(DocumentEmbeddedObjectRun* run)override
|
|
{
|
|
}
|
|
|
|
void Visit(DocumentParagraphRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
};
|
|
}
|
|
using namespace document_operation_visitors;
|
|
|
|
namespace document_editor
|
|
{
|
|
void ClearUnnecessaryRun(DocumentParagraphRun* run, DocumentModel* model)
|
|
{
|
|
{
|
|
ClearRunVisitor visitor;
|
|
run->Accept(&visitor);
|
|
}
|
|
{
|
|
CompressStyleRunVisitor visitor(model);
|
|
run->Accept(&visitor);
|
|
}
|
|
{
|
|
MergeSiblingRunRecursivelyVisitor visitor;
|
|
run->Accept(&visitor);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\RESOURCES\GUIDOCUMENTEDITOR_CLONERUN.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
using namespace collections;
|
|
using namespace document_editor;
|
|
|
|
/***********************************************************************
|
|
Clone the current run without its children
|
|
If clonedRun field is assigned then it will be added to the cloned container run
|
|
***********************************************************************/
|
|
|
|
namespace document_operation_visitors
|
|
{
|
|
class CloneRunVisitor : public Object, public DocumentRun::IVisitor
|
|
{
|
|
public:
|
|
Ptr<DocumentRun> clonedRun;
|
|
|
|
CloneRunVisitor(Ptr<DocumentRun> subRun)
|
|
:clonedRun(subRun)
|
|
{
|
|
}
|
|
|
|
void VisitContainer(Ptr<DocumentContainerRun> cloned)
|
|
{
|
|
if (clonedRun)
|
|
{
|
|
cloned->runs.Add(clonedRun);
|
|
}
|
|
clonedRun = cloned;
|
|
}
|
|
|
|
void Visit(DocumentTextRun* run)override
|
|
{
|
|
Ptr<DocumentTextRun> cloned = new DocumentTextRun;
|
|
cloned->text = run->text;
|
|
clonedRun = cloned;
|
|
}
|
|
|
|
void Visit(DocumentStylePropertiesRun* run)override
|
|
{
|
|
Ptr<DocumentStylePropertiesRun> cloned = new DocumentStylePropertiesRun;
|
|
cloned->style = CopyStyle(run->style);
|
|
VisitContainer(cloned);
|
|
}
|
|
|
|
void Visit(DocumentStyleApplicationRun* run)override
|
|
{
|
|
Ptr<DocumentStyleApplicationRun> cloned = new DocumentStyleApplicationRun;
|
|
cloned->styleName = run->styleName;
|
|
|
|
VisitContainer(cloned);
|
|
}
|
|
|
|
void Visit(DocumentHyperlinkRun* run)override
|
|
{
|
|
Ptr<DocumentHyperlinkRun> cloned = new DocumentHyperlinkRun;
|
|
cloned->styleName = run->styleName;
|
|
cloned->normalStyleName = run->normalStyleName;
|
|
cloned->activeStyleName = run->activeStyleName;
|
|
cloned->reference = run->reference;
|
|
|
|
VisitContainer(cloned);
|
|
}
|
|
|
|
void Visit(DocumentImageRun* run)override
|
|
{
|
|
Ptr<DocumentImageRun> cloned = new DocumentImageRun;
|
|
cloned->size = run->size;
|
|
cloned->baseline = run->baseline;
|
|
cloned->image = run->image;
|
|
cloned->frameIndex = run->frameIndex;
|
|
cloned->source = run->source;
|
|
clonedRun = cloned;
|
|
}
|
|
|
|
void Visit(DocumentEmbeddedObjectRun* run)override
|
|
{
|
|
Ptr<DocumentEmbeddedObjectRun> cloned = new DocumentEmbeddedObjectRun;
|
|
cloned->name = run->name;
|
|
clonedRun = cloned;
|
|
}
|
|
|
|
void Visit(DocumentParagraphRun* run)override
|
|
{
|
|
Ptr<DocumentParagraphRun> cloned = new DocumentParagraphRun;
|
|
cloned->alignment = run->alignment;
|
|
|
|
VisitContainer(cloned);
|
|
}
|
|
};
|
|
}
|
|
using namespace document_operation_visitors;
|
|
|
|
/***********************************************************************
|
|
Clone the current run with its children
|
|
***********************************************************************/
|
|
|
|
namespace document_operation_visitors
|
|
{
|
|
class CloneRunRecursivelyVisitor : public Object, public DocumentRun::IVisitor
|
|
{
|
|
public:
|
|
Ptr<DocumentRun> clonedRun;
|
|
RunRangeMap& runRanges;
|
|
vint start;
|
|
vint end;
|
|
bool deepCopy;
|
|
|
|
CloneRunRecursivelyVisitor(RunRangeMap& _runRanges, vint _start, vint _end, bool _deepCopy)
|
|
:runRanges(_runRanges)
|
|
, start(_start)
|
|
, end(_end)
|
|
, deepCopy(_deepCopy)
|
|
{
|
|
}
|
|
|
|
void VisitContainer(DocumentContainerRun* run)
|
|
{
|
|
clonedRun = 0;
|
|
RunRange range = runRanges[run];
|
|
if (range.start <= end && start <= range.end)
|
|
{
|
|
if (start <= range.start && range.end <= end && !deepCopy)
|
|
{
|
|
clonedRun = run;
|
|
}
|
|
else
|
|
{
|
|
Ptr<DocumentContainerRun> containerRun = CopyRun(run).Cast<DocumentContainerRun>();
|
|
FOREACH(Ptr<DocumentRun>, subRun, run->runs)
|
|
{
|
|
subRun->Accept(this);
|
|
if (clonedRun)
|
|
{
|
|
containerRun->runs.Add(clonedRun);
|
|
}
|
|
}
|
|
clonedRun = containerRun;
|
|
}
|
|
}
|
|
}
|
|
|
|
void Visit(DocumentTextRun* run)override
|
|
{
|
|
clonedRun = 0;
|
|
RunRange range = runRanges[run];
|
|
if (range.start<end && start<range.end)
|
|
{
|
|
if (start <= range.start && range.end <= end)
|
|
{
|
|
if (deepCopy)
|
|
{
|
|
clonedRun = CopyRun(run);
|
|
}
|
|
else
|
|
{
|
|
clonedRun = run;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
Ptr<DocumentTextRun> textRun = new DocumentTextRun;
|
|
vint copyStart = start>range.start ? start : range.start;
|
|
vint copyEnd = end<range.end ? end : range.end;
|
|
if (copyStart<copyEnd)
|
|
{
|
|
textRun->text = run->text.Sub(copyStart - range.start, copyEnd - copyStart);
|
|
}
|
|
clonedRun = textRun;
|
|
}
|
|
}
|
|
}
|
|
|
|
void Visit(DocumentStylePropertiesRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
|
|
void Visit(DocumentStyleApplicationRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
|
|
void Visit(DocumentHyperlinkRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
|
|
void Visit(DocumentImageRun* run)override
|
|
{
|
|
clonedRun = 0;
|
|
RunRange range = runRanges[run];
|
|
if (range.start<end && start<range.end)
|
|
{
|
|
if (deepCopy)
|
|
{
|
|
clonedRun = CopyRun(run);
|
|
}
|
|
else
|
|
{
|
|
clonedRun = run;
|
|
}
|
|
}
|
|
}
|
|
|
|
void Visit(DocumentEmbeddedObjectRun* run)override
|
|
{
|
|
clonedRun = 0;
|
|
RunRange range = runRanges[run];
|
|
if (range.start<end && start<range.end)
|
|
{
|
|
if (deepCopy)
|
|
{
|
|
clonedRun = CopyRun(run);
|
|
}
|
|
else
|
|
{
|
|
clonedRun = run;
|
|
}
|
|
}
|
|
}
|
|
|
|
void Visit(DocumentParagraphRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
};
|
|
}
|
|
using namespace document_operation_visitors;
|
|
|
|
namespace document_editor
|
|
{
|
|
Ptr<DocumentStyleProperties> CopyStyle(Ptr<DocumentStyleProperties> style)
|
|
{
|
|
if (!style) return nullptr;
|
|
Ptr<DocumentStyleProperties> newStyle = new DocumentStyleProperties;
|
|
|
|
newStyle->face = style->face;
|
|
newStyle->size = style->size;
|
|
newStyle->color = style->color;
|
|
newStyle->backgroundColor = style->backgroundColor;
|
|
newStyle->bold = style->bold;
|
|
newStyle->italic = style->italic;
|
|
newStyle->underline = style->underline;
|
|
newStyle->strikeline = style->strikeline;
|
|
newStyle->antialias = style->antialias;
|
|
newStyle->verticalAntialias = style->verticalAntialias;
|
|
|
|
return newStyle;
|
|
}
|
|
|
|
Ptr<DocumentRun> CopyRun(DocumentRun* run)
|
|
{
|
|
CloneRunVisitor visitor(0);
|
|
run->Accept(&visitor);
|
|
return visitor.clonedRun;
|
|
}
|
|
|
|
Ptr<DocumentRun> CopyStyledText(List<DocumentContainerRun*>& styleRuns, const WString& text)
|
|
{
|
|
Ptr<DocumentTextRun> textRun = new DocumentTextRun;
|
|
textRun->text = text;
|
|
|
|
CloneRunVisitor visitor(textRun);
|
|
for (vint i = styleRuns.Count() - 1; i >= 0; i--)
|
|
{
|
|
styleRuns[i]->Accept(&visitor);
|
|
}
|
|
|
|
return visitor.clonedRun;
|
|
}
|
|
|
|
Ptr<DocumentRun> CopyRunRecursively(DocumentParagraphRun* run, RunRangeMap& runRanges, vint start, vint end, bool deepCopy)
|
|
{
|
|
CloneRunRecursivelyVisitor visitor(runRanges, start, end, deepCopy);
|
|
run->Accept(&visitor);
|
|
return visitor.clonedRun;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\RESOURCES\GUIDOCUMENTEDITOR_COLLECTSTYLE.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
using namespace collections;
|
|
|
|
/***********************************************************************
|
|
Search all used style names
|
|
***********************************************************************/
|
|
|
|
namespace document_operation_visitors
|
|
{
|
|
class CollectStyleNameVisitor : public Object, public DocumentRun::IVisitor
|
|
{
|
|
public:
|
|
List<WString>& styleNames;
|
|
|
|
CollectStyleNameVisitor(List<WString>& _styleNames)
|
|
:styleNames(_styleNames)
|
|
{
|
|
}
|
|
|
|
void VisitContainer(DocumentContainerRun* run)
|
|
{
|
|
FOREACH(Ptr<DocumentRun>, subRun, run->runs)
|
|
{
|
|
subRun->Accept(this);
|
|
}
|
|
}
|
|
|
|
void Visit(DocumentTextRun* run)override
|
|
{
|
|
}
|
|
|
|
void Visit(DocumentStylePropertiesRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
|
|
void Visit(DocumentStyleApplicationRun* run)override
|
|
{
|
|
if (!styleNames.Contains(run->styleName))
|
|
{
|
|
styleNames.Add(run->styleName);
|
|
}
|
|
VisitContainer(run);
|
|
}
|
|
|
|
void Visit(DocumentHyperlinkRun* run)override
|
|
{
|
|
if (!styleNames.Contains(run->normalStyleName))
|
|
{
|
|
styleNames.Add(run->normalStyleName);
|
|
}
|
|
if (!styleNames.Contains(run->activeStyleName))
|
|
{
|
|
styleNames.Add(run->activeStyleName);
|
|
}
|
|
VisitContainer(run);
|
|
}
|
|
|
|
void Visit(DocumentImageRun* run)override
|
|
{
|
|
}
|
|
|
|
void Visit(DocumentEmbeddedObjectRun* run)override
|
|
{
|
|
}
|
|
|
|
void Visit(DocumentParagraphRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
};
|
|
}
|
|
using namespace document_operation_visitors;
|
|
|
|
namespace document_editor
|
|
{
|
|
void CollectStyleName(DocumentParagraphRun* run, List<WString>& styleNames)
|
|
{
|
|
CollectStyleNameVisitor visitor(styleNames);
|
|
run->Accept(&visitor);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\RESOURCES\GUIDOCUMENTEDITOR_CUTRUN.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
using namespace collections;
|
|
using namespace document_editor;
|
|
|
|
/***********************************************************************
|
|
Cut all runs into pieces so that a run either completely outside or inside the specified range
|
|
If a run decides that itself should be cut, then leftRun and rightRun contains new run that will be inserted before and after it
|
|
***********************************************************************/
|
|
|
|
namespace document_operation_visitors
|
|
{
|
|
class CutRunVisitor : public Object, public DocumentRun::IVisitor
|
|
{
|
|
public:
|
|
RunRangeMap& runRanges;
|
|
vint position;
|
|
Ptr<DocumentRun> leftRun;
|
|
Ptr<DocumentRun> rightRun;
|
|
|
|
CutRunVisitor(RunRangeMap& _runRanges, vint _position)
|
|
:runRanges(_runRanges)
|
|
, position(_position)
|
|
{
|
|
}
|
|
|
|
void VisitContainer(DocumentContainerRun* run)
|
|
{
|
|
vint leftCount = 0;
|
|
Ptr<DocumentRun> selectedRun;
|
|
|
|
FOREACH(Ptr<DocumentRun>, subRun, run->runs)
|
|
{
|
|
RunRange range = runRanges[subRun.Obj()];
|
|
if (range.start<position)
|
|
{
|
|
leftCount++;
|
|
if (position<range.end)
|
|
{
|
|
selectedRun = subRun;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (selectedRun)
|
|
{
|
|
selectedRun->Accept(this);
|
|
if (leftRun && rightRun)
|
|
{
|
|
run->runs.RemoveAt(leftCount - 1);
|
|
run->runs.Insert(leftCount - 1, leftRun);
|
|
run->runs.Insert(leftCount, rightRun);
|
|
}
|
|
}
|
|
|
|
Ptr<DocumentContainerRun> leftContainer = CopyRun(run).Cast<DocumentContainerRun>();
|
|
Ptr<DocumentContainerRun> rightContainer = CopyRun(run).Cast<DocumentContainerRun>();
|
|
for (vint i = 0; i<run->runs.Count(); i++)
|
|
{
|
|
(i<leftCount ? leftContainer : rightContainer)->runs.Add(run->runs[i]);
|
|
}
|
|
leftRun = leftContainer;
|
|
rightRun = rightContainer;
|
|
}
|
|
|
|
void Visit(DocumentTextRun* run)override
|
|
{
|
|
RunRange range = runRanges[run];
|
|
|
|
Ptr<DocumentTextRun> leftText = new DocumentTextRun;
|
|
leftText->text = run->text.Sub(0, position - range.start);
|
|
|
|
Ptr<DocumentTextRun> rightText = new DocumentTextRun;
|
|
rightText->text = run->text.Sub(position - range.start, range.end - position);
|
|
|
|
leftRun = leftText;
|
|
rightRun = rightText;
|
|
}
|
|
|
|
void Visit(DocumentStylePropertiesRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
|
|
void Visit(DocumentStyleApplicationRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
|
|
void Visit(DocumentHyperlinkRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
|
|
void Visit(DocumentImageRun* run)override
|
|
{
|
|
leftRun = 0;
|
|
rightRun = 0;
|
|
}
|
|
|
|
void Visit(DocumentEmbeddedObjectRun* run)override
|
|
{
|
|
leftRun = 0;
|
|
rightRun = 0;
|
|
}
|
|
|
|
void Visit(DocumentParagraphRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
};
|
|
}
|
|
using namespace document_operation_visitors;
|
|
|
|
namespace document_editor
|
|
{
|
|
void CutRun(DocumentParagraphRun* run, RunRangeMap& runRanges, vint position, Ptr<DocumentRun>& leftRun, Ptr<DocumentRun>& rightRun)
|
|
{
|
|
CutRunVisitor visitor(runRanges, position);
|
|
run->Accept(&visitor);
|
|
leftRun = visitor.leftRun;
|
|
rightRun = visitor.rightRun;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\RESOURCES\GUIDOCUMENTEDITOR_GETRUNRANGE.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
using namespace collections;
|
|
|
|
/***********************************************************************
|
|
Calculate range informations for each run object
|
|
***********************************************************************/
|
|
|
|
namespace document_operation_visitors
|
|
{
|
|
class GetRunRangeVisitor : public Object, public DocumentRun::IVisitor
|
|
{
|
|
public:
|
|
RunRangeMap& runRanges;
|
|
vint start;
|
|
|
|
GetRunRangeVisitor(RunRangeMap& _runRanges)
|
|
:runRanges(_runRanges)
|
|
, start(0)
|
|
{
|
|
}
|
|
|
|
void VisitContainer(DocumentContainerRun* run)
|
|
{
|
|
RunRange range;
|
|
range.start = start;
|
|
FOREACH(Ptr<DocumentRun>, subRun, run->runs)
|
|
{
|
|
subRun->Accept(this);
|
|
}
|
|
range.end = start;
|
|
runRanges.Add(run, range);
|
|
}
|
|
|
|
void VisitContent(DocumentContentRun* run)
|
|
{
|
|
RunRange range;
|
|
range.start = start;
|
|
start += run->GetRepresentationText().Length();
|
|
range.end = start;
|
|
runRanges.Add(run, range);
|
|
}
|
|
|
|
void Visit(DocumentTextRun* run)override
|
|
{
|
|
VisitContent(run);
|
|
}
|
|
|
|
void Visit(DocumentStylePropertiesRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
|
|
void Visit(DocumentStyleApplicationRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
|
|
void Visit(DocumentHyperlinkRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
|
|
void Visit(DocumentImageRun* run)override
|
|
{
|
|
VisitContent(run);
|
|
}
|
|
|
|
void Visit(DocumentEmbeddedObjectRun* run)override
|
|
{
|
|
VisitContent(run);
|
|
}
|
|
|
|
void Visit(DocumentParagraphRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
};
|
|
}
|
|
using namespace document_operation_visitors;
|
|
|
|
namespace document_editor
|
|
{
|
|
void GetRunRange(DocumentParagraphRun* run, RunRangeMap& runRanges)
|
|
{
|
|
GetRunRangeVisitor visitor(runRanges);
|
|
run->Accept(&visitor);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\RESOURCES\GUIDOCUMENTEDITOR_LOCALEHYPERLINK.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
using namespace collections;
|
|
|
|
/***********************************************************************
|
|
Get the hyperlink run that contains the specified position
|
|
***********************************************************************/
|
|
|
|
namespace document_operation_visitors
|
|
{
|
|
class LocateHyperlinkVisitor : public Object, public DocumentRun::IVisitor
|
|
{
|
|
public:
|
|
Ptr<DocumentHyperlinkRun::Package> package;
|
|
RunRangeMap& runRanges;
|
|
vint start;
|
|
vint end;
|
|
|
|
LocateHyperlinkVisitor(RunRangeMap& _runRanges, Ptr<DocumentHyperlinkRun::Package> _package, vint _start, vint _end)
|
|
:runRanges(_runRanges)
|
|
, package(_package)
|
|
, start(_start)
|
|
, end(_end)
|
|
{
|
|
}
|
|
|
|
void VisitContainer(DocumentContainerRun* run)
|
|
{
|
|
Ptr<DocumentRun> selectedRun;
|
|
FOREACH(Ptr<DocumentRun>, subRun, run->runs)
|
|
{
|
|
RunRange range = runRanges[subRun.Obj()];
|
|
if (range.start <= start && end <= range.end)
|
|
{
|
|
subRun->Accept(this);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
void Visit(DocumentTextRun* run)override
|
|
{
|
|
}
|
|
|
|
void Visit(DocumentStylePropertiesRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
|
|
void Visit(DocumentStyleApplicationRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
|
|
void Visit(DocumentHyperlinkRun* run)override
|
|
{
|
|
package->hyperlinks.Add(run);
|
|
}
|
|
|
|
void Visit(DocumentImageRun* run)override
|
|
{
|
|
}
|
|
|
|
void Visit(DocumentEmbeddedObjectRun* run)override
|
|
{
|
|
}
|
|
|
|
void Visit(DocumentParagraphRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
};
|
|
}
|
|
using namespace document_operation_visitors;
|
|
|
|
namespace document_editor
|
|
{
|
|
Ptr<DocumentHyperlinkRun::Package> LocateHyperlink(DocumentParagraphRun* run, RunRangeMap& runRanges, vint row, vint start, vint end)
|
|
{
|
|
auto package = MakePtr<DocumentHyperlinkRun::Package>();
|
|
package->row = row;
|
|
{
|
|
LocateHyperlinkVisitor visitor(runRanges, package, start, end);
|
|
run->Accept(&visitor);
|
|
}
|
|
|
|
Ptr<DocumentHyperlinkRun> startRun, endRun;
|
|
FOREACH(Ptr<DocumentHyperlinkRun>, run, package->hyperlinks)
|
|
{
|
|
auto range = runRanges[run.Obj()];
|
|
if (package->start == -1 || range.start < package->start)
|
|
{
|
|
package->start = range.start;
|
|
startRun = run;
|
|
}
|
|
if (package->end == -1 || range.end > package->end)
|
|
{
|
|
package->end = range.end;
|
|
endRun = run;
|
|
}
|
|
}
|
|
|
|
while (startRun)
|
|
{
|
|
vint pos = runRanges[startRun.Obj()].start;
|
|
if (pos == 0) break;
|
|
|
|
auto newPackage = MakePtr<DocumentHyperlinkRun::Package>();
|
|
LocateHyperlinkVisitor visitor(runRanges, newPackage, pos - 1, pos);
|
|
run->Accept(&visitor);
|
|
if (newPackage->hyperlinks.Count() == 0) break;
|
|
|
|
auto newRun = newPackage->hyperlinks[0];
|
|
if (startRun->reference != newRun->reference) break;
|
|
|
|
auto range = runRanges[newRun.Obj()];
|
|
package->hyperlinks.Add(newRun);
|
|
package->start = range.start;
|
|
startRun = newRun;
|
|
}
|
|
|
|
vint length = runRanges[run].end;
|
|
while (endRun)
|
|
{
|
|
vint pos = runRanges[endRun.Obj()].end;
|
|
if (pos == length) break;
|
|
|
|
auto newPackage = MakePtr<DocumentHyperlinkRun::Package>();
|
|
LocateHyperlinkVisitor visitor(runRanges, newPackage, pos, pos + 1);
|
|
run->Accept(&visitor);
|
|
if (newPackage->hyperlinks.Count() == 0) break;
|
|
|
|
auto newRun = newPackage->hyperlinks[0];
|
|
if (endRun->reference != newRun->reference) break;
|
|
|
|
auto range = runRanges[newRun.Obj()];
|
|
package->hyperlinks.Add(newRun);
|
|
package->end = range.end;
|
|
endRun = newRun;
|
|
}
|
|
|
|
return package;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\RESOURCES\GUIDOCUMENTEDITOR_LOCALESTYLE.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
using namespace collections;
|
|
|
|
/***********************************************************************
|
|
Get all container runs that contain the specified position from top to bottom
|
|
***********************************************************************/
|
|
|
|
namespace document_operation_visitors
|
|
{
|
|
class LocateStyleVisitor : public Object, public DocumentRun::IVisitor
|
|
{
|
|
public:
|
|
List<DocumentContainerRun*>& locatedRuns;
|
|
RunRangeMap& runRanges;
|
|
vint position;
|
|
bool frontSide;
|
|
|
|
LocateStyleVisitor(List<DocumentContainerRun*>& _locatedRuns, RunRangeMap& _runRanges, vint _position, bool _frontSide)
|
|
:locatedRuns(_locatedRuns)
|
|
, runRanges(_runRanges)
|
|
, position(_position)
|
|
, frontSide(_frontSide)
|
|
{
|
|
}
|
|
|
|
void VisitContainer(DocumentContainerRun* run)
|
|
{
|
|
locatedRuns.Add(run);
|
|
Ptr<DocumentRun> selectedRun;
|
|
FOREACH(Ptr<DocumentRun>, subRun, run->runs)
|
|
{
|
|
RunRange range = runRanges[subRun.Obj()];
|
|
if (position == range.start)
|
|
{
|
|
if (!frontSide)
|
|
{
|
|
selectedRun = subRun;
|
|
break;
|
|
}
|
|
}
|
|
else if (position == range.end)
|
|
{
|
|
if (frontSide)
|
|
{
|
|
selectedRun = subRun;
|
|
break;
|
|
}
|
|
}
|
|
else if (range.start<position && position<range.end)
|
|
{
|
|
selectedRun = subRun;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (selectedRun)
|
|
{
|
|
selectedRun->Accept(this);
|
|
}
|
|
}
|
|
|
|
void Visit(DocumentTextRun* run)override
|
|
{
|
|
}
|
|
|
|
void Visit(DocumentStylePropertiesRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
|
|
void Visit(DocumentStyleApplicationRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
|
|
void Visit(DocumentHyperlinkRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
|
|
void Visit(DocumentImageRun* run)override
|
|
{
|
|
}
|
|
|
|
void Visit(DocumentEmbeddedObjectRun* run)override
|
|
{
|
|
}
|
|
|
|
void Visit(DocumentParagraphRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
};
|
|
}
|
|
using namespace document_operation_visitors;
|
|
|
|
namespace document_editor
|
|
{
|
|
void LocateStyle(DocumentParagraphRun* run, RunRangeMap& runRanges, vint position, bool frontSide, List<DocumentContainerRun*>& locatedRuns)
|
|
{
|
|
LocateStyleVisitor visitor(locatedRuns, runRanges, position, frontSide);
|
|
run->Accept(&visitor);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\RESOURCES\GUIDOCUMENTEDITOR_REMOVECONTAINER.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
using namespace collections;
|
|
|
|
/***********************************************************************
|
|
Remove some containers that intersect with the specified range
|
|
If a run decides that itself should be removed, then replacedRuns contains all runs to replace itself
|
|
RemoveHyperlinkVisitor : Remove all hyperlinks that intersect with the specified range
|
|
RemoveStyleNameVisitor : Remove all style names that intersect with the specified range
|
|
ClearStyleVisitor : Remove all styles that intersect with the specified range
|
|
***********************************************************************/
|
|
|
|
namespace document_operation_visitors
|
|
{
|
|
class RemoveContainerVisitor : public Object, public DocumentRun::IVisitor
|
|
{
|
|
public:
|
|
RunRangeMap& runRanges;
|
|
vint start;
|
|
vint end;
|
|
List<Ptr<DocumentRun>> replacedRuns;
|
|
|
|
RemoveContainerVisitor(RunRangeMap& _runRanges, vint _start, vint _end)
|
|
:runRanges(_runRanges)
|
|
, start(_start)
|
|
, end(_end)
|
|
{
|
|
}
|
|
|
|
void VisitContainer(DocumentContainerRun* run)
|
|
{
|
|
for (vint i = run->runs.Count() - 1; i >= 0; i--)
|
|
{
|
|
Ptr<DocumentRun> subRun = run->runs[i];
|
|
RunRange range = runRanges[subRun.Obj()];
|
|
if (range.start<end && start<range.end)
|
|
{
|
|
replacedRuns.Clear();
|
|
subRun->Accept(this);
|
|
if (replacedRuns.Count() != 1 || replacedRuns[0] != subRun)
|
|
{
|
|
run->runs.RemoveAt(i);
|
|
for (vint j = 0; j<replacedRuns.Count(); j++)
|
|
{
|
|
run->runs.Insert(i + j, replacedRuns[j]);
|
|
}
|
|
i += replacedRuns.Count();
|
|
}
|
|
}
|
|
}
|
|
replacedRuns.Clear();
|
|
replacedRuns.Add(run);
|
|
}
|
|
|
|
void VisitContent(DocumentContentRun* run)
|
|
{
|
|
replacedRuns.Add(run);
|
|
}
|
|
|
|
void RemoveContainer(DocumentContainerRun* run)
|
|
{
|
|
CopyFrom(replacedRuns, run->runs);
|
|
}
|
|
|
|
void Visit(DocumentTextRun* run)override
|
|
{
|
|
VisitContent(run);
|
|
}
|
|
|
|
void Visit(DocumentStylePropertiesRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
|
|
void Visit(DocumentStyleApplicationRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
|
|
void Visit(DocumentHyperlinkRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
|
|
void Visit(DocumentImageRun* run)override
|
|
{
|
|
VisitContent(run);
|
|
}
|
|
|
|
void Visit(DocumentEmbeddedObjectRun* run)override
|
|
{
|
|
VisitContent(run);
|
|
}
|
|
|
|
void Visit(DocumentParagraphRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
};
|
|
|
|
class RemoveHyperlinkVisitor : public RemoveContainerVisitor
|
|
{
|
|
public:
|
|
RemoveHyperlinkVisitor(RunRangeMap& _runRanges, vint _start, vint _end)
|
|
:RemoveContainerVisitor(_runRanges, _start, _end)
|
|
{
|
|
}
|
|
|
|
void Visit(DocumentHyperlinkRun* run)override
|
|
{
|
|
RemoveContainer(run);
|
|
}
|
|
};
|
|
|
|
class RemoveStyleNameVisitor : public RemoveContainerVisitor
|
|
{
|
|
public:
|
|
RemoveStyleNameVisitor(RunRangeMap& _runRanges, vint _start, vint _end)
|
|
:RemoveContainerVisitor(_runRanges, _start, _end)
|
|
{
|
|
}
|
|
|
|
void Visit(DocumentStyleApplicationRun* run)override
|
|
{
|
|
RemoveContainer(run);
|
|
}
|
|
};
|
|
|
|
class ClearStyleVisitor : public RemoveContainerVisitor
|
|
{
|
|
public:
|
|
ClearStyleVisitor(RunRangeMap& _runRanges, vint _start, vint _end)
|
|
:RemoveContainerVisitor(_runRanges, _start, _end)
|
|
{
|
|
}
|
|
|
|
void Visit(DocumentStylePropertiesRun* run)override
|
|
{
|
|
RemoveContainer(run);
|
|
}
|
|
|
|
void Visit(DocumentStyleApplicationRun* run)override
|
|
{
|
|
RemoveContainer(run);
|
|
}
|
|
};
|
|
}
|
|
using namespace document_operation_visitors;
|
|
|
|
namespace document_editor
|
|
{
|
|
void RemoveHyperlink(DocumentParagraphRun* run, RunRangeMap& runRanges, vint start, vint end)
|
|
{
|
|
RemoveHyperlinkVisitor visitor(runRanges, start, end);
|
|
run->Accept(&visitor);
|
|
}
|
|
|
|
void RemoveStyleName(DocumentParagraphRun* run, RunRangeMap& runRanges, vint start, vint end)
|
|
{
|
|
RemoveStyleNameVisitor visitor(runRanges, start, end);
|
|
run->Accept(&visitor);
|
|
}
|
|
|
|
void ClearStyle(DocumentParagraphRun* run, RunRangeMap& runRanges, vint start, vint end)
|
|
{
|
|
ClearStyleVisitor visitor(runRanges, start, end);
|
|
run->Accept(&visitor);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\RESOURCES\GUIDOCUMENTEDITOR_REMOVERUN.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
using namespace collections;
|
|
|
|
/***********************************************************************
|
|
Remove text run contents with the specified range, or other content runs that intersect with the range
|
|
If a run decides that itself should be removed, then replacedRuns contains all runs to replace itself
|
|
***********************************************************************/
|
|
|
|
namespace document_operation_visitors
|
|
{
|
|
class RemoveRunVisitor : public Object, public DocumentRun::IVisitor
|
|
{
|
|
public:
|
|
RunRangeMap& runRanges;
|
|
vint start;
|
|
vint end;
|
|
List<Ptr<DocumentRun>> replacedRuns;
|
|
|
|
RemoveRunVisitor(RunRangeMap& _runRanges, vint _start, vint _end)
|
|
:runRanges(_runRanges)
|
|
, start(_start)
|
|
, end(_end)
|
|
{
|
|
}
|
|
|
|
void VisitContainer(DocumentContainerRun* run)
|
|
{
|
|
if (start == end) return;
|
|
for (vint i = run->runs.Count() - 1; i >= 0; i--)
|
|
{
|
|
Ptr<DocumentRun> subRun = run->runs[i];
|
|
RunRange range = runRanges[subRun.Obj()];
|
|
|
|
vint maxStart = range.start > start ? range.start : start;
|
|
vint minEnd = range.end < end ? range.end : end;
|
|
if (maxStart < minEnd)
|
|
{
|
|
subRun->Accept(this);
|
|
if (replacedRuns.Count() == 0 || subRun != replacedRuns[0])
|
|
{
|
|
run->runs.RemoveAt(i);
|
|
for (vint j = 0; j<replacedRuns.Count(); j++)
|
|
{
|
|
run->runs.Insert(i + j, replacedRuns[j]);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
replacedRuns.Clear();
|
|
replacedRuns.Add(run);
|
|
}
|
|
|
|
void Visit(DocumentTextRun* run)override
|
|
{
|
|
replacedRuns.Clear();
|
|
RunRange range = runRanges[run];
|
|
|
|
if (start <= range.start)
|
|
{
|
|
if (end<range.end)
|
|
{
|
|
run->text = run->text.Sub(end - range.start, range.end - end);
|
|
replacedRuns.Add(run);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (end<range.end)
|
|
{
|
|
DocumentTextRun* firstRun = new DocumentTextRun;
|
|
DocumentTextRun* secondRun = new DocumentTextRun;
|
|
|
|
firstRun->text = run->text.Sub(0, start - range.start);
|
|
secondRun->text = run->text.Sub(end - range.start, range.end - end);
|
|
|
|
replacedRuns.Add(firstRun);
|
|
replacedRuns.Add(secondRun);
|
|
}
|
|
else
|
|
{
|
|
run->text = run->text.Sub(0, start - range.start);
|
|
replacedRuns.Add(run);
|
|
}
|
|
}
|
|
}
|
|
|
|
void Visit(DocumentStylePropertiesRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
|
|
void Visit(DocumentStyleApplicationRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
|
|
void Visit(DocumentHyperlinkRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
|
|
void Visit(DocumentImageRun* run)override
|
|
{
|
|
replacedRuns.Clear();
|
|
}
|
|
|
|
void Visit(DocumentEmbeddedObjectRun* run)override
|
|
{
|
|
replacedRuns.Clear();
|
|
}
|
|
|
|
void Visit(DocumentParagraphRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
};
|
|
}
|
|
using namespace document_operation_visitors;
|
|
|
|
namespace document_editor
|
|
{
|
|
void RemoveRun(DocumentParagraphRun* run, RunRangeMap& runRanges, vint start, vint end)
|
|
{
|
|
RemoveRunVisitor visitor(runRanges, start, end);
|
|
run->Accept(&visitor);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\RESOURCES\GUIDOCUMENTEDITOR_REPLACESTYLENAME.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
using namespace collections;
|
|
|
|
/***********************************************************************
|
|
Replace a style name with another one
|
|
***********************************************************************/
|
|
|
|
namespace document_operation_visitors
|
|
{
|
|
class ReplaceStyleNameVisitor : public Object, public DocumentRun::IVisitor
|
|
{
|
|
public:
|
|
WString oldStyleName;
|
|
WString newStyleName;
|
|
|
|
ReplaceStyleNameVisitor(const WString& _oldStyleName, const WString& _newStyleName)
|
|
:oldStyleName(_oldStyleName)
|
|
, newStyleName(_newStyleName)
|
|
{
|
|
}
|
|
|
|
void VisitContainer(DocumentContainerRun* run)
|
|
{
|
|
FOREACH(Ptr<DocumentRun>, subRun, run->runs)
|
|
{
|
|
subRun->Accept(this);
|
|
}
|
|
}
|
|
|
|
void Visit(DocumentTextRun* run)override
|
|
{
|
|
}
|
|
|
|
void Visit(DocumentStylePropertiesRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
|
|
void Visit(DocumentStyleApplicationRun* run)override
|
|
{
|
|
if (run->styleName == oldStyleName) run->styleName = newStyleName;
|
|
VisitContainer(run);
|
|
}
|
|
|
|
void Visit(DocumentHyperlinkRun* run)override
|
|
{
|
|
if (run->styleName == oldStyleName) run->styleName = newStyleName;
|
|
if (run->normalStyleName == oldStyleName) run->normalStyleName = newStyleName;
|
|
if (run->activeStyleName == oldStyleName) run->activeStyleName = newStyleName;
|
|
VisitContainer(run);
|
|
}
|
|
|
|
void Visit(DocumentImageRun* run)override
|
|
{
|
|
}
|
|
|
|
void Visit(DocumentEmbeddedObjectRun* run)override
|
|
{
|
|
}
|
|
|
|
void Visit(DocumentParagraphRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
};
|
|
}
|
|
using namespace document_operation_visitors;
|
|
|
|
namespace document_editor
|
|
{
|
|
void ReplaceStyleName(DocumentParagraphRun* run, const WString& oldStyleName, const WString& newStyleName)
|
|
{
|
|
ReplaceStyleNameVisitor visitor(oldStyleName, newStyleName);
|
|
run->Accept(&visitor);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\RESOURCES\GUIDOCUMENTEDITOR_SUMMERIZESTYLE.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
using namespace collections;
|
|
|
|
namespace document_operation_visitors
|
|
{
|
|
|
|
/***********************************************************************
|
|
Calculate if all text in the specified range has some common styles
|
|
***********************************************************************/
|
|
|
|
class SummarizeStyleVisitor : public Object, public DocumentRun::IVisitor
|
|
{
|
|
public:
|
|
RunRangeMap& runRanges;
|
|
DocumentModel* model;
|
|
vint start;
|
|
vint end;
|
|
Ptr<DocumentStyleProperties> style;
|
|
List<DocumentModel::ResolvedStyle> resolvedStyles;
|
|
|
|
SummarizeStyleVisitor(RunRangeMap& _runRanges, DocumentModel* _model, vint _start, vint _end)
|
|
:runRanges(_runRanges)
|
|
, model(_model)
|
|
, start(_start)
|
|
, end(_end)
|
|
{
|
|
DocumentModel::ResolvedStyle resolvedStyle;
|
|
resolvedStyle = model->GetStyle(DocumentModel::DefaultStyleName, resolvedStyle);
|
|
resolvedStyles.Add(resolvedStyle);
|
|
}
|
|
|
|
const DocumentModel::ResolvedStyle& GetCurrentResolvedStyle()
|
|
{
|
|
return resolvedStyles[resolvedStyles.Count() - 1];
|
|
}
|
|
|
|
// ---------------------------------------------------------
|
|
|
|
template<typename T>
|
|
void SetStyleItem(Nullable<T> DocumentStyleProperties::* dstField, T FontProperties::* srcField)
|
|
{
|
|
const DocumentModel::ResolvedStyle& src = GetCurrentResolvedStyle();
|
|
if (style.Obj()->*dstField && (style.Obj()->*dstField).Value() != src.style.*srcField)
|
|
{
|
|
style.Obj()->*dstField = Nullable<T>();
|
|
}
|
|
}
|
|
|
|
template<typename T>
|
|
void SetStyleItem(Nullable<T> DocumentStyleProperties::* dstField, T DocumentModel::ResolvedStyle::* srcField)
|
|
{
|
|
const DocumentModel::ResolvedStyle& src = GetCurrentResolvedStyle();
|
|
if (style.Obj()->*dstField && (style.Obj()->*dstField).Value() != src.*srcField)
|
|
{
|
|
style.Obj()->*dstField = Nullable<T>();
|
|
}
|
|
}
|
|
|
|
void SetStyleItem(Nullable<DocumentFontSize> DocumentStyleProperties::* dstField, vint FontProperties::* srcField)
|
|
{
|
|
const DocumentModel::ResolvedStyle& src = GetCurrentResolvedStyle();
|
|
if (style.Obj()->*dstField)
|
|
{
|
|
auto dfs = (style.Obj()->*dstField).Value();
|
|
if (dfs.relative || dfs.size != src.style.*srcField)
|
|
{
|
|
style.Obj()->*dstField = Nullable<DocumentFontSize>();
|
|
}
|
|
}
|
|
}
|
|
|
|
// ---------------------------------------------------------
|
|
|
|
template<typename T>
|
|
void OverrideStyleItem(Nullable<T> DocumentStyleProperties::* dstField, T FontProperties::* srcField)
|
|
{
|
|
const DocumentModel::ResolvedStyle& src = GetCurrentResolvedStyle();
|
|
style.Obj()->*dstField = src.style.*srcField;
|
|
}
|
|
|
|
template<typename T>
|
|
void OverrideStyleItem(Nullable<T> DocumentStyleProperties::* dstField, T DocumentModel::ResolvedStyle::* srcField)
|
|
{
|
|
const DocumentModel::ResolvedStyle& src = GetCurrentResolvedStyle();
|
|
style.Obj()->*dstField = src.*srcField;
|
|
}
|
|
|
|
void OverrideStyleItem(Nullable<DocumentFontSize> DocumentStyleProperties::* dstField, vint FontProperties::* srcField)
|
|
{
|
|
const DocumentModel::ResolvedStyle& src = GetCurrentResolvedStyle();
|
|
style.Obj()->*dstField = DocumentFontSize((double)(src.style.*srcField), false);
|
|
}
|
|
|
|
// ---------------------------------------------------------
|
|
|
|
void VisitContainer(DocumentContainerRun* run)
|
|
{
|
|
for (vint i = run->runs.Count() - 1; i >= 0; i--)
|
|
{
|
|
Ptr<DocumentRun> subRun = run->runs[i];
|
|
RunRange range = runRanges[subRun.Obj()];
|
|
if (range.start<end && start<range.end)
|
|
{
|
|
subRun->Accept(this);
|
|
}
|
|
}
|
|
}
|
|
|
|
void Visit(DocumentTextRun* run)override
|
|
{
|
|
const DocumentModel::ResolvedStyle& currentResolvedStyle = GetCurrentResolvedStyle();
|
|
if (style)
|
|
{
|
|
SetStyleItem(&DocumentStyleProperties::face, &FontProperties::fontFamily);
|
|
SetStyleItem(&DocumentStyleProperties::size, &FontProperties::size);
|
|
SetStyleItem(&DocumentStyleProperties::color, &DocumentModel::ResolvedStyle::color);
|
|
SetStyleItem(&DocumentStyleProperties::backgroundColor, &DocumentModel::ResolvedStyle::backgroundColor);
|
|
SetStyleItem(&DocumentStyleProperties::bold, &FontProperties::bold);
|
|
SetStyleItem(&DocumentStyleProperties::italic, &FontProperties::italic);
|
|
SetStyleItem(&DocumentStyleProperties::underline, &FontProperties::underline);
|
|
SetStyleItem(&DocumentStyleProperties::strikeline, &FontProperties::strikeline);
|
|
SetStyleItem(&DocumentStyleProperties::antialias, &FontProperties::antialias);
|
|
SetStyleItem(&DocumentStyleProperties::verticalAntialias, &FontProperties::verticalAntialias);
|
|
}
|
|
else
|
|
{
|
|
style = new DocumentStyleProperties;
|
|
OverrideStyleItem(&DocumentStyleProperties::face, &FontProperties::fontFamily);
|
|
OverrideStyleItem(&DocumentStyleProperties::size, &FontProperties::size);
|
|
OverrideStyleItem(&DocumentStyleProperties::color, &DocumentModel::ResolvedStyle::color);
|
|
OverrideStyleItem(&DocumentStyleProperties::backgroundColor, &DocumentModel::ResolvedStyle::backgroundColor);
|
|
OverrideStyleItem(&DocumentStyleProperties::bold, &FontProperties::bold);
|
|
OverrideStyleItem(&DocumentStyleProperties::italic, &FontProperties::italic);
|
|
OverrideStyleItem(&DocumentStyleProperties::underline, &FontProperties::underline);
|
|
OverrideStyleItem(&DocumentStyleProperties::strikeline, &FontProperties::strikeline);
|
|
OverrideStyleItem(&DocumentStyleProperties::antialias, &FontProperties::antialias);
|
|
OverrideStyleItem(&DocumentStyleProperties::verticalAntialias, &FontProperties::verticalAntialias);
|
|
}
|
|
}
|
|
|
|
void Visit(DocumentStylePropertiesRun* run)override
|
|
{
|
|
const DocumentModel::ResolvedStyle& currentResolvedStyle = GetCurrentResolvedStyle();
|
|
DocumentModel::ResolvedStyle resolvedStyle = model->GetStyle(run->style, currentResolvedStyle);
|
|
resolvedStyles.Add(resolvedStyle);
|
|
VisitContainer(run);
|
|
resolvedStyles.RemoveAt(resolvedStyles.Count() - 1);
|
|
}
|
|
|
|
void Visit(DocumentStyleApplicationRun* run)override
|
|
{
|
|
const DocumentModel::ResolvedStyle& currentResolvedStyle = GetCurrentResolvedStyle();
|
|
DocumentModel::ResolvedStyle resolvedStyle = model->GetStyle(run->styleName, currentResolvedStyle);
|
|
resolvedStyles.Add(resolvedStyle);
|
|
VisitContainer(run);
|
|
resolvedStyles.RemoveAt(resolvedStyles.Count() - 1);
|
|
}
|
|
|
|
void Visit(DocumentHyperlinkRun* run)override
|
|
{
|
|
const DocumentModel::ResolvedStyle& currentResolvedStyle = GetCurrentResolvedStyle();
|
|
DocumentModel::ResolvedStyle resolvedStyle = model->GetStyle(run->styleName, currentResolvedStyle);
|
|
resolvedStyles.Add(resolvedStyle);
|
|
VisitContainer(run);
|
|
resolvedStyles.RemoveAt(resolvedStyles.Count() - 1);
|
|
}
|
|
|
|
void Visit(DocumentImageRun* run)override
|
|
{
|
|
}
|
|
|
|
void Visit(DocumentEmbeddedObjectRun* run)override
|
|
{
|
|
}
|
|
|
|
void Visit(DocumentParagraphRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
};
|
|
|
|
/***********************************************************************
|
|
Calculate if all text in the specified range has a common style name
|
|
***********************************************************************/
|
|
|
|
class SummarizeStyleNameVisitor : public Object, public DocumentRun::IVisitor
|
|
{
|
|
public:
|
|
RunRangeMap& runRanges;
|
|
DocumentModel* model;
|
|
vint start;
|
|
vint end;
|
|
Nullable<WString> currentStyleName;
|
|
Nullable<WString> styleName;
|
|
bool assignedStyleName = false;
|
|
|
|
SummarizeStyleNameVisitor(RunRangeMap& _runRanges, DocumentModel* _model, vint _start, vint _end)
|
|
:runRanges(_runRanges)
|
|
, model(_model)
|
|
, start(_start)
|
|
, end(_end)
|
|
{
|
|
}
|
|
|
|
void VisitContentRun(DocumentContentRun* run)
|
|
{
|
|
if (!assignedStyleName)
|
|
{
|
|
styleName = currentStyleName;
|
|
assignedStyleName = true;
|
|
}
|
|
else if (styleName && (!currentStyleName || styleName.Value() != currentStyleName.Value()))
|
|
{
|
|
styleName = Nullable<WString>();
|
|
}
|
|
}
|
|
|
|
void VisitContainer(DocumentContainerRun* run)
|
|
{
|
|
for (vint i = run->runs.Count() - 1; i >= 0; i--)
|
|
{
|
|
Ptr<DocumentRun> subRun = run->runs[i];
|
|
RunRange range = runRanges[subRun.Obj()];
|
|
if (range.start<end && start<range.end)
|
|
{
|
|
subRun->Accept(this);
|
|
}
|
|
}
|
|
}
|
|
|
|
void Visit(DocumentTextRun* run)override
|
|
{
|
|
VisitContentRun(run);
|
|
}
|
|
|
|
void Visit(DocumentStylePropertiesRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
|
|
void Visit(DocumentStyleApplicationRun* run)override
|
|
{
|
|
auto oldStyleName = currentStyleName;
|
|
currentStyleName = run->styleName;
|
|
VisitContainer(run);
|
|
currentStyleName = oldStyleName;
|
|
}
|
|
|
|
void Visit(DocumentHyperlinkRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
|
|
void Visit(DocumentImageRun* run)override
|
|
{
|
|
VisitContentRun(run);
|
|
}
|
|
|
|
void Visit(DocumentEmbeddedObjectRun* run)override
|
|
{
|
|
VisitContentRun(run);
|
|
}
|
|
|
|
void Visit(DocumentParagraphRun* run)override
|
|
{
|
|
VisitContainer(run);
|
|
}
|
|
};
|
|
}
|
|
using namespace document_operation_visitors;
|
|
|
|
namespace document_editor
|
|
{
|
|
Ptr<DocumentStyleProperties> SummarizeStyle(DocumentParagraphRun* run, RunRangeMap& runRanges, DocumentModel* model, vint start, vint end)
|
|
{
|
|
SummarizeStyleVisitor visitor(runRanges, model, start, end);
|
|
run->Accept(&visitor);
|
|
return visitor.style;
|
|
}
|
|
|
|
Nullable<WString> SummarizeStyleName(DocumentParagraphRun* run, RunRangeMap& runRanges, DocumentModel* model, vint start, vint end)
|
|
{
|
|
SummarizeStyleNameVisitor visitor(runRanges, model, start, end);
|
|
run->Accept(&visitor);
|
|
return visitor.styleName;
|
|
}
|
|
|
|
template<typename T>
|
|
void AggregateStyleItem(Ptr<DocumentStyleProperties>& dst, Ptr<DocumentStyleProperties> src, Nullable<T> DocumentStyleProperties::* field)
|
|
{
|
|
if (dst.Obj()->*field && (!(src.Obj()->*field) || (dst.Obj()->*field).Value() != (src.Obj()->*field).Value()))
|
|
{
|
|
dst.Obj()->*field = Nullable<T>();
|
|
}
|
|
}
|
|
|
|
void AggregateStyle(Ptr<DocumentStyleProperties>& dst, Ptr<DocumentStyleProperties> src)
|
|
{
|
|
AggregateStyleItem(dst, src, &DocumentStyleProperties::face);
|
|
AggregateStyleItem(dst, src, &DocumentStyleProperties::size);
|
|
AggregateStyleItem(dst, src, &DocumentStyleProperties::color);
|
|
AggregateStyleItem(dst, src, &DocumentStyleProperties::backgroundColor);
|
|
AggregateStyleItem(dst, src, &DocumentStyleProperties::bold);
|
|
AggregateStyleItem(dst, src, &DocumentStyleProperties::italic);
|
|
AggregateStyleItem(dst, src, &DocumentStyleProperties::underline);
|
|
AggregateStyleItem(dst, src, &DocumentStyleProperties::strikeline);
|
|
AggregateStyleItem(dst, src, &DocumentStyleProperties::antialias);
|
|
AggregateStyleItem(dst, src, &DocumentStyleProperties::verticalAntialias);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
.\RESOURCES\GUIDOCUMENT_EDIT.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
using namespace collections;
|
|
using namespace document_editor;
|
|
|
|
/***********************************************************************
|
|
DocumentModel::EditRangeOperations
|
|
***********************************************************************/
|
|
|
|
bool DocumentModel::CheckEditRange(TextPos begin, TextPos end, RunRangeMap& relatedRanges)
|
|
{
|
|
// check caret range
|
|
if(begin>end) return false;
|
|
if(begin.row<0 || begin.row>=paragraphs.Count()) return false;
|
|
if(end.row<0 || end.row>=paragraphs.Count()) return false;
|
|
|
|
// determine run ranges
|
|
GetRunRange(paragraphs[begin.row].Obj(), relatedRanges);
|
|
if(begin.row!=end.row)
|
|
{
|
|
GetRunRange(paragraphs[end.row].Obj(), relatedRanges);
|
|
}
|
|
|
|
// check caret range
|
|
RunRange beginRange=relatedRanges[paragraphs[begin.row].Obj()];
|
|
RunRange endRange=relatedRanges[paragraphs[end.row].Obj()];
|
|
if(begin.column<0 || begin.column>beginRange.end) return false;
|
|
if(end.column<0 || end.column>endRange.end) return false;
|
|
|
|
return true;
|
|
}
|
|
|
|
Ptr<DocumentModel> DocumentModel::CopyDocument(TextPos begin, TextPos end, bool deepCopy)
|
|
{
|
|
// check caret range
|
|
RunRangeMap runRanges;
|
|
if(!CheckEditRange(begin, end, runRanges)) return nullptr;
|
|
|
|
// get ranges
|
|
for(vint i=begin.row+1;i<end.row;i++)
|
|
{
|
|
GetRunRange(paragraphs[i].Obj(), runRanges);
|
|
}
|
|
|
|
Ptr<DocumentModel> newDocument=new DocumentModel;
|
|
|
|
// copy paragraphs
|
|
if(begin.row==end.row)
|
|
{
|
|
newDocument->paragraphs.Add(CopyRunRecursively(paragraphs[begin.row].Obj(), runRanges, begin.column, end.column, deepCopy).Cast<DocumentParagraphRun>());
|
|
}
|
|
else
|
|
{
|
|
for(vint i=begin.row;i<=end.row;i++)
|
|
{
|
|
Ptr<DocumentParagraphRun> paragraph=paragraphs[i];
|
|
RunRange range=runRanges[paragraph.Obj()];
|
|
if(i==begin.row)
|
|
{
|
|
newDocument->paragraphs.Add(CopyRunRecursively(paragraph.Obj(), runRanges, begin.column, range.end, deepCopy).Cast<DocumentParagraphRun>());
|
|
}
|
|
else if(i==end.row)
|
|
{
|
|
newDocument->paragraphs.Add(CopyRunRecursively(paragraph.Obj(), runRanges, range.start, end.column, deepCopy).Cast<DocumentParagraphRun>());
|
|
}
|
|
else if(deepCopy)
|
|
{
|
|
newDocument->paragraphs.Add(CopyRunRecursively(paragraph.Obj(), runRanges, range.start, range.end, deepCopy).Cast<DocumentParagraphRun>());
|
|
}
|
|
else
|
|
{
|
|
newDocument->paragraphs.Add(paragraph);
|
|
}
|
|
}
|
|
}
|
|
|
|
// copy styles
|
|
List<WString> styleNames;
|
|
FOREACH(Ptr<DocumentParagraphRun>, paragraph, newDocument->paragraphs)
|
|
{
|
|
CollectStyleName(paragraph.Obj(), styleNames);
|
|
}
|
|
|
|
for(vint i=0;i<styleNames.Count();i++)
|
|
{
|
|
WString styleName=styleNames[i];
|
|
if(!newDocument->styles.Keys().Contains(styleName))
|
|
{
|
|
Ptr<DocumentStyle> style=styles[styleName];
|
|
if(deepCopy)
|
|
{
|
|
Ptr<DocumentStyle> newStyle=new DocumentStyle;
|
|
newStyle->parentStyleName=style->parentStyleName;
|
|
newStyle->styles=CopyStyle(style->styles);
|
|
newStyle->resolvedStyles=CopyStyle(style->resolvedStyles);
|
|
newDocument->styles.Add(styleName, newStyle);
|
|
}
|
|
else
|
|
{
|
|
newDocument->styles.Add(styleName, style);
|
|
}
|
|
|
|
if(!styleNames.Contains(style->parentStyleName))
|
|
{
|
|
styleNames.Add(style->parentStyleName);
|
|
}
|
|
}
|
|
}
|
|
|
|
return newDocument;
|
|
}
|
|
|
|
Ptr<DocumentModel> DocumentModel::CopyDocument()
|
|
{
|
|
// determine run ranges
|
|
RunRangeMap runRanges;
|
|
vint lastParagraphIndex = paragraphs.Count() - 1;
|
|
GetRunRange(paragraphs[lastParagraphIndex].Obj(), runRanges);
|
|
|
|
TextPos begin(0, 0);
|
|
TextPos end(lastParagraphIndex, runRanges[paragraphs[lastParagraphIndex].Obj()].end);
|
|
return CopyDocument(begin, end, true);
|
|
}
|
|
|
|
bool DocumentModel::CutParagraph(TextPos position)
|
|
{
|
|
if(position.row<0 || position.row>=paragraphs.Count()) return false;
|
|
|
|
Ptr<DocumentParagraphRun> paragraph=paragraphs[position.row];
|
|
RunRangeMap runRanges;
|
|
Ptr<DocumentRun> leftRun, rightRun;
|
|
|
|
GetRunRange(paragraph.Obj(), runRanges);
|
|
CutRun(paragraph.Obj(), runRanges, position.column, leftRun, rightRun);
|
|
|
|
CopyFrom(paragraph->runs, leftRun.Cast<DocumentParagraphRun>()->runs);
|
|
CopyFrom(paragraph->runs, rightRun.Cast<DocumentParagraphRun>()->runs, true);
|
|
|
|
return true;
|
|
}
|
|
|
|
bool DocumentModel::CutEditRange(TextPos begin, TextPos end)
|
|
{
|
|
// check caret range
|
|
if(begin>end) return false;
|
|
if(begin.row<0 || begin.row>=paragraphs.Count()) return false;
|
|
if(end.row<0 || end.row>=paragraphs.Count()) return false;
|
|
|
|
// cut paragraphs
|
|
CutParagraph(begin);
|
|
if(begin!=end)
|
|
{
|
|
CutParagraph(end);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool DocumentModel::EditContainer(TextPos begin, TextPos end, const Func<void(DocumentParagraphRun*, RunRangeMap&, vint, vint)>& editor)
|
|
{
|
|
if(begin==end) return false;
|
|
|
|
// cut paragraphs
|
|
if(!CutEditRange(begin, end)) return false;
|
|
|
|
// check caret range
|
|
RunRangeMap runRanges;
|
|
if(!CheckEditRange(begin, end, runRanges)) return false;
|
|
|
|
// edit container
|
|
if(begin.row==end.row)
|
|
{
|
|
editor(paragraphs[begin.row].Obj(), runRanges, begin.column, end.column);
|
|
}
|
|
else
|
|
{
|
|
for(vint i=begin.row;i<=end.row;i++)
|
|
{
|
|
Ptr<DocumentParagraphRun> paragraph=paragraphs[i];
|
|
if(begin.row<i && i<end.row)
|
|
{
|
|
GetRunRange(paragraph.Obj(), runRanges);
|
|
}
|
|
RunRange range=runRanges[paragraph.Obj()];
|
|
if(i==begin.row)
|
|
{
|
|
editor(paragraph.Obj(), runRanges, begin.column, range.end);
|
|
}
|
|
else if(i==end.row)
|
|
{
|
|
editor(paragraph.Obj(), runRanges, range.start, end.column);
|
|
}
|
|
else
|
|
{
|
|
editor(paragraph.Obj(), runRanges, range.start, range.end);
|
|
}
|
|
}
|
|
}
|
|
|
|
// clear paragraphs
|
|
for(vint i=begin.row;i<=end.row;i++)
|
|
{
|
|
ClearUnnecessaryRun(paragraphs[i].Obj(), this);
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
/***********************************************************************
|
|
DocumentModel::EditRun
|
|
***********************************************************************/
|
|
|
|
vint DocumentModel::EditRun(TextPos begin, TextPos end, Ptr<DocumentModel> replaceToModel, bool copy)
|
|
{
|
|
// check caret range
|
|
RunRangeMap runRanges;
|
|
if(!CheckEditRange(begin, end, runRanges)) return -1;
|
|
|
|
auto model = replaceToModel;
|
|
if (copy)
|
|
{
|
|
model = replaceToModel->CopyDocument();
|
|
}
|
|
|
|
// calculate new names for the model's styles to prevent conflicting
|
|
List<WString> oldNames, newNames;
|
|
CopyFrom(oldNames, model->styles.Keys());
|
|
CopyFrom(newNames, model->styles.Keys());
|
|
for(vint i=0;i<newNames.Count();i++)
|
|
{
|
|
WString name=newNames[i];
|
|
if((name.Length()==0 || name[0]!=L'#') && styles.Keys().Contains(name))
|
|
{
|
|
vint index=2;
|
|
while(true)
|
|
{
|
|
WString newName=name+L"_"+itow(index++);
|
|
if(!styles.Keys().Contains(newName) && !model->styles.Keys().Contains(newName))
|
|
{
|
|
newNames[i]=newName;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// rename model's styles
|
|
typedef Pair<WString, WString> NamePair;
|
|
FOREACH(NamePair, name, From(oldNames).Pairwise(newNames))
|
|
{
|
|
model->RenameStyle(name.key, name.value);
|
|
}
|
|
FOREACH(WString, name, newNames)
|
|
{
|
|
if((name.Length()==0 || name[0]!=L'#') && !styles.Keys().Contains(name))
|
|
{
|
|
styles.Add(name, model->styles[name]);
|
|
}
|
|
}
|
|
|
|
// edit runs
|
|
Array<Ptr<DocumentParagraphRun>> runs;
|
|
CopyFrom(runs, model->paragraphs);
|
|
return EditRunNoCopy(begin, end, runs);
|
|
}
|
|
|
|
vint DocumentModel::EditRunNoCopy(TextPos begin, TextPos end, const collections::Array<Ptr<DocumentParagraphRun>>& runs)
|
|
{
|
|
// check caret range
|
|
RunRangeMap runRanges;
|
|
if(!CheckEditRange(begin, end, runRanges)) return -1;
|
|
|
|
// remove unnecessary paragraphs
|
|
if(begin.row!=end.row)
|
|
{
|
|
for(vint i=end.row-1;i>begin.row;i--)
|
|
{
|
|
paragraphs.RemoveAt(i);
|
|
}
|
|
end.row=begin.row+1;
|
|
}
|
|
|
|
// remove unnecessary runs and ensure begin.row!=end.row
|
|
if(begin.row==end.row)
|
|
{
|
|
RemoveRun(paragraphs[begin.row].Obj(), runRanges, begin.column, end.column);
|
|
|
|
Ptr<DocumentRun> leftRun, rightRun;
|
|
runRanges.Clear();
|
|
GetRunRange(paragraphs[begin.row].Obj(), runRanges);
|
|
CutRun(paragraphs[begin.row].Obj(), runRanges, begin.column, leftRun, rightRun);
|
|
|
|
paragraphs.RemoveAt(begin.row);
|
|
paragraphs.Insert(begin.row, leftRun.Cast<DocumentParagraphRun>());
|
|
paragraphs.Insert(begin.row+1, rightRun.Cast<DocumentParagraphRun>());
|
|
end.row=begin.row+1;
|
|
}
|
|
else
|
|
{
|
|
RemoveRun(paragraphs[begin.row].Obj(), runRanges, begin.column, runRanges[paragraphs[begin.row].Obj()].end);
|
|
RemoveRun(paragraphs[end.row].Obj(), runRanges, 0, end.column);
|
|
}
|
|
|
|
// insert new paragraphs
|
|
Ptr<DocumentParagraphRun> beginParagraph=paragraphs[begin.row];
|
|
Ptr<DocumentParagraphRun> endParagraph=paragraphs[end.row];
|
|
if(runs.Count()==0)
|
|
{
|
|
CopyFrom(beginParagraph->runs, endParagraph->runs, true);
|
|
paragraphs.RemoveAt(end.row);
|
|
}
|
|
else if(runs.Count()==1)
|
|
{
|
|
CopyFrom(beginParagraph->runs, runs[0]->runs, true);
|
|
CopyFrom(beginParagraph->runs, endParagraph->runs, true);
|
|
paragraphs.RemoveAt(end.row);
|
|
}
|
|
else
|
|
{
|
|
Ptr<DocumentParagraphRun> newBeginRuns=runs[0];
|
|
CopyFrom(beginParagraph->runs, newBeginRuns->runs, true);
|
|
|
|
Ptr<DocumentParagraphRun> newEndRuns=runs[runs.Count()-1];
|
|
if (newEndRuns->alignment)
|
|
{
|
|
endParagraph->alignment = newEndRuns->alignment;
|
|
}
|
|
for(vint i=0;i<newEndRuns->runs.Count();i++)
|
|
{
|
|
endParagraph->runs.Insert(i, newEndRuns->runs[i]);
|
|
}
|
|
|
|
for(vint i=1;i<runs.Count()-1;i++)
|
|
{
|
|
paragraphs.Insert(begin.row+i, runs[i]);
|
|
}
|
|
}
|
|
|
|
// clear unnecessary runs
|
|
vint rows=runs.Count()==0?1:runs.Count();
|
|
for(vint i=0;i<rows;i++)
|
|
{
|
|
ClearUnnecessaryRun(paragraphs[begin.row+i].Obj(), this);
|
|
}
|
|
return rows;
|
|
}
|
|
|
|
/***********************************************************************
|
|
DocumentModel::EditText
|
|
***********************************************************************/
|
|
|
|
vint DocumentModel::EditText(TextPos begin, TextPos end, bool frontSide, const collections::Array<WString>& text)
|
|
{
|
|
// check caret range
|
|
RunRangeMap runRanges;
|
|
if(!CheckEditRange(begin, end, runRanges)) return -1;
|
|
|
|
// calcuate the position to get the text style
|
|
TextPos stylePosition;
|
|
if(frontSide)
|
|
{
|
|
stylePosition=begin;
|
|
if(stylePosition.column==0)
|
|
{
|
|
frontSide=false;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
stylePosition=end;
|
|
if(stylePosition.column==runRanges[paragraphs[end.row].Obj()].end)
|
|
{
|
|
frontSide=true;
|
|
}
|
|
}
|
|
|
|
// copy runs that contains the target style for new text
|
|
List<DocumentContainerRun*> styleRuns;
|
|
LocateStyle(paragraphs[stylePosition.row].Obj(), runRanges, stylePosition.column, frontSide, styleRuns);
|
|
|
|
// create paragraphs
|
|
Array<Ptr<DocumentParagraphRun>> runs(text.Count());
|
|
for(vint i=0;i<text.Count();i++)
|
|
{
|
|
Ptr<DocumentRun> paragraph=CopyStyledText(styleRuns, text[i]);
|
|
runs[i]=paragraph.Cast<DocumentParagraphRun>();
|
|
}
|
|
|
|
// replace the paragraphs
|
|
return EditRunNoCopy(begin, end, runs);
|
|
}
|
|
|
|
/***********************************************************************
|
|
DocumentModel::EditStyle
|
|
***********************************************************************/
|
|
|
|
bool DocumentModel::EditStyle(TextPos begin, TextPos end, Ptr<DocumentStyleProperties> style)
|
|
{
|
|
return EditContainer(begin, end, [=](DocumentParagraphRun* paragraph, RunRangeMap& runRanges, vint start, vint end)
|
|
{
|
|
AddStyle(paragraph, runRanges, start, end, style);
|
|
});
|
|
}
|
|
|
|
/***********************************************************************
|
|
DocumentModel::EditImage
|
|
***********************************************************************/
|
|
|
|
Ptr<DocumentImageRun> DocumentModel::EditImage(TextPos begin, TextPos end, Ptr<GuiImageData> image)
|
|
{
|
|
Ptr<DocumentImageRun> imageRun=new DocumentImageRun;
|
|
imageRun->size=image->GetImage()->GetFrame(image->GetFrameIndex())->GetSize();
|
|
imageRun->baseline=imageRun->size.y;
|
|
imageRun->image=image->GetImage();
|
|
imageRun->frameIndex=image->GetFrameIndex();
|
|
|
|
Ptr<DocumentParagraphRun> paragraph=new DocumentParagraphRun;
|
|
paragraph->runs.Add(imageRun);
|
|
|
|
Array<Ptr<DocumentParagraphRun>> runs(1);
|
|
runs[0]=paragraph;
|
|
if(EditRunNoCopy(begin, end, runs))
|
|
{
|
|
return imageRun;
|
|
}
|
|
else
|
|
{
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
DocumentModel::EditHyperlink
|
|
***********************************************************************/
|
|
|
|
bool DocumentModel::EditHyperlink(vint paragraphIndex, vint begin, vint end, const WString& reference, const WString& normalStyleName, const WString& activeStyleName)
|
|
{
|
|
auto package = GetHyperlink(paragraphIndex, begin, end);
|
|
if (package->hyperlinks.Count() > 0)
|
|
{
|
|
FOREACH(Ptr<DocumentHyperlinkRun>, run, package->hyperlinks)
|
|
{
|
|
run->reference = reference;
|
|
run->normalStyleName = normalStyleName;
|
|
run->activeStyleName = activeStyleName;
|
|
run->styleName = normalStyleName;
|
|
}
|
|
return true;
|
|
}
|
|
else if (RemoveHyperlink(paragraphIndex, begin, end))
|
|
{
|
|
CutEditRange(TextPos(paragraphIndex, begin), TextPos(paragraphIndex, end));
|
|
|
|
RunRangeMap runRanges;
|
|
Ptr<DocumentParagraphRun> paragraph = paragraphs[paragraphIndex];
|
|
GetRunRange(paragraph.Obj(), runRanges);
|
|
AddHyperlink(paragraph.Obj(), runRanges, begin, end, reference, normalStyleName, activeStyleName);
|
|
|
|
ClearUnnecessaryRun(paragraph.Obj(), this);
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool DocumentModel::RemoveHyperlink(vint paragraphIndex, vint begin, vint end)
|
|
{
|
|
RunRangeMap runRanges;
|
|
if (!CheckEditRange(TextPos(paragraphIndex, begin), TextPos(paragraphIndex, end), runRanges)) return 0;
|
|
|
|
auto paragraph = paragraphs[paragraphIndex];
|
|
auto package = LocateHyperlink(paragraph.Obj(), runRanges, paragraphIndex, begin, end);
|
|
document_editor::RemoveHyperlink(paragraph.Obj(), runRanges, package->start, package->end);
|
|
ClearUnnecessaryRun(paragraph.Obj(), this);
|
|
return true;
|
|
}
|
|
|
|
Ptr<DocumentHyperlinkRun::Package> DocumentModel::GetHyperlink(vint paragraphIndex, vint begin, vint end)
|
|
{
|
|
RunRangeMap runRanges;
|
|
if (!CheckEditRange(TextPos(paragraphIndex, begin), TextPos(paragraphIndex, end), runRanges)) return 0;
|
|
|
|
auto paragraph = paragraphs[paragraphIndex];
|
|
return LocateHyperlink(paragraph.Obj(), runRanges, paragraphIndex, begin, end);
|
|
}
|
|
|
|
/***********************************************************************
|
|
DocumentModel::EditStyleName
|
|
***********************************************************************/
|
|
|
|
bool DocumentModel::EditStyleName(TextPos begin, TextPos end, const WString& styleName)
|
|
{
|
|
return EditContainer(begin, end, [=](DocumentParagraphRun* paragraph, RunRangeMap& runRanges, vint start, vint end)
|
|
{
|
|
AddStyleName(paragraph, runRanges, start, end, styleName);
|
|
});
|
|
}
|
|
|
|
bool DocumentModel::RemoveStyleName(TextPos begin, TextPos end)
|
|
{
|
|
return EditContainer(begin, end, [=](DocumentParagraphRun* paragraph, RunRangeMap& runRanges, vint start, vint end)
|
|
{
|
|
document_editor::RemoveStyleName(paragraph, runRanges, start, end);
|
|
});
|
|
}
|
|
|
|
bool DocumentModel::RenameStyle(const WString& oldStyleName, const WString& newStyleName)
|
|
{
|
|
vint index=styles.Keys().IndexOf(oldStyleName);
|
|
if(index==-1) return false;
|
|
if(styles.Keys().Contains(newStyleName)) return false;
|
|
|
|
Ptr<DocumentStyle> style=styles.Values()[index];
|
|
styles.Remove(oldStyleName);
|
|
styles.Add(newStyleName, style);
|
|
|
|
FOREACH(Ptr<DocumentStyle>, subStyle, styles.Values())
|
|
{
|
|
if(subStyle->parentStyleName==oldStyleName)
|
|
{
|
|
subStyle->parentStyleName=newStyleName;
|
|
}
|
|
}
|
|
|
|
FOREACH(Ptr<DocumentParagraphRun>, paragraph, paragraphs)
|
|
{
|
|
ReplaceStyleName(paragraph.Obj(), oldStyleName, newStyleName);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
/***********************************************************************
|
|
DocumentModel::ClearStyle
|
|
***********************************************************************/
|
|
|
|
bool DocumentModel::ClearStyle(TextPos begin, TextPos end)
|
|
{
|
|
return EditContainer(begin, end, [=](DocumentParagraphRun* paragraph, RunRangeMap& runRanges, vint start, vint end)
|
|
{
|
|
document_editor::ClearStyle(paragraph, runRanges, start, end);
|
|
});
|
|
}
|
|
|
|
/***********************************************************************
|
|
DocumentModel::ClearStyle
|
|
***********************************************************************/
|
|
|
|
Ptr<DocumentStyleProperties> DocumentModel::SummarizeStyle(TextPos begin, TextPos end)
|
|
{
|
|
Ptr<DocumentStyleProperties> style;
|
|
RunRangeMap runRanges;
|
|
|
|
if (begin == end) goto END_OF_SUMMERIZING;
|
|
|
|
// check caret range
|
|
if (!CheckEditRange(begin, end, runRanges)) return nullptr;
|
|
|
|
// Summarize container
|
|
if (begin.row == end.row)
|
|
{
|
|
style = document_editor::SummarizeStyle(paragraphs[begin.row].Obj(), runRanges, this, begin.column, end.column);
|
|
}
|
|
else
|
|
{
|
|
for (vint i = begin.row; i <= end.row; i++)
|
|
{
|
|
Ptr<DocumentParagraphRun> paragraph = paragraphs[i];
|
|
if (begin.row < i && i < end.row)
|
|
{
|
|
GetRunRange(paragraph.Obj(), runRanges);
|
|
}
|
|
RunRange range = runRanges[paragraph.Obj()];
|
|
Ptr<DocumentStyleProperties> paragraphStyle;
|
|
if (i == begin.row)
|
|
{
|
|
paragraphStyle = document_editor::SummarizeStyle(paragraph.Obj(), runRanges, this, begin.column, range.end);
|
|
}
|
|
else if (i == end.row)
|
|
{
|
|
paragraphStyle = document_editor::SummarizeStyle(paragraph.Obj(), runRanges, this, range.start, end.column);
|
|
}
|
|
else
|
|
{
|
|
paragraphStyle = document_editor::SummarizeStyle(paragraph.Obj(), runRanges, this, range.start, range.end);
|
|
}
|
|
|
|
if (!style)
|
|
{
|
|
style = paragraphStyle;
|
|
}
|
|
else if (paragraphStyle)
|
|
{
|
|
AggregateStyle(style, paragraphStyle);
|
|
}
|
|
}
|
|
}
|
|
|
|
END_OF_SUMMERIZING:
|
|
if (!style)
|
|
{
|
|
style = new DocumentStyleProperties;
|
|
}
|
|
return style;
|
|
}
|
|
|
|
Nullable<WString> DocumentModel::SummarizeStyleName(TextPos begin, TextPos end)
|
|
{
|
|
if (begin == end) return {};
|
|
|
|
// check caret range
|
|
RunRangeMap runRanges;
|
|
if (!CheckEditRange(begin, end, runRanges)) return {};
|
|
|
|
// Summarize container
|
|
Nullable<WString> styleName;
|
|
|
|
if (begin.row == end.row)
|
|
{
|
|
styleName = document_editor::SummarizeStyleName(paragraphs[begin.row].Obj(), runRanges, this, begin.column, end.column);
|
|
}
|
|
else
|
|
{
|
|
for (vint i = begin.row; i <= end.row; i++)
|
|
{
|
|
Ptr<DocumentParagraphRun> paragraph = paragraphs[i];
|
|
if (begin.row < i && i < end.row)
|
|
{
|
|
GetRunRange(paragraph.Obj(), runRanges);
|
|
}
|
|
RunRange range = runRanges[paragraph.Obj()];
|
|
Nullable<WString> newStyleName;
|
|
if (i == begin.row)
|
|
{
|
|
newStyleName = document_editor::SummarizeStyleName(paragraph.Obj(), runRanges, this, begin.column, range.end);
|
|
}
|
|
else if (i == end.row)
|
|
{
|
|
newStyleName = document_editor::SummarizeStyleName(paragraph.Obj(), runRanges, this, range.start, end.column);
|
|
}
|
|
else
|
|
{
|
|
newStyleName = document_editor::SummarizeStyleName(paragraph.Obj(), runRanges, this, range.start, end.column);
|
|
}
|
|
|
|
if (i == begin.row)
|
|
{
|
|
styleName = newStyleName;
|
|
}
|
|
else if (!styleName || !newStyleName || styleName.Value() != newStyleName.Value())
|
|
{
|
|
styleName = Nullable<WString>();
|
|
}
|
|
}
|
|
}
|
|
return styleName;
|
|
}
|
|
|
|
Nullable<Alignment> DocumentModel::SummarizeParagraphAlignment(TextPos begin, TextPos end)
|
|
{
|
|
bool left = false;
|
|
bool center = false;
|
|
bool right = false;
|
|
|
|
RunRangeMap runRanges;
|
|
if (!CheckEditRange(begin, end, runRanges)) return {};
|
|
|
|
for (vint i = begin.row; i <= end.row; i++)
|
|
{
|
|
auto paragraph = paragraphs[i];
|
|
if (paragraph->alignment)
|
|
{
|
|
switch (paragraph->alignment.Value())
|
|
{
|
|
case Alignment::Left:
|
|
left = true;
|
|
break;
|
|
case Alignment::Center:
|
|
center = true;
|
|
break;
|
|
case Alignment::Right:
|
|
right = true;
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
left = true;
|
|
}
|
|
}
|
|
|
|
if (left && !center && !right) return Alignment::Left;
|
|
if (!left && center && !right) return Alignment::Center;
|
|
if (!left && !center && right) return Alignment::Right;
|
|
return {};
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
.\RESOURCES\GUIDOCUMENT_LOAD.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
using namespace collections;
|
|
using namespace parsing::tabling;
|
|
using namespace parsing::xml;
|
|
using namespace regex;
|
|
|
|
/***********************************************************************
|
|
document_operation_visitors::DeserializeNodeVisitor
|
|
***********************************************************************/
|
|
|
|
namespace document_operation_visitors
|
|
{
|
|
class DeserializeNodeVisitor : public XmlNode::IVisitor
|
|
{
|
|
public:
|
|
Ptr<DocumentModel> model;
|
|
Ptr<DocumentContainerRun> container;
|
|
vint paragraphIndex;
|
|
Ptr<GuiResourceItem> resource;
|
|
Ptr<GuiResourcePathResolver> resolver;
|
|
Regex regexAttributeApply;
|
|
GuiResourceError::List& errors;
|
|
|
|
DeserializeNodeVisitor(Ptr<DocumentModel> _model, Ptr<DocumentParagraphRun> _paragraph, vint _paragraphIndex, Ptr<GuiResourceItem> _resource, Ptr<GuiResourcePathResolver> _resolver, GuiResourceError::List& _errors)
|
|
:model(_model)
|
|
, container(_paragraph)
|
|
, paragraphIndex(_paragraphIndex)
|
|
, resource(_resource)
|
|
, resolver(_resolver)
|
|
, regexAttributeApply(L"/{@(<value>[^{}]+)/}")
|
|
, errors(_errors)
|
|
{
|
|
}
|
|
|
|
void PrintText(const WString& text)
|
|
{
|
|
Ptr<DocumentTextRun> run = new DocumentTextRun;
|
|
run->text = text;
|
|
container->runs.Add(run);
|
|
}
|
|
|
|
void Visit(XmlText* node)override
|
|
{
|
|
PrintText(node->content.value);
|
|
}
|
|
|
|
void Visit(XmlCData* node)override
|
|
{
|
|
PrintText(node->content.value);
|
|
}
|
|
|
|
void Visit(XmlAttribute* node)override
|
|
{
|
|
}
|
|
|
|
void Visit(XmlComment* node)override
|
|
{
|
|
}
|
|
|
|
void Visit(XmlElement* node)override
|
|
{
|
|
Ptr<DocumentContainerRun> createdContainer;
|
|
bool useTemplateInfo = false;
|
|
XmlElement* subNodeContainer = node;
|
|
|
|
if (node->name.value == L"br")
|
|
{
|
|
PrintText(L"\r\n");
|
|
}
|
|
else if (node->name.value == L"sp")
|
|
{
|
|
PrintText(L" ");
|
|
}
|
|
else if (node->name.value == L"tab")
|
|
{
|
|
PrintText(L"\t");
|
|
}
|
|
else if (node->name.value == L"img")
|
|
{
|
|
Ptr<DocumentImageRun> run = new DocumentImageRun;
|
|
run->baseline = -1;
|
|
|
|
if (Ptr<XmlAttribute> source = XmlGetAttribute(node, L"source"))
|
|
{
|
|
run->source = source->value.value;
|
|
WString protocol, path;
|
|
if (IsResourceUrl(run->source, protocol, path))
|
|
{
|
|
Ptr<GuiImageData> imageData = resolver->ResolveResource(protocol, path).Cast<GuiImageData>();
|
|
if (imageData)
|
|
{
|
|
run->image = imageData->GetImage();
|
|
}
|
|
if (run->image && run->image->GetFrameCount() > 0)
|
|
{
|
|
run->size = run->image->GetFrame(0)->GetSize();
|
|
run->frameIndex = 0;
|
|
}
|
|
}
|
|
|
|
FOREACH(Ptr<XmlAttribute>, att, node->attributes)
|
|
{
|
|
if (att->name.value == L"width")
|
|
{
|
|
run->size.x = wtoi(att->value.value);
|
|
}
|
|
else if (att->name.value == L"height")
|
|
{
|
|
run->size.y = wtoi(att->value.value);
|
|
}
|
|
else if (att->name.value == L"baseline")
|
|
{
|
|
run->baseline = wtoi(att->value.value);
|
|
}
|
|
else if (att->name.value == L"frameIndex")
|
|
{
|
|
run->frameIndex = wtoi(att->value.value);
|
|
}
|
|
else if (att->name.value != L"source")
|
|
{
|
|
errors.Add(GuiResourceError({ {resource},att->name.codeRange.start }, L"Unknown attribute in <img>: \"" + att->name.value + L"\"."));
|
|
}
|
|
}
|
|
|
|
container->runs.Add(run);
|
|
}
|
|
else
|
|
{
|
|
errors.Add(GuiResourceError({ {resource},node->codeRange.start }, L"Attribute \"source\" is missing in <img>."));
|
|
}
|
|
}
|
|
else if (node->name.value == L"object")
|
|
{
|
|
Ptr<DocumentEmbeddedObjectRun> run = new DocumentEmbeddedObjectRun;
|
|
run->baseline = -1;
|
|
|
|
if (auto name = XmlGetAttribute(node, L"name"))
|
|
{
|
|
run->name = name->value.value;
|
|
container->runs.Add(run);
|
|
}
|
|
else
|
|
{
|
|
errors.Add(GuiResourceError({ {resource},node->codeRange.start }, L"The \"name\" attribute in <object> is missing."));
|
|
}
|
|
}
|
|
else if (node->name.value == L"font")
|
|
{
|
|
Ptr<DocumentStylePropertiesRun> run = new DocumentStylePropertiesRun();
|
|
Ptr<DocumentStyleProperties> sp = new DocumentStyleProperties;
|
|
run->style = sp;
|
|
|
|
FOREACH(Ptr<XmlAttribute>, att, node->attributes)
|
|
{
|
|
if (att->name.value == L"face")
|
|
{
|
|
sp->face = att->value.value;
|
|
}
|
|
else if (att->name.value == L"size")
|
|
{
|
|
sp->size = DocumentFontSize::Parse(att->value.value);
|
|
}
|
|
else if (att->name.value == L"color")
|
|
{
|
|
sp->color = Color::Parse(att->value.value);
|
|
}
|
|
else if (att->name.value == L"bkcolor")
|
|
{
|
|
sp->backgroundColor = Color::Parse(att->value.value);
|
|
}
|
|
else
|
|
{
|
|
errors.Add(GuiResourceError({ {resource},att->name.codeRange.start }, L"Unknown attribute in <font>: \"" + att->name.value + L"\"."));
|
|
}
|
|
}
|
|
container->runs.Add(run);
|
|
createdContainer = run;
|
|
}
|
|
else if (node->name.value == L"b" || node->name.value == L"b-")
|
|
{
|
|
Ptr<DocumentStylePropertiesRun> run = new DocumentStylePropertiesRun();
|
|
run->style = new DocumentStyleProperties;
|
|
run->style->bold = node->name.value == L"b";
|
|
container->runs.Add(run);
|
|
createdContainer = run;
|
|
}
|
|
else if (node->name.value == L"i" || node->name.value == L"i-")
|
|
{
|
|
Ptr<DocumentStylePropertiesRun> run = new DocumentStylePropertiesRun();
|
|
run->style = new DocumentStyleProperties;
|
|
run->style->italic = node->name.value == L"i";
|
|
container->runs.Add(run);
|
|
createdContainer = run;
|
|
}
|
|
else if (node->name.value == L"u" || node->name.value == L"u-")
|
|
{
|
|
Ptr<DocumentStylePropertiesRun> run = new DocumentStylePropertiesRun();
|
|
run->style = new DocumentStyleProperties;
|
|
run->style->underline = node->name.value == L"u";
|
|
container->runs.Add(run);
|
|
createdContainer = run;
|
|
}
|
|
else if (node->name.value == L"s" || node->name.value == L"s-")
|
|
{
|
|
Ptr<DocumentStylePropertiesRun> run = new DocumentStylePropertiesRun();
|
|
run->style = new DocumentStyleProperties;
|
|
run->style->strikeline = node->name.value == L"s";
|
|
container->runs.Add(run);
|
|
createdContainer = run;
|
|
}
|
|
else if (node->name.value == L"ha")
|
|
{
|
|
Ptr<DocumentStylePropertiesRun> run = new DocumentStylePropertiesRun();
|
|
run->style = new DocumentStyleProperties;
|
|
run->style->antialias = true;
|
|
run->style->verticalAntialias = false;
|
|
container->runs.Add(run);
|
|
createdContainer = run;
|
|
}
|
|
else if (node->name.value == L"va")
|
|
{
|
|
Ptr<DocumentStylePropertiesRun> run = new DocumentStylePropertiesRun();
|
|
run->style = new DocumentStyleProperties;
|
|
run->style->antialias = true;
|
|
run->style->verticalAntialias = true;
|
|
container->runs.Add(run);
|
|
createdContainer = run;
|
|
}
|
|
else if (node->name.value == L"na")
|
|
{
|
|
Ptr<DocumentStylePropertiesRun> run = new DocumentStylePropertiesRun();
|
|
run->style = new DocumentStyleProperties;
|
|
run->style->antialias = false;
|
|
run->style->verticalAntialias = false;
|
|
container->runs.Add(run);
|
|
createdContainer = run;
|
|
}
|
|
else if (node->name.value == L"div")
|
|
{
|
|
if (Ptr<XmlAttribute> att = XmlGetAttribute(node, L"style"))
|
|
{
|
|
WString styleName = att->value.value;
|
|
|
|
Ptr<DocumentStyleApplicationRun> run = new DocumentStyleApplicationRun;
|
|
run->styleName = styleName;
|
|
container->runs.Add(run);
|
|
createdContainer = run;
|
|
}
|
|
else
|
|
{
|
|
createdContainer = container;
|
|
}
|
|
}
|
|
else if (node->name.value == L"a")
|
|
{
|
|
Ptr<DocumentHyperlinkRun> run = new DocumentHyperlinkRun;
|
|
run->normalStyleName = L"#NormalLink";
|
|
run->activeStyleName = L"#ActiveLink";
|
|
if (Ptr<XmlAttribute> att = XmlGetAttribute(node, L"normal"))
|
|
{
|
|
run->normalStyleName = att->value.value;
|
|
}
|
|
if (Ptr<XmlAttribute> att = XmlGetAttribute(node, L"active"))
|
|
{
|
|
run->activeStyleName = att->value.value;
|
|
}
|
|
if (Ptr<XmlAttribute> att = XmlGetAttribute(node, L"href"))
|
|
{
|
|
run->reference = att->value.value;
|
|
}
|
|
run->styleName = run->normalStyleName;
|
|
container->runs.Add(run);
|
|
createdContainer = run;
|
|
}
|
|
else if (node->name.value == L"p")
|
|
{
|
|
FOREACH(Ptr<XmlNode>, sub, node->subNodes)
|
|
{
|
|
sub->Accept(this);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (node->name.value != L"nop")
|
|
{
|
|
errors.Add(GuiResourceError({ {resource},node->codeRange.start }, L"Unknown element in <p>: \"" + node->name.value + L"\"."));
|
|
}
|
|
FOREACH(Ptr<XmlNode>, sub, node->subNodes)
|
|
{
|
|
sub->Accept(this);
|
|
}
|
|
}
|
|
|
|
if (createdContainer)
|
|
{
|
|
Ptr<DocumentContainerRun> oldContainer = container;
|
|
container = createdContainer;
|
|
FOREACH(Ptr<XmlNode>, subNode, subNodeContainer->subNodes)
|
|
{
|
|
subNode->Accept(this);
|
|
}
|
|
container = oldContainer;
|
|
}
|
|
}
|
|
|
|
void Visit(XmlInstruction* node)override
|
|
{
|
|
}
|
|
|
|
void Visit(XmlDocument* node)override
|
|
{
|
|
}
|
|
};
|
|
|
|
Ptr<DocumentStyle> ParseDocumentStyle(Ptr<GuiResourceItem> resource, Ptr<XmlElement> styleElement, GuiResourceError::List& errors)
|
|
{
|
|
Ptr<DocumentStyle> style=new DocumentStyle;
|
|
|
|
if(Ptr<XmlAttribute> parent=XmlGetAttribute(styleElement, L"parent"))
|
|
{
|
|
style->parentStyleName=parent->value.value;
|
|
}
|
|
|
|
Ptr<DocumentStyleProperties> sp=new DocumentStyleProperties;
|
|
style->styles=sp;
|
|
|
|
FOREACH(Ptr<XmlElement>, att, XmlGetElements(styleElement))
|
|
{
|
|
if(att->name.value==L"face")
|
|
{
|
|
sp->face=XmlGetValue(att);
|
|
}
|
|
else if(att->name.value==L"size")
|
|
{
|
|
sp->size=DocumentFontSize::Parse(XmlGetValue(att));
|
|
}
|
|
else if(att->name.value==L"color")
|
|
{
|
|
sp->color=Color::Parse(XmlGetValue(att));
|
|
}
|
|
else if(att->name.value==L"bkcolor")
|
|
{
|
|
sp->backgroundColor=Color::Parse(XmlGetValue(att));
|
|
}
|
|
else if(att->name.value==L"b")
|
|
{
|
|
sp->bold=XmlGetValue(att)==L"true";
|
|
}
|
|
else if(att->name.value==L"i")
|
|
{
|
|
sp->italic=XmlGetValue(att)==L"true";
|
|
}
|
|
else if(att->name.value==L"u")
|
|
{
|
|
sp->underline=XmlGetValue(att)==L"true";
|
|
}
|
|
else if(att->name.value==L"s")
|
|
{
|
|
sp->strikeline=XmlGetValue(att)==L"true";
|
|
}
|
|
else if(att->name.value==L"antialias")
|
|
{
|
|
WString value=XmlGetValue(att);
|
|
if(value==L"horizontal" || value==L"default")
|
|
{
|
|
sp->antialias=true;
|
|
sp->verticalAntialias=false;
|
|
}
|
|
else if(value==L"no")
|
|
{
|
|
sp->antialias=false;
|
|
sp->verticalAntialias=false;
|
|
}
|
|
else if(value==L"vertical")
|
|
{
|
|
sp->antialias=true;
|
|
sp->verticalAntialias=true;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
errors.Add(GuiResourceError({ {resource},att->codeRange.start }, L"Unknown element in <Style>: \"" + att->name.value + L"\"."));
|
|
}
|
|
}
|
|
|
|
return style;
|
|
}
|
|
}
|
|
using namespace document_operation_visitors;
|
|
|
|
/***********************************************************************
|
|
DocumentModel
|
|
***********************************************************************/
|
|
|
|
Ptr<DocumentModel> DocumentModel::LoadFromXml(Ptr<GuiResourceItem> resource, Ptr<parsing::xml::XmlDocument> xml, Ptr<GuiResourcePathResolver> resolver, GuiResourceError::List& errors)
|
|
{
|
|
Ptr<DocumentModel> model = new DocumentModel;
|
|
if (xml->rootElement->name.value == L"Doc")
|
|
{
|
|
FOREACH(Ptr<XmlElement>, partElement, XmlGetElements(xml->rootElement))
|
|
{
|
|
if (partElement->name.value == L"Styles")
|
|
{
|
|
FOREACH(Ptr<XmlElement>, styleElement, XmlGetElements(partElement))
|
|
{
|
|
if (styleElement->name.value == L"Style")
|
|
{
|
|
if (Ptr<XmlAttribute> name = XmlGetAttribute(styleElement, L"name"))
|
|
{
|
|
auto style = ParseDocumentStyle(resource, styleElement, errors);
|
|
auto styleName = name->value.value;
|
|
if (!model->styles.Keys().Contains(styleName))
|
|
{
|
|
model->styles.Add(styleName, style);
|
|
if (styleName.Length() > 9 && styleName.Right(9) == L"-Override")
|
|
{
|
|
auto overridedStyle = MakePtr<DocumentStyle>();
|
|
overridedStyle->styles = new DocumentStyleProperties;
|
|
MergeStyle(overridedStyle->styles, style->styles);
|
|
|
|
styleName = styleName.Left(styleName.Length() - 9);
|
|
auto index = model->styles.Keys().IndexOf(styleName);
|
|
if (index == -1)
|
|
{
|
|
model->styles.Add(styleName, overridedStyle);
|
|
}
|
|
else
|
|
{
|
|
auto originalStyle = model->styles.Values()[index];
|
|
MergeStyle(overridedStyle->styles, originalStyle->styles);
|
|
originalStyle->styles = overridedStyle->styles;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
errors.Add(GuiResourceError({ {resource},styleElement->codeRange.start }, L"Attribute \"name\" is missing in <Style>."));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
errors.Add(GuiResourceError({ {resource},styleElement->codeRange.start }, L"Unknown element in <Styles>: \"" + styleElement->name.value + L"\"."));
|
|
}
|
|
}
|
|
}
|
|
else if (partElement->name.value == L"Content")
|
|
{
|
|
FOREACH_INDEXER(Ptr<XmlElement>, p, i, XmlGetElements(partElement))
|
|
{
|
|
if (p->name.value == L"p")
|
|
{
|
|
Ptr<DocumentParagraphRun> paragraph = new DocumentParagraphRun;
|
|
if (Ptr<XmlAttribute> att = XmlGetAttribute(p, L"align"))
|
|
{
|
|
if (att->value.value == L"Left")
|
|
{
|
|
paragraph->alignment = Alignment::Left;
|
|
}
|
|
else if (att->value.value == L"Center")
|
|
{
|
|
paragraph->alignment = Alignment::Center;
|
|
}
|
|
else if (att->value.value == L"Right")
|
|
{
|
|
paragraph->alignment = Alignment::Right;
|
|
}
|
|
else
|
|
{
|
|
errors.Add(GuiResourceError({ {resource},att->value.codeRange.start }, L"Unknown value in align attribute \"" + att->value.value + L"\"."));
|
|
}
|
|
}
|
|
model->paragraphs.Add(paragraph);
|
|
DeserializeNodeVisitor visitor(model, paragraph, i, resource, resolver, errors);
|
|
p->Accept(&visitor);
|
|
}
|
|
else
|
|
{
|
|
errors.Add(GuiResourceError({ {resource},p->codeRange.start }, L"Unknown element in <Content>: \"" + p->name.value + L"\"."));
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
errors.Add(GuiResourceError({ {resource},partElement->codeRange.start }, L"Unknown element in <Doc>: \"" + partElement->name.value + L"\"."));
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
errors.Add(GuiResourceError({ {resource},xml->rootElement->codeRange.start }, L"The root element of document should be \"Doc\"."));
|
|
}
|
|
return model;
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\RESOURCES\GUIDOCUMENT_SAVE.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
using namespace collections;
|
|
using namespace parsing::xml;
|
|
|
|
/***********************************************************************
|
|
document_operation_visitors::SerializeRunVisitor
|
|
***********************************************************************/
|
|
|
|
namespace document_operation_visitors
|
|
{
|
|
class SerializeRunVisitor : public Object, public DocumentRun::IVisitor
|
|
{
|
|
protected:
|
|
Ptr<XmlElement> parent;
|
|
|
|
public:
|
|
SerializeRunVisitor(Ptr<XmlElement> _parent)
|
|
:parent(_parent)
|
|
{
|
|
}
|
|
|
|
void VisitContainer(Ptr<XmlElement> replacedParent, DocumentContainerRun* run)
|
|
{
|
|
if (replacedParent)
|
|
{
|
|
parent->subNodes.Add(replacedParent);
|
|
Ptr<XmlElement> oldParent = parent;
|
|
parent = replacedParent;
|
|
FOREACH(Ptr<DocumentRun>, subRun, run->runs)
|
|
{
|
|
subRun->Accept(this);
|
|
}
|
|
parent = oldParent;
|
|
}
|
|
else
|
|
{
|
|
FOREACH(Ptr<DocumentRun>, subRun, run->runs)
|
|
{
|
|
subRun->Accept(this);
|
|
}
|
|
}
|
|
}
|
|
|
|
void Visit(DocumentTextRun* run)override
|
|
{
|
|
if (run->text != L"")
|
|
{
|
|
auto writer = XmlElementWriter(parent).Element(L"nop");
|
|
auto begin = run->text.Buffer();
|
|
auto reading = begin;
|
|
auto last = reading;
|
|
while (true)
|
|
{
|
|
const wchar_t* tag = nullptr;
|
|
auto c = *reading;
|
|
switch (c)
|
|
{
|
|
case L'\n':
|
|
tag = L"br";
|
|
break;
|
|
case L' ':
|
|
tag = L"sp";
|
|
break;
|
|
case L'\t':
|
|
tag = L"tab";
|
|
break;
|
|
}
|
|
|
|
if (tag || c == 0)
|
|
{
|
|
if (reading > last)
|
|
{
|
|
auto end = reading[-1] == L'\r' ? reading - 1 : reading;
|
|
if (end > last)
|
|
{
|
|
writer.Text(run->text.Sub(last - begin, end - last));
|
|
}
|
|
last = reading;
|
|
}
|
|
}
|
|
|
|
if (tag)
|
|
{
|
|
writer.Element(tag);
|
|
}
|
|
else if (c == 0)
|
|
{
|
|
break;
|
|
}
|
|
reading++;
|
|
}
|
|
}
|
|
}
|
|
|
|
void Visit(DocumentStylePropertiesRun* run)override
|
|
{
|
|
Ptr<DocumentStyleProperties> sp = run->style;
|
|
Ptr<XmlElement> oldParent = parent;
|
|
if (sp->face || sp->size || sp->color)
|
|
{
|
|
Ptr<XmlElement> element = new XmlElement;
|
|
element->name.value = L"font";
|
|
parent->subNodes.Add(element);
|
|
|
|
XmlElementWriter writer(element);
|
|
if (sp->face)
|
|
{
|
|
writer.Attribute(L"face", sp->face.Value());
|
|
}
|
|
if (sp->size)
|
|
{
|
|
writer.Attribute(L"size", sp->size.Value().ToString());
|
|
}
|
|
if (sp->color)
|
|
{
|
|
writer.Attribute(L"color", sp->color.Value().ToString());
|
|
}
|
|
if (sp->backgroundColor)
|
|
{
|
|
writer.Attribute(L"bkcolor", sp->backgroundColor.Value().ToString());
|
|
}
|
|
parent = element;
|
|
}
|
|
if (sp->bold)
|
|
{
|
|
Ptr<XmlElement> element = new XmlElement;
|
|
element->name.value = sp->bold.Value() ? L"b" : L"b-";
|
|
parent->subNodes.Add(element);
|
|
parent = element;
|
|
}
|
|
if (sp->italic)
|
|
{
|
|
Ptr<XmlElement> element = new XmlElement;
|
|
element->name.value = sp->italic.Value() ? L"i" : L"i-";
|
|
parent->subNodes.Add(element);
|
|
parent = element;
|
|
}
|
|
if (sp->underline)
|
|
{
|
|
Ptr<XmlElement> element = new XmlElement;
|
|
element->name.value = sp->underline.Value() ? L"u" : L"u-";
|
|
parent->subNodes.Add(element);
|
|
parent = element;
|
|
}
|
|
if (sp->strikeline)
|
|
{
|
|
Ptr<XmlElement> element = new XmlElement;
|
|
element->name.value = sp->strikeline.Value() ? L"s" : L"s-";
|
|
parent->subNodes.Add(element);
|
|
parent = element;
|
|
}
|
|
if (sp->antialias || sp->verticalAntialias)
|
|
{
|
|
bool ha = sp->antialias ? sp->antialias.Value() : true;
|
|
bool va = sp->verticalAntialias ? sp->verticalAntialias.Value() : false;
|
|
if (!ha)
|
|
{
|
|
Ptr<XmlElement> element = new XmlElement;
|
|
element->name.value = L"ha";
|
|
parent->subNodes.Add(element);
|
|
parent = element;
|
|
}
|
|
else if (!va)
|
|
{
|
|
Ptr<XmlElement> element = new XmlElement;
|
|
element->name.value = L"va";
|
|
parent->subNodes.Add(element);
|
|
parent = element;
|
|
}
|
|
else
|
|
{
|
|
Ptr<XmlElement> element = new XmlElement;
|
|
element->name.value = L"na";
|
|
parent->subNodes.Add(element);
|
|
parent = element;
|
|
}
|
|
}
|
|
VisitContainer(0, run);
|
|
parent = oldParent;
|
|
}
|
|
|
|
void Visit(DocumentStyleApplicationRun* run)override
|
|
{
|
|
Ptr<XmlElement> element = new XmlElement;
|
|
element->name.value = L"div";
|
|
XmlElementWriter(element).Attribute(L"style", run->styleName);
|
|
VisitContainer(element, run);
|
|
}
|
|
|
|
void Visit(DocumentHyperlinkRun* run)override
|
|
{
|
|
Ptr<XmlElement> element = new XmlElement;
|
|
element->name.value = L"a";
|
|
XmlElementWriter writer(element);
|
|
if (run->normalStyleName != L"#NormalLink")
|
|
{
|
|
writer.Attribute(L"normal", run->normalStyleName);
|
|
}
|
|
if (run->activeStyleName != L"#ActiveLink")
|
|
{
|
|
writer.Attribute(L"active", run->activeStyleName);
|
|
}
|
|
if (run->reference != L"")
|
|
{
|
|
writer.Attribute(L"href", run->reference);
|
|
}
|
|
VisitContainer(element, run);
|
|
}
|
|
|
|
void Visit(DocumentImageRun* run)override
|
|
{
|
|
XmlElementWriter writer(parent);
|
|
writer
|
|
.Element(L"img")
|
|
.Attribute(L"width", itow(run->size.x))
|
|
.Attribute(L"height", itow(run->size.y))
|
|
.Attribute(L"baseline", itow(run->baseline))
|
|
.Attribute(L"frameIndex", itow(run->frameIndex))
|
|
.Attribute(L"source", run->source)
|
|
;
|
|
}
|
|
|
|
void Visit(DocumentEmbeddedObjectRun* run)override
|
|
{
|
|
XmlElementWriter writer(parent);
|
|
writer
|
|
.Element(L"object")
|
|
.Attribute(L"name", run->name)
|
|
;
|
|
}
|
|
|
|
void Visit(DocumentParagraphRun* run)override
|
|
{
|
|
Ptr<XmlElement> element = new XmlElement;
|
|
element->name.value = L"p";
|
|
|
|
XmlElementWriter writer(element);
|
|
if (run->alignment)
|
|
{
|
|
switch (run->alignment.Value())
|
|
{
|
|
case Alignment::Left:
|
|
writer.Attribute(L"align", L"Left");
|
|
break;
|
|
case Alignment::Center:
|
|
writer.Attribute(L"align", L"Center");
|
|
break;
|
|
case Alignment::Right:
|
|
writer.Attribute(L"align", L"Right");
|
|
break;
|
|
}
|
|
}
|
|
VisitContainer(element, run);
|
|
}
|
|
};
|
|
}
|
|
using namespace document_operation_visitors;
|
|
|
|
/***********************************************************************
|
|
DocumentModel
|
|
***********************************************************************/
|
|
|
|
Ptr<parsing::xml::XmlDocument> DocumentModel::SaveToXml()
|
|
{
|
|
Ptr<XmlDocument> xml=new XmlDocument;
|
|
Ptr<XmlElement> doc=new XmlElement;
|
|
doc->name.value=L"Doc";
|
|
xml->rootElement=doc;
|
|
{
|
|
Ptr<XmlElement> content=new XmlElement;
|
|
content->name.value=L"Content";
|
|
doc->subNodes.Add(content);
|
|
|
|
FOREACH(Ptr<DocumentParagraphRun>, p, paragraphs)
|
|
{
|
|
SerializeRunVisitor visitor(content);
|
|
p->Accept(&visitor);
|
|
}
|
|
}
|
|
{
|
|
Ptr<XmlElement> stylesElement=new XmlElement;
|
|
stylesElement->name.value=L"Styles";
|
|
doc->subNodes.Add(stylesElement);
|
|
|
|
for(vint i=0;i<styles.Count();i++)
|
|
{
|
|
WString name=styles.Keys()[i];
|
|
if (name.Length()>0 && name[0] == L'#' && (name.Length() <= 9 || name.Right(9) != L"-Override")) continue;
|
|
|
|
Ptr<DocumentStyle> style=styles.Values().Get(i);
|
|
Ptr<DocumentStyleProperties> sp=style->styles;
|
|
Ptr<XmlElement> styleElement=new XmlElement;
|
|
styleElement->name.value=L"Style";
|
|
stylesElement->subNodes.Add(styleElement);
|
|
|
|
XmlElementWriter(styleElement).Attribute(L"name", name);
|
|
if(style->parentStyleName!=L"")
|
|
{
|
|
XmlElementWriter(styleElement).Attribute(L"parent", style->parentStyleName);
|
|
}
|
|
|
|
if(sp->face) XmlElementWriter(styleElement).Element(L"face").Text( sp->face.Value() );
|
|
if(sp->size) XmlElementWriter(styleElement).Element(L"size").Text( sp->size.Value().ToString() );
|
|
if(sp->color) XmlElementWriter(styleElement).Element(L"color").Text( sp->color.Value().ToString() );
|
|
if(sp->backgroundColor) XmlElementWriter(styleElement).Element(L"bkcolor").Text( sp->backgroundColor.Value().ToString() );
|
|
if(sp->bold) XmlElementWriter(styleElement).Element(L"b").Text( sp->bold.Value()?L"true":L"false" );
|
|
if(sp->italic) XmlElementWriter(styleElement).Element(L"i").Text( sp->italic.Value()?L"true":L"false" );
|
|
if(sp->underline) XmlElementWriter(styleElement).Element(L"u").Text( sp->underline.Value()?L"true":L"false" );
|
|
if(sp->strikeline) XmlElementWriter(styleElement).Element(L"s").Text( sp->strikeline.Value()?L"true":L"false" );
|
|
if(sp->antialias && sp->verticalAntialias)
|
|
{
|
|
bool h=sp->antialias;
|
|
bool v=sp->verticalAntialias;
|
|
if(!h)
|
|
{
|
|
XmlElementWriter(styleElement).Element(L"antialias").Text(L"no");
|
|
}
|
|
else if(!v)
|
|
{
|
|
XmlElementWriter(styleElement).Element(L"antialias").Text(L"horizontal");
|
|
}
|
|
else
|
|
{
|
|
XmlElementWriter(styleElement).Element(L"antialias").Text(L"vertical");
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return xml;
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\RESOURCES\GUIPARSERMANAGER.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
using namespace collections;
|
|
using namespace controls;
|
|
using namespace parsing::tabling;
|
|
using namespace parsing::xml;
|
|
using namespace parsing::json;
|
|
using namespace regex;
|
|
|
|
/***********************************************************************
|
|
IGuiParserManager
|
|
***********************************************************************/
|
|
|
|
IGuiParserManager* parserManager=0;
|
|
|
|
IGuiParserManager* GetParserManager()
|
|
{
|
|
return parserManager;
|
|
}
|
|
|
|
class GuiParserManager : public Object, public IGuiParserManager, public IGuiPlugin
|
|
{
|
|
protected:
|
|
Dictionary<WString, Ptr<Table>> tables;
|
|
Dictionary<WString, Func<Ptr<Table>()>> loaders;
|
|
SpinLock lock;
|
|
|
|
Dictionary<WString, Ptr<IGuiGeneralParser>> parsers;
|
|
public:
|
|
|
|
GUI_PLUGIN_NAME(GacUI_Parser)
|
|
{
|
|
}
|
|
|
|
void Load()override
|
|
{
|
|
parserManager=this;
|
|
SetParsingTable(L"XML", &XmlLoadTable);
|
|
SetParsingTable(L"JSON", &JsonLoadTable);
|
|
SetTableParser(L"XML", L"XML", &XmlParseDocument);
|
|
SetTableParser(L"JSON", L"JSON", &JsonParse);
|
|
}
|
|
|
|
void Unload()override
|
|
{
|
|
parserManager=0;
|
|
}
|
|
|
|
Ptr<Table> GetParsingTable(const WString& name)override
|
|
{
|
|
SPIN_LOCK(lock)
|
|
{
|
|
vint index=tables.Keys().IndexOf(name);
|
|
if(index!=-1)
|
|
{
|
|
return tables.Values()[index];
|
|
}
|
|
|
|
index=loaders.Keys().IndexOf(name);
|
|
if(index!=-1)
|
|
{
|
|
Ptr<Table> table=loaders.Values()[index]();
|
|
tables.Add(name, table);
|
|
return table;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
bool SetParsingTable(const WString& name, Func<Ptr<Table>()> loader)override
|
|
{
|
|
if(loaders.Keys().Contains(name)) return false;
|
|
loaders.Add(name, loader);
|
|
return true;
|
|
}
|
|
|
|
Ptr<IGuiGeneralParser> GetParser(const WString& name)override
|
|
{
|
|
vint index=parsers.Keys().IndexOf(name);
|
|
return index == -1 ? nullptr : parsers.Values()[index];
|
|
}
|
|
|
|
bool SetParser(const WString& name, Ptr<IGuiGeneralParser> parser)override
|
|
{
|
|
if(parsers.Keys().Contains(name)) return false;
|
|
parsers.Add(name, parser);
|
|
return true;
|
|
}
|
|
};
|
|
GUI_REGISTER_PLUGIN(GuiParserManager)
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
.\RESOURCES\GUIRESOURCE.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
using namespace controls;
|
|
using namespace collections;
|
|
using namespace parsing;
|
|
using namespace parsing::xml;
|
|
using namespace stream;
|
|
using namespace filesystem;
|
|
|
|
WString GetFolderPath(const WString& filePath)
|
|
{
|
|
auto path = FilePath(filePath).GetFolder().GetFullPath();
|
|
if (path != L"")
|
|
{
|
|
if (path[path.Length() - 1] != FilePath::Delimiter)
|
|
{
|
|
path += FilePath::Delimiter;
|
|
}
|
|
}
|
|
return path;
|
|
}
|
|
|
|
WString GetFileName(const WString& filePath)
|
|
{
|
|
return FilePath(filePath).GetName();
|
|
}
|
|
|
|
bool LoadTextFile(const WString& filePath, WString& text)
|
|
{
|
|
BomEncoder::Encoding encoding;
|
|
bool bom;
|
|
return File(filePath).ReadAllTextWithEncodingTesting(text, encoding, bom);
|
|
}
|
|
|
|
bool IsResourceUrl(const WString& text, WString& protocol, WString& path)
|
|
{
|
|
Pair<vint, vint> index = INVLOC.FindFirst(text, L"://", Locale::None);
|
|
if (index.key != -1)
|
|
{
|
|
protocol = INVLOC.ToLower(text.Sub(0, index.key));
|
|
path = text.Sub(index.key + index.value, text.Length() - index.key - index.value);
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
vint HexToInt(wchar_t c)
|
|
{
|
|
if (L'0' <= c && c <= L'9')
|
|
{
|
|
return c - L'0';
|
|
}
|
|
else if (L'A' <= c && c <= L'F')
|
|
{
|
|
return c - L'A' + 10;
|
|
}
|
|
else
|
|
{
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
void HexToBinary(stream::IStream& binaryStream, const WString& hexText)
|
|
{
|
|
const wchar_t* buffer = hexText.Buffer();
|
|
vint count = hexText.Length() / 2;
|
|
for (vint i = 0; i < count; i++)
|
|
{
|
|
vuint8_t byte = (vuint8_t)(HexToInt(buffer[0]) * 16 + HexToInt(buffer[1]));
|
|
buffer += 2;
|
|
binaryStream.Write(&byte, 1);
|
|
}
|
|
}
|
|
|
|
WString BinaryToHex(stream::IStream& binaryStream)
|
|
{
|
|
stream::MemoryStream memoryStream;
|
|
{
|
|
stream::StreamWriter writer(memoryStream);
|
|
vuint8_t byte;
|
|
while (binaryStream.Read(&byte, 1) == 1)
|
|
{
|
|
writer.WriteChar(L"0123456789ABCDEF"[byte / 16]);
|
|
writer.WriteChar(L"0123456789ABCDEF"[byte % 16]);
|
|
}
|
|
}
|
|
memoryStream.SeekFromBegin(0);
|
|
{
|
|
stream::StreamReader reader(memoryStream);
|
|
return reader.ReadToEnd();
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GlobalStringKey
|
|
***********************************************************************/
|
|
|
|
GlobalStringKey GlobalStringKey::Empty;
|
|
GlobalStringKey GlobalStringKey::_InferType;
|
|
GlobalStringKey GlobalStringKey::_Set;
|
|
GlobalStringKey GlobalStringKey::_Ref;
|
|
GlobalStringKey GlobalStringKey::_Bind;
|
|
GlobalStringKey GlobalStringKey::_Format;
|
|
GlobalStringKey GlobalStringKey::_Str;
|
|
GlobalStringKey GlobalStringKey::_Eval;
|
|
GlobalStringKey GlobalStringKey::_Uri;
|
|
GlobalStringKey GlobalStringKey::_ControlTemplate;
|
|
|
|
class GlobalStringKeyManager
|
|
{
|
|
public:
|
|
Dictionary<WString, vint> stoi;
|
|
List<WString> itos;
|
|
|
|
void InitializeConstants()
|
|
{
|
|
GlobalStringKey::_Set = GlobalStringKey::Get(L"set");
|
|
GlobalStringKey::_InferType = GlobalStringKey::Get(L"_");
|
|
GlobalStringKey::_Ref = GlobalStringKey::Get(L"ref");
|
|
GlobalStringKey::_Bind = GlobalStringKey::Get(L"bind");
|
|
GlobalStringKey::_Format = GlobalStringKey::Get(L"format");
|
|
GlobalStringKey::_Str = GlobalStringKey::Get(L"str");
|
|
GlobalStringKey::_Eval = GlobalStringKey::Get(L"eval");
|
|
GlobalStringKey::_Uri = GlobalStringKey::Get(L"uri");
|
|
GlobalStringKey::_ControlTemplate = GlobalStringKey::Get(L"ControlTemplate");
|
|
}
|
|
}*globalStringKeyManager = 0;
|
|
|
|
GlobalStringKey GlobalStringKey::Get(const WString& string)
|
|
{
|
|
GlobalStringKey key;
|
|
if (string != L"")
|
|
{
|
|
vint index = globalStringKeyManager->stoi.Keys().IndexOf(string);
|
|
if (index == -1)
|
|
{
|
|
key.key = globalStringKeyManager->itos.Add(string);
|
|
globalStringKeyManager->stoi.Add(string, key.key);
|
|
}
|
|
else
|
|
{
|
|
key.key = globalStringKeyManager->stoi.Values()[index];
|
|
}
|
|
}
|
|
return key;
|
|
}
|
|
|
|
vint GlobalStringKey::ToKey()const
|
|
{
|
|
return key;
|
|
}
|
|
|
|
WString GlobalStringKey::ToString()const
|
|
{
|
|
return *this == GlobalStringKey::Empty
|
|
? L""
|
|
: globalStringKeyManager->itos[key];
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiImageData
|
|
***********************************************************************/
|
|
|
|
GuiImageData::GuiImageData()
|
|
:frameIndex(-1)
|
|
{
|
|
}
|
|
|
|
GuiImageData::GuiImageData(Ptr<INativeImage> _image, vint _frameIndex)
|
|
: image(_image)
|
|
, frameIndex(_frameIndex)
|
|
{
|
|
}
|
|
|
|
GuiImageData::~GuiImageData()
|
|
{
|
|
}
|
|
|
|
Ptr<INativeImage> GuiImageData::GetImage()
|
|
{
|
|
return image;
|
|
}
|
|
|
|
vint GuiImageData::GetFrameIndex()
|
|
{
|
|
return frameIndex;
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiTextData
|
|
***********************************************************************/
|
|
|
|
GuiTextData::GuiTextData()
|
|
{
|
|
}
|
|
|
|
GuiTextData::GuiTextData(const WString& _text)
|
|
:text(_text)
|
|
{
|
|
}
|
|
|
|
WString GuiTextData::GetText()
|
|
{
|
|
return text;
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiResourceNodeBase
|
|
***********************************************************************/
|
|
|
|
GuiResourceNodeBase::GuiResourceNodeBase()
|
|
:parent(0)
|
|
{
|
|
}
|
|
|
|
GuiResourceNodeBase::~GuiResourceNodeBase()
|
|
{
|
|
}
|
|
|
|
const WString& GuiResourceNodeBase::GetName()
|
|
{
|
|
return name;
|
|
}
|
|
|
|
WString GuiResourceNodeBase::GetResourcePath()
|
|
{
|
|
auto resourcePath = name;
|
|
auto current = parent;
|
|
while (current && current->GetParent())
|
|
{
|
|
resourcePath = current->GetName() + L"/" + resourcePath;
|
|
current = current->GetParent();
|
|
}
|
|
return resourcePath;
|
|
}
|
|
|
|
const WString& GuiResourceNodeBase::GetFileContentPath()
|
|
{
|
|
return fileContentPath;
|
|
}
|
|
|
|
const WString& GuiResourceNodeBase::GetFileAbsolutePath()
|
|
{
|
|
return fileAbsolutePath;
|
|
}
|
|
|
|
void GuiResourceNodeBase::SetFileContentPath(const WString& content, const WString& absolute)
|
|
{
|
|
fileContentPath = content;
|
|
fileAbsolutePath = absolute;
|
|
}
|
|
|
|
GuiResourceFolder* GuiResourceNodeBase::GetParent()
|
|
{
|
|
return parent;
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiResourceLocation
|
|
***********************************************************************/
|
|
|
|
GuiResourceLocation::GuiResourceLocation(const WString& _resourcePath, const WString& _filePath)
|
|
:resourcePath(_resourcePath)
|
|
, filePath(_filePath)
|
|
{
|
|
}
|
|
|
|
GuiResourceLocation::GuiResourceLocation(Ptr<GuiResourceNodeBase> node)
|
|
{
|
|
if (node)
|
|
{
|
|
resourcePath = node->GetResourcePath();
|
|
|
|
auto current = node.Obj();
|
|
while (current)
|
|
{
|
|
if (current->GetFileContentPath() != L"")
|
|
{
|
|
filePath = current->GetFileAbsolutePath();
|
|
break;
|
|
}
|
|
current = current->GetParent();
|
|
}
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiResourceTextPos
|
|
***********************************************************************/
|
|
|
|
GuiResourceTextPos::GuiResourceTextPos(GuiResourceLocation location, parsing::ParsingTextPos position)
|
|
:originalLocation(location)
|
|
, row(position.row)
|
|
, column(position.column)
|
|
{
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiResourceError
|
|
***********************************************************************/
|
|
|
|
GuiResourceError::GuiResourceError(GuiResourceTextPos _position, const WString& _message)
|
|
:location(_position.originalLocation)
|
|
, position(_position)
|
|
, message(_message)
|
|
{
|
|
}
|
|
|
|
GuiResourceError::GuiResourceError(GuiResourceLocation _location, const WString& _message)
|
|
:location(_location)
|
|
, position(_location, {})
|
|
, message(_message)
|
|
{
|
|
}
|
|
|
|
GuiResourceError::GuiResourceError(GuiResourceLocation _location, GuiResourceTextPos _position, const WString& _message)
|
|
:location(_location)
|
|
, position(_position)
|
|
, message(_message)
|
|
{
|
|
}
|
|
|
|
template<typename TCallback>
|
|
void TransformErrors(GuiResourceError::List& errors, collections::List<Ptr<parsing::ParsingError>>& parsingErrors, GuiResourceTextPos offset, const TCallback& callback)
|
|
{
|
|
if (offset.row < 0 || offset.column < 0)
|
|
{
|
|
offset.row = 0;
|
|
offset.column = 0;
|
|
}
|
|
|
|
FOREACH(Ptr<ParsingError>, error, parsingErrors)
|
|
{
|
|
auto pos = error->codeRange.start;
|
|
if (pos.row < 0 || pos.column < 0)
|
|
{
|
|
pos = { offset.row,offset.column };
|
|
}
|
|
else
|
|
{
|
|
if (pos.row == 0)
|
|
{
|
|
pos.column += offset.column;
|
|
}
|
|
pos.row += offset.row;
|
|
}
|
|
errors.Add(callback({ offset.originalLocation,pos }, error->errorMessage));
|
|
}
|
|
}
|
|
|
|
void GuiResourceError::Transform(GuiResourceLocation _location, GuiResourceError::List& errors, collections::List<Ptr<parsing::ParsingError>>& parsingErrors)
|
|
{
|
|
Transform(_location, errors, parsingErrors, { _location,{ 0,0 } });
|
|
}
|
|
|
|
void GuiResourceError::Transform(GuiResourceLocation _location, GuiResourceError::List& errors, collections::List<Ptr<parsing::ParsingError>>& parsingErrors, parsing::ParsingTextPos offset)
|
|
{
|
|
Transform(_location, errors, parsingErrors, { _location,offset });
|
|
}
|
|
|
|
void GuiResourceError::Transform(GuiResourceLocation _location, GuiResourceError::List& errors, collections::List<Ptr<parsing::ParsingError>>& parsingErrors, GuiResourceTextPos offset)
|
|
{
|
|
TransformErrors(errors, parsingErrors, offset, [&](GuiResourceTextPos pos, const WString& message)
|
|
{
|
|
return GuiResourceError(_location, pos, message);
|
|
});
|
|
}
|
|
|
|
void GuiResourceError::SortAndLog(List& errors, collections::List<WString>& output, const WString& workingDirectory)
|
|
{
|
|
if (errors.Count() == 0) return;
|
|
SortLambda(&errors[0], errors.Count(), [](const GuiResourceError& a, const GuiResourceError& b)
|
|
{
|
|
vint result = 0;
|
|
if (result == 0) result = WString::Compare(a.location.resourcePath, b.location.resourcePath);
|
|
if (result == 0) result = WString::Compare(a.location.filePath, b.location.filePath);
|
|
if (result == 0) result = WString::Compare(a.position.originalLocation.resourcePath, b.position.originalLocation.resourcePath);
|
|
if (result == 0) result = WString::Compare(a.position.originalLocation.filePath, b.position.originalLocation.filePath);
|
|
if (result == 0) result = a.position.row - b.position.row;
|
|
if (result == 0) result = a.position.column - b.position.column;
|
|
return result;
|
|
});
|
|
|
|
FOREACH_INDEXER(GuiResourceError, error, index, errors)
|
|
{
|
|
bool needHeader = index == 0;
|
|
if (index > 0)
|
|
{
|
|
auto previousError = errors[index - 1];
|
|
if (error.location != previousError.location || error.position.originalLocation != previousError.position.originalLocation)
|
|
{
|
|
needHeader = true;
|
|
}
|
|
}
|
|
|
|
#define CONVERT_FILEPATH(FILEPATH) (workingDirectory == L"" ? FILEPATH : filesystem::FilePath(workingDirectory).GetRelativePathFor(FILEPATH))
|
|
#define CONVERT_LOCATION(LOCATION) (LOCATION).resourcePath + L" # " + CONVERT_FILEPATH((LOCATION).filePath)
|
|
if (needHeader)
|
|
{
|
|
output.Add(CONVERT_LOCATION(error.location));
|
|
if (error.location != error.position.originalLocation)
|
|
{
|
|
output.Add(L" Original: " + CONVERT_LOCATION(error.position.originalLocation));
|
|
}
|
|
}
|
|
|
|
WString prefix = L"Failed to load file \"";
|
|
WString postfix = L"\".";
|
|
if (INVLOC.StartsWith(error.message, prefix, Locale::Normalization::None) && INVLOC.EndsWith(error.message, postfix, Locale::Normalization::None))
|
|
{
|
|
auto path = error.message.Sub(prefix.Length(), error.message.Length() - prefix.Length() - postfix.Length());
|
|
path = CONVERT_FILEPATH(path);
|
|
error.message = prefix + path + postfix;
|
|
}
|
|
|
|
auto row = error.position.row;
|
|
if (row >= 0) row++;
|
|
auto column = error.position.column;
|
|
if (column >= 0) column++;
|
|
output.Add(L"(" + itow(row) + L", " + itow(column) + L"): " + error.message);
|
|
#undef CONVERT_FILEPATH
|
|
#undef CONVERT_LOCATION
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiResourceItem
|
|
***********************************************************************/
|
|
|
|
GuiResourceItem::GuiResourceItem()
|
|
{
|
|
}
|
|
|
|
GuiResourceItem::~GuiResourceItem()
|
|
{
|
|
}
|
|
|
|
const WString& GuiResourceItem::GetTypeName()
|
|
{
|
|
return typeName;
|
|
}
|
|
|
|
Ptr<DescriptableObject> GuiResourceItem::GetContent()
|
|
{
|
|
return content;
|
|
}
|
|
|
|
void GuiResourceItem::SetContent(const WString& _typeName, Ptr<DescriptableObject> value)
|
|
{
|
|
typeName = _typeName;
|
|
content = value;
|
|
}
|
|
|
|
Ptr<GuiImageData> GuiResourceItem::AsImage()
|
|
{
|
|
return content.Cast<GuiImageData>();
|
|
}
|
|
|
|
Ptr<parsing::xml::XmlDocument> GuiResourceItem::AsXml()
|
|
{
|
|
return content.Cast<XmlDocument>();
|
|
}
|
|
|
|
Ptr<GuiTextData> GuiResourceItem::AsString()
|
|
{
|
|
return content.Cast<GuiTextData>();
|
|
}
|
|
|
|
Ptr<DocumentModel> GuiResourceItem::AsDocument()
|
|
{
|
|
return content.Cast<DocumentModel>();
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiResourceFolder
|
|
***********************************************************************/
|
|
|
|
void GuiResourceFolder::LoadResourceFolderFromXml(DelayLoadingList& delayLoadings, const WString& containingFolder, Ptr<parsing::xml::XmlElement> folderXml, GuiResourceError::List& errors)
|
|
{
|
|
ClearItems();
|
|
ClearFolders();
|
|
FOREACH(Ptr<XmlElement>, element, XmlGetElements(folderXml))
|
|
{
|
|
WString name;
|
|
if (Ptr<XmlAttribute> nameAtt = XmlGetAttribute(element, L"name"))
|
|
{
|
|
name = nameAtt->value.value;
|
|
}
|
|
if (element->name.value == L"Folder")
|
|
{
|
|
if (name == L"")
|
|
{
|
|
errors.Add(GuiResourceError({ {this},element->codeRange.start }, L"A resource folder should have a name."));
|
|
}
|
|
else
|
|
{
|
|
Ptr<GuiResourceFolder> folder = new GuiResourceFolder;
|
|
if (AddFolder(name, folder))
|
|
{
|
|
WString newContainingFolder = containingFolder;
|
|
Ptr<XmlElement> newFolderXml = element;
|
|
if (Ptr<XmlAttribute> contentAtt = XmlGetAttribute(element, L"content"))
|
|
{
|
|
if (contentAtt->value.value == L"Link")
|
|
{
|
|
auto fileContentPath = XmlGetValue(element);
|
|
auto fileAbsolutePath = containingFolder + fileContentPath;
|
|
folder->SetFileContentPath(fileContentPath, fileAbsolutePath);
|
|
|
|
WString text;
|
|
if (LoadTextFile(fileAbsolutePath, text))
|
|
{
|
|
if (auto parser = GetParserManager()->GetParser<XmlDocument>(L"XML"))
|
|
{
|
|
if (auto xml = parser->Parse({ WString::Empty,fileAbsolutePath }, text, errors))
|
|
{
|
|
newContainingFolder = GetFolderPath(fileAbsolutePath);
|
|
newFolderXml = xml->rootElement;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
errors.Add(GuiResourceError({ {this},element->codeRange.start }, L"Failed to load file \"" + fileAbsolutePath + L"\"."));
|
|
}
|
|
}
|
|
else if (contentAtt->value.value == L"Import")
|
|
{
|
|
auto importUri = XmlGetValue(element);
|
|
folder->ImportFromUri(importUri, { { this },element->codeRange.start }, errors);
|
|
}
|
|
else
|
|
{
|
|
errors.Add(GuiResourceError({ { this },element->codeRange.start }, L"Folder's content attributes can only be \"Link\"."));
|
|
}
|
|
}
|
|
if (folder->GetImportUri() == L"")
|
|
{
|
|
folder->LoadResourceFolderFromXml(delayLoadings, newContainingFolder, newFolderXml, errors);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
errors.Add(GuiResourceError({ {this},element->codeRange.start }, L"Duplicated resource folder name \"" + name + L"\"."));
|
|
}
|
|
}
|
|
}
|
|
else if (element->name.value.Length() <= 3 || element->name.value.Sub(0, 4) != L"ref.")
|
|
{
|
|
WString fileContentPath;
|
|
WString fileAbsolutePath;
|
|
if (Ptr<XmlAttribute> contentAtt = XmlGetAttribute(element, L"content"))
|
|
{
|
|
if (contentAtt->value.value == L"File")
|
|
{
|
|
fileContentPath = XmlGetValue(element);
|
|
fileAbsolutePath = containingFolder + fileContentPath;
|
|
if (name == L"")
|
|
{
|
|
name = GetFileName(fileAbsolutePath);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
errors.Add(GuiResourceError({ { this },element->codeRange.start }, L"File's content attributes can only be \"File\"."));
|
|
}
|
|
}
|
|
|
|
Ptr<GuiResourceItem> item = new GuiResourceItem;
|
|
if (AddItem(name, item))
|
|
{
|
|
WString type = element->name.value;
|
|
IGuiResourceTypeResolver* typeResolver = GetResourceResolverManager()->GetTypeResolver(type);
|
|
IGuiResourceTypeResolver* preloadResolver = typeResolver;
|
|
|
|
if (typeResolver)
|
|
{
|
|
if (!typeResolver->DirectLoadXml())
|
|
{
|
|
WString preloadType = typeResolver->IndirectLoad()->GetPreloadType();
|
|
preloadResolver = GetResourceResolverManager()->GetTypeResolver(preloadType);
|
|
if (!preloadResolver)
|
|
{
|
|
errors.Add(GuiResourceError({ {this}, element->codeRange.start }, L"[INTERNAL-ERROR] Unknown resource resolver \"" + preloadType + L"\" of resource type \"" + type + L"\"."));
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
errors.Add(GuiResourceError({ {this}, element->codeRange.start }, L"Unknown resource type \"" + type + L"\"."));
|
|
}
|
|
|
|
if (typeResolver && preloadResolver)
|
|
{
|
|
if (auto directLoad = preloadResolver->DirectLoadXml())
|
|
{
|
|
{
|
|
Ptr<DescriptableObject> resource;
|
|
if (fileAbsolutePath == L"")
|
|
{
|
|
resource = directLoad->ResolveResource(item, element, errors);
|
|
}
|
|
else
|
|
{
|
|
item->SetFileContentPath(fileContentPath, fileAbsolutePath);
|
|
resource = directLoad->ResolveResource(item, fileAbsolutePath, errors);
|
|
}
|
|
item->SetContent(preloadResolver->GetType(), resource);
|
|
}
|
|
|
|
if (typeResolver != preloadResolver)
|
|
{
|
|
if (auto indirectLoad = typeResolver->IndirectLoad())
|
|
{
|
|
if (indirectLoad->IsDelayLoad())
|
|
{
|
|
DelayLoading delayLoading;
|
|
delayLoading.type = type;
|
|
delayLoading.workingDirectory = containingFolder;
|
|
delayLoading.preloadResource = item;
|
|
delayLoadings.Add(delayLoading);
|
|
}
|
|
else if (item->GetContent())
|
|
{
|
|
auto resource = indirectLoad->ResolveResource(item, 0, errors);
|
|
item->SetContent(typeResolver->GetType(), resource);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
item->SetContent(typeResolver->GetType(), nullptr);
|
|
errors.Add(GuiResourceError({ {this},element->codeRange.start }, L"[INTERNAL-ERROR] Resource type \"" + typeResolver->GetType() + L"\" is not a indirect load resource type.")); }
|
|
}
|
|
}
|
|
else
|
|
{
|
|
errors.Add(GuiResourceError({ {this},element->codeRange.start }, L"[INTERNAL-ERROR] Resource type \"" + preloadResolver->GetType() + L"\" is not a direct load resource type."));
|
|
}
|
|
}
|
|
|
|
if (!item->GetContent())
|
|
{
|
|
RemoveItem(name);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
errors.Add(GuiResourceError({ {this},element->codeRange.start }, L"Duplicated resource item name \"" + name + L"\"."));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiResourceFolder::SaveResourceFolderToXml(Ptr<parsing::xml::XmlElement> xmlParent)
|
|
{
|
|
FOREACH(Ptr<GuiResourceItem>, item, items.Values())
|
|
{
|
|
auto resolver = GetResourceResolverManager()->GetTypeResolver(item->GetTypeName());
|
|
if (resolver->XmlSerializable())
|
|
{
|
|
auto attName = MakePtr<XmlAttribute>();
|
|
attName->name.value = L"name";
|
|
attName->value.value = item->GetName();
|
|
|
|
if (item->GetFileContentPath() == L"")
|
|
{
|
|
Ptr<XmlElement> xmlElement;
|
|
|
|
if (auto directLoad = resolver->DirectLoadXml())
|
|
{
|
|
xmlElement = directLoad->Serialize(item, item->GetContent());
|
|
}
|
|
else if (auto indirectLoad = resolver->IndirectLoad())
|
|
{
|
|
if (auto preloadResolver = GetResourceResolverManager()->GetTypeResolver(indirectLoad->GetPreloadType()))
|
|
{
|
|
if (auto directLoad = preloadResolver->DirectLoadXml())
|
|
{
|
|
if (auto resource = indirectLoad->Serialize(item, item->GetContent()))
|
|
{
|
|
xmlElement = directLoad->Serialize(item, resource);
|
|
xmlElement->name.value = resolver->GetType();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (xmlElement)
|
|
{
|
|
xmlElement->attributes.Add(attName);
|
|
xmlParent->subNodes.Add(xmlElement);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
auto xmlElement = MakePtr<XmlElement>();
|
|
xmlElement->name.value = item->GetTypeName();
|
|
xmlParent->subNodes.Add(xmlElement);
|
|
|
|
auto attContent = MakePtr<XmlAttribute>();
|
|
attContent->name.value = L"content";
|
|
attContent->value.value = L"File";
|
|
xmlElement->attributes.Add(attContent);
|
|
|
|
auto xmlText = MakePtr<XmlText>();
|
|
xmlText->content.value = item->GetFileContentPath();
|
|
xmlElement->subNodes.Add(xmlText);
|
|
}
|
|
}
|
|
}
|
|
|
|
FOREACH(Ptr<GuiResourceFolder>, folder, folders.Values())
|
|
{
|
|
auto attName = MakePtr<XmlAttribute>();
|
|
attName->name.value = L"name";
|
|
attName->value.value = folder->GetName();
|
|
|
|
auto xmlFolder = MakePtr<XmlElement>();
|
|
xmlFolder->name.value = L"Folder";
|
|
xmlFolder->attributes.Add(attName);
|
|
xmlParent->subNodes.Add(xmlFolder);
|
|
|
|
if (folder->GetImportUri() != L"")
|
|
{
|
|
auto attContent = MakePtr<XmlAttribute>();
|
|
attContent->name.value = L"content";
|
|
attContent->value.value = L"Import";
|
|
xmlFolder->attributes.Add(attContent);
|
|
|
|
auto xmlText = MakePtr<XmlText>();
|
|
xmlText->content.value = folder->GetImportUri();
|
|
xmlFolder->subNodes.Add(xmlText);
|
|
}
|
|
else if (folder->GetFileContentPath() != L"")
|
|
{
|
|
auto attContent = MakePtr<XmlAttribute>();
|
|
attContent->name.value = L"content";
|
|
attContent->value.value = L"Link";
|
|
xmlFolder->attributes.Add(attContent);
|
|
|
|
auto xmlText = MakePtr<XmlText>();
|
|
xmlText->content.value = folder->GetFileContentPath();
|
|
xmlFolder->subNodes.Add(xmlText);
|
|
}
|
|
else
|
|
{
|
|
folder->SaveResourceFolderToXml(xmlFolder);
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiResourceFolder::CollectTypeNames(collections::List<WString>& typeNames)
|
|
{
|
|
if (importUri != L"") return;
|
|
FOREACH(Ptr<GuiResourceItem>, item, items.Values())
|
|
{
|
|
if (!typeNames.Contains(item->GetTypeName()))
|
|
{
|
|
typeNames.Add(item->GetTypeName());
|
|
}
|
|
}
|
|
FOREACH(Ptr<GuiResourceFolder>, folder, folders.Values())
|
|
{
|
|
folder->CollectTypeNames(typeNames);
|
|
}
|
|
}
|
|
|
|
void GuiResourceFolder::LoadResourceFolderFromBinary(DelayLoadingList& delayLoadings, stream::internal::ContextFreeReader& reader, collections::List<WString>& typeNames, GuiResourceError::List& errors)
|
|
{
|
|
vint count = 0;
|
|
reader << count;
|
|
for (vint i = 0; i < count; i++)
|
|
{
|
|
vint typeName = 0;
|
|
WString name;
|
|
reader << typeName << name;
|
|
|
|
auto resolver = GetResourceResolverManager()->GetTypeResolver(typeNames[typeName]);
|
|
Ptr<GuiResourceItem> item = new GuiResourceItem;
|
|
if(AddItem(name, item))
|
|
{
|
|
WString type = typeNames[typeName];
|
|
IGuiResourceTypeResolver* typeResolver = GetResourceResolverManager()->GetTypeResolver(type);
|
|
IGuiResourceTypeResolver* preloadResolver = typeResolver;
|
|
|
|
if(typeResolver)
|
|
{
|
|
if (!typeResolver->DirectLoadStream())
|
|
{
|
|
WString preloadType = typeResolver->IndirectLoad()->GetPreloadType();
|
|
if (preloadType != L"")
|
|
{
|
|
preloadResolver = GetResourceResolverManager()->GetTypeResolver(preloadType);
|
|
if (!preloadResolver)
|
|
{
|
|
errors.Add(GuiResourceError({ item }, L"[INTERNAL-ERROR] Unknown resource resolver \"" + preloadType + L"\" of resource type \"" + type + L"\"."));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
errors.Add(GuiResourceError({ item }, L"[BINARY] Unknown resource type \"" + type + L"\"."));
|
|
}
|
|
|
|
if(typeResolver && preloadResolver)
|
|
{
|
|
if (auto directLoad = preloadResolver->DirectLoadStream())
|
|
{
|
|
{
|
|
auto resource = directLoad->ResolveResourcePrecompiled(item, reader.input, errors);
|
|
item->SetContent(preloadResolver->GetType(), resource);
|
|
}
|
|
|
|
if (typeResolver != preloadResolver)
|
|
{
|
|
if (auto indirectLoad = typeResolver->IndirectLoad())
|
|
{
|
|
if(indirectLoad->IsDelayLoad())
|
|
{
|
|
DelayLoading delayLoading;
|
|
delayLoading.type = type;
|
|
delayLoading.preloadResource = item;
|
|
delayLoadings.Add(delayLoading);
|
|
}
|
|
else if(item->GetContent())
|
|
{
|
|
auto resource = indirectLoad->ResolveResource(item, nullptr, errors);
|
|
item->SetContent(typeResolver->GetType(), resource);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
item->SetContent(typeResolver->GetType(), nullptr);
|
|
errors.Add(GuiResourceError({ item }, L"[INTERNAL-ERROR] Resource type \"" + typeResolver->GetType() + L"\" is not a indirect load resource type."));
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
errors.Add(GuiResourceError({ item }, L"[INTERNAL-ERROR] Resource type \"" + preloadResolver->GetType() + L"\" is not a direct load resource type."));
|
|
}
|
|
}
|
|
|
|
if(!item->GetContent())
|
|
{
|
|
RemoveItem(name);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
errors.Add(GuiResourceError({ this }, L"[BINARY] Duplicated resource item name \"" + name + L"\"."));
|
|
}
|
|
}
|
|
|
|
reader << count;
|
|
for (vint i = 0; i < count; i++)
|
|
{
|
|
WString name, importUri;
|
|
reader << name << importUri;
|
|
|
|
auto folder = MakePtr<GuiResourceFolder>();
|
|
if (importUri == L"")
|
|
{
|
|
folder->LoadResourceFolderFromBinary(delayLoadings, reader, typeNames, errors);
|
|
}
|
|
else
|
|
{
|
|
folder->ImportFromUri(importUri, { { this },{0,0} }, errors);
|
|
}
|
|
AddFolder(name, folder);
|
|
}
|
|
}
|
|
|
|
void GuiResourceFolder::SaveResourceFolderToBinary(stream::internal::ContextFreeWriter& writer, collections::List<WString>& typeNames)
|
|
{
|
|
typedef Tuple<vint, WString, IGuiResourceTypeResolver_DirectLoadStream*, Ptr<GuiResourceItem>, Ptr<DescriptableObject>> ItemTuple;
|
|
List<ItemTuple> itemTuples;
|
|
|
|
FOREACH(Ptr<GuiResourceItem>, item, items.Values())
|
|
{
|
|
auto resolver = GetResourceResolverManager()->GetTypeResolver(item->GetTypeName());
|
|
if (resolver->StreamSerializable())
|
|
{
|
|
vint typeName = typeNames.IndexOf(item->GetTypeName());
|
|
WString name = item->GetName();
|
|
|
|
if (auto directLoad = resolver->DirectLoadStream())
|
|
{
|
|
itemTuples.Add(ItemTuple(typeName, name, directLoad, item, item->GetContent()));
|
|
}
|
|
else if (auto indirectLoad = resolver->IndirectLoad())
|
|
{
|
|
if (auto preloadResolver = GetResourceResolverManager()->GetTypeResolver(indirectLoad->GetPreloadType()))
|
|
{
|
|
if (auto directLoad = preloadResolver->DirectLoadStream())
|
|
{
|
|
if (auto resource = indirectLoad->Serialize(item, item->GetContent()))
|
|
{
|
|
itemTuples.Add(ItemTuple(typeName, name, directLoad, item, resource));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
vint count = itemTuples.Count();
|
|
writer << count;
|
|
FOREACH(ItemTuple, item, itemTuples)
|
|
{
|
|
vint typeName = item.f0;
|
|
WString name = item.f1;
|
|
writer << typeName << name;
|
|
|
|
auto directLoad = item.f2;
|
|
auto resource = item.f3;
|
|
auto content = item.f4;
|
|
directLoad->SerializePrecompiled(resource, content, writer.output);
|
|
}
|
|
|
|
count = folders.Count();
|
|
writer << count;
|
|
FOREACH(Ptr<GuiResourceFolder>, folder, folders.Values())
|
|
{
|
|
WString name = folder->GetName();
|
|
WString importUri = folder->GetImportUri();
|
|
writer << name << importUri;
|
|
if (importUri == L"")
|
|
{
|
|
folder->SaveResourceFolderToBinary(writer, typeNames);
|
|
}
|
|
}
|
|
}
|
|
|
|
void GuiResourceFolder::PrecompileResourceFolder(GuiResourcePrecompileContext& context, IGuiResourcePrecompileCallback* callback, GuiResourceError::List& errors)
|
|
{
|
|
if (importUri != L"") return;
|
|
FOREACH(Ptr<GuiResourceItem>, item, items.Values())
|
|
{
|
|
auto typeResolver = GetResourceResolverManager()->GetTypeResolver(item->GetTypeName());
|
|
if (auto precompile = typeResolver->Precompile())
|
|
{
|
|
if (precompile->GetPassSupport(context.passIndex) == IGuiResourceTypeResolver_Precompile::PerResource)
|
|
{
|
|
if (callback)
|
|
{
|
|
callback->OnPerResource(context.passIndex, item);
|
|
}
|
|
precompile->PerResourcePrecompile(item, context, errors);
|
|
}
|
|
}
|
|
}
|
|
|
|
FOREACH(Ptr<GuiResourceFolder>, folder, folders.Values())
|
|
{
|
|
folder->PrecompileResourceFolder(context, callback, errors);
|
|
}
|
|
}
|
|
|
|
void GuiResourceFolder::InitializeResourceFolder(GuiResourceInitializeContext& context, GuiResourceError::List& errors)
|
|
{
|
|
if (importUri != L"") return;
|
|
FOREACH(Ptr<GuiResourceItem>, item, items.Values())
|
|
{
|
|
auto typeResolver = GetResourceResolverManager()->GetTypeResolver(item->GetTypeName());
|
|
if (auto initialize = typeResolver->Initialize())
|
|
{
|
|
initialize->Initialize(item, context, errors);
|
|
}
|
|
}
|
|
|
|
FOREACH(Ptr<GuiResourceFolder>, folder, folders.Values())
|
|
{
|
|
folder->InitializeResourceFolder(context, errors);
|
|
}
|
|
}
|
|
|
|
void GuiResourceFolder::ImportFromUri(const WString& uri, GuiResourceTextPos position, GuiResourceError::List& errors)
|
|
{
|
|
SetImportUri(uri);
|
|
if (importUri.Length() == 0 || importUri[importUri.Length() - 1] != L'/')
|
|
{
|
|
errors.Add(GuiResourceError(position, L"Path of imported folder should ends with L\"/\"."));
|
|
}
|
|
else
|
|
{
|
|
WString protocol, path;
|
|
if (IsResourceUrl(importUri, protocol, path))
|
|
{
|
|
if (protocol == L"import-res")
|
|
{
|
|
auto factory = GetResourceResolverManager()->GetPathResolverFactory(protocol);
|
|
auto resolver = factory->CreateResolver(nullptr, L"");
|
|
if (auto sourceFolder = resolver->ResolveResource(path).Cast<GuiResourceFolder>())
|
|
{
|
|
CopyFrom(items, sourceFolder->items);
|
|
CopyFrom(folders, sourceFolder->folders);
|
|
}
|
|
else
|
|
{
|
|
errors.Add(GuiResourceError(position, L"Path of imported folder does not exist: \"" + importUri + L"\"."));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
errors.Add(GuiResourceError(position, L"Path of imported folder should begin with \"import-res://\"."));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
errors.Add(GuiResourceError(position, L"Invalid path of imported folder : \"" + importUri + L"\"."));
|
|
}
|
|
}
|
|
}
|
|
|
|
GuiResourceFolder::GuiResourceFolder()
|
|
{
|
|
}
|
|
|
|
GuiResourceFolder::~GuiResourceFolder()
|
|
{
|
|
}
|
|
|
|
const WString& GuiResourceFolder::GetImportUri()
|
|
{
|
|
return importUri;
|
|
}
|
|
|
|
void GuiResourceFolder::SetImportUri(const WString& uri)
|
|
{
|
|
importUri = uri;
|
|
}
|
|
|
|
const GuiResourceFolder::ItemList& GuiResourceFolder::GetItems()
|
|
{
|
|
return items.Values();
|
|
}
|
|
|
|
Ptr<GuiResourceItem> GuiResourceFolder::GetItem(const WString& name)
|
|
{
|
|
vint index=items.Keys().IndexOf(name);
|
|
return index == -1 ? nullptr : items.Values().Get(index);
|
|
}
|
|
|
|
bool GuiResourceFolder::AddItem(const WString& name, Ptr<GuiResourceItem> item)
|
|
{
|
|
if (item->GetParent() != 0 || items.Keys().Contains(name)) return false;
|
|
items.Add(name, item);
|
|
item->parent = this;
|
|
item->name = name;
|
|
return true;
|
|
}
|
|
|
|
Ptr<GuiResourceItem> GuiResourceFolder::RemoveItem(const WString& name)
|
|
{
|
|
Ptr<GuiResourceItem> item = GetItem(name);
|
|
if (!item) return 0;
|
|
items.Remove(name);
|
|
item->parent = nullptr;
|
|
item->name = L"";
|
|
return item;
|
|
}
|
|
|
|
void GuiResourceFolder::ClearItems()
|
|
{
|
|
items.Clear();
|
|
}
|
|
|
|
const GuiResourceFolder::FolderList& GuiResourceFolder::GetFolders()
|
|
{
|
|
return folders.Values();
|
|
}
|
|
|
|
Ptr<GuiResourceFolder> GuiResourceFolder::GetFolder(const WString& name)
|
|
{
|
|
vint index=folders.Keys().IndexOf(name);
|
|
return index == -1 ? nullptr : folders.Values().Get(index);
|
|
}
|
|
|
|
bool GuiResourceFolder::AddFolder(const WString& name, Ptr<GuiResourceFolder> folder)
|
|
{
|
|
if (folder->GetParent() != 0 || folders.Keys().Contains(name)) return false;
|
|
folders.Add(name, folder);
|
|
folder->parent = this;
|
|
folder->name = name;
|
|
return true;
|
|
}
|
|
|
|
Ptr<GuiResourceFolder> GuiResourceFolder::RemoveFolder(const WString& name)
|
|
{
|
|
Ptr<GuiResourceFolder> folder = GetFolder(name);
|
|
if (!folder) return 0;
|
|
folders.Remove(name);
|
|
folder->parent = nullptr;
|
|
folder->name = L"";
|
|
return folder;
|
|
}
|
|
|
|
void GuiResourceFolder::ClearFolders()
|
|
{
|
|
folders.Clear();
|
|
}
|
|
|
|
Ptr<DescriptableObject> GuiResourceFolder::GetValueByPath(const WString& path)
|
|
{
|
|
const wchar_t* buffer=path.Buffer();
|
|
const wchar_t* index=wcschr(buffer, L'\\');
|
|
if(!index) index=wcschr(buffer, '/');
|
|
|
|
if(index)
|
|
{
|
|
WString name=path.Sub(0, index-buffer);
|
|
Ptr<GuiResourceFolder> folder=GetFolder(name);
|
|
if(folder)
|
|
{
|
|
vint start=index-buffer+1;
|
|
return folder->GetValueByPath(path.Sub(start, path.Length()-start));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
Ptr<GuiResourceItem> item=GetItem(path);
|
|
if(item)
|
|
{
|
|
return item->GetContent();
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
Ptr<GuiResourceFolder> GuiResourceFolder::GetFolderByPath(const WString& path)
|
|
{
|
|
const wchar_t* buffer=path.Buffer();
|
|
const wchar_t* index=wcschr(buffer, L'\\');
|
|
if(!index) index=wcschr(buffer, '/');
|
|
if(!index) return 0;
|
|
|
|
WString name=path.Sub(0, index-buffer);
|
|
Ptr<GuiResourceFolder> folder=GetFolder(name);
|
|
|
|
if(index-buffer==path.Length()-1)
|
|
{
|
|
return folder;
|
|
}
|
|
|
|
if(folder)
|
|
{
|
|
vint start=index-buffer+1;
|
|
return folder->GetFolderByPath(path.Sub(start, path.Length()-start));
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
bool GuiResourceFolder::CreateValueByPath(const WString& path, const WString& typeName, Ptr<DescriptableObject> value)
|
|
{
|
|
const wchar_t* buffer = path.Buffer();
|
|
const wchar_t* index = wcschr(buffer, L'\\');
|
|
if (!index) index = wcschr(buffer, '/');
|
|
|
|
if(index)
|
|
{
|
|
WString name = path.Sub(0, index - buffer);
|
|
Ptr<GuiResourceFolder> folder = GetFolder(name);
|
|
if (!folder)
|
|
{
|
|
folder = new GuiResourceFolder;
|
|
AddFolder(name, folder);
|
|
}
|
|
vint start = index - buffer + 1;
|
|
return folder->CreateValueByPath(path.Sub(start, path.Length() - start), typeName, value);
|
|
}
|
|
else
|
|
{
|
|
if(GetItem(path))
|
|
{
|
|
return false;
|
|
}
|
|
|
|
auto item = new GuiResourceItem;
|
|
item->SetContent(typeName, value);
|
|
return AddItem(path, item);
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiResourceMetadata
|
|
***********************************************************************/
|
|
|
|
void GuiResourceMetadata::LoadFromXml(Ptr<parsing::xml::XmlDocument> xml, GuiResourceLocation location, GuiResourceError::List& errors)
|
|
{
|
|
auto attrName = XmlGetAttribute(xml->rootElement, L"Name");
|
|
auto attrVersion = XmlGetAttribute(xml->rootElement, L"Version");
|
|
if (!attrName || !attrVersion)
|
|
{
|
|
errors.Add(GuiResourceError(location, L"[INTERNAL-ERROR] Resource metadata lacks of Name or Version attribute."));
|
|
return;
|
|
}
|
|
name = attrName->value.value;
|
|
version = attrVersion->value.value;
|
|
dependencies.Clear();
|
|
|
|
if (auto xmlDeps = XmlGetElement(xml->rootElement, L"Dependencies"))
|
|
{
|
|
FOREACH(Ptr<XmlElement>, xmlDep, XmlGetElements(xmlDeps, L"Resource"))
|
|
{
|
|
auto attrDep = XmlGetAttribute(xmlDep, L"Name");
|
|
if (!attrDep)
|
|
{
|
|
errors.Add(GuiResourceError(location, L"[INTERNAL-ERROR] Resource dependency lacks of Name attribute."));
|
|
}
|
|
dependencies.Add(attrDep->value.value);
|
|
}
|
|
}
|
|
}
|
|
|
|
Ptr<parsing::xml::XmlDocument> GuiResourceMetadata::SaveToXml()
|
|
{
|
|
auto root = MakePtr<XmlElement>();
|
|
root->name.value = L"ResourceMetadata";
|
|
{
|
|
auto attr = MakePtr<XmlAttribute>();
|
|
attr->name.value = L"Name";
|
|
attr->value.value = name;
|
|
root->attributes.Add(attr);
|
|
}
|
|
{
|
|
auto attr = MakePtr<XmlAttribute>();
|
|
attr->name.value = L"Version";
|
|
attr->value.value = version;
|
|
root->attributes.Add(attr);
|
|
}
|
|
{
|
|
auto xmlDeps = MakePtr<XmlElement>();
|
|
xmlDeps->name.value = L"Dependencies";
|
|
root->subNodes.Add(xmlDeps);
|
|
|
|
FOREACH(WString, dep, dependencies)
|
|
{
|
|
auto xmlDep = MakePtr<XmlElement>();
|
|
xmlDep->name.value = L"Resource";
|
|
xmlDeps->subNodes.Add(xmlDep);
|
|
{
|
|
auto attr = MakePtr<XmlAttribute>();
|
|
attr->name.value = L"Name";
|
|
attr->value.value = dep;
|
|
xmlDep->attributes.Add(attr);
|
|
}
|
|
}
|
|
}
|
|
|
|
auto doc = MakePtr<XmlDocument>();
|
|
doc->rootElement = root;
|
|
return doc;
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiResource
|
|
***********************************************************************/
|
|
|
|
const wchar_t* GuiResource::CurrentVersionString = L"1.0";
|
|
|
|
void GuiResource::ProcessDelayLoading(Ptr<GuiResource> resource, DelayLoadingList& delayLoadings, GuiResourceError::List& errors)
|
|
{
|
|
FOREACH(DelayLoading, delay, delayLoadings)
|
|
{
|
|
WString type = delay.type;
|
|
WString folder = delay.workingDirectory;
|
|
Ptr<GuiResourceItem> item = delay.preloadResource;
|
|
|
|
if (auto typeResolver = GetResourceResolverManager()->GetTypeResolver(type))
|
|
{
|
|
if (auto indirectLoad = typeResolver->IndirectLoad())
|
|
{
|
|
if (item->GetContent())
|
|
{
|
|
Ptr<GuiResourcePathResolver> pathResolver = new GuiResourcePathResolver(resource, folder);
|
|
Ptr<DescriptableObject> resource = indirectLoad->ResolveResource(item, pathResolver, errors);
|
|
if (resource)
|
|
{
|
|
item->SetContent(typeResolver->GetType(), resource);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
errors.Add(GuiResourceError({ item }, L"[INTERNAL-ERROR] Resource type \"" + type + L"\" is not a indirect load resource type."));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
errors.Add(GuiResourceError({ item }, L"[INTERNAL-ERROR] Unknown resource type \"" + type + L"\"."));
|
|
}
|
|
}
|
|
}
|
|
|
|
GuiResource::GuiResource()
|
|
{
|
|
metadata = MakePtr<GuiResourceMetadata>();
|
|
metadata->version = CurrentVersionString;
|
|
}
|
|
|
|
GuiResource::~GuiResource()
|
|
{
|
|
}
|
|
|
|
Ptr<GuiResourceMetadata> GuiResource::GetMetadata()
|
|
{
|
|
return metadata;
|
|
}
|
|
|
|
WString GuiResource::GetWorkingDirectory()
|
|
{
|
|
return workingDirectory;
|
|
}
|
|
|
|
Ptr<GuiResource> GuiResource::LoadFromXml(Ptr<parsing::xml::XmlDocument> xml, const WString& filePath, const WString& workingDirectory, GuiResourceError::List& errors)
|
|
{
|
|
Ptr<GuiResource> resource = new GuiResource;
|
|
resource->SetFileContentPath(filePath, filePath);
|
|
resource->workingDirectory = workingDirectory;
|
|
DelayLoadingList delayLoadings;
|
|
resource->LoadResourceFolderFromXml(delayLoadings, resource->workingDirectory, xml->rootElement, errors);
|
|
|
|
ProcessDelayLoading(resource, delayLoadings, errors);
|
|
return resource;
|
|
}
|
|
|
|
Ptr<GuiResource> GuiResource::LoadFromXml(const WString& filePath, GuiResourceError::List& errors)
|
|
{
|
|
Ptr<XmlDocument> xml;
|
|
if(auto parser=GetParserManager()->GetParser<XmlDocument>(L"XML"))
|
|
{
|
|
WString text;
|
|
if(LoadTextFile(filePath, text))
|
|
{
|
|
xml = parser->Parse({ WString::Empty,filePath }, text, errors);
|
|
}
|
|
else
|
|
{
|
|
errors.Add(GuiResourceError({ WString::Empty,filePath }, L"Failed to load file \"" + filePath + L"\"."));
|
|
}
|
|
}
|
|
if(xml)
|
|
{
|
|
return LoadFromXml(xml, filePath, GetFolderPath(filePath), errors);
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
Ptr<parsing::xml::XmlDocument> GuiResource::SaveToXml()
|
|
{
|
|
auto xmlRoot = MakePtr<XmlElement>();
|
|
xmlRoot->name.value = L"Resource";
|
|
SaveResourceFolderToXml(xmlRoot);
|
|
|
|
auto doc = MakePtr<XmlDocument>();
|
|
doc->rootElement = xmlRoot;
|
|
return doc;
|
|
}
|
|
|
|
Ptr<GuiResource> GuiResource::LoadPrecompiledBinary(stream::IStream& binaryStream, GuiResourceError::List& errors)
|
|
{
|
|
stream::internal::ContextFreeReader reader(binaryStream);
|
|
auto resource = MakePtr<GuiResource>();
|
|
{
|
|
WString metadata;
|
|
reader << metadata;
|
|
|
|
auto parser = GetParserManager()->GetParser<XmlDocument>(L"XML");
|
|
auto xmlMetadata = parser->Parse({ resource }, metadata, errors);
|
|
if (!xmlMetadata) return nullptr;
|
|
|
|
resource->metadata->LoadFromXml(xmlMetadata, { resource }, errors);
|
|
if (errors.Count() != 0) return nullptr;
|
|
|
|
if (resource->metadata->version != CurrentVersionString)
|
|
{
|
|
errors.Add(GuiResourceError({ resource }, L"Only resource binary of version \"" + WString(CurrentVersionString) + L"\" is accepted. Please recompile the resource before loading it."));
|
|
return nullptr;
|
|
}
|
|
}
|
|
|
|
List<WString> typeNames;
|
|
reader << typeNames;
|
|
|
|
DelayLoadingList delayLoadings;
|
|
resource->LoadResourceFolderFromBinary(delayLoadings, reader, typeNames, errors);
|
|
|
|
ProcessDelayLoading(resource, delayLoadings, errors);
|
|
return resource;
|
|
}
|
|
|
|
Ptr<GuiResource> GuiResource::LoadPrecompiledBinary(stream::IStream& binaryStream)
|
|
{
|
|
GuiResourceError::List errors;
|
|
auto resource = LoadPrecompiledBinary(binaryStream, errors);
|
|
CHECK_ERROR(errors.Count() == 0, L"GuiResource::LoadPrecompiledBinary(IStream&)#There are errors.");
|
|
return resource;
|
|
}
|
|
|
|
void GuiResource::SavePrecompiledBinary(stream::IStream& binaryStream)
|
|
{
|
|
stream::internal::ContextFreeWriter writer(binaryStream);
|
|
{
|
|
auto xmlMetadata = metadata->SaveToXml();
|
|
WString xml = GenerateToStream([&](StreamWriter& writer)
|
|
{
|
|
XmlPrint(xmlMetadata, writer);
|
|
});
|
|
writer << xml;
|
|
}
|
|
List<WString> typeNames;
|
|
CollectTypeNames(typeNames);
|
|
writer << typeNames;
|
|
SaveResourceFolderToBinary(writer, typeNames);
|
|
}
|
|
|
|
Ptr<GuiResourceFolder> GuiResource::Precompile(IGuiResourcePrecompileCallback* callback, GuiResourceError::List& errors)
|
|
{
|
|
if (GetFolder(L"Precompiled"))
|
|
{
|
|
errors.Add(GuiResourceError({ this }, L"A precompiled resource cannot be compiled again."));
|
|
return nullptr;
|
|
}
|
|
|
|
GuiResourcePrecompileContext context;
|
|
context.compilerCallback = callback ? callback->GetCompilerCallback() : nullptr;
|
|
context.rootResource = this;
|
|
context.resolver = new GuiResourcePathResolver(this, workingDirectory);
|
|
context.targetFolder = new GuiResourceFolder;
|
|
|
|
auto manager = GetResourceResolverManager();
|
|
vint maxPass = manager->GetMaxPrecompilePassIndex();
|
|
List<WString> resolvers;
|
|
for (vint i = 0; i <= maxPass; i++)
|
|
{
|
|
context.passIndex = i;
|
|
{
|
|
manager->GetPerResourceResolverNames(i, resolvers);
|
|
if (resolvers.Count() > 0)
|
|
{
|
|
PrecompileResourceFolder(context, callback, errors);
|
|
}
|
|
}
|
|
{
|
|
manager->GetPerPassResolverNames(i, resolvers);
|
|
if (resolvers.Count() > 0)
|
|
{
|
|
if (callback)
|
|
{
|
|
callback->OnPerPass(i);
|
|
}
|
|
FOREACH(WString, name, resolvers)
|
|
{
|
|
auto resolver = manager->GetTypeResolver(name);
|
|
resolver->Precompile()->PerPassPrecompile(context, errors);
|
|
}
|
|
}
|
|
}
|
|
if (errors.Count() > 0)
|
|
{
|
|
return context.targetFolder;
|
|
}
|
|
}
|
|
AddFolder(L"Precompiled", context.targetFolder);
|
|
return context.targetFolder;
|
|
}
|
|
|
|
void GuiResource::Initialize(GuiResourceUsage usage, GuiResourceError::List& errors)
|
|
{
|
|
auto precompiledFolder = GetFolder(L"Precompiled");
|
|
if (!precompiledFolder)
|
|
{
|
|
CHECK_FAIL(L"GuiResource::Initialize()#Cannot initialize a non-precompiled resource.");
|
|
return;
|
|
}
|
|
|
|
GuiResourceInitializeContext context;
|
|
context.rootResource = this;
|
|
context.resolver = new GuiResourcePathResolver(this, workingDirectory);
|
|
context.targetFolder = precompiledFolder;
|
|
context.usage = usage;
|
|
|
|
vint maxPass = GetResourceResolverManager()->GetMaxInitializePassIndex();
|
|
for (vint i = 0; i <= maxPass; i++)
|
|
{
|
|
context.passIndex = i;
|
|
InitializeResourceFolder(context, errors);
|
|
}
|
|
}
|
|
|
|
Ptr<DocumentModel> GuiResource::GetDocumentByPath(const WString& path)
|
|
{
|
|
Ptr<DocumentModel> result=GetValueByPath(path).Cast<DocumentModel>();
|
|
if(!result) throw ArgumentException(L"Path not exists.", L"GuiResource::GetDocumentByPath", L"path");
|
|
return result;
|
|
}
|
|
|
|
Ptr<GuiImageData> GuiResource::GetImageByPath(const WString& path)
|
|
{
|
|
Ptr<GuiImageData> result=GetValueByPath(path).Cast<GuiImageData>();
|
|
if(!result) throw ArgumentException(L"Path not exists.", L"GuiResource::GetImageByPath", L"path");
|
|
return result;
|
|
}
|
|
|
|
Ptr<parsing::xml::XmlDocument> GuiResource::GetXmlByPath(const WString& path)
|
|
{
|
|
Ptr<XmlDocument> result=GetValueByPath(path).Cast<XmlDocument>();
|
|
if(!result) throw ArgumentException(L"Path not exists.", L"GuiResource::GetXmlByPath", L"path");
|
|
return result;
|
|
}
|
|
|
|
WString GuiResource::GetStringByPath(const WString& path)
|
|
{
|
|
Ptr<ObjectBox<WString>> result=GetValueByPath(path).Cast<ObjectBox<WString>>();
|
|
if(!result) throw ArgumentException(L"Path not exists.", L"GuiResource::GetStringByPath", L"path");
|
|
return result->Unbox();
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiResourcePathResolver
|
|
***********************************************************************/
|
|
|
|
GuiResourcePathResolver::GuiResourcePathResolver(Ptr<GuiResource> _resource, const WString& _workingDirectory)
|
|
:resource(_resource)
|
|
,workingDirectory(_workingDirectory)
|
|
{
|
|
}
|
|
|
|
GuiResourcePathResolver::~GuiResourcePathResolver()
|
|
{
|
|
}
|
|
|
|
Ptr<DescriptableObject> GuiResourcePathResolver::ResolveResource(const WString& protocol, const WString& path)
|
|
{
|
|
Ptr<IGuiResourcePathResolver> resolver;
|
|
vint index=resolvers.Keys().IndexOf(protocol);
|
|
if(index==-1)
|
|
{
|
|
IGuiResourcePathResolverFactory* factory=GetResourceResolverManager()->GetPathResolverFactory(protocol);
|
|
if(factory)
|
|
{
|
|
resolver=factory->CreateResolver(resource, workingDirectory);
|
|
}
|
|
resolvers.Add(protocol, resolver);
|
|
}
|
|
else
|
|
{
|
|
resolver=resolvers.Values()[index];
|
|
}
|
|
|
|
if(resolver)
|
|
{
|
|
return resolver->ResolveResource(path);
|
|
}
|
|
else
|
|
{
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
/***********************************************************************
|
|
GuiResourcePathResResolver
|
|
***********************************************************************/
|
|
|
|
class GuiResourcePathResResolver : public Object, public IGuiResourcePathResolver
|
|
{
|
|
protected:
|
|
Ptr<GuiResource> resource;
|
|
|
|
public:
|
|
GuiResourcePathResResolver(Ptr<GuiResource> _resource)
|
|
:resource(_resource)
|
|
{
|
|
}
|
|
|
|
Ptr<DescriptableObject> ResolveResource(const WString& path)
|
|
{
|
|
if (resource)
|
|
{
|
|
if (path.Length() > 0)
|
|
{
|
|
switch (path[path.Length() - 1])
|
|
{
|
|
case L'\\':case L'/':
|
|
return resource->GetFolderByPath(path);
|
|
default:
|
|
return resource->GetValueByPath(path);
|
|
}
|
|
}
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
class Factory : public Object, public IGuiResourcePathResolverFactory
|
|
{
|
|
public:
|
|
WString GetProtocol()override
|
|
{
|
|
return L"res";
|
|
}
|
|
|
|
Ptr<IGuiResourcePathResolver> CreateResolver(Ptr<GuiResource> resource, const WString& workingDirectory)override
|
|
{
|
|
return new GuiResourcePathResResolver(resource);
|
|
}
|
|
};
|
|
};
|
|
|
|
/***********************************************************************
|
|
GuiImportResourcePathResResolver
|
|
***********************************************************************/
|
|
|
|
class GuiImportResourcePathResResolver : public Object, public IGuiResourcePathResolver
|
|
{
|
|
public:
|
|
GuiImportResourcePathResResolver()
|
|
{
|
|
}
|
|
|
|
Ptr<DescriptableObject> ResolveResource(const WString& path)
|
|
{
|
|
const wchar_t* buffer = path.Buffer();
|
|
const wchar_t* d1 = wcschr(buffer, L'\\');
|
|
const wchar_t* d2 = wcschr(buffer, L'/');
|
|
const wchar_t* d =
|
|
d1 == nullptr&&d2 == nullptr ? nullptr :
|
|
d1 == nullptr ? d2 :
|
|
d2 == nullptr ? d1 :
|
|
d1 < d2 ? d1 : d2;
|
|
|
|
if (!d) return nullptr;
|
|
WString resourceName(buffer, d - buffer);
|
|
WString resourcePath(path.Right(path.Length() - resourceName.Length() - 1));
|
|
if (auto resource = GetResourceManager()->GetResource(resourceName))
|
|
{
|
|
switch (path[path.Length() - 1])
|
|
{
|
|
case L'\\':case L'/':
|
|
return resource->GetFolderByPath(resourcePath);
|
|
default:
|
|
return resource->GetValueByPath(resourcePath);
|
|
}
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
class Factory : public Object, public IGuiResourcePathResolverFactory
|
|
{
|
|
public:
|
|
WString GetProtocol()override
|
|
{
|
|
return L"import-res";
|
|
}
|
|
|
|
Ptr<IGuiResourcePathResolver> CreateResolver(Ptr<GuiResource> resource, const WString& workingDirectory)override
|
|
{
|
|
return new GuiImportResourcePathResResolver;
|
|
}
|
|
};
|
|
};
|
|
|
|
/***********************************************************************
|
|
IGuiResourceResolverManager
|
|
***********************************************************************/
|
|
|
|
IGuiResourceResolverManager* resourceResolverManager=0;
|
|
|
|
IGuiResourceResolverManager* GetResourceResolverManager()
|
|
{
|
|
return resourceResolverManager;
|
|
}
|
|
|
|
class GuiResourceResolverManager : public Object, public IGuiResourceResolverManager, public IGuiPlugin
|
|
{
|
|
typedef Dictionary<WString, Ptr<IGuiResourcePathResolverFactory>> PathFactoryMap;
|
|
typedef Dictionary<WString, Ptr<IGuiResourceTypeResolver>> TypeResolverMap;
|
|
typedef Group<vint, WString> ResolverGroup;
|
|
protected:
|
|
PathFactoryMap pathFactories;
|
|
TypeResolverMap typeResolvers;
|
|
ResolverGroup perResourceResolvers;
|
|
ResolverGroup perPassResolvers;
|
|
|
|
public:
|
|
|
|
GUI_PLUGIN_NAME(GacUI_Res_ResourceResolver)
|
|
{
|
|
}
|
|
|
|
void Load()override
|
|
{
|
|
globalStringKeyManager = new GlobalStringKeyManager();
|
|
globalStringKeyManager->InitializeConstants();
|
|
|
|
resourceResolverManager = this;
|
|
SetPathResolverFactory(new GuiResourcePathResResolver::Factory);
|
|
SetPathResolverFactory(new GuiImportResourcePathResResolver::Factory);
|
|
}
|
|
|
|
void Unload()override
|
|
{
|
|
delete globalStringKeyManager;
|
|
globalStringKeyManager = 0;
|
|
resourceResolverManager = 0;
|
|
}
|
|
|
|
IGuiResourcePathResolverFactory* GetPathResolverFactory(const WString& protocol)override
|
|
{
|
|
vint index=pathFactories.Keys().IndexOf(protocol);
|
|
return index==-1?0:pathFactories.Values()[index].Obj();
|
|
}
|
|
|
|
bool SetPathResolverFactory(Ptr<IGuiResourcePathResolverFactory> factory)override
|
|
{
|
|
if(pathFactories.Keys().Contains(factory->GetProtocol())) return false;
|
|
pathFactories.Add(factory->GetProtocol(), factory);
|
|
return true;
|
|
}
|
|
|
|
IGuiResourceTypeResolver* GetTypeResolver(const WString& type)override
|
|
{
|
|
vint index=typeResolvers.Keys().IndexOf(type);
|
|
return index==-1?0:typeResolvers.Values()[index].Obj();
|
|
}
|
|
|
|
bool SetTypeResolver(Ptr<IGuiResourceTypeResolver> resolver)override
|
|
{
|
|
if(typeResolvers.Keys().Contains(resolver->GetType())) return false;
|
|
typeResolvers.Add(resolver->GetType(), resolver);
|
|
|
|
if (auto precompile = resolver->Precompile())
|
|
{
|
|
vint maxPassIndex = precompile->GetMaxPassIndex();
|
|
for (vint i = 0; i <= maxPassIndex; i++)
|
|
{
|
|
switch (precompile->GetPassSupport(i))
|
|
{
|
|
case IGuiResourceTypeResolver_Precompile::PerResource:
|
|
perResourceResolvers.Add(i, resolver->GetType());
|
|
break;
|
|
case IGuiResourceTypeResolver_Precompile::PerPass:
|
|
perPassResolvers.Add(i, resolver->GetType());
|
|
break;
|
|
default:;
|
|
}
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
vint GetMaxPrecompilePassIndex()override
|
|
{
|
|
vint maxPass = -1;
|
|
FOREACH(Ptr<IGuiResourceTypeResolver>, resolver, typeResolvers.Values())
|
|
{
|
|
if (auto precompile = resolver->Precompile())
|
|
{
|
|
vint pass = precompile->GetMaxPassIndex();
|
|
if (maxPass < pass)
|
|
{
|
|
maxPass = pass;
|
|
}
|
|
}
|
|
}
|
|
return maxPass;
|
|
}
|
|
|
|
vint GetMaxInitializePassIndex()override
|
|
{
|
|
vint maxPass = -1;
|
|
FOREACH(Ptr<IGuiResourceTypeResolver>, resolver, typeResolvers.Values())
|
|
{
|
|
if (auto initialize = resolver->Initialize())
|
|
{
|
|
vint pass = initialize->GetMaxPassIndex();
|
|
if (maxPass < pass)
|
|
{
|
|
maxPass = pass;
|
|
}
|
|
}
|
|
}
|
|
return maxPass;
|
|
}
|
|
|
|
void GetPerResourceResolverNames(vint passIndex, collections::List<WString>& names)override
|
|
{
|
|
names.Clear();
|
|
vint index = perResourceResolvers.Keys().IndexOf(passIndex);
|
|
if (index != -1)
|
|
{
|
|
CopyFrom(names, perResourceResolvers.GetByIndex(index));
|
|
}
|
|
}
|
|
|
|
void GetPerPassResolverNames(vint passIndex, collections::List<WString>& names)override
|
|
{
|
|
names.Clear();
|
|
vint index = perPassResolvers.Keys().IndexOf(passIndex);
|
|
if (index != -1)
|
|
{
|
|
CopyFrom(names, perPassResolvers.GetByIndex(index));
|
|
}
|
|
}
|
|
};
|
|
GUI_REGISTER_PLUGIN(GuiResourceResolverManager)
|
|
|
|
/***********************************************************************
|
|
Helpers
|
|
***********************************************************************/
|
|
|
|
void DecompressStream(const char** buffer, bool decompress, vint rows, vint block, vint remain, stream::IStream& outputStream)
|
|
{
|
|
if (decompress)
|
|
{
|
|
MemoryStream compressedStream;
|
|
DecompressStream(buffer, false, rows, block, remain, compressedStream);
|
|
compressedStream.SeekFromBegin(0);
|
|
DecompressStream(compressedStream, outputStream);
|
|
}
|
|
else
|
|
{
|
|
for (vint i = 0; i < rows; i++)
|
|
{
|
|
vint size = i == rows - 1 ? remain : block;
|
|
outputStream.Write((void*)buffer[i], size);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
.\RESOURCES\GUIRESOURCEMANAGER.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
using namespace collections;
|
|
using namespace stream;
|
|
using namespace parsing::xml;
|
|
using namespace reflection::description;
|
|
using namespace controls;
|
|
|
|
/***********************************************************************
|
|
Class Name Record (ClassNameRecord)
|
|
***********************************************************************/
|
|
|
|
class GuiResourceClassNameRecordTypeResolver
|
|
: public Object
|
|
, public IGuiResourceTypeResolver
|
|
, private IGuiResourceTypeResolver_DirectLoadStream
|
|
{
|
|
public:
|
|
WString GetType()override
|
|
{
|
|
return L"ClassNameRecord";
|
|
}
|
|
|
|
bool XmlSerializable()override
|
|
{
|
|
return false;
|
|
}
|
|
|
|
bool StreamSerializable()override
|
|
{
|
|
return true;
|
|
}
|
|
|
|
IGuiResourceTypeResolver_DirectLoadStream* DirectLoadStream()override
|
|
{
|
|
return this;
|
|
}
|
|
|
|
void SerializePrecompiled(Ptr<GuiResourceItem> resource, Ptr<DescriptableObject> content, stream::IStream& binaryStream)override
|
|
{
|
|
if (auto obj = content.Cast<GuiResourceClassNameRecord>())
|
|
{
|
|
internal::ContextFreeWriter writer(binaryStream);
|
|
writer << obj->classNames;
|
|
}
|
|
}
|
|
|
|
Ptr<DescriptableObject> ResolveResourcePrecompiled(Ptr<GuiResourceItem> resource, stream::IStream& binaryStream, GuiResourceError::List& errors)override
|
|
{
|
|
internal::ContextFreeReader reader(binaryStream);
|
|
|
|
auto obj = MakePtr<GuiResourceClassNameRecord>();
|
|
reader << obj->classNames;
|
|
return obj;
|
|
}
|
|
};
|
|
|
|
/***********************************************************************
|
|
IGuiInstanceResourceManager
|
|
***********************************************************************/
|
|
|
|
IGuiResourceManager* resourceManager = nullptr;
|
|
|
|
IGuiResourceManager* GetResourceManager()
|
|
{
|
|
return resourceManager;
|
|
}
|
|
|
|
class GuiResourceManager : public Object, public IGuiResourceManager, public IGuiPlugin
|
|
{
|
|
protected:
|
|
typedef Dictionary<WString, Ptr<GuiResource>> ResourceMap;
|
|
|
|
List<Ptr<GuiResource>> anonymousResources;
|
|
ResourceMap resources;
|
|
ResourceMap instanceResources;
|
|
|
|
class PendingResource : public Object
|
|
{
|
|
public:
|
|
Ptr<GuiResourceMetadata> metadata;
|
|
GuiResourceUsage usage;
|
|
MemoryStream memoryStream;
|
|
SortedList<WString> dependencies;
|
|
|
|
Ptr<GuiResource> LoadResource()
|
|
{
|
|
memoryStream.SeekFromBegin(0);
|
|
List<GuiResourceError> errors;
|
|
auto resource = GuiResource::LoadPrecompiledBinary(memoryStream, errors);
|
|
CHECK_ERROR(errors.Count() == 0, L"PendingResource::LoadResource()#Failed to load the resource.");
|
|
return resource;
|
|
}
|
|
};
|
|
Group<WString, Ptr<PendingResource>> depToPendings;
|
|
SortedList<Ptr<PendingResource>> pendingResources;
|
|
|
|
public:
|
|
|
|
GUI_PLUGIN_NAME(GacUI_Res_Resource)
|
|
{
|
|
GUI_PLUGIN_DEPEND(GacUI_Res_ResourceResolver);
|
|
}
|
|
|
|
void Load()override
|
|
{
|
|
resourceManager = this;
|
|
IGuiResourceResolverManager* manager = GetResourceResolverManager();
|
|
manager->SetTypeResolver(new GuiResourceClassNameRecordTypeResolver);
|
|
}
|
|
|
|
void Unload()override
|
|
{
|
|
resourceManager = nullptr;
|
|
}
|
|
|
|
void SetResource(Ptr<GuiResource> resource, GuiResourceError::List& errors, GuiResourceUsage usage)override
|
|
{
|
|
auto metadata = resource->GetMetadata();
|
|
if (metadata->name == L"")
|
|
{
|
|
if (anonymousResources.Contains(resource.Obj())) return;
|
|
resource->Initialize(usage, errors);
|
|
if (errors.Count() > 0)
|
|
{
|
|
return;
|
|
}
|
|
anonymousResources.Add(resource);
|
|
}
|
|
else
|
|
{
|
|
CHECK_ERROR(!resources.Keys().Contains(metadata->name), L"GuiResourceManager::SetResource(Ptr<GuiResource>, GuiResourceUsage)#A resource with the same name has been loaded.");
|
|
|
|
resource->Initialize(usage, errors);
|
|
if (errors.Count() > 0)
|
|
{
|
|
return;
|
|
}
|
|
resources.Add(metadata->name, resource);
|
|
}
|
|
|
|
if (auto record = resource->GetValueByPath(L"Precompiled/ClassNameRecord").Cast<GuiResourceClassNameRecord>())
|
|
{
|
|
FOREACH(WString, className, record->classNames)
|
|
{
|
|
instanceResources.Add(className, resource);
|
|
}
|
|
}
|
|
|
|
if (metadata->name != L"")
|
|
{
|
|
vint index = depToPendings.Keys().IndexOf(metadata->name);
|
|
if (index != -1)
|
|
{
|
|
List<Ptr<PendingResource>> prs;
|
|
CopyFrom(prs, depToPendings.GetByIndex(index));
|
|
depToPendings.Remove(metadata->name);
|
|
|
|
FOREACH(Ptr<PendingResource>, pr, prs)
|
|
{
|
|
pr->dependencies.Remove(metadata->name);
|
|
if (pr->dependencies.Count() == 0)
|
|
{
|
|
pendingResources.Remove(pr.Obj());
|
|
SetResource(pr->LoadResource(), errors, pr->usage);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
Ptr<GuiResource> GetResource(const WString& name)override
|
|
{
|
|
vint index = resources.Keys().IndexOf(name);
|
|
return index == -1 ? nullptr : resources.Values()[index];
|
|
}
|
|
|
|
Ptr<GuiResource> GetResourceFromClassName(const WString& classFullName)override
|
|
{
|
|
vint index = instanceResources.Keys().IndexOf(classFullName);
|
|
if (index == -1) return nullptr;
|
|
return instanceResources.Values()[index];
|
|
}
|
|
|
|
void UnloadResource(const WString& name)override
|
|
{
|
|
vint index = resources.Keys().IndexOf(name);
|
|
if (index != -1)
|
|
{
|
|
auto resource = resources.Values()[index];
|
|
resources.Remove(name);
|
|
|
|
if (auto record = resource->GetValueByPath(L"Precompiled/ClassNameRecord").Cast<GuiResourceClassNameRecord>())
|
|
{
|
|
FOREACH(WString, className, record->classNames)
|
|
{
|
|
instanceResources.Remove(className);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void LoadResourceOrPending(stream::IStream& resourceStream, GuiResourceError::List& errors, GuiResourceUsage usage)override
|
|
{
|
|
auto pr = MakePtr<PendingResource>();
|
|
pr->usage = usage;
|
|
CopyStream(resourceStream, pr->memoryStream);
|
|
|
|
pr->metadata = MakePtr<GuiResourceMetadata>();
|
|
{
|
|
pr->memoryStream.SeekFromBegin(0);
|
|
stream::internal::ContextFreeReader reader(pr->memoryStream);
|
|
WString metadata;
|
|
reader << metadata;
|
|
|
|
List<GuiResourceError> errors;
|
|
auto parser = GetParserManager()->GetParser<XmlDocument>(L"XML");
|
|
auto xmlMetadata = parser->Parse({}, metadata, errors);
|
|
CHECK_ERROR(xmlMetadata, L"GuiResourceManager::LoadResourceOrPending(stream::IStream&, GuiResourceUsage)#This resource does not contain a valid metadata.");
|
|
pr->metadata->LoadFromXml(xmlMetadata, {}, errors);
|
|
CHECK_ERROR(errors.Count() == 0, L"GuiResourceManager::LoadResourceOrPending(stream::IStream&, GuiResourceUsage)#This resource does not contain a valid metadata.");
|
|
}
|
|
|
|
CHECK_ERROR(
|
|
pr->metadata->name != L"" || pr->dependencies.Count() == 0,
|
|
L"GuiResourceManager::LoadResourceOrPending(stream::IStream&, GuiResourceUsage)#The name of this resource cannot be empty because it has dependencies."
|
|
);
|
|
CopyFrom(pr->dependencies, From(pr->metadata->dependencies).Except(resources.Keys()));
|
|
|
|
if (pr->dependencies.Count() == 0)
|
|
{
|
|
SetResource(pr->LoadResource(), errors, pr->usage);
|
|
}
|
|
else
|
|
{
|
|
pendingResources.Add(pr);
|
|
FOREACH(WString, dep, pr->dependencies)
|
|
{
|
|
depToPendings.Add(dep, pr);
|
|
}
|
|
}
|
|
}
|
|
|
|
void LoadResourceOrPending(stream::IStream& resourceStream, GuiResourceUsage usage)override
|
|
{
|
|
GuiResourceError::List errors;
|
|
LoadResourceOrPending(resourceStream, errors, usage);
|
|
CHECK_ERROR(errors.Count() == 0, L"GuiResourceManager::LoadResourceOrPending(stream::IStream&, GuiResourceUsage)#Error happened.");
|
|
}
|
|
|
|
void GetPendingResourceNames(collections::List<WString>& names)override
|
|
{
|
|
CopyFrom(names, From(pendingResources).Select([](Ptr<PendingResource> pr) {return pr->metadata->name; }));
|
|
}
|
|
};
|
|
GUI_REGISTER_PLUGIN(GuiResourceManager)
|
|
}
|
|
}
|
|
|
|
|
|
/***********************************************************************
|
|
.\RESOURCES\GUIRESOURCETYPERESOLVERS.CPP
|
|
***********************************************************************/
|
|
|
|
namespace vl
|
|
{
|
|
namespace presentation
|
|
{
|
|
using namespace collections;
|
|
using namespace controls;
|
|
using namespace parsing;
|
|
using namespace parsing::tabling;
|
|
using namespace parsing::xml;
|
|
using namespace stream;
|
|
|
|
/***********************************************************************
|
|
Image Type Resolver (Image)
|
|
***********************************************************************/
|
|
|
|
class GuiResourceImageTypeResolver
|
|
: public Object
|
|
, public IGuiResourceTypeResolver
|
|
, private IGuiResourceTypeResolver_DirectLoadXml
|
|
, private IGuiResourceTypeResolver_DirectLoadStream
|
|
{
|
|
public:
|
|
WString GetType()override
|
|
{
|
|
return L"Image";
|
|
}
|
|
|
|
bool XmlSerializable()override
|
|
{
|
|
return true;
|
|
}
|
|
|
|
bool StreamSerializable()override
|
|
{
|
|
return true;
|
|
}
|
|
|
|
IGuiResourceTypeResolver_DirectLoadXml* DirectLoadXml()override
|
|
{
|
|
return this;
|
|
}
|
|
|
|
IGuiResourceTypeResolver_DirectLoadStream* DirectLoadStream()override
|
|
{
|
|
return this;
|
|
}
|
|
|
|
Ptr<parsing::xml::XmlElement> Serialize(Ptr<GuiResourceItem> resource, Ptr<DescriptableObject> content)override
|
|
{
|
|
return nullptr;
|
|
}
|
|
|
|
void SerializePrecompiled(Ptr<GuiResourceItem> resource, Ptr<DescriptableObject> content, stream::IStream& binaryStream)override
|
|
{
|
|
auto obj = content.Cast<GuiImageData>();
|
|
stream::internal::ContextFreeWriter writer(binaryStream);
|
|
FileStream fileStream(resource->GetFileAbsolutePath(), FileStream::ReadOnly);
|
|
writer << (stream::IStream&)fileStream;
|
|
}
|
|
|
|
Ptr<DescriptableObject> ResolveResource(Ptr<GuiResourceItem> resource, Ptr<parsing::xml::XmlElement> element, GuiResourceError::List& errors)override
|
|
{
|
|
errors.Add(GuiResourceError({ resource }, L"Image should load from file."));
|
|
return nullptr;
|
|
}
|
|
|
|
Ptr<DescriptableObject> ResolveResource(Ptr<GuiResourceItem> resource, const WString& path, GuiResourceError::List& errors)override
|
|
{
|
|
Ptr<INativeImage> image = GetCurrentController()->ImageService()->CreateImageFromFile(path);
|
|
if(image)
|
|
{
|
|
return new GuiImageData(image, 0);
|
|
}
|
|
else
|
|
{
|
|
errors.Add(GuiResourceError({ resource }, L"Failed to load file \"" + path + L"\"."));
|
|
return nullptr;
|
|
}
|
|
}
|
|
|
|
Ptr<DescriptableObject> ResolveResourcePrecompiled(Ptr<GuiResourceItem> resource, stream::IStream& binaryStream, GuiResourceError::List& errors)override
|
|
{
|
|
stream::internal::ContextFreeReader reader(binaryStream);
|
|
MemoryStream memoryStream;
|
|
reader << (stream::IStream&)memoryStream;
|
|
|
|
auto image = GetCurrentController()->ImageService()->CreateImageFromStream(memoryStream);
|
|
if (image)
|
|
{
|
|
return new GuiImageData(image, 0);
|
|
}
|
|
else
|
|
{
|
|
errors.Add(GuiResourceError({ resource }, L"[BINARY] Failed to load an image from binary data in a stream."));
|
|
return nullptr;
|
|
}
|
|
}
|
|
};
|
|
|
|
/***********************************************************************
|
|
Text Type Resolver (Text)
|
|
***********************************************************************/
|
|
|
|
class GuiResourceTextTypeResolver
|
|
: public Object
|
|
, public IGuiResourceTypeResolver
|
|
, private IGuiResourceTypeResolver_DirectLoadXml
|
|
, private IGuiResourceTypeResolver_DirectLoadStream
|
|
{
|
|
public:
|
|
WString GetType()override
|
|
{
|
|
return L"Text";
|
|
}
|
|
|
|
bool XmlSerializable()override
|
|
{
|
|
return true;
|
|
}
|
|
|
|
bool StreamSerializable()override
|
|
{
|
|
return true;
|
|
}
|
|
|
|
IGuiResourceTypeResolver_DirectLoadXml* DirectLoadXml()override
|
|
{
|
|
return this;
|
|
}
|
|
|
|
IGuiResourceTypeResolver_DirectLoadStream* DirectLoadStream()override
|
|
{
|
|
return this;
|
|
}
|
|
|
|
Ptr<parsing::xml::XmlElement> Serialize(Ptr<GuiResourceItem> resource, Ptr<DescriptableObject> content)override
|
|
{
|
|
if (auto obj = content.Cast<GuiTextData>())
|
|
{
|
|
auto xmlContent = MakePtr<XmlText>();
|
|
xmlContent->content.value = obj->GetText();
|
|
|
|
auto xmlText = MakePtr<XmlElement>();
|
|
xmlText->name.value = L"Text";
|
|
xmlText->subNodes.Add(xmlContent);
|
|
|
|
return xmlText;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
void SerializePrecompiled(Ptr<GuiResourceItem> resource, Ptr<DescriptableObject> content, stream::IStream& binaryStream)override
|
|
{
|
|
auto obj = content.Cast<GuiTextData>();
|
|
stream::internal::ContextFreeWriter writer(binaryStream);
|
|
WString text = obj->GetText();
|
|
writer << text;
|
|
}
|
|
|
|
Ptr<DescriptableObject> ResolveResource(Ptr<GuiResourceItem> resource, Ptr<parsing::xml::XmlElement> element, GuiResourceError::List& errors)override
|
|
{
|
|
return new GuiTextData(XmlGetValue(element));
|
|
}
|
|
|
|
Ptr<DescriptableObject> ResolveResource(Ptr<GuiResourceItem> resource, const WString& path, GuiResourceError::List& errors)override
|
|
{
|
|
WString text;
|
|
if(LoadTextFile(path, text))
|
|
{
|
|
return new GuiTextData(text);
|
|
}
|
|
else
|
|
{
|
|
errors.Add(GuiResourceError({ resource }, L"Failed to load file \"" + path + L"\"."));
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
Ptr<DescriptableObject> ResolveResourcePrecompiled(Ptr<GuiResourceItem> resource, stream::IStream& binaryStream, GuiResourceError::List& errors)override
|
|
{
|
|
stream::internal::ContextFreeReader reader(binaryStream);
|
|
WString text;
|
|
reader << text;
|
|
return new GuiTextData(text);
|
|
}
|
|
};
|
|
|
|
/***********************************************************************
|
|
Xml Type Resolver (Xml)
|
|
***********************************************************************/
|
|
|
|
class GuiResourceXmlTypeResolver
|
|
: public Object
|
|
, public IGuiResourceTypeResolver
|
|
, private IGuiResourceTypeResolver_DirectLoadXml
|
|
, private IGuiResourceTypeResolver_DirectLoadStream
|
|
{
|
|
public:
|
|
WString GetType()override
|
|
{
|
|
return L"Xml";
|
|
}
|
|
|
|
bool XmlSerializable()override
|
|
{
|
|
return true;
|
|
}
|
|
|
|
bool StreamSerializable()override
|
|
{
|
|
return true;
|
|
}
|
|
|
|
IGuiResourceTypeResolver_DirectLoadXml* DirectLoadXml()override
|
|
{
|
|
return this;
|
|
}
|
|
|
|
IGuiResourceTypeResolver_DirectLoadStream* DirectLoadStream()override
|
|
{
|
|
return this;
|
|
}
|
|
|
|
Ptr<parsing::xml::XmlElement> Serialize(Ptr<GuiResourceItem> resource, Ptr<DescriptableObject> content)override
|
|
{
|
|
if (auto obj = content.Cast<XmlDocument>())
|
|
{
|
|
auto xmlXml = MakePtr<XmlElement>();
|
|
xmlXml->name.value = L"Xml";
|
|
xmlXml->subNodes.Add(obj->rootElement);
|
|
return xmlXml;
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
void SerializePrecompiled(Ptr<GuiResourceItem> resource, Ptr<DescriptableObject> content, stream::IStream& binaryStream)override
|
|
{
|
|
auto obj = content.Cast<XmlDocument>();
|
|
WString text = GenerateToStream([&](StreamWriter& writer)
|
|
{
|
|
XmlPrint(obj, writer);
|
|
});
|
|
stream::internal::ContextFreeWriter writer(binaryStream);
|
|
writer << text;
|
|
}
|
|
|
|
Ptr<DescriptableObject> ResolveResource(Ptr<GuiResourceItem> resource, Ptr<parsing::xml::XmlElement> element, GuiResourceError::List& errors)override
|
|
{
|
|
Ptr<XmlElement> root = XmlGetElements(element).First(0);
|
|
if(root)
|
|
{
|
|
Ptr<XmlDocument> xml=new XmlDocument;
|
|
xml->rootElement=root;
|
|
return xml;
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
Ptr<DescriptableObject> ResolveResource(Ptr<GuiResourceItem> resource, const WString& path, GuiResourceError::List& errors)override
|
|
{
|
|
if (auto parser = GetParserManager()->GetParser<XmlDocument>(L"XML"))
|
|
{
|
|
WString text;
|
|
if (LoadTextFile(path, text))
|
|
{
|
|
return parser->Parse({ resource }, text, errors);
|
|
}
|
|
else
|
|
{
|
|
errors.Add(GuiResourceError({ resource }, L"Failed to load file \"" + path + L"\"."));
|
|
}
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
Ptr<DescriptableObject> ResolveResourcePrecompiled(Ptr<GuiResourceItem> resource, stream::IStream& binaryStream, GuiResourceError::List& errors)override
|
|
{
|
|
if (auto parser = GetParserManager()->GetParser<XmlDocument>(L"XML"))
|
|
{
|
|
stream::internal::ContextFreeReader reader(binaryStream);
|
|
WString text;
|
|
reader << text;
|
|
|
|
return parser->Parse({ resource }, text, errors);
|
|
}
|
|
return nullptr;
|
|
}
|
|
};
|
|
|
|
/***********************************************************************
|
|
Doc Type Resolver (Doc)
|
|
***********************************************************************/
|
|
|
|
class GuiResourceDocTypeResolver
|
|
: public Object
|
|
, public IGuiResourceTypeResolver
|
|
, private IGuiResourceTypeResolver_IndirectLoad
|
|
{
|
|
public:
|
|
WString GetType()override
|
|
{
|
|
return L"Doc";
|
|
}
|
|
|
|
bool XmlSerializable()override
|
|
{
|
|
return true;
|
|
}
|
|
|
|
bool StreamSerializable()override
|
|
{
|
|
return true;
|
|
}
|
|
|
|
WString GetPreloadType()override
|
|
{
|
|
return L"Xml";
|
|
}
|
|
|
|
bool IsDelayLoad()override
|
|
{
|
|
return true;
|
|
}
|
|
|
|
IGuiResourceTypeResolver_IndirectLoad* IndirectLoad()override
|
|
{
|
|
return this;
|
|
}
|
|
|
|
Ptr<DescriptableObject> Serialize(Ptr<GuiResourceItem> resource, Ptr<DescriptableObject> content)override
|
|
{
|
|
if (auto obj = content.Cast<DocumentModel>())
|
|
{
|
|
return obj->SaveToXml();
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
Ptr<DescriptableObject> ResolveResource(Ptr<GuiResourceItem> resource, Ptr<GuiResourcePathResolver> resolver, GuiResourceError::List& errors)override
|
|
{
|
|
if(auto xml = resource->GetContent().Cast<XmlDocument>())
|
|
{
|
|
Ptr<DocumentModel> model = DocumentModel::LoadFromXml(resource, xml, resolver, errors);
|
|
return model;
|
|
}
|
|
return nullptr;
|
|
}
|
|
};
|
|
|
|
/***********************************************************************
|
|
Type Resolver Plugin
|
|
***********************************************************************/
|
|
|
|
class GuiResourceTypeResolversPlugin : public Object, public IGuiPlugin
|
|
{
|
|
public:
|
|
|
|
GUI_PLUGIN_NAME(GacUI_Res_TypeResolvers)
|
|
{
|
|
GUI_PLUGIN_DEPEND(GacUI_Res_ResourceResolver);
|
|
}
|
|
|
|
void Load()override
|
|
{
|
|
IGuiResourceResolverManager* manager=GetResourceResolverManager();
|
|
manager->SetTypeResolver(new GuiResourceImageTypeResolver);
|
|
manager->SetTypeResolver(new GuiResourceTextTypeResolver);
|
|
manager->SetTypeResolver(new GuiResourceXmlTypeResolver);
|
|
manager->SetTypeResolver(new GuiResourceDocTypeResolver);
|
|
}
|
|
|
|
void Unload()override
|
|
{
|
|
}
|
|
};
|
|
GUI_REGISTER_PLUGIN(GuiResourceTypeResolversPlugin)
|
|
}
|
|
}
|
|
|