forked from microsoft/microsoft-ui-xaml
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathRecyclingElementFactory.cpp
125 lines (98 loc) · 3.76 KB
/
RecyclingElementFactory.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
// Copyright (c) Microsoft Corporation. All rights reserved.
// Licensed under the MIT License. See LICENSE in the project root for license information.
#include <pch.h>
#include <common.h>
#include "ItemsRepeater.common.h"
#include <HashMap.h>
#include "SelectTemplateEventArgs.h"
#include "RecyclingElementFactory.h"
#include "ItemsRepeater.h"
#include "RecyclePool.h"
#include "RecyclingElementFactory.properties.cpp"
RecyclingElementFactory::RecyclingElementFactory()
{
m_templates.set(winrt::make<HashMap<winrt::hstring, winrt::DataTemplate>>());
}
#pragma region IRecyclingElementFactory
winrt::RecyclePool RecyclingElementFactory::RecyclePool()
{
return m_recyclePool.get();
}
void RecyclingElementFactory::RecyclePool(winrt::RecyclePool const& value)
{
m_recyclePool.set(value);
}
winrt::IMap<winrt::hstring, winrt::DataTemplate> RecyclingElementFactory::Templates()
{
return m_templates.get();
}
void RecyclingElementFactory::Templates(winrt::IMap<winrt::hstring, winrt::DataTemplate> const& value)
{
m_templates.set(value);
}
#pragma endregion
#pragma region IRecyclingElementFactoryOverrides
winrt::hstring RecyclingElementFactory::OnSelectTemplateKeyCore(
winrt::IInspectable const& dataContext,
winrt::UIElement const& owner)
{
if (!m_args)
{
m_args.set(winrt::make<SelectTemplateEventArgs>());
}
auto args = winrt::get_self<SelectTemplateEventArgs>(m_args.get());
args->TemplateKey({});
args->DataContext(dataContext);
args->Owner(owner);
m_selectTemplateKeyEventSource(*this, *args);
auto templateKey = args->TemplateKey();
if (templateKey.empty())
{
throw winrt::hresult_error(E_FAIL, L"Please provide a valid template identifier in the handler for the SelectTemplateKey event.");
}
return templateKey;
}
#pragma endregion
#pragma region IElementFactoryOverrides
winrt::UIElement RecyclingElementFactory::GetElementCore(winrt::ElementFactoryGetArgs const& args)
{
if (!m_templates || m_templates.get().Size() == 0)
{
throw winrt::hresult_error(E_FAIL, L"Templates property cannot be null or empty.");
}
const auto winrtOwner = args.Parent();
const auto templateKey =
m_templates.get().Size() == 1 ?
m_templates.get().First().Current().Key() :
OnSelectTemplateKeyCore(args.Data(), winrtOwner);
if (templateKey.empty())
{
// Note: We could allow null/whitespace, which would work as long as
// the recycle pool is not shared. in order to make this work in all cases
// currently we validate that a valid template key is provided.
throw winrt::hresult_error(E_FAIL, L"Template key cannot be empty or null.");
}
// Get an element from the Recycle Pool or create one
auto element = m_recyclePool.get().TryGetElement(templateKey, winrtOwner).as<winrt::FrameworkElement>();
if (!element)
{
// No need to call HasKey if there is only one template.
if (m_templates.get().Size() > 1 && !m_templates.get().HasKey(templateKey))
{
std::wstring message = L"No templates of key " + std::wstring(templateKey.data()) + L" were found in the templates collection.";
throw winrt::hresult_error(E_FAIL, message.c_str());
}
auto dataTemplate = m_templates.get().Lookup(templateKey);
element = dataTemplate.LoadContent().as<winrt::FrameworkElement>();
// Associate ReuseKey with element
RecyclePool::SetReuseKey(element, templateKey);
}
return element;
}
void RecyclingElementFactory::RecycleElementCore(winrt::ElementFactoryRecycleArgs const& args)
{
auto element = args.Element();
auto key = RecyclePool::GetReuseKey(element);
m_recyclePool.get().PutElement(element, key, args.Parent());
}
#pragma endregion