2019-12-04 13:29:26 +08:00
|
|
|
/*
|
|
|
|
* Copyright [2019] [Doric.Pub]
|
|
|
|
*
|
|
|
|
* 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
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
//
|
|
|
|
// DoricImageNode.m
|
|
|
|
// Doric
|
|
|
|
//
|
|
|
|
// Created by pengfei.zhou on 2019/8/6.
|
|
|
|
//
|
|
|
|
|
|
|
|
#import "DoricImageNode.h"
|
|
|
|
#import "YYWebImage.h"
|
2020-04-03 17:58:47 +08:00
|
|
|
#import "DoricExtensions.h"
|
|
|
|
#import "DoricUtil.h"
|
2020-04-08 16:06:57 +08:00
|
|
|
#import "DoricSuperNode.h"
|
2019-12-04 13:29:26 +08:00
|
|
|
|
2020-03-25 22:49:47 +08:00
|
|
|
@interface DoricImageView : YYAnimatedImageView
|
|
|
|
@end
|
|
|
|
|
|
|
|
@implementation DoricImageView
|
|
|
|
@end
|
|
|
|
|
2019-12-04 13:29:26 +08:00
|
|
|
@interface DoricImageNode ()
|
|
|
|
@property(nonatomic, copy) NSString *loadCallbackId;
|
2020-03-05 11:01:10 +08:00
|
|
|
@property(nonatomic, assign) UIViewContentMode contentMode;
|
|
|
|
@property(nonatomic, strong) NSNumber *placeHolderColor;
|
|
|
|
@property(nonatomic, strong) NSString *placeHolderImage;
|
|
|
|
@property(nonatomic, strong) NSNumber *errorColor;
|
|
|
|
@property(nonatomic, strong) NSString *errorImage;
|
2020-03-11 13:29:22 +08:00
|
|
|
@property(nonatomic, strong) UIVisualEffectView *blurEffectView;
|
2020-04-17 14:40:33 +08:00
|
|
|
@property(nonatomic, strong) NSDictionary *stretchInsetDic;
|
2019-12-04 13:29:26 +08:00
|
|
|
@end
|
|
|
|
|
|
|
|
@implementation DoricImageNode
|
|
|
|
|
|
|
|
- (UIImageView *)build {
|
2020-03-25 22:49:47 +08:00
|
|
|
return [[DoricImageView new] also:^(UIImageView *it) {
|
2019-12-04 13:29:26 +08:00
|
|
|
it.clipsToBounds = YES;
|
2020-03-24 11:19:22 +08:00
|
|
|
it.contentMode = UIViewContentModeScaleAspectFill;
|
2019-12-04 13:29:26 +08:00
|
|
|
}];
|
|
|
|
}
|
|
|
|
|
2019-12-18 13:03:20 +08:00
|
|
|
- (void)blend:(NSDictionary *)props {
|
2020-03-05 11:01:10 +08:00
|
|
|
[props[@"placeHolderColor"] also:^(id it) {
|
|
|
|
self.placeHolderColor = it;
|
|
|
|
}];
|
|
|
|
[props[@"placeHolderImage"] also:^(id it) {
|
|
|
|
self.placeHolderImage = it;
|
|
|
|
}];
|
|
|
|
[props[@"errorColor"] also:^(id it) {
|
|
|
|
self.errorColor = it;
|
|
|
|
}];
|
|
|
|
[props[@"errorImage"] also:^(id it) {
|
|
|
|
self.errorImage = it;
|
|
|
|
}];
|
2019-12-18 13:03:20 +08:00
|
|
|
[super blend:props];
|
|
|
|
}
|
|
|
|
|
2020-03-05 11:01:10 +08:00
|
|
|
- (UIImage *)currentPlaceHolderImage {
|
|
|
|
if (self.placeHolderImage) {
|
|
|
|
return [UIImage imageNamed:self.placeHolderImage];
|
|
|
|
}
|
|
|
|
if (self.placeHolderColor) {
|
|
|
|
UIColor *color = DoricColor(self.placeHolderColor);
|
|
|
|
CGRect rect = CGRectMake(0, 0, 1, 1);
|
|
|
|
self.view.contentMode = UIViewContentModeScaleToFill;
|
|
|
|
UIGraphicsBeginImageContextWithOptions(rect.size, NO, [UIScreen mainScreen].scale);
|
|
|
|
|
|
|
|
CGContextRef context = UIGraphicsGetCurrentContext();
|
|
|
|
CGContextSetFillColorWithColor(context, color.CGColor);
|
|
|
|
CGContextFillRect(context, rect);
|
|
|
|
|
|
|
|
UIImage *image = UIGraphicsGetImageFromCurrentImageContext();
|
|
|
|
UIGraphicsEndImageContext();
|
|
|
|
return image;
|
|
|
|
}
|
|
|
|
return self.doricContext.driver.registry.defaultPlaceHolderImage;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (UIImage *)currentErrorImage {
|
|
|
|
if (self.errorImage) {
|
|
|
|
return [UIImage imageNamed:self.errorImage];
|
|
|
|
}
|
|
|
|
if (self.errorColor) {
|
|
|
|
UIColor *color = DoricColor(self.errorColor);
|
|
|
|
CGRect rect = CGRectMake(0, 0, 1, 1);
|
|
|
|
self.view.contentMode = UIViewContentModeScaleToFill;
|
|
|
|
UIGraphicsBeginImageContextWithOptions(rect.size, NO, [UIScreen mainScreen].scale);
|
|
|
|
|
|
|
|
CGContextRef context = UIGraphicsGetCurrentContext();
|
|
|
|
CGContextSetFillColorWithColor(context, color.CGColor);
|
|
|
|
CGContextFillRect(context, rect);
|
|
|
|
|
|
|
|
UIImage *image = UIGraphicsGetImageFromCurrentImageContext();
|
|
|
|
UIGraphicsEndImageContext();
|
|
|
|
return image;
|
|
|
|
}
|
|
|
|
return self.doricContext.driver.registry.defaultErrorImage;
|
|
|
|
}
|
|
|
|
|
2019-12-04 13:29:26 +08:00
|
|
|
- (void)blendView:(UIImageView *)view forPropName:(NSString *)name propValue:(id)prop {
|
|
|
|
if ([@"imageUrl" isEqualToString:name]) {
|
|
|
|
__weak typeof(self) _self = self;
|
2020-04-09 13:30:31 +08:00
|
|
|
__block BOOL async = NO;
|
2020-03-05 11:01:10 +08:00
|
|
|
[view yy_setImageWithURL:[NSURL URLWithString:prop] placeholder:[self currentPlaceHolderImage] options:0 completion:^(UIImage *image, NSURL *url, YYWebImageFromType from, YYWebImageStage stage, NSError *error) {
|
2019-12-04 13:29:26 +08:00
|
|
|
__strong typeof(_self) self = _self;
|
2020-03-05 11:01:10 +08:00
|
|
|
if (self.placeHolderColor || self.errorColor) {
|
|
|
|
self.view.contentMode = self.contentMode;
|
|
|
|
}
|
2019-12-04 13:29:26 +08:00
|
|
|
if (error) {
|
2020-03-05 11:01:10 +08:00
|
|
|
[[self currentErrorImage] also:^(UIImage *it) {
|
2020-03-21 18:20:32 +08:00
|
|
|
self.view.image = it;
|
2020-03-05 11:01:10 +08:00
|
|
|
}];
|
2019-12-04 13:29:26 +08:00
|
|
|
if (self.loadCallbackId.length > 0) {
|
|
|
|
[self callJSResponse:self.loadCallbackId, nil];
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (self.loadCallbackId.length > 0) {
|
|
|
|
[self callJSResponse:self.loadCallbackId,
|
|
|
|
@{@"width": @(image.size.width), @"height": @(image.size.height)},
|
|
|
|
nil];
|
|
|
|
}
|
2020-04-08 16:06:57 +08:00
|
|
|
DoricSuperNode *node = self.superNode;
|
|
|
|
while (node.superNode != nil) {
|
|
|
|
node = node.superNode;
|
|
|
|
}
|
2020-04-09 11:21:41 +08:00
|
|
|
if (async) {
|
|
|
|
[node requestLayout];
|
|
|
|
}
|
2019-12-04 13:29:26 +08:00
|
|
|
}
|
|
|
|
}];
|
2020-04-09 11:21:41 +08:00
|
|
|
async = YES;
|
2019-12-04 13:29:26 +08:00
|
|
|
} else if ([@"scaleType" isEqualToString:name]) {
|
|
|
|
switch ([prop integerValue]) {
|
|
|
|
case 1: {
|
|
|
|
self.view.contentMode = UIViewContentModeScaleAspectFit;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case 2: {
|
|
|
|
self.view.contentMode = UIViewContentModeScaleAspectFill;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
default: {
|
|
|
|
self.view.contentMode = UIViewContentModeScaleToFill;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2020-03-05 11:01:10 +08:00
|
|
|
self.contentMode = self.view.contentMode;
|
2019-12-04 13:29:26 +08:00
|
|
|
} else if ([@"loadCallback" isEqualToString:name]) {
|
|
|
|
self.loadCallbackId = prop;
|
|
|
|
} else if ([@"imageBase64" isEqualToString:name]) {
|
|
|
|
NSString *base64 = prop;
|
|
|
|
if (YES == [base64 hasPrefix:@"data:image"]) {
|
|
|
|
base64 = [base64 componentsSeparatedByString:@","].lastObject;
|
|
|
|
}
|
|
|
|
NSData *imageData = [[NSData alloc] initWithBase64EncodedString:base64
|
|
|
|
options:NSDataBase64DecodingIgnoreUnknownCharacters];
|
2020-04-17 12:06:00 +08:00
|
|
|
YYImage *image = [YYImage imageWithData:imageData scale:UIScreen.mainScreen.scale];
|
2020-03-11 13:29:22 +08:00
|
|
|
view.image = image;
|
|
|
|
} else if ([@"isBlur" isEqualToString:name]) {
|
|
|
|
NSInteger value = [prop intValue];
|
|
|
|
if (value == 1) {
|
|
|
|
if (!self.blurEffectView) {
|
|
|
|
UIBlurEffect *blurEffect = [UIBlurEffect effectWithStyle:UIBlurEffectStyleLight];
|
|
|
|
self.blurEffectView = [[UIVisualEffectView alloc] initWithEffect:blurEffect];
|
|
|
|
[view addSubview:self.blurEffectView];
|
|
|
|
dispatch_async(dispatch_get_main_queue(), ^{
|
|
|
|
self.blurEffectView.translatesAutoresizingMaskIntoConstraints = NO;
|
|
|
|
NSLayoutConstraint *widthConstraint = [NSLayoutConstraint
|
|
|
|
constraintWithItem:self.blurEffectView
|
|
|
|
attribute:(NSLayoutAttributeWidth)
|
|
|
|
relatedBy:(NSLayoutRelationEqual)
|
|
|
|
toItem:view
|
|
|
|
attribute:(NSLayoutAttributeWidth)
|
|
|
|
multiplier:1
|
|
|
|
constant:0];
|
|
|
|
NSLayoutConstraint *heightConstraint = [NSLayoutConstraint
|
|
|
|
constraintWithItem:self.blurEffectView
|
|
|
|
attribute:(NSLayoutAttributeHeight)
|
|
|
|
relatedBy:(NSLayoutRelationEqual)
|
|
|
|
toItem:view
|
|
|
|
attribute:(NSLayoutAttributeHeight)
|
|
|
|
multiplier:1
|
|
|
|
constant:0];
|
|
|
|
[NSLayoutConstraint activateConstraints:@[widthConstraint, heightConstraint,]];
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
2020-03-12 19:56:47 +08:00
|
|
|
} else if ([@"imageRes" isEqualToString:name]) {
|
2020-03-27 16:02:58 +08:00
|
|
|
YYImage *image = [YYImage imageNamed:prop];
|
2020-03-12 19:56:47 +08:00
|
|
|
view.image = image;
|
2020-03-27 16:02:58 +08:00
|
|
|
|
2020-03-12 19:56:47 +08:00
|
|
|
if (self.loadCallbackId.length > 0) {
|
|
|
|
if (image) {
|
|
|
|
[self callJSResponse:self.loadCallbackId,
|
|
|
|
@{@"width": @(image.size.width), @"height": @(image.size.height)},
|
|
|
|
nil];
|
|
|
|
} else {
|
|
|
|
[self callJSResponse:self.loadCallbackId, nil];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
} else if ([@"imagePath" isEqualToString:name]) {
|
|
|
|
NSString *path = [[NSBundle mainBundle] bundlePath];
|
|
|
|
NSString *fullPath = [path stringByAppendingPathComponent:prop];
|
2020-03-27 16:02:58 +08:00
|
|
|
YYImage *image = [YYImage imageWithContentsOfFile:fullPath];
|
2020-03-12 19:56:47 +08:00
|
|
|
view.image = image;
|
|
|
|
if (self.loadCallbackId.length > 0) {
|
|
|
|
if (image) {
|
|
|
|
[self callJSResponse:self.loadCallbackId,
|
|
|
|
@{@"width": @(image.size.width), @"height": @(image.size.height)},
|
|
|
|
nil];
|
|
|
|
} else {
|
|
|
|
[self callJSResponse:self.loadCallbackId, nil];
|
|
|
|
}
|
|
|
|
}
|
2020-04-16 19:32:25 +08:00
|
|
|
} else if ([@"stretchInset" isEqualToString:name]) {
|
2020-04-17 14:40:33 +08:00
|
|
|
self.stretchInsetDic = (NSDictionary *)prop;
|
2019-12-04 13:29:26 +08:00
|
|
|
} else {
|
|
|
|
[super blendView:view forPropName:name propValue:prop];
|
|
|
|
}
|
|
|
|
}
|
2020-04-17 14:40:33 +08:00
|
|
|
|
|
|
|
- (void)afterBlended:(NSDictionary *)props {
|
|
|
|
if (self.stretchInsetDic != nil) {
|
|
|
|
CGFloat left = [self.stretchInsetDic[@"left"] floatValue];
|
|
|
|
CGFloat top = [self.stretchInsetDic[@"top"] floatValue];
|
|
|
|
CGFloat right = [self.stretchInsetDic[@"right"] floatValue];
|
|
|
|
CGFloat bottom = [self.stretchInsetDic[@"bottom"] floatValue];
|
2020-04-17 15:18:01 +08:00
|
|
|
UIImage *result = [self.view.image resizableImageWithCapInsets:UIEdgeInsetsMake(top * self.view.image.size.height, left * self.view.image.size.width, bottom * self.view.image.size.height, right * self.view.image.size.width) resizingMode:UIImageResizingModeStretch];
|
2020-04-17 14:40:33 +08:00
|
|
|
self.view.image = result;
|
|
|
|
}
|
|
|
|
}
|
2019-12-04 13:29:26 +08:00
|
|
|
@end
|