mirror of
https://github.com/Swiftgram/Telegram-iOS.git
synced 2025-09-04 20:00:53 +00:00
Add basic infrastructure for layout validation
This commit is contained in:
parent
cb67671f1d
commit
197950f39b
@ -285,6 +285,10 @@
|
||||
68FC85EB1CE29C7D00EDD713 /* ASVisibilityProtocols.m in Sources */ = {isa = PBXBuildFile; fileRef = 68FC85E81CE29C7D00EDD713 /* ASVisibilityProtocols.m */; };
|
||||
68FC85EC1CE29C7D00EDD713 /* ASVisibilityProtocols.m in Sources */ = {isa = PBXBuildFile; fileRef = 68FC85E81CE29C7D00EDD713 /* ASVisibilityProtocols.m */; };
|
||||
697B315A1CFE4B410049936F /* ASEditableTextNodeTests.m in Sources */ = {isa = PBXBuildFile; fileRef = 697B31591CFE4B410049936F /* ASEditableTextNodeTests.m */; };
|
||||
697C0DE31CF38F28001DE0D4 /* ASLayoutValidation.h in Headers */ = {isa = PBXBuildFile; fileRef = 697C0DE11CF38F28001DE0D4 /* ASLayoutValidation.h */; };
|
||||
697C0DE41CF38F28001DE0D4 /* ASLayoutValidation.h in Headers */ = {isa = PBXBuildFile; fileRef = 697C0DE11CF38F28001DE0D4 /* ASLayoutValidation.h */; };
|
||||
697C0DE51CF38F28001DE0D4 /* ASLayoutValidation.mm in Sources */ = {isa = PBXBuildFile; fileRef = 697C0DE21CF38F28001DE0D4 /* ASLayoutValidation.mm */; };
|
||||
697C0DE61CF38F28001DE0D4 /* ASLayoutValidation.mm in Sources */ = {isa = PBXBuildFile; fileRef = 697C0DE21CF38F28001DE0D4 /* ASLayoutValidation.mm */; };
|
||||
698548631CA9E025008A345F /* ASEnvironment.h in Headers */ = {isa = PBXBuildFile; fileRef = 698548611CA9E025008A345F /* ASEnvironment.h */; settings = {ATTRIBUTES = (Public, ); }; };
|
||||
698548641CA9E025008A345F /* ASEnvironment.h in Headers */ = {isa = PBXBuildFile; fileRef = 698548611CA9E025008A345F /* ASEnvironment.h */; settings = {ATTRIBUTES = (Public, ); }; };
|
||||
698C8B611CAB49FC0052DC3F /* ASLayoutableExtensibility.h in Headers */ = {isa = PBXBuildFile; fileRef = 698C8B601CAB49FC0052DC3F /* ASLayoutableExtensibility.h */; settings = {ATTRIBUTES = (Public, ); }; };
|
||||
@ -815,6 +819,8 @@
|
||||
68FC85E71CE29C7D00EDD713 /* ASVisibilityProtocols.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; path = ASVisibilityProtocols.h; sourceTree = "<group>"; };
|
||||
68FC85E81CE29C7D00EDD713 /* ASVisibilityProtocols.m */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.objc; path = ASVisibilityProtocols.m; sourceTree = "<group>"; };
|
||||
697B31591CFE4B410049936F /* ASEditableTextNodeTests.m */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.objc; path = ASEditableTextNodeTests.m; sourceTree = "<group>"; };
|
||||
697C0DE11CF38F28001DE0D4 /* ASLayoutValidation.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = ASLayoutValidation.h; path = AsyncDisplayKit/Layout/ASLayoutValidation.h; sourceTree = "<group>"; };
|
||||
697C0DE21CF38F28001DE0D4 /* ASLayoutValidation.mm */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.cpp.objcpp; name = ASLayoutValidation.mm; path = AsyncDisplayKit/Layout/ASLayoutValidation.mm; sourceTree = "<group>"; };
|
||||
698548611CA9E025008A345F /* ASEnvironment.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; path = ASEnvironment.h; sourceTree = "<group>"; };
|
||||
698C8B601CAB49FC0052DC3F /* ASLayoutableExtensibility.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = ASLayoutableExtensibility.h; path = AsyncDisplayKit/Layout/ASLayoutableExtensibility.h; sourceTree = "<group>"; };
|
||||
69CB62A91CB8165900024920 /* _ASDisplayViewAccessiblity.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; path = _ASDisplayViewAccessiblity.h; sourceTree = "<group>"; };
|
||||
@ -1454,6 +1460,8 @@
|
||||
9CDC18CB1B910E12004965E2 /* ASLayoutablePrivate.h */,
|
||||
ACF6ED0D1B17843500DA7C62 /* ASLayoutSpec.h */,
|
||||
ACF6ED0E1B17843500DA7C62 /* ASLayoutSpec.mm */,
|
||||
697C0DE11CF38F28001DE0D4 /* ASLayoutValidation.h */,
|
||||
697C0DE21CF38F28001DE0D4 /* ASLayoutValidation.mm */,
|
||||
ACF6ED121B17843500DA7C62 /* ASOverlayLayoutSpec.h */,
|
||||
ACF6ED131B17843500DA7C62 /* ASOverlayLayoutSpec.mm */,
|
||||
ACF6ED141B17843500DA7C62 /* ASRatioLayoutSpec.h */,
|
||||
@ -1572,6 +1580,7 @@
|
||||
AC7A2C171BDE11DF0093FE1A /* ASTableViewInternal.h in Headers */,
|
||||
058D0A4D195D05CB00B7D73C /* ASDisplayNodeExtras.h in Headers */,
|
||||
92074A671CC8BADA00918F75 /* ASControlNode+tvOS.h in Headers */,
|
||||
697C0DE31CF38F28001DE0D4 /* ASLayoutValidation.h in Headers */,
|
||||
68355B3B1CB57A5A001D4E68 /* ASImageContainerProtocolCategories.h in Headers */,
|
||||
68B0277A1C1A79CC0041016B /* ASDisplayNode+Beta.h in Headers */,
|
||||
767E7F8D1C9019130066C000 /* AsyncDisplayKit+Debug.h in Headers */,
|
||||
@ -1758,6 +1767,7 @@
|
||||
9CDC18CD1B910E12004965E2 /* ASLayoutablePrivate.h in Headers */,
|
||||
68B8A4E21CBDB958007E4543 /* ASWeakProxy.h in Headers */,
|
||||
B35062201B010EFD0018CF92 /* ASLayoutController.h in Headers */,
|
||||
697C0DE41CF38F28001DE0D4 /* ASLayoutValidation.h in Headers */,
|
||||
B35062211B010EFD0018CF92 /* ASLayoutRangeType.h in Headers */,
|
||||
34EFC76A1B701CE600AD841F /* ASLayoutSpec.h in Headers */,
|
||||
34EFC7791B701D3600AD841F /* ASLayoutSpecUtilities.h in Headers */,
|
||||
@ -2121,6 +2131,7 @@
|
||||
ACF6ED301B17843500DA7C62 /* ASStackLayoutSpec.mm in Sources */,
|
||||
257754BE1BEE458E00737CA5 /* ASTextKitComponents.m in Sources */,
|
||||
257754A91BEE44CD00737CA5 /* ASTextKitContext.mm in Sources */,
|
||||
697C0DE51CF38F28001DE0D4 /* ASLayoutValidation.mm in Sources */,
|
||||
ACF6ED501B17847A00DA7C62 /* ASStackPositionedLayout.mm in Sources */,
|
||||
ACF6ED521B17847A00DA7C62 /* ASStackUnpositionedLayout.mm in Sources */,
|
||||
257754A61BEE44CD00737CA5 /* ASTextKitAttributes.mm in Sources */,
|
||||
@ -2285,6 +2296,7 @@
|
||||
34EFC7721B701D0300AD841F /* ASStackLayoutSpec.mm in Sources */,
|
||||
34EFC7761B701D2A00AD841F /* ASStackPositionedLayout.mm in Sources */,
|
||||
7AB338661C55B3420055FDE8 /* ASRelativeLayoutSpec.mm in Sources */,
|
||||
697C0DE61CF38F28001DE0D4 /* ASLayoutValidation.mm in Sources */,
|
||||
9C70F2051CDA4F06007D6C76 /* ASTraitCollection.m in Sources */,
|
||||
34EFC7781B701D3100AD841F /* ASStackUnpositionedLayout.mm in Sources */,
|
||||
DE84918E1C8FFF9F003D89E9 /* ASRunLoopQueue.mm in Sources */,
|
||||
|
@ -26,6 +26,7 @@
|
||||
#import "ASEqualityHelpers.h"
|
||||
#import "ASRunLoopQueue.h"
|
||||
#import "ASEnvironmentInternal.h"
|
||||
#import "ASLayoutValidation.h"
|
||||
|
||||
#import "ASInternalHelpers.h"
|
||||
#import "ASLayout.h"
|
||||
@ -854,6 +855,16 @@ static ASDisplayNodeMethodOverrides GetASDisplayNodeMethodOverrides(Class c)
|
||||
_pendingLayoutTransition = nil;
|
||||
}
|
||||
|
||||
#pragma mark - Layout Validation
|
||||
|
||||
- (void)validateLayout:(ASLayout *)layout
|
||||
{
|
||||
ASLayoutableValidation *validation = [[ASLayoutableValidation alloc] init];
|
||||
[validation registerValidator:[[ASLayoutableStaticValidator alloc] init]];
|
||||
[validation registerValidator:[[ASLayoutableStackValidator alloc] init]];
|
||||
[validation validateLayout:layout];
|
||||
}
|
||||
|
||||
|
||||
#pragma mark - _ASTransitionContextCompletionDelegate
|
||||
|
||||
@ -1905,12 +1916,13 @@ void recursivelyTriggerDisplayForLayer(CALayer *layer, BOOL shouldBlock)
|
||||
layoutSpec.isMutable = NO;
|
||||
ASLayout *layout = [layoutSpec measureWithSizeRange:constrainedSize];
|
||||
// Make sure layoutableObject of the root layout is `self`, so that the flattened layout will be structurally correct.
|
||||
if (layout.layoutableObject != self) {
|
||||
BOOL isRootLayout = (layout.layoutableObject != self);
|
||||
if (isRootLayout) {
|
||||
layout.position = CGPointZero;
|
||||
layout = [ASLayout layoutWithLayoutableObject:self
|
||||
constrainedSizeRange:constrainedSize
|
||||
size:layout.size
|
||||
sublayouts:@[layout]];
|
||||
layout = [ASLayout layoutWithLayoutableObject:self constrainedSizeRange:constrainedSize size:layout.size sublayouts:@[layout]];
|
||||
#if DEBUG
|
||||
[self validateLayout:layout];
|
||||
#endif
|
||||
}
|
||||
return [layout flattenedLayoutUsingPredicateBlock:^BOOL(ASLayout *evaluatedLayout) {
|
||||
if (self.usesImplicitHierarchyManagement) {
|
||||
|
73
AsyncDisplayKit/Layout/ASLayoutValidation.h
Normal file
73
AsyncDisplayKit/Layout/ASLayoutValidation.h
Normal file
@ -0,0 +1,73 @@
|
||||
/*
|
||||
* 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 <Foundation/Foundation.h>
|
||||
|
||||
NS_ASSUME_NONNULL_BEGIN
|
||||
|
||||
@class ASLayout;
|
||||
|
||||
#pragma mark - ASLayoutableValidator
|
||||
|
||||
@protocol ASLayoutableValidator <NSObject>
|
||||
- (void)validateLayout:(ASLayout *)layout;
|
||||
@end
|
||||
|
||||
typedef void (^ASLayoutableBlockValidatorBlock)(id layout);
|
||||
|
||||
@interface ASLayoutableBlockValidator : NSObject<ASLayoutableValidator>
|
||||
@property (nonatomic, copy) ASLayoutableBlockValidatorBlock block;
|
||||
- (instancetype)initWithBlock:(ASLayoutableBlockValidatorBlock)block NS_DESIGNATED_INITIALIZER;
|
||||
- (instancetype)init NS_UNAVAILABLE;
|
||||
@end
|
||||
|
||||
/*
|
||||
* ASLayoutables that have sizeRange or layoutPosition set needs to be wrapped into a ASStaticLayoutSpec. This
|
||||
* validator checks if sublayouts has sizeRange or layoutPosition set and is wrapped in a ASStaticLayoutSpec
|
||||
*/
|
||||
@interface ASLayoutableStaticValidator : NSObject<ASLayoutableValidator>
|
||||
|
||||
@end
|
||||
|
||||
/*
|
||||
* ASLayoutables that have spacingBefore, spacingAfter, flexGrow, flexShrink, flexBasis, alignSelf, ascender or descender
|
||||
* set needs to be wrapped into a ASStackLayout. This validator checks if sublayouts has set one of this properties and
|
||||
* asserts if it's not wrapped in a ASStackLayout if so.
|
||||
*/
|
||||
@interface ASLayoutableStackValidator : NSObject<ASLayoutableValidator>
|
||||
|
||||
@end
|
||||
|
||||
@interface ASLayoutablePreferredSizeValidator : NSObject<ASLayoutableValidator>
|
||||
|
||||
@end
|
||||
|
||||
|
||||
#pragma mark - ASLayoutableValidation
|
||||
|
||||
@interface ASLayoutableValidation : NSObject
|
||||
|
||||
/// Currently registered validators
|
||||
@property (copy, nonatomic, readonly) NSMutableArray<id<ASLayoutableValidator>> *validators;
|
||||
|
||||
/// Start from given layout and validates each layout in the layout tree with registered validators
|
||||
- (void)validateLayout:(ASLayout *)layout;
|
||||
|
||||
/// Register a layout validator
|
||||
- (void)registerValidator:(id<ASLayoutableValidator>)validator;
|
||||
|
||||
/// Register a layout validator with a block. Method returns the registered ASLayoutableValidator object that can be used to store somewhere and unregister
|
||||
- (id<ASLayoutableValidator>)registerValidatorWithBlock:(ASLayoutableBlockValidatorBlock)block;
|
||||
|
||||
/// Unregister a validtor
|
||||
- (void)unregisterValidator:(id<ASLayoutableValidator>)validator;
|
||||
@end
|
||||
|
||||
NS_ASSUME_NONNULL_END
|
197
AsyncDisplayKit/Layout/ASLayoutValidation.mm
Normal file
197
AsyncDisplayKit/Layout/ASLayoutValidation.mm
Normal file
@ -0,0 +1,197 @@
|
||||
/*
|
||||
* 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 "ASLayoutValidation.h"
|
||||
#import "ASLayout.h"
|
||||
#import "ASDisplayNode.h"
|
||||
|
||||
#import "ASStaticLayoutSpec.h"
|
||||
#import "ASStackLayoutSpec.h"
|
||||
|
||||
#import <queue>
|
||||
|
||||
#pragma mark ASLayoutableBlockValidator
|
||||
|
||||
@implementation ASLayoutableBlockValidator
|
||||
|
||||
#pragma mark Lifecycle
|
||||
|
||||
- (instancetype)initWithBlock:(ASLayoutableBlockValidatorBlock)block
|
||||
{
|
||||
self = [super init];
|
||||
if (self) {
|
||||
_block = [block copy];
|
||||
}
|
||||
return self;
|
||||
}
|
||||
|
||||
#pragma mark <ASLayoutableValidator>
|
||||
|
||||
- (void)validateLayout:(ASLayout *)layout
|
||||
{
|
||||
if (self.block) {
|
||||
self.block(layout);
|
||||
}
|
||||
}
|
||||
|
||||
@end
|
||||
|
||||
#pragma mark ASLayoutableStaticValidator
|
||||
|
||||
@implementation ASLayoutableStaticValidator
|
||||
|
||||
- (void)validateLayout:(ASLayout *)layout
|
||||
{
|
||||
id<ASLayoutable> layoutable = layout.layoutableObject;
|
||||
for (ASLayout * sublayout in layout.sublayouts) {
|
||||
id<ASLayoutable> sublayoutLayoutable = sublayout.layoutableObject;
|
||||
|
||||
// Check for default sizeRange and layoutPosition
|
||||
ASRelativeSizeRange sizeRange = sublayoutLayoutable.sizeRange;
|
||||
ASRelativeSizeRange zeroSizeRange = ASRelativeSizeRangeMake(ASRelativeSizeMakeWithCGSize(CGSizeZero),
|
||||
ASRelativeSizeMakeWithCGSize(CGSizeZero));
|
||||
|
||||
// Currently setting the preferredFrameSize also updates the sizeRange. Create a size range based on the
|
||||
// preferredFrameSize and check it if it's the same as the current sizeRange to be sure it was not changed manually
|
||||
CGSize preferredFrameSize = CGSizeZero;
|
||||
if ([sublayoutLayoutable respondsToSelector:@selector(preferredFrameSize)]) {
|
||||
preferredFrameSize = [((ASDisplayNode *)sublayoutLayoutable) preferredFrameSize];
|
||||
}
|
||||
ASRelativeSizeRange preferredFrameSizeRange = ASRelativeSizeRangeMakeWithExactCGSize(preferredFrameSize);
|
||||
|
||||
if ((ASRelativeSizeRangeEqualToRelativeSizeRange(sizeRange, zeroSizeRange) ||
|
||||
ASRelativeSizeRangeEqualToRelativeSizeRange(sizeRange, preferredFrameSizeRange))
|
||||
&& CGPointEqualToPoint(sublayoutLayoutable.layoutPosition, CGPointZero)) {
|
||||
continue;
|
||||
}
|
||||
|
||||
// Sublayout layoutable needs to be wrapped in a ASStaticLayoutSpec
|
||||
if ([layoutable isKindOfClass:[ASStaticLayoutSpec class]]) {
|
||||
continue;
|
||||
}
|
||||
|
||||
ASDisplayNodeCAssert(NO, @"A property was set that requires the ASLayoutable to be wrapped in a ASStaticLayoutSpec");
|
||||
}
|
||||
}
|
||||
|
||||
@end
|
||||
|
||||
#pragma mark ASLayoutableStackValidator
|
||||
|
||||
@implementation ASLayoutableStackValidator
|
||||
|
||||
#pragma mark <ASLayoutableValidator>
|
||||
|
||||
- (void)validateLayout:(ASLayout *)layout
|
||||
{
|
||||
id<ASLayoutable> layoutable = layout.layoutableObject;
|
||||
for (ASLayout *sublayout in layout.sublayouts) {
|
||||
id<ASLayoutable> sublayoutLayoutable = sublayout.layoutableObject;
|
||||
|
||||
// Check if default values related to ASStackLayoutSpec have changed
|
||||
if (sublayoutLayoutable.spacingBefore == 0 &&
|
||||
sublayoutLayoutable.spacingAfter == 0 &&
|
||||
!sublayoutLayoutable.flexGrow &&
|
||||
!sublayoutLayoutable.flexShrink &&
|
||||
ASRelativeDimensionEqualToRelativeDimension([sublayoutLayoutable flexBasis], ASRelativeDimensionUnconstrained) &&
|
||||
sublayoutLayoutable.alignSelf == ASStackLayoutAlignSelfAuto)
|
||||
{
|
||||
continue;
|
||||
}
|
||||
|
||||
// Sublayout layoutable needs to be wrapped in a ASStackLayoutSpec
|
||||
if ([layoutable isKindOfClass:[ASStackLayoutSpec class]]) {
|
||||
continue;
|
||||
}
|
||||
|
||||
ASDisplayNodeCAssert(NO, @"A property was set that requires the ASLayoutable to be wrapped in a ASStackLayoutSpec");
|
||||
}
|
||||
}
|
||||
|
||||
@end
|
||||
|
||||
#pragma mark ASLayoutablePreferredSizeValidator
|
||||
|
||||
@implementation ASLayoutablePreferredSizeValidator
|
||||
|
||||
#pragma mark <ASLayoutableValidator>
|
||||
|
||||
- (void)validateLayout:(ASLayout *)layout
|
||||
{
|
||||
// TODO: Implement validation that certain node classes need to have a preferredSize set e.g. ASVideoNode
|
||||
}
|
||||
|
||||
@end
|
||||
|
||||
|
||||
#pragma mark - ASLayoutableValidation
|
||||
|
||||
@interface ASLayoutableValidation ()
|
||||
@property (copy, nonatomic) NSMutableArray<id<ASLayoutableValidator>> *validators;
|
||||
@end
|
||||
|
||||
@implementation ASLayoutableValidation
|
||||
|
||||
#pragma mark Lifecycle
|
||||
|
||||
- (instancetype)init
|
||||
{
|
||||
self = [super init];
|
||||
if (self) {
|
||||
_validators = [NSMutableArray array];
|
||||
}
|
||||
return self;
|
||||
}
|
||||
|
||||
#pragma mark Validator Management
|
||||
|
||||
- (void)registerValidator:(id<ASLayoutableValidator>)validator
|
||||
{
|
||||
[self.validators addObject:validator];
|
||||
}
|
||||
|
||||
- (id<ASLayoutableValidator>)registerValidatorWithBlock:(ASLayoutableBlockValidatorBlock)block
|
||||
{
|
||||
ASLayoutableBlockValidator *blockValidator = [[ASLayoutableBlockValidator alloc] initWithBlock:block];
|
||||
[self.validators addObject:blockValidator];
|
||||
return blockValidator;
|
||||
}
|
||||
|
||||
- (void)unregisterValidator:(id<ASLayoutableValidator>)validator
|
||||
{
|
||||
[self.validators removeObject:validator];
|
||||
}
|
||||
|
||||
#pragma mark Validation Process
|
||||
|
||||
- (void)validateLayout:(ASLayout *)layout
|
||||
{
|
||||
// Queue used to keep track of sublayouts while traversing this layout in a BFS fashion.
|
||||
std::queue<ASLayout *> queue;
|
||||
queue.push(layout);
|
||||
|
||||
while (!queue.empty()) {
|
||||
layout = queue.front();
|
||||
queue.pop();
|
||||
|
||||
// Validate layout with all registered validators
|
||||
for (id<ASLayoutableValidator> validator in self.validators) {
|
||||
[validator validateLayout:layout];
|
||||
}
|
||||
|
||||
// Push sublayouts to queue for validation
|
||||
for (id sublayout in [layout sublayouts]) {
|
||||
queue.push(sublayout);
|
||||
}
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
@end
|
Loading…
x
Reference in New Issue
Block a user