forked from TextureGroup/Texture
-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathASCollectionLayoutCache.mm
More file actions
90 lines (77 loc) · 2.69 KB
/
Copy pathASCollectionLayoutCache.mm
File metadata and controls
90 lines (77 loc) · 2.69 KB
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
//
// ASCollectionLayoutCache.mm
// Texture
//
// Copyright (c) 2017-present, Pinterest, Inc. All rights reserved.
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
#import <AsyncDisplayKit/ASCollectionLayoutCache.h>
#import <AsyncDisplayKit/ASCollectionLayoutContext.h>
#import <AsyncDisplayKit/ASCollectionLayoutState.h>
#import <AsyncDisplayKit/ASElementMap.h>
#import <AsyncDisplayKit/ASThread.h>
@implementation ASCollectionLayoutCache {
ASDN::Mutex __instanceLock__;
/**
* The underlying data structure of this cache.
*
* The outer map table is a weak to strong table. That is because ASCollectionLayoutContext doesn't (and shouldn't)
* hold a strong reference on its element map. As a result, this cache should handle the case in which
* an element map no longer exists and all contexts and layouts associated with it should be cleared.
*
* The inner map table is a standard strong to strong map.
* Since different ASCollectionLayoutContext objects with the same content are considered equal,
* "object pointer personality" can't be used as a key option.
*/
NSMapTable<ASElementMap *, NSMapTable<ASCollectionLayoutContext *, ASCollectionLayoutState *> *> *_map;
}
- (instancetype)init
{
self = [super init];
if (self) {
_map = [NSMapTable mapTableWithKeyOptions:(NSMapTableWeakMemory | NSMapTableObjectPointerPersonality) valueOptions:NSMapTableStrongMemory];
}
return self;
}
- (ASCollectionLayoutState *)layoutForContext:(ASCollectionLayoutContext *)context
{
ASElementMap *elements = context.elements;
if (elements == nil) {
return nil;
}
ASDN::MutexLocker l(__instanceLock__);
return [[_map objectForKey:elements] objectForKey:context];
}
- (void)setLayout:(ASCollectionLayoutState *)layout forContext:(ASCollectionLayoutContext *)context
{
ASElementMap *elements = context.elements;
if (layout == nil || elements == nil) {
return;
}
ASDN::MutexLocker l(__instanceLock__);
auto innerMap = [_map objectForKey:elements];
if (innerMap == nil) {
innerMap = [NSMapTable strongToStrongObjectsMapTable];
[_map setObject:innerMap forKey:elements];
}
[innerMap setObject:layout forKey:context];
}
- (void)removeLayoutForContext:(ASCollectionLayoutContext *)context
{
ASElementMap *elements = context.elements;
if (elements == nil) {
return;
}
ASDN::MutexLocker l(__instanceLock__);
[[_map objectForKey:elements] removeObjectForKey:context];
}
- (void)removeAllLayouts
{
ASDN::MutexLocker l(__instanceLock__);
[_map removeAllObjects];
}
@end