mirror of
https://github.com/Swiftgram/Telegram-iOS.git
synced 2025-09-07 05:02:56 +00:00
Introduces ASLayoutRangeModeVisibleOnly, allowing the preservation of decoded backing stores without any extra padding to strictly minimize memory usage while supporting immediate re-display of content. Set visible range controllers to this mode upon app suspend / memory warning, while more aggressively clearing others to the ASLayoutRangeModeLowMemory mode. By default, when the app is running and recieves a memory warning, we set the range mode for non-visible controllers to ASLayoutRangeModeVisibleOnly. This is because, unlike in the app suspend case where on app resume we can restore controllers from LowMemory to VisibleOnly, the memory warning doesn't provide a good opportunity to do this. A new +Beta API to control this behavior is called +setRangeModeForMemoryWarnings:, as some apps may prefer to use LowMemory in the memory warning scenario. For these apps, optimal user experience will require manually setting the range mode back to some larger value as the user navigates the app, or they will encounter controllers that are temporarily blank and need a moment to re-display their contents as they start to become visible.
1012 lines
27 KiB
Plaintext
1012 lines
27 KiB
Plaintext
/* Copyright (c) 2014-present, Facebook, Inc.
|
|
* All rights reserved.
|
|
*
|
|
* This source code is licensed under the BSD-style license found in the
|
|
* LICENSE file in the root directory of this source tree. An additional grant
|
|
* of patent rights can be found in the PATENTS file in the same directory.
|
|
*/
|
|
|
|
#import "_ASPendingState.h"
|
|
|
|
#import "_ASCoreAnimationExtras.h"
|
|
#import "_ASAsyncTransactionContainer.h"
|
|
#import "ASAssert.h"
|
|
#import "ASInternalHelpers.h"
|
|
#import "ASDisplayNodeInternal.h"
|
|
|
|
#define __shouldSetNeedsDisplay(layer) (flags.needsDisplay \
|
|
|| (flags.setOpaque && opaque != (layer).opaque)\
|
|
|| (flags.setBackgroundColor && !CGColorEqualToColor(backgroundColor, (layer).backgroundColor)))
|
|
|
|
typedef struct {
|
|
// Properties
|
|
int needsDisplay:1;
|
|
int needsLayout:1;
|
|
|
|
// Flags indicating that a given property should be applied to the view at creation
|
|
int setClipsToBounds:1;
|
|
int setOpaque:1;
|
|
int setNeedsDisplayOnBoundsChange:1;
|
|
int setAutoresizesSubviews:1;
|
|
int setAutoresizingMask:1;
|
|
int setFrame:1;
|
|
int setBounds:1;
|
|
int setBackgroundColor:1;
|
|
int setTintColor:1;
|
|
int setContents:1;
|
|
int setHidden:1;
|
|
int setAlpha:1;
|
|
int setCornerRadius:1;
|
|
int setContentMode:1;
|
|
int setNeedsDisplay:1;
|
|
int setAnchorPoint:1;
|
|
int setPosition:1;
|
|
int setZPosition:1;
|
|
int setContentsScale:1;
|
|
int setTransform:1;
|
|
int setSublayerTransform:1;
|
|
int setUserInteractionEnabled:1;
|
|
int setExclusiveTouch:1;
|
|
int setShadowColor:1;
|
|
int setShadowOpacity:1;
|
|
int setShadowOffset:1;
|
|
int setShadowRadius:1;
|
|
int setBorderWidth:1;
|
|
int setBorderColor:1;
|
|
int setAsyncTransactionContainer:1;
|
|
int setAllowsEdgeAntialiasing:1;
|
|
int setEdgeAntialiasingMask:1;
|
|
int setIsAccessibilityElement:1;
|
|
int setAccessibilityLabel:1;
|
|
int setAccessibilityHint:1;
|
|
int setAccessibilityValue:1;
|
|
int setAccessibilityTraits:1;
|
|
int setAccessibilityFrame:1;
|
|
int setAccessibilityLanguage:1;
|
|
int setAccessibilityElementsHidden:1;
|
|
int setAccessibilityViewIsModal:1;
|
|
int setShouldGroupAccessibilityChildren:1;
|
|
int setAccessibilityIdentifier:1;
|
|
} ASPendingStateFlags;
|
|
|
|
@implementation _ASPendingState
|
|
{
|
|
@package //Expose all ivars for ASDisplayNode to bypass getters for efficiency
|
|
|
|
UIViewAutoresizing autoresizingMask;
|
|
unsigned int edgeAntialiasingMask;
|
|
CGRect frame; // Frame is only to be used for synchronous views wrapped by nodes (see setFrame:)
|
|
CGRect bounds;
|
|
CGColorRef backgroundColor;
|
|
id contents;
|
|
CGFloat alpha;
|
|
CGFloat cornerRadius;
|
|
UIViewContentMode contentMode;
|
|
CGPoint anchorPoint;
|
|
CGPoint position;
|
|
CGFloat zPosition;
|
|
CGFloat contentsScale;
|
|
CATransform3D transform;
|
|
CATransform3D sublayerTransform;
|
|
CGColorRef shadowColor;
|
|
CGFloat shadowOpacity;
|
|
CGSize shadowOffset;
|
|
CGFloat shadowRadius;
|
|
CGFloat borderWidth;
|
|
CGColorRef borderColor;
|
|
BOOL asyncTransactionContainer;
|
|
BOOL isAccessibilityElement;
|
|
NSString *accessibilityLabel;
|
|
NSString *accessibilityHint;
|
|
NSString *accessibilityValue;
|
|
UIAccessibilityTraits accessibilityTraits;
|
|
CGRect accessibilityFrame;
|
|
NSString *accessibilityLanguage;
|
|
BOOL accessibilityElementsHidden;
|
|
BOOL accessibilityViewIsModal;
|
|
BOOL shouldGroupAccessibilityChildren;
|
|
NSString *accessibilityIdentifier;
|
|
|
|
ASPendingStateFlags _flags;
|
|
}
|
|
|
|
/**
|
|
* Apply the state's frame, bounds, and position to layer. This will not
|
|
* be called on synchronous view-backed nodes which require we directly
|
|
* call [view setFrame:].
|
|
*
|
|
* FIXME: How should we reconcile order-of-operations between setting frame, bounds, position?
|
|
* Note we can't read bounds and position in the background, so we have to keep the frame
|
|
* value intact until application time (now).
|
|
*/
|
|
ASDISPLAYNODE_INLINE void ASPendingStateApplyMetricsToLayer(_ASPendingState *state, CALayer *layer) {
|
|
ASPendingStateFlags flags = state->_flags;
|
|
if (flags.setFrame) {
|
|
CGRect _bounds = CGRectZero;
|
|
CGPoint _position = CGPointZero;
|
|
ASBoundsAndPositionForFrame(state->frame, layer.bounds.origin, layer.anchorPoint, &_bounds, &_position);
|
|
layer.bounds = _bounds;
|
|
layer.position = _position;
|
|
} else {
|
|
if (flags.setBounds)
|
|
layer.bounds = state->bounds;
|
|
if (flags.setPosition)
|
|
layer.position = state->position;
|
|
}
|
|
}
|
|
|
|
@synthesize clipsToBounds=clipsToBounds;
|
|
@synthesize opaque=opaque;
|
|
@synthesize frame=frame;
|
|
@synthesize bounds=bounds;
|
|
@synthesize backgroundColor=backgroundColor;
|
|
@synthesize contents=contents;
|
|
@synthesize hidden=isHidden;
|
|
@synthesize needsDisplayOnBoundsChange=needsDisplayOnBoundsChange;
|
|
@synthesize allowsEdgeAntialiasing=allowsEdgeAntialiasing;
|
|
@synthesize edgeAntialiasingMask=edgeAntialiasingMask;
|
|
@synthesize autoresizesSubviews=autoresizesSubviews;
|
|
@synthesize autoresizingMask=autoresizingMask;
|
|
@synthesize tintColor=tintColor;
|
|
@synthesize alpha=alpha;
|
|
@synthesize cornerRadius=cornerRadius;
|
|
@synthesize contentMode=contentMode;
|
|
@synthesize anchorPoint=anchorPoint;
|
|
@synthesize position=position;
|
|
@synthesize zPosition=zPosition;
|
|
@synthesize contentsScale=contentsScale;
|
|
@synthesize transform=transform;
|
|
@synthesize sublayerTransform=sublayerTransform;
|
|
@synthesize userInteractionEnabled=userInteractionEnabled;
|
|
@synthesize exclusiveTouch=exclusiveTouch;
|
|
@synthesize shadowColor=shadowColor;
|
|
@synthesize shadowOpacity=shadowOpacity;
|
|
@synthesize shadowOffset=shadowOffset;
|
|
@synthesize shadowRadius=shadowRadius;
|
|
@synthesize borderWidth=borderWidth;
|
|
@synthesize borderColor=borderColor;
|
|
@synthesize asyncdisplaykit_asyncTransactionContainer=asyncTransactionContainer;
|
|
|
|
|
|
static CGColorRef blackColorRef = NULL;
|
|
static UIColor *defaultTintColor = nil;
|
|
|
|
- (id)init
|
|
{
|
|
if (!(self = [super init]))
|
|
return nil;
|
|
|
|
|
|
static dispatch_once_t onceToken;
|
|
dispatch_once(&onceToken, ^{
|
|
// Default UIKit color is an RGB color
|
|
CGColorSpaceRef colorSpace = CGColorSpaceCreateDeviceRGB();
|
|
blackColorRef = CGColorCreate(colorSpace, (CGFloat[]){0,0,0,1} );
|
|
CFRetain(blackColorRef);
|
|
CGColorSpaceRelease(colorSpace);
|
|
defaultTintColor = [UIColor colorWithRed:0.0 green:0.478 blue:1.0 alpha:1.0];
|
|
});
|
|
|
|
// Set defaults, these come from the defaults specified in CALayer and UIView
|
|
clipsToBounds = NO;
|
|
opaque = YES;
|
|
frame = CGRectZero;
|
|
bounds = CGRectZero;
|
|
backgroundColor = nil;
|
|
tintColor = defaultTintColor;
|
|
contents = nil;
|
|
isHidden = NO;
|
|
needsDisplayOnBoundsChange = NO;
|
|
autoresizesSubviews = YES;
|
|
alpha = 1.0f;
|
|
cornerRadius = 0.0f;
|
|
contentMode = UIViewContentModeScaleToFill;
|
|
_flags.needsDisplay = NO;
|
|
anchorPoint = CGPointMake(0.5, 0.5);
|
|
position = CGPointZero;
|
|
zPosition = 0.0;
|
|
contentsScale = 1.0f;
|
|
transform = CATransform3DIdentity;
|
|
sublayerTransform = CATransform3DIdentity;
|
|
userInteractionEnabled = YES;
|
|
shadowColor = blackColorRef;
|
|
shadowOpacity = 0.0;
|
|
shadowOffset = CGSizeMake(0, -3);
|
|
shadowRadius = 3;
|
|
borderWidth = 0;
|
|
borderColor = blackColorRef;
|
|
isAccessibilityElement = NO;
|
|
accessibilityLabel = nil;
|
|
accessibilityHint = nil;
|
|
accessibilityValue = nil;
|
|
accessibilityTraits = UIAccessibilityTraitNone;
|
|
accessibilityFrame = CGRectZero;
|
|
accessibilityLanguage = nil;
|
|
accessibilityElementsHidden = NO;
|
|
accessibilityViewIsModal = NO;
|
|
shouldGroupAccessibilityChildren = NO;
|
|
accessibilityIdentifier = nil;
|
|
edgeAntialiasingMask = (kCALayerLeftEdge | kCALayerRightEdge | kCALayerTopEdge | kCALayerBottomEdge);
|
|
|
|
return self;
|
|
}
|
|
|
|
- (void)setNeedsDisplay
|
|
{
|
|
_flags.needsDisplay = YES;
|
|
}
|
|
|
|
- (void)setNeedsLayout
|
|
{
|
|
_flags.needsLayout = YES;
|
|
}
|
|
|
|
- (void)setClipsToBounds:(BOOL)flag
|
|
{
|
|
clipsToBounds = flag;
|
|
_flags.setClipsToBounds = YES;
|
|
}
|
|
|
|
- (void)setOpaque:(BOOL)flag
|
|
{
|
|
opaque = flag;
|
|
_flags.setOpaque = YES;
|
|
}
|
|
|
|
- (void)setNeedsDisplayOnBoundsChange:(BOOL)flag
|
|
{
|
|
needsDisplayOnBoundsChange = flag;
|
|
_flags.setNeedsDisplayOnBoundsChange = YES;
|
|
}
|
|
|
|
- (void)setAllowsEdgeAntialiasing:(BOOL)flag
|
|
{
|
|
allowsEdgeAntialiasing = flag;
|
|
_flags.setAllowsEdgeAntialiasing = YES;
|
|
}
|
|
|
|
- (void)setEdgeAntialiasingMask:(unsigned int)mask
|
|
{
|
|
edgeAntialiasingMask = mask;
|
|
_flags.setEdgeAntialiasingMask = YES;
|
|
}
|
|
|
|
- (void)setAutoresizesSubviews:(BOOL)flag
|
|
{
|
|
autoresizesSubviews = flag;
|
|
_flags.setAutoresizesSubviews = YES;
|
|
}
|
|
|
|
- (void)setAutoresizingMask:(UIViewAutoresizing)mask
|
|
{
|
|
autoresizingMask = mask;
|
|
_flags.setAutoresizingMask = YES;
|
|
}
|
|
|
|
- (void)setFrame:(CGRect)newFrame
|
|
{
|
|
frame = newFrame;
|
|
_flags.setFrame = YES;
|
|
}
|
|
|
|
- (void)setBounds:(CGRect)newBounds
|
|
{
|
|
ASDisplayNodeAssert(!isnan(newBounds.size.width) && !isnan(newBounds.size.height), @"Invalid bounds %@ provided to %@", NSStringFromCGRect(newBounds), self);
|
|
if (isnan(newBounds.size.width))
|
|
newBounds.size.width = 0.0;
|
|
if (isnan(newBounds.size.height))
|
|
newBounds.size.height = 0.0;
|
|
bounds = newBounds;
|
|
_flags.setBounds = YES;
|
|
}
|
|
|
|
- (CGColorRef)backgroundColor
|
|
{
|
|
return backgroundColor;
|
|
}
|
|
|
|
- (void)setBackgroundColor:(CGColorRef)color
|
|
{
|
|
if (color == backgroundColor) {
|
|
return;
|
|
}
|
|
|
|
CGColorRelease(backgroundColor);
|
|
backgroundColor = CGColorRetain(color);
|
|
_flags.setBackgroundColor = YES;
|
|
}
|
|
|
|
- (void)setTintColor:(UIColor *)newTintColor
|
|
{
|
|
tintColor = newTintColor;
|
|
_flags.setTintColor = YES;
|
|
}
|
|
|
|
- (void)setContents:(id)newContents
|
|
{
|
|
if (contents == newContents) {
|
|
return;
|
|
}
|
|
|
|
contents = newContents;
|
|
_flags.setContents = YES;
|
|
}
|
|
|
|
- (void)setHidden:(BOOL)flag
|
|
{
|
|
isHidden = flag;
|
|
_flags.setHidden = YES;
|
|
}
|
|
|
|
- (void)setAlpha:(CGFloat)newAlpha
|
|
{
|
|
alpha = newAlpha;
|
|
_flags.setAlpha = YES;
|
|
}
|
|
|
|
- (void)setCornerRadius:(CGFloat)newCornerRadius
|
|
{
|
|
cornerRadius = newCornerRadius;
|
|
_flags.setCornerRadius = YES;
|
|
}
|
|
|
|
- (void)setContentMode:(UIViewContentMode)newContentMode
|
|
{
|
|
contentMode = newContentMode;
|
|
_flags.setContentMode = YES;
|
|
}
|
|
|
|
- (void)setAnchorPoint:(CGPoint)newAnchorPoint
|
|
{
|
|
anchorPoint = newAnchorPoint;
|
|
_flags.setAnchorPoint = YES;
|
|
}
|
|
|
|
- (void)setPosition:(CGPoint)newPosition
|
|
{
|
|
ASDisplayNodeAssert(!isnan(newPosition.x) && !isnan(newPosition.y), @"Invalid position %@ provided to %@", NSStringFromCGPoint(newPosition), self);
|
|
if (isnan(newPosition.x))
|
|
newPosition.x = 0.0;
|
|
if (isnan(newPosition.y))
|
|
newPosition.y = 0.0;
|
|
position = newPosition;
|
|
_flags.setPosition = YES;
|
|
}
|
|
|
|
- (void)setZPosition:(CGFloat)newPosition
|
|
{
|
|
zPosition = newPosition;
|
|
_flags.setZPosition = YES;
|
|
}
|
|
|
|
- (void)setContentsScale:(CGFloat)newContentsScale
|
|
{
|
|
contentsScale = newContentsScale;
|
|
_flags.setContentsScale = YES;
|
|
}
|
|
|
|
- (void)setTransform:(CATransform3D)newTransform
|
|
{
|
|
transform = newTransform;
|
|
_flags.setTransform = YES;
|
|
}
|
|
|
|
- (void)setSublayerTransform:(CATransform3D)newSublayerTransform
|
|
{
|
|
sublayerTransform = newSublayerTransform;
|
|
_flags.setSublayerTransform = YES;
|
|
}
|
|
|
|
- (void)setUserInteractionEnabled:(BOOL)flag
|
|
{
|
|
userInteractionEnabled = flag;
|
|
_flags.setUserInteractionEnabled = YES;
|
|
}
|
|
|
|
- (void)setExclusiveTouch:(BOOL)flag
|
|
{
|
|
exclusiveTouch = flag;
|
|
_flags.setExclusiveTouch = YES;
|
|
}
|
|
|
|
- (void)setShadowColor:(CGColorRef)color
|
|
{
|
|
if (shadowColor == color) {
|
|
return;
|
|
}
|
|
|
|
if (shadowColor != blackColorRef) {
|
|
CGColorRelease(shadowColor);
|
|
}
|
|
shadowColor = color;
|
|
CGColorRetain(shadowColor);
|
|
|
|
_flags.setShadowColor = YES;
|
|
}
|
|
|
|
- (void)setShadowOpacity:(CGFloat)newOpacity
|
|
{
|
|
shadowOpacity = newOpacity;
|
|
_flags.setShadowOpacity = YES;
|
|
}
|
|
|
|
- (void)setShadowOffset:(CGSize)newOffset
|
|
{
|
|
shadowOffset = newOffset;
|
|
_flags.setShadowOffset = YES;
|
|
}
|
|
|
|
- (void)setShadowRadius:(CGFloat)newRadius
|
|
{
|
|
shadowRadius = newRadius;
|
|
_flags.setShadowRadius = YES;
|
|
}
|
|
|
|
- (void)setBorderWidth:(CGFloat)newWidth
|
|
{
|
|
borderWidth = newWidth;
|
|
_flags.setBorderWidth = YES;
|
|
}
|
|
|
|
- (void)setBorderColor:(CGColorRef)color
|
|
{
|
|
if (borderColor == color) {
|
|
return;
|
|
}
|
|
|
|
if (borderColor != blackColorRef) {
|
|
CGColorRelease(borderColor);
|
|
}
|
|
borderColor = color;
|
|
CGColorRetain(borderColor);
|
|
|
|
_flags.setBorderColor = YES;
|
|
}
|
|
|
|
- (void)asyncdisplaykit_setAsyncTransactionContainer:(BOOL)flag
|
|
{
|
|
asyncTransactionContainer = flag;
|
|
_flags.setAsyncTransactionContainer = YES;
|
|
}
|
|
|
|
- (BOOL)isAccessibilityElement
|
|
{
|
|
return isAccessibilityElement;
|
|
}
|
|
|
|
- (void)setIsAccessibilityElement:(BOOL)newIsAccessibilityElement
|
|
{
|
|
isAccessibilityElement = newIsAccessibilityElement;
|
|
_flags.setIsAccessibilityElement = YES;
|
|
}
|
|
|
|
- (NSString *)accessibilityLabel
|
|
{
|
|
return accessibilityLabel;
|
|
}
|
|
|
|
- (void)setAccessibilityLabel:(NSString *)newAccessibilityLabel
|
|
{
|
|
_flags.setAccessibilityLabel = YES;
|
|
if (accessibilityLabel != newAccessibilityLabel) {
|
|
accessibilityLabel = [newAccessibilityLabel copy];
|
|
}
|
|
}
|
|
|
|
- (NSString *)accessibilityHint
|
|
{
|
|
return accessibilityHint;
|
|
}
|
|
|
|
- (void)setAccessibilityHint:(NSString *)newAccessibilityHint
|
|
{
|
|
_flags.setAccessibilityHint = YES;
|
|
accessibilityHint = [newAccessibilityHint copy];
|
|
}
|
|
|
|
- (NSString *)accessibilityValue
|
|
{
|
|
return accessibilityValue;
|
|
}
|
|
|
|
- (void)setAccessibilityValue:(NSString *)newAccessibilityValue
|
|
{
|
|
_flags.setAccessibilityValue = YES;
|
|
accessibilityValue = [newAccessibilityValue copy];
|
|
}
|
|
|
|
- (UIAccessibilityTraits)accessibilityTraits
|
|
{
|
|
return accessibilityTraits;
|
|
}
|
|
|
|
- (void)setAccessibilityTraits:(UIAccessibilityTraits)newAccessibilityTraits
|
|
{
|
|
accessibilityTraits = newAccessibilityTraits;
|
|
_flags.setAccessibilityTraits = YES;
|
|
}
|
|
|
|
- (CGRect)accessibilityFrame
|
|
{
|
|
return accessibilityFrame;
|
|
}
|
|
|
|
- (void)setAccessibilityFrame:(CGRect)newAccessibilityFrame
|
|
{
|
|
accessibilityFrame = newAccessibilityFrame;
|
|
_flags.setAccessibilityFrame = YES;
|
|
}
|
|
|
|
- (NSString *)accessibilityLanguage
|
|
{
|
|
return accessibilityLanguage;
|
|
}
|
|
|
|
- (void)setAccessibilityLanguage:(NSString *)newAccessibilityLanguage
|
|
{
|
|
_flags.setAccessibilityLanguage = YES;
|
|
accessibilityLanguage = newAccessibilityLanguage;
|
|
}
|
|
|
|
- (BOOL)accessibilityElementsHidden
|
|
{
|
|
return accessibilityElementsHidden;
|
|
}
|
|
|
|
- (void)setAccessibilityElementsHidden:(BOOL)newAccessibilityElementsHidden
|
|
{
|
|
accessibilityElementsHidden = newAccessibilityElementsHidden;
|
|
_flags.setAccessibilityElementsHidden = YES;
|
|
}
|
|
|
|
- (BOOL)accessibilityViewIsModal
|
|
{
|
|
return accessibilityViewIsModal;
|
|
}
|
|
|
|
- (void)setAccessibilityViewIsModal:(BOOL)newAccessibilityViewIsModal
|
|
{
|
|
accessibilityViewIsModal = newAccessibilityViewIsModal;
|
|
_flags.setAccessibilityViewIsModal = YES;
|
|
}
|
|
|
|
- (BOOL)shouldGroupAccessibilityChildren
|
|
{
|
|
return shouldGroupAccessibilityChildren;
|
|
}
|
|
|
|
- (void)setShouldGroupAccessibilityChildren:(BOOL)newShouldGroupAccessibilityChildren
|
|
{
|
|
shouldGroupAccessibilityChildren = newShouldGroupAccessibilityChildren;
|
|
_flags.setShouldGroupAccessibilityChildren = YES;
|
|
}
|
|
|
|
- (NSString *)accessibilityIdentifier
|
|
{
|
|
return accessibilityIdentifier;
|
|
}
|
|
|
|
- (void)setAccessibilityIdentifier:(NSString *)newAccessibilityIdentifier
|
|
{
|
|
_flags.setAccessibilityIdentifier = YES;
|
|
if (accessibilityIdentifier != newAccessibilityIdentifier) {
|
|
accessibilityIdentifier = [newAccessibilityIdentifier copy];
|
|
}
|
|
}
|
|
|
|
- (void)applyToLayer:(CALayer *)layer
|
|
{
|
|
ASPendingStateFlags flags = _flags;
|
|
|
|
if (__shouldSetNeedsDisplay(layer)) {
|
|
[layer setNeedsDisplay];
|
|
}
|
|
|
|
if (flags.setAnchorPoint)
|
|
layer.anchorPoint = anchorPoint;
|
|
|
|
if (flags.setZPosition)
|
|
layer.zPosition = zPosition;
|
|
|
|
if (flags.setContentsScale)
|
|
layer.contentsScale = contentsScale;
|
|
|
|
if (flags.setTransform)
|
|
layer.transform = transform;
|
|
|
|
if (flags.setSublayerTransform)
|
|
layer.sublayerTransform = sublayerTransform;
|
|
|
|
if (flags.setContents)
|
|
layer.contents = contents;
|
|
|
|
if (flags.setClipsToBounds)
|
|
layer.masksToBounds = clipsToBounds;
|
|
|
|
if (flags.setBackgroundColor)
|
|
layer.backgroundColor = backgroundColor;
|
|
|
|
if (flags.setOpaque)
|
|
layer.opaque = opaque;
|
|
|
|
if (flags.setHidden)
|
|
layer.hidden = isHidden;
|
|
|
|
if (flags.setAlpha)
|
|
layer.opacity = alpha;
|
|
|
|
if (flags.setCornerRadius)
|
|
layer.cornerRadius = cornerRadius;
|
|
|
|
if (flags.setContentMode)
|
|
layer.contentsGravity = ASDisplayNodeCAContentsGravityFromUIContentMode(contentMode);
|
|
|
|
if (flags.setShadowColor)
|
|
layer.shadowColor = shadowColor;
|
|
|
|
if (flags.setShadowOpacity)
|
|
layer.shadowOpacity = shadowOpacity;
|
|
|
|
if (flags.setShadowOffset)
|
|
layer.shadowOffset = shadowOffset;
|
|
|
|
if (flags.setShadowRadius)
|
|
layer.shadowRadius = shadowRadius;
|
|
|
|
if (flags.setBorderWidth)
|
|
layer.borderWidth = borderWidth;
|
|
|
|
if (flags.setBorderColor)
|
|
layer.borderColor = borderColor;
|
|
|
|
if (flags.setNeedsDisplayOnBoundsChange)
|
|
layer.needsDisplayOnBoundsChange = needsDisplayOnBoundsChange;
|
|
|
|
if (flags.setAllowsEdgeAntialiasing)
|
|
layer.allowsEdgeAntialiasing = allowsEdgeAntialiasing;
|
|
|
|
if (flags.setEdgeAntialiasingMask)
|
|
layer.edgeAntialiasingMask = edgeAntialiasingMask;
|
|
|
|
if (flags.needsLayout)
|
|
[layer setNeedsLayout];
|
|
|
|
if (flags.setAsyncTransactionContainer)
|
|
layer.asyncdisplaykit_asyncTransactionContainer = asyncTransactionContainer;
|
|
|
|
if (flags.setOpaque)
|
|
ASDisplayNodeAssert(layer.opaque == opaque, @"Didn't set opaque as desired");
|
|
|
|
ASPendingStateApplyMetricsToLayer(self, layer);
|
|
}
|
|
|
|
- (void)applyToView:(UIView *)view setFrameDirectly:(BOOL)setFrameDirectly
|
|
{
|
|
/*
|
|
Use our convenience setters blah here instead of layer.blah
|
|
We were accidentally setting some properties on layer here, but view in UIViewBridgeOptimizations.
|
|
|
|
That could easily cause bugs where it mattered whether you set something up on a bg thread on in -didLoad
|
|
because a different setter would be called.
|
|
*/
|
|
|
|
CALayer *layer = view.layer;
|
|
|
|
ASPendingStateFlags flags = _flags;
|
|
if (__shouldSetNeedsDisplay(layer)) {
|
|
[view setNeedsDisplay];
|
|
}
|
|
|
|
if (flags.setAnchorPoint)
|
|
layer.anchorPoint = anchorPoint;
|
|
|
|
if (flags.setPosition)
|
|
layer.position = position;
|
|
|
|
if (flags.setZPosition)
|
|
layer.zPosition = zPosition;
|
|
|
|
if (flags.setBounds)
|
|
view.bounds = bounds;
|
|
|
|
if (flags.setContentsScale)
|
|
layer.contentsScale = contentsScale;
|
|
|
|
if (flags.setTransform)
|
|
layer.transform = transform;
|
|
|
|
if (flags.setSublayerTransform)
|
|
layer.sublayerTransform = sublayerTransform;
|
|
|
|
if (flags.setContents)
|
|
layer.contents = contents;
|
|
|
|
if (flags.setClipsToBounds)
|
|
view.clipsToBounds = clipsToBounds;
|
|
|
|
if (flags.setBackgroundColor)
|
|
layer.backgroundColor = backgroundColor;
|
|
|
|
if (flags.setTintColor)
|
|
view.tintColor = self.tintColor;
|
|
|
|
if (flags.setOpaque)
|
|
view.layer.opaque = opaque;
|
|
|
|
if (flags.setHidden)
|
|
view.hidden = isHidden;
|
|
|
|
if (flags.setAlpha)
|
|
view.alpha = alpha;
|
|
|
|
if (flags.setCornerRadius)
|
|
layer.cornerRadius = cornerRadius;
|
|
|
|
if (flags.setContentMode)
|
|
view.contentMode = contentMode;
|
|
|
|
if (flags.setUserInteractionEnabled)
|
|
view.userInteractionEnabled = userInteractionEnabled;
|
|
|
|
#if TARGET_OS_IOS
|
|
if (flags.setExclusiveTouch)
|
|
view.exclusiveTouch = exclusiveTouch;
|
|
#endif
|
|
|
|
if (flags.setShadowColor)
|
|
layer.shadowColor = shadowColor;
|
|
|
|
if (flags.setShadowOpacity)
|
|
layer.shadowOpacity = shadowOpacity;
|
|
|
|
if (flags.setShadowOffset)
|
|
layer.shadowOffset = shadowOffset;
|
|
|
|
if (flags.setShadowRadius)
|
|
layer.shadowRadius = shadowRadius;
|
|
|
|
if (flags.setBorderWidth)
|
|
layer.borderWidth = borderWidth;
|
|
|
|
if (flags.setBorderColor)
|
|
layer.borderColor = borderColor;
|
|
|
|
if (flags.setAutoresizingMask)
|
|
view.autoresizingMask = autoresizingMask;
|
|
|
|
if (flags.setAutoresizesSubviews)
|
|
view.autoresizesSubviews = autoresizesSubviews;
|
|
|
|
if (flags.setNeedsDisplayOnBoundsChange)
|
|
layer.needsDisplayOnBoundsChange = needsDisplayOnBoundsChange;
|
|
|
|
if (flags.setAllowsEdgeAntialiasing)
|
|
layer.allowsEdgeAntialiasing = allowsEdgeAntialiasing;
|
|
|
|
if (flags.setEdgeAntialiasingMask)
|
|
layer.edgeAntialiasingMask = edgeAntialiasingMask;
|
|
|
|
if (flags.needsLayout)
|
|
[view setNeedsLayout];
|
|
|
|
if (flags.setAsyncTransactionContainer)
|
|
view.asyncdisplaykit_asyncTransactionContainer = asyncTransactionContainer;
|
|
|
|
if (flags.setOpaque)
|
|
ASDisplayNodeAssert(view.layer.opaque == opaque, @"Didn't set opaque as desired");
|
|
|
|
if (flags.setIsAccessibilityElement)
|
|
view.isAccessibilityElement = isAccessibilityElement;
|
|
|
|
if (flags.setAccessibilityLabel)
|
|
view.accessibilityLabel = accessibilityLabel;
|
|
|
|
if (flags.setAccessibilityHint)
|
|
view.accessibilityHint = accessibilityHint;
|
|
|
|
if (flags.setAccessibilityValue)
|
|
view.accessibilityValue = accessibilityValue;
|
|
|
|
if (flags.setAccessibilityTraits)
|
|
view.accessibilityTraits = accessibilityTraits;
|
|
|
|
if (flags.setAccessibilityFrame)
|
|
view.accessibilityFrame = accessibilityFrame;
|
|
|
|
if (flags.setAccessibilityLanguage)
|
|
view.accessibilityLanguage = accessibilityLanguage;
|
|
|
|
if (flags.setAccessibilityElementsHidden)
|
|
view.accessibilityElementsHidden = accessibilityElementsHidden;
|
|
|
|
if (flags.setAccessibilityViewIsModal)
|
|
view.accessibilityViewIsModal = accessibilityViewIsModal;
|
|
|
|
if (flags.setShouldGroupAccessibilityChildren)
|
|
view.shouldGroupAccessibilityChildren = shouldGroupAccessibilityChildren;
|
|
|
|
if (flags.setAccessibilityIdentifier)
|
|
view.accessibilityIdentifier = accessibilityIdentifier;
|
|
|
|
// For classes like ASTableNode, ASCollectionNode, ASScrollNode and similar - make sure UIView gets setFrame:
|
|
if (flags.setFrame && setFrameDirectly) {
|
|
// Frame is only defined when transform is identity because we explicitly diverge from CALayer behavior and define frame without transform
|
|
#if DEBUG
|
|
// Checking if the transform is identity is expensive, so disable when unnecessary. We have assertions on in Release, so DEBUG is the only way I know of.
|
|
ASDisplayNodeAssert(CATransform3DIsIdentity(layer.transform), @"-[ASDisplayNode setFrame:] - self.transform must be identity in order to set the frame property. (From Apple's UIView documentation: If the transform property is not the identity transform, the value of this property is undefined and therefore should be ignored.)");
|
|
#endif
|
|
view.frame = frame;
|
|
} else {
|
|
ASPendingStateApplyMetricsToLayer(self, layer);
|
|
}
|
|
}
|
|
|
|
// FIXME: Make this more efficient by tracking which properties are set rather than reading everything.
|
|
+ (_ASPendingState *)pendingViewStateFromLayer:(CALayer *)layer
|
|
{
|
|
if (!layer) {
|
|
return nil;
|
|
}
|
|
_ASPendingState *pendingState = [[_ASPendingState alloc] init];
|
|
pendingState.anchorPoint = layer.anchorPoint;
|
|
pendingState.position = layer.position;
|
|
pendingState.zPosition = layer.zPosition;
|
|
pendingState.bounds = layer.bounds;
|
|
pendingState.contentsScale = layer.contentsScale;
|
|
pendingState.transform = layer.transform;
|
|
pendingState.sublayerTransform = layer.sublayerTransform;
|
|
pendingState.contents = layer.contents;
|
|
pendingState.clipsToBounds = layer.masksToBounds;
|
|
pendingState.backgroundColor = layer.backgroundColor;
|
|
pendingState.opaque = layer.opaque;
|
|
pendingState.hidden = layer.hidden;
|
|
pendingState.alpha = layer.opacity;
|
|
pendingState.cornerRadius = layer.cornerRadius;
|
|
pendingState.contentMode = ASDisplayNodeUIContentModeFromCAContentsGravity(layer.contentsGravity);
|
|
pendingState.shadowColor = layer.shadowColor;
|
|
pendingState.shadowOpacity = layer.shadowOpacity;
|
|
pendingState.shadowOffset = layer.shadowOffset;
|
|
pendingState.shadowRadius = layer.shadowRadius;
|
|
pendingState.borderWidth = layer.borderWidth;
|
|
pendingState.borderColor = layer.borderColor;
|
|
pendingState.needsDisplayOnBoundsChange = layer.needsDisplayOnBoundsChange;
|
|
pendingState.allowsEdgeAntialiasing = layer.allowsEdgeAntialiasing;
|
|
pendingState.edgeAntialiasingMask = layer.edgeAntialiasingMask;
|
|
return pendingState;
|
|
}
|
|
|
|
// FIXME: Make this more efficient by tracking which properties are set rather than reading everything.
|
|
+ (_ASPendingState *)pendingViewStateFromView:(UIView *)view
|
|
{
|
|
if (!view) {
|
|
return nil;
|
|
}
|
|
_ASPendingState *pendingState = [[_ASPendingState alloc] init];
|
|
|
|
CALayer *layer = view.layer;
|
|
pendingState.anchorPoint = layer.anchorPoint;
|
|
pendingState.position = layer.position;
|
|
pendingState.zPosition = layer.zPosition;
|
|
pendingState.bounds = view.bounds;
|
|
pendingState.contentsScale = layer.contentsScale;
|
|
pendingState.transform = layer.transform;
|
|
pendingState.sublayerTransform = layer.sublayerTransform;
|
|
pendingState.contents = layer.contents;
|
|
pendingState.clipsToBounds = view.clipsToBounds;
|
|
pendingState.backgroundColor = layer.backgroundColor;
|
|
pendingState.tintColor = view.tintColor;
|
|
pendingState.opaque = layer.opaque;
|
|
pendingState.hidden = view.hidden;
|
|
pendingState.alpha = view.alpha;
|
|
pendingState.cornerRadius = layer.cornerRadius;
|
|
pendingState.contentMode = view.contentMode;
|
|
pendingState.userInteractionEnabled = view.userInteractionEnabled;
|
|
#if TARGET_OS_IOS
|
|
pendingState.exclusiveTouch = view.exclusiveTouch;
|
|
#endif
|
|
pendingState.shadowColor = layer.shadowColor;
|
|
pendingState.shadowOpacity = layer.shadowOpacity;
|
|
pendingState.shadowOffset = layer.shadowOffset;
|
|
pendingState.shadowRadius = layer.shadowRadius;
|
|
pendingState.borderWidth = layer.borderWidth;
|
|
pendingState.borderColor = layer.borderColor;
|
|
pendingState.autoresizingMask = view.autoresizingMask;
|
|
pendingState.autoresizesSubviews = view.autoresizesSubviews;
|
|
pendingState.needsDisplayOnBoundsChange = layer.needsDisplayOnBoundsChange;
|
|
pendingState.allowsEdgeAntialiasing = layer.allowsEdgeAntialiasing;
|
|
pendingState.edgeAntialiasingMask = layer.edgeAntialiasingMask;
|
|
pendingState.isAccessibilityElement = view.isAccessibilityElement;
|
|
pendingState.accessibilityLabel = view.accessibilityLabel;
|
|
pendingState.accessibilityHint = view.accessibilityHint;
|
|
pendingState.accessibilityValue = view.accessibilityValue;
|
|
pendingState.accessibilityTraits = view.accessibilityTraits;
|
|
pendingState.accessibilityFrame = view.accessibilityFrame;
|
|
pendingState.accessibilityLanguage = view.accessibilityLanguage;
|
|
pendingState.accessibilityElementsHidden = view.accessibilityElementsHidden;
|
|
pendingState.accessibilityViewIsModal = view.accessibilityViewIsModal;
|
|
pendingState.shouldGroupAccessibilityChildren = view.shouldGroupAccessibilityChildren;
|
|
pendingState.accessibilityIdentifier = view.accessibilityIdentifier;
|
|
return pendingState;
|
|
}
|
|
|
|
- (void)clearChanges
|
|
{
|
|
_flags = (ASPendingStateFlags){ 0 };
|
|
}
|
|
|
|
- (BOOL)hasSetNeedsLayout
|
|
{
|
|
return _flags.needsLayout;
|
|
}
|
|
|
|
- (BOOL)hasSetNeedsDisplay
|
|
{
|
|
return _flags.needsDisplay;
|
|
}
|
|
|
|
- (BOOL)hasChanges
|
|
{
|
|
ASPendingStateFlags flags = _flags;
|
|
|
|
return (flags.setAnchorPoint
|
|
|| flags.setPosition
|
|
|| flags.setZPosition
|
|
|| flags.setFrame
|
|
|| flags.setBounds
|
|
|| flags.setPosition
|
|
|| flags.setContentsScale
|
|
|| flags.setTransform
|
|
|| flags.setSublayerTransform
|
|
|| flags.setContents
|
|
|| flags.setClipsToBounds
|
|
|| flags.setBackgroundColor
|
|
|| flags.setTintColor
|
|
|| flags.setHidden
|
|
|| flags.setAlpha
|
|
|| flags.setCornerRadius
|
|
|| flags.setContentMode
|
|
|| flags.setUserInteractionEnabled
|
|
|| flags.setExclusiveTouch
|
|
|| flags.setShadowOpacity
|
|
|| flags.setShadowOffset
|
|
|| flags.setShadowRadius
|
|
|| flags.setShadowColor
|
|
|| flags.setBorderWidth
|
|
|| flags.setBorderColor
|
|
|| flags.setAutoresizingMask
|
|
|| flags.setAutoresizesSubviews
|
|
|| flags.setNeedsDisplayOnBoundsChange
|
|
|| flags.setAllowsEdgeAntialiasing
|
|
|| flags.setEdgeAntialiasingMask
|
|
|| flags.needsDisplay
|
|
|| flags.needsLayout
|
|
|| flags.setAsyncTransactionContainer
|
|
|| flags.setOpaque
|
|
|| flags.setIsAccessibilityElement
|
|
|| flags.setAccessibilityLabel
|
|
|| flags.setAccessibilityHint
|
|
|| flags.setAccessibilityValue
|
|
|| flags.setAccessibilityTraits
|
|
|| flags.setAccessibilityFrame
|
|
|| flags.setAccessibilityLanguage
|
|
|| flags.setAccessibilityElementsHidden
|
|
|| flags.setAccessibilityViewIsModal
|
|
|| flags.setShouldGroupAccessibilityChildren
|
|
|| flags.setAccessibilityIdentifier);
|
|
}
|
|
|
|
- (void)dealloc
|
|
{
|
|
CGColorRelease(backgroundColor);
|
|
|
|
if (shadowColor != blackColorRef) {
|
|
CGColorRelease(shadowColor);
|
|
}
|
|
|
|
if (borderColor != blackColorRef) {
|
|
CGColorRelease(borderColor);
|
|
}
|
|
}
|
|
|
|
@end
|