mirror of
https://github.com/Swiftgram/Telegram-iOS.git
synced 2025-06-16 05:55:20 +00:00

git-subtree-dir: submodules/AsyncDisplayKit git-subtree-mainline: d06f423e0ed3df1fed9bd10d79ee312a9179b632 git-subtree-split: 02bedc12816e251ad71777f9d2578329b6d2bef6
342 lines
14 KiB
Objective-C
342 lines
14 KiB
Objective-C
//
|
|
// PhotoCellNode.m
|
|
// Texture
|
|
//
|
|
// Copyright (c) Facebook, Inc. and its affiliates. All rights reserved.
|
|
// Changes after 4/13/2017 are: Copyright (c) Pinterest, Inc. All rights reserved.
|
|
// Licensed under Apache 2.0: http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
|
|
#import "PhotoCellNode.h"
|
|
|
|
#import <AsyncDisplayKit/AsyncDisplayKit.h>
|
|
#import <AsyncDisplayKit/ASDisplayNode+Beta.h>
|
|
|
|
#import "Availability.h"
|
|
#import "PINImageView+PINRemoteImage.h"
|
|
#import "PINButton+PINRemoteImage.h"
|
|
#import "Utilities.h"
|
|
|
|
#define DEBUG_PHOTOCELL_LAYOUT 0
|
|
|
|
#define HEADER_HEIGHT 50
|
|
#define USER_IMAGE_HEIGHT 30
|
|
#define HORIZONTAL_BUFFER 10
|
|
#define VERTICAL_BUFFER 5
|
|
#define FONT_SIZE 14
|
|
|
|
#define InsetForAvatar UIEdgeInsetsMake(HORIZONTAL_BUFFER, 0, HORIZONTAL_BUFFER, HORIZONTAL_BUFFER)
|
|
#define InsetForHeader UIEdgeInsetsMake(0, HORIZONTAL_BUFFER, 0, HORIZONTAL_BUFFER)
|
|
#define InsetForFooter UIEdgeInsetsMake(VERTICAL_BUFFER, HORIZONTAL_BUFFER, VERTICAL_BUFFER, HORIZONTAL_BUFFER)
|
|
|
|
@interface PhotoCellNode () <ASNetworkImageNodeDelegate>
|
|
@end
|
|
|
|
@implementation PhotoCellNode
|
|
{
|
|
PhotoModel *_photoModel;
|
|
ASNetworkImageNode *_userAvatarImageNode;
|
|
ASNetworkImageNode *_photoImageNode;
|
|
ASTextNode *_userNameLabel;
|
|
ASTextNode *_photoLocationLabel;
|
|
ASTextNode *_photoTimeIntervalSincePostLabel;
|
|
ASTextNode *_photoLikesLabel;
|
|
ASTextNode *_photoDescriptionLabel;
|
|
}
|
|
|
|
#pragma mark - Lifecycle
|
|
|
|
- (instancetype)initWithPhotoObject:(PhotoModel *)photo;
|
|
{
|
|
self = [super init];
|
|
|
|
if (self) {
|
|
|
|
_photoModel = photo;
|
|
|
|
_userAvatarImageNode = [[ASNetworkImageNode alloc] init];
|
|
_userAvatarImageNode.URL = photo.ownerUserProfile.userPicURL; // FIXME: make round
|
|
|
|
// FIXME: autocomplete for this line seems broken
|
|
[_userAvatarImageNode setImageModificationBlock:^UIImage *(UIImage *image) {
|
|
CGSize profileImageSize = CGSizeMake(USER_IMAGE_HEIGHT, USER_IMAGE_HEIGHT);
|
|
return [image makeCircularImageWithSize:profileImageSize];
|
|
}];
|
|
|
|
_photoImageNode = [[ASNetworkImageNode alloc] init];
|
|
_photoImageNode.delegate = self;
|
|
_photoImageNode.URL = photo.URL;
|
|
_photoImageNode.layerBacked = YES;
|
|
|
|
_userNameLabel = [[ASTextNode alloc] init];
|
|
_userNameLabel.attributedText = [photo.ownerUserProfile usernameAttributedStringWithFontSize:FONT_SIZE];
|
|
|
|
_photoLocationLabel = [[ASTextNode alloc] init];
|
|
_photoLocationLabel.maximumNumberOfLines = 1;
|
|
_photoLocationLabel.attributedText = [photo locationAttributedStringWithFontSize:FONT_SIZE];
|
|
|
|
_photoTimeIntervalSincePostLabel = [self createLayerBackedTextNodeWithString:[photo uploadDateAttributedStringWithFontSize:FONT_SIZE]];
|
|
_photoLikesLabel = [self createLayerBackedTextNodeWithString:[photo likesAttributedStringWithFontSize:FONT_SIZE]];
|
|
_photoDescriptionLabel = [self createLayerBackedTextNodeWithString:[photo descriptionAttributedStringWithFontSize:FONT_SIZE]];
|
|
_photoDescriptionLabel.maximumNumberOfLines = 3;
|
|
|
|
// instead of adding everything addSubnode:
|
|
self.automaticallyManagesSubnodes = YES;
|
|
|
|
[self setupYogaLayoutIfNeeded];
|
|
|
|
#if DEBUG_PHOTOCELL_LAYOUT
|
|
_userAvatarImageNode.backgroundColor = [UIColor greenColor];
|
|
_userNameLabel.backgroundColor = [UIColor greenColor];
|
|
_photoLocationLabel.backgroundColor = [UIColor greenColor];
|
|
_photoTimeIntervalSincePostLabel.backgroundColor = [UIColor greenColor];
|
|
_photoCommentsNode.backgroundColor = [UIColor greenColor];
|
|
_photoDescriptionLabel.backgroundColor = [UIColor greenColor];
|
|
_photoLikesLabel.backgroundColor = [UIColor greenColor];
|
|
#endif
|
|
}
|
|
|
|
return self;
|
|
}
|
|
|
|
#if !YOGA_LAYOUT
|
|
- (ASLayoutSpec *)layoutSpecThatFits:(ASSizeRange)constrainedSize
|
|
{
|
|
// There are many ways to format ASLayoutSpec code. In this example, we offer two different formats:
|
|
// A flatter, more ordinary Objective-C style; or a more structured, "visually" declarative style.
|
|
if (FLAT_LAYOUT) {
|
|
// This layout has a horizontal stack of header items at the top, set within a vertical stack of items.
|
|
NSMutableArray *headerChildren = [NSMutableArray array];
|
|
NSMutableArray *verticalChildren = [NSMutableArray array];
|
|
|
|
// Header stack
|
|
ASStackLayoutSpec *headerStack = [ASStackLayoutSpec horizontalStackLayoutSpec];
|
|
headerStack.alignItems = ASStackLayoutAlignItemsCenter;
|
|
|
|
// Avatar Image, with inset - first thing in the header stack.
|
|
_userAvatarImageNode.style.preferredSize = CGSizeMake(USER_IMAGE_HEIGHT, USER_IMAGE_HEIGHT);
|
|
[headerChildren addObject:[ASInsetLayoutSpec insetLayoutSpecWithInsets:InsetForAvatar child:_userAvatarImageNode]];
|
|
|
|
// User Name and Photo Location stack is next
|
|
ASStackLayoutSpec *userPhotoLocationStack = [ASStackLayoutSpec verticalStackLayoutSpec];
|
|
userPhotoLocationStack.style.flexShrink = 1.0;
|
|
[headerChildren addObject:userPhotoLocationStack];
|
|
|
|
// Setup the inside of the User Name and Photo Location stack.
|
|
_userNameLabel.style.flexShrink = 1.0;
|
|
[userPhotoLocationStack setChildren:@[_userNameLabel]];
|
|
|
|
if (_photoLocationLabel.attributedText) {
|
|
_photoLocationLabel.style.flexShrink = 1.0;
|
|
[userPhotoLocationStack setChildren:[userPhotoLocationStack.children arrayByAddingObject:_photoLocationLabel]];
|
|
}
|
|
|
|
// Add a spacer to allow a flexible space between the User Name / Location stack, and the Timestamp.
|
|
ASLayoutSpec *spacer = [ASLayoutSpec new];
|
|
spacer.style.flexGrow = 1.0;
|
|
[headerChildren addObject:spacer];
|
|
|
|
// Photo Timestamp Label.
|
|
_photoTimeIntervalSincePostLabel.style.spacingBefore = HORIZONTAL_BUFFER;
|
|
[headerChildren addObject:_photoTimeIntervalSincePostLabel];
|
|
|
|
// Add all of the above items to the horizontal header stack
|
|
headerStack.children = headerChildren;
|
|
|
|
// Create the last stack before assembling everything: the Footer Stack contains the description and comments.
|
|
ASStackLayoutSpec *footerStack = [ASStackLayoutSpec verticalStackLayoutSpec];
|
|
footerStack.spacing = VERTICAL_BUFFER;
|
|
footerStack.children = @[_photoLikesLabel, _photoDescriptionLabel];
|
|
|
|
// Main Vertical Stack: contains header, large main photo with fixed aspect ratio, and footer.
|
|
ASStackLayoutSpec *verticalStack = [ASStackLayoutSpec verticalStackLayoutSpec];
|
|
|
|
[verticalChildren addObject:[ASInsetLayoutSpec insetLayoutSpecWithInsets:InsetForHeader child:headerStack]];
|
|
[verticalChildren addObject:[ASRatioLayoutSpec ratioLayoutSpecWithRatio :1.0 child:_photoImageNode]];
|
|
[verticalChildren addObject:[ASInsetLayoutSpec insetLayoutSpecWithInsets:InsetForFooter child:footerStack]];
|
|
|
|
verticalStack.children = verticalChildren;
|
|
|
|
return verticalStack;
|
|
|
|
} else { // The style below is the more structured, visual, and declarative style. It is functionally identical.
|
|
|
|
return
|
|
// Main stack
|
|
[ASStackLayoutSpec
|
|
stackLayoutSpecWithDirection:ASStackLayoutDirectionVertical
|
|
spacing:0
|
|
justifyContent:ASStackLayoutJustifyContentStart
|
|
alignItems:ASStackLayoutAlignItemsStretch
|
|
children:@[
|
|
|
|
// Header stack with inset
|
|
[ASInsetLayoutSpec
|
|
insetLayoutSpecWithInsets:InsetForHeader
|
|
child:
|
|
// Header stack
|
|
[ASStackLayoutSpec
|
|
stackLayoutSpecWithDirection:ASStackLayoutDirectionHorizontal
|
|
spacing:0.0
|
|
justifyContent:ASStackLayoutJustifyContentStart
|
|
alignItems:ASStackLayoutAlignItemsCenter
|
|
children:@[
|
|
// Avatar image with inset
|
|
[ASInsetLayoutSpec
|
|
insetLayoutSpecWithInsets:InsetForAvatar
|
|
child:
|
|
[_userAvatarImageNode styledWithBlock:^(ASLayoutElementStyle *style) {
|
|
style.preferredSize = CGSizeMake(USER_IMAGE_HEIGHT, USER_IMAGE_HEIGHT);
|
|
}]
|
|
],
|
|
// User and photo location stack
|
|
[[ASStackLayoutSpec
|
|
stackLayoutSpecWithDirection:ASStackLayoutDirectionVertical
|
|
spacing:0.0
|
|
justifyContent:ASStackLayoutJustifyContentStart
|
|
alignItems:ASStackLayoutAlignItemsStretch
|
|
children:_photoLocationLabel.attributedText ? @[
|
|
[_userNameLabel styledWithBlock:^(ASLayoutElementStyle *style) {
|
|
style.flexShrink = 1.0;
|
|
}],
|
|
[_photoLocationLabel styledWithBlock:^(ASLayoutElementStyle *style) {
|
|
style.flexShrink = 1.0;
|
|
}]
|
|
] :
|
|
@[
|
|
[_userNameLabel styledWithBlock:^(ASLayoutElementStyle *style) {
|
|
style.flexShrink = 1.0;
|
|
}]
|
|
]]
|
|
styledWithBlock:^(ASLayoutElementStyle *style) {
|
|
style.flexShrink = 1.0;
|
|
}],
|
|
// Spacer between user / photo location and photo time inverval
|
|
[[ASLayoutSpec new] styledWithBlock:^(ASLayoutElementStyle *style) {
|
|
style.flexGrow = 1.0;
|
|
}],
|
|
// Photo and time interval node
|
|
[_photoTimeIntervalSincePostLabel styledWithBlock:^(ASLayoutElementStyle *style) {
|
|
// to remove double spaces around spacer
|
|
style.spacingBefore = HORIZONTAL_BUFFER;
|
|
}]
|
|
]]
|
|
],
|
|
|
|
// Center photo with ratio
|
|
[ASRatioLayoutSpec
|
|
ratioLayoutSpecWithRatio:1.0
|
|
child:_photoImageNode],
|
|
|
|
// Footer stack with inset
|
|
[ASInsetLayoutSpec
|
|
insetLayoutSpecWithInsets:InsetForFooter
|
|
child:
|
|
[ASStackLayoutSpec
|
|
stackLayoutSpecWithDirection:ASStackLayoutDirectionVertical
|
|
spacing:VERTICAL_BUFFER
|
|
justifyContent:ASStackLayoutJustifyContentStart
|
|
alignItems:ASStackLayoutAlignItemsStretch
|
|
children:@[
|
|
_photoLikesLabel,
|
|
_photoDescriptionLabel
|
|
]]
|
|
]
|
|
]];
|
|
}
|
|
}
|
|
#endif
|
|
|
|
#pragma mark - Instance Methods
|
|
|
|
- (void)didEnterPreloadState
|
|
{
|
|
[super didEnterPreloadState];
|
|
}
|
|
|
|
#pragma mark - Network Image Delegate
|
|
|
|
- (void)imageNode:(ASNetworkImageNode *)imageNode didLoadImage:(UIImage *)image info:(ASNetworkImageLoadInfo *)info
|
|
{
|
|
// Docs say method is called from bg but right now it's called from main.
|
|
// Save main thread time by shunting this.
|
|
if (info.sourceType == ASNetworkImageSourceDownload) {
|
|
ASPerformBlockOnBackgroundThread(^{
|
|
NSLog(@"Received image %@ from %@ with userInfo %@", image, info.url.path, ASObjectDescriptionMakeTiny(info.userInfo));
|
|
});
|
|
}
|
|
}
|
|
|
|
#pragma mark - Helper Methods
|
|
|
|
- (ASTextNode *)createLayerBackedTextNodeWithString:(NSAttributedString *)attributedString
|
|
{
|
|
ASTextNode *textNode = [[ASTextNode alloc] init];
|
|
textNode.layerBacked = YES;
|
|
textNode.attributedText = attributedString;
|
|
return textNode;
|
|
}
|
|
|
|
- (void)setupYogaLayoutIfNeeded
|
|
{
|
|
#if YOGA_LAYOUT
|
|
[self.style yogaNodeCreateIfNeeded];
|
|
[_userAvatarImageNode.style yogaNodeCreateIfNeeded];
|
|
[_userNameLabel.style yogaNodeCreateIfNeeded];
|
|
[_photoImageNode.style yogaNodeCreateIfNeeded];
|
|
[_photoLikesLabel.style yogaNodeCreateIfNeeded];
|
|
[_photoDescriptionLabel.style yogaNodeCreateIfNeeded];
|
|
[_photoLocationLabel.style yogaNodeCreateIfNeeded];
|
|
[_photoTimeIntervalSincePostLabel.style yogaNodeCreateIfNeeded];
|
|
|
|
ASDisplayNode *headerStack = [ASDisplayNode yogaHorizontalStack];
|
|
headerStack.style.margin = ASEdgeInsetsMake(InsetForHeader);
|
|
headerStack.style.alignItems = ASStackLayoutAlignItemsCenter;
|
|
headerStack.style.flexGrow = 1.0;
|
|
|
|
// Avatar Image, with inset - first thing in the header stack.
|
|
_userAvatarImageNode.style.preferredSize = CGSizeMake(USER_IMAGE_HEIGHT, USER_IMAGE_HEIGHT);
|
|
_userAvatarImageNode.style.margin = ASEdgeInsetsMake(InsetForAvatar);
|
|
[headerStack addYogaChild:_userAvatarImageNode];
|
|
|
|
// User Name and Photo Location stack is next
|
|
ASDisplayNode *userPhotoLocationStack = [ASDisplayNode yogaVerticalStack];
|
|
userPhotoLocationStack.style.flexShrink = 1.0;
|
|
[headerStack addYogaChild:userPhotoLocationStack];
|
|
|
|
// Setup the inside of the User Name and Photo Location stack.
|
|
_userNameLabel.style.flexShrink = 1.0;
|
|
[userPhotoLocationStack addYogaChild:_userNameLabel];
|
|
|
|
if (_photoLocationLabel.attributedText) {
|
|
_photoLocationLabel.style.flexShrink = 1.0;
|
|
[userPhotoLocationStack addYogaChild:_photoLocationLabel];
|
|
}
|
|
|
|
// Add a spacer to allow a flexible space between the User Name / Location stack, and the Timestamp.
|
|
[headerStack addYogaChild:[ASDisplayNode yogaSpacerNode]];
|
|
|
|
// Photo Timestamp Label.
|
|
_photoTimeIntervalSincePostLabel.style.spacingBefore = HORIZONTAL_BUFFER;
|
|
[headerStack addYogaChild:_photoTimeIntervalSincePostLabel];
|
|
|
|
// Create the last stack before assembling everything: the Footer Stack contains the description and comments.
|
|
ASDisplayNode *footerStack = [ASDisplayNode yogaVerticalStack];
|
|
footerStack.style.margin = ASEdgeInsetsMake(InsetForFooter);
|
|
footerStack.style.padding = ASEdgeInsetsMake(UIEdgeInsetsMake(0.0, 0.0, VERTICAL_BUFFER, 0.0));
|
|
footerStack.yogaChildren = @[_photoLikesLabel, _photoDescriptionLabel];
|
|
|
|
// Main Vertical Stack: contains header, large main photo with fixed aspect ratio, and footer.
|
|
_photoImageNode.style.aspectRatio = 1.0;
|
|
|
|
ASDisplayNode *verticalStack = self;
|
|
self.style.flexDirection = ASStackLayoutDirectionVertical;
|
|
|
|
[verticalStack addYogaChild:headerStack];
|
|
[verticalStack addYogaChild:_photoImageNode];
|
|
[verticalStack addYogaChild:footerStack];
|
|
#endif
|
|
}
|
|
|
|
@end
|