185 lines
7.0 KiB
Objective-C
185 lines
7.0 KiB
Objective-C
/*
|
|
* 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.
|
|
*/
|
|
//
|
|
// DoricGroupNode.m
|
|
// Doric
|
|
//
|
|
// Created by pengfei.zhou on 2019/7/30.
|
|
//
|
|
|
|
#import "DoricGroupNode.h"
|
|
|
|
@implementation DoricGroupNode
|
|
|
|
- (instancetype)initWithContext:(DoricContext *)doricContext {
|
|
if (self = [super initWithContext:doricContext]) {
|
|
_childNodes = @[];
|
|
_childViewIds = @[];
|
|
}
|
|
return self;
|
|
}
|
|
|
|
- (UIView *)build {
|
|
UIView *ret = [[UIView alloc] init];
|
|
ret.clipsToBounds = YES;
|
|
return ret;
|
|
}
|
|
|
|
- (void)blendView:(UIView *)view forPropName:(NSString *)name propValue:(id)prop {
|
|
if ([@"children" isEqualToString:name]) {
|
|
self.childViewIds = prop;
|
|
} else {
|
|
[super blendView:view forPropName:name propValue:prop];
|
|
}
|
|
}
|
|
|
|
- (void)afterBlended:(NSDictionary *)props {
|
|
[self configChildNodes];
|
|
}
|
|
|
|
- (void)configChildNodes {
|
|
NSMutableArray *childNodes = [self.childNodes mutableCopy];
|
|
for (NSUInteger idx = 0; idx < self.childViewIds.count; idx++) {
|
|
NSString *viewId = self.childViewIds[idx];
|
|
NSDictionary *model = [self subModelOf:viewId];
|
|
NSString *type = model[@"type"];
|
|
if (idx < self.childNodes.count) {
|
|
DoricViewNode *oldNode = childNodes[idx];
|
|
if ([viewId isEqualToString:oldNode.viewId]) {
|
|
///Same,skip
|
|
} else {
|
|
if (self.reusable) {
|
|
if ([oldNode.type isEqualToString:type]) {
|
|
///Same type,can be reused
|
|
oldNode.viewId = viewId;
|
|
[oldNode blend:model[@"props"]];
|
|
} else {
|
|
///Replace this view
|
|
[childNodes removeObjectAtIndex:idx];
|
|
[oldNode.view removeFromSuperview];
|
|
DoricViewNode *viewNode = [DoricViewNode create:self.doricContext withType:type];
|
|
if ([viewNode isKindOfClass:[DoricGroupNode class]]) {
|
|
((DoricGroupNode *) viewNode).reusable = self.reusable;
|
|
}
|
|
viewNode.viewId = viewId;
|
|
[viewNode initWithSuperNode:self];
|
|
[viewNode blend:model[@"props"]];
|
|
if (idx >= childNodes.count) {
|
|
[childNodes addObject:viewNode];
|
|
} else {
|
|
[childNodes insertObject:viewNode atIndex:idx];
|
|
}
|
|
[self.view insertSubview:viewNode.view atIndex:idx];
|
|
}
|
|
} else {
|
|
///Find in remain nodes
|
|
NSInteger position = -1;
|
|
for (NSUInteger start = idx + 1; start < childNodes.count; start++) {
|
|
DoricViewNode *node = childNodes[start];
|
|
if ([viewId isEqualToString:node.viewId]) {
|
|
position = start;
|
|
break;
|
|
}
|
|
}
|
|
if (position >= 0) {
|
|
///Found ,swap idx,position
|
|
DoricViewNode *reused = childNodes[(NSUInteger) position];
|
|
[childNodes removeObjectAtIndex:(NSUInteger) position];
|
|
[childNodes removeObjectAtIndex:idx];
|
|
if (idx >= childNodes.count) {
|
|
[childNodes addObject:reused];
|
|
} else {
|
|
[childNodes insertObject:reused atIndex:idx];
|
|
}
|
|
if (position >= childNodes.count) {
|
|
[childNodes addObject:oldNode];
|
|
} else {
|
|
[childNodes insertObject:oldNode atIndex:(NSUInteger) position];
|
|
}
|
|
///View swap index
|
|
[reused.view removeFromSuperview];
|
|
[oldNode.view removeFromSuperview];
|
|
[self.view insertSubview:reused.view atIndex:idx];
|
|
[self.view insertSubview:oldNode.view atIndex:position];
|
|
} else {
|
|
///Not found,insert
|
|
DoricViewNode *viewNode = [DoricViewNode create:self.doricContext withType:type];
|
|
viewNode.viewId = viewId;
|
|
[viewNode initWithSuperNode:self];
|
|
[viewNode blend:model[@"props"]];
|
|
if (idx >= childNodes.count) {
|
|
[childNodes addObject:viewNode];
|
|
} else {
|
|
[childNodes insertObject:viewNode atIndex:idx];
|
|
}
|
|
[self.view insertSubview:viewNode.view atIndex:idx];
|
|
}
|
|
}
|
|
}
|
|
} else {
|
|
/// Insert
|
|
DoricViewNode *viewNode = [DoricViewNode create:self.doricContext withType:type];
|
|
if ([viewNode isKindOfClass:[DoricGroupNode class]]) {
|
|
((DoricGroupNode *) viewNode).reusable = self.reusable;
|
|
}
|
|
viewNode.viewId = viewId;
|
|
[viewNode initWithSuperNode:self];
|
|
[viewNode blend:model[@"props"]];
|
|
if (idx >= childNodes.count) {
|
|
[childNodes addObject:viewNode];
|
|
} else {
|
|
[childNodes insertObject:viewNode atIndex:idx];
|
|
}
|
|
[self.view insertSubview:viewNode.view atIndex:idx];
|
|
}
|
|
}
|
|
|
|
NSUInteger count = childNodes.count;
|
|
for (NSUInteger idx = self.childViewIds.count; idx < count; idx++) {
|
|
DoricViewNode *viewNode = childNodes.lastObject;
|
|
[childNodes removeLastObject];
|
|
[viewNode.view removeFromSuperview];
|
|
}
|
|
self.childNodes = [childNodes copy];
|
|
}
|
|
|
|
- (void)blendSubNode:(NSDictionary *)subModel {
|
|
NSString *viewId = subModel[@"id"];
|
|
[self.childNodes enumerateObjectsUsingBlock:^(DoricViewNode *obj, NSUInteger idx, BOOL *stop) {
|
|
if ([viewId isEqualToString:obj.viewId]) {
|
|
[obj blend:subModel[@"props"]];
|
|
*stop = YES;
|
|
}
|
|
}];
|
|
}
|
|
|
|
- (DoricViewNode *)subNodeWithViewId:(NSString *)viewId {
|
|
for (DoricViewNode *viewNode in self.childNodes) {
|
|
if ([viewId isEqualToString:viewNode.viewId]) {
|
|
return viewNode;
|
|
}
|
|
}
|
|
return nil;
|
|
}
|
|
|
|
- (void)requestLayout {
|
|
[super requestLayout];
|
|
for (DoricViewNode *node in self.childNodes) {
|
|
[node requestLayout];
|
|
}
|
|
}
|
|
@end
|