feat:fix ListView's auto resize height in iOS
This commit is contained in:
parent
1d5b312922
commit
db55cb30fb
@ -32,8 +32,8 @@ class ListPanel extends Panel {
|
|||||||
heightSpec: LayoutSpec.WRAP_CONTENT,
|
heightSpec: LayoutSpec.WRAP_CONTENT,
|
||||||
margin: {
|
margin: {
|
||||||
left: 10,
|
left: 10,
|
||||||
right: 10,
|
right: 50,
|
||||||
top: 10,
|
top: 50,
|
||||||
bottom: 10,
|
bottom: 10,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
@ -48,7 +48,7 @@ class ListPanel extends Panel {
|
|||||||
widthSpec: LayoutSpec.AT_MOST,
|
widthSpec: LayoutSpec.AT_MOST,
|
||||||
heightSpec: LayoutSpec.EXACTLY,
|
heightSpec: LayoutSpec.EXACTLY,
|
||||||
}
|
}
|
||||||
it.height = 50
|
it.height = 100
|
||||||
it.onClick = () => {
|
it.onClick = () => {
|
||||||
log(`Click item at ${idx}`)
|
log(`Click item at ${idx}`)
|
||||||
it.bgColor = Color.parse('#000000')
|
it.bgColor = Color.parse('#000000')
|
||||||
|
@ -69,6 +69,10 @@ typedef NS_ENUM(NSInteger, DoricGravity) {
|
|||||||
|
|
||||||
@interface DoricLayoutContainer : UIView
|
@interface DoricLayoutContainer : UIView
|
||||||
@property(nonatomic, assign) DoricGravity gravity;
|
@property(nonatomic, assign) DoricGravity gravity;
|
||||||
|
|
||||||
|
- (void)layout:(CGSize)targetSize;
|
||||||
|
|
||||||
|
- (CGSize)sizeContent:(CGSize)size;
|
||||||
@end
|
@end
|
||||||
|
|
||||||
@interface DoricStackView : DoricLayoutContainer
|
@interface DoricStackView : DoricLayoutContainer
|
||||||
|
@ -62,10 +62,6 @@ @interface DoricLayoutContainer ()
|
|||||||
@property(nonatomic, assign) CGFloat contentWidth;
|
@property(nonatomic, assign) CGFloat contentWidth;
|
||||||
@property(nonatomic, assign) CGFloat contentHeight;
|
@property(nonatomic, assign) CGFloat contentHeight;
|
||||||
@property(nonatomic, assign) NSUInteger contentWeight;
|
@property(nonatomic, assign) NSUInteger contentWeight;
|
||||||
|
|
||||||
- (void)layout:(CGSize)targetSize;
|
|
||||||
|
|
||||||
- (CGSize)sizeContent:(CGSize)size;
|
|
||||||
@end
|
@end
|
||||||
|
|
||||||
@implementation DoricLayoutContainer
|
@implementation DoricLayoutContainer
|
||||||
|
@ -21,6 +21,5 @@
|
|||||||
|
|
||||||
#import "DoricStackNode.h"
|
#import "DoricStackNode.h"
|
||||||
|
|
||||||
@interface DoricListItemNode : DoricGroupNode<UITableViewCell *>
|
@interface DoricListItemNode : DoricStackNode
|
||||||
|
|
||||||
@end
|
@end
|
@ -18,10 +18,27 @@
|
|||||||
//
|
//
|
||||||
|
|
||||||
#import "DoricListItemNode.h"
|
#import "DoricListItemNode.h"
|
||||||
|
#import "DoricExtensions.h"
|
||||||
|
|
||||||
|
@interface DoricListItemNode ()
|
||||||
|
@end
|
||||||
|
|
||||||
|
@interface DoricListItemView : DoricStackView
|
||||||
|
@end
|
||||||
|
|
||||||
|
@implementation DoricListItemView
|
||||||
|
@end
|
||||||
|
|
||||||
|
|
||||||
@implementation DoricListItemNode
|
@implementation DoricListItemNode
|
||||||
- (UITableViewCell *)build {
|
- (instancetype)initWithContext:(DoricContext *)doricContext {
|
||||||
return nil;
|
if (self = [super initWithContext:doricContext]) {
|
||||||
|
self.reusable = YES;
|
||||||
|
}
|
||||||
|
return self;
|
||||||
}
|
}
|
||||||
@end
|
|
||||||
|
- (DoricStackView *)build {
|
||||||
|
return [DoricListItemView new];
|
||||||
|
}
|
||||||
|
@end
|
||||||
|
@ -19,6 +19,6 @@
|
|||||||
|
|
||||||
#import <Foundation/Foundation.h>
|
#import <Foundation/Foundation.h>
|
||||||
#import "DoricSuperNode.h"
|
#import "DoricSuperNode.h"
|
||||||
@interface DoricListNode : DoricSuperNode<UITableView *>
|
|
||||||
|
|
||||||
|
@interface DoricListNode : DoricSuperNode<UITableView *>
|
||||||
@end
|
@end
|
@ -22,8 +22,16 @@
|
|||||||
#import "DoricExtensions.h"
|
#import "DoricExtensions.h"
|
||||||
#import "DoricListItemNode.h"
|
#import "DoricListItemNode.h"
|
||||||
|
|
||||||
|
@interface DoricTableViewCell : UITableViewCell
|
||||||
|
@property(nonatomic, strong) DoricListItemNode *doricListItemNode;
|
||||||
|
@end
|
||||||
|
|
||||||
|
@implementation DoricTableViewCell
|
||||||
|
@end
|
||||||
|
|
||||||
@interface DoricListNode () <UITableViewDataSource, UITableViewDelegate>
|
@interface DoricListNode () <UITableViewDataSource, UITableViewDelegate>
|
||||||
@property(nonatomic, strong) NSMutableDictionary <NSNumber *, DoricListItemNode *> *tempNodes;
|
@property(nonatomic, strong) NSMutableDictionary <NSNumber *, NSString *> *itemViewIds;
|
||||||
|
@property(nonatomic, strong) NSMutableDictionary <NSNumber *, NSNumber *> *itemHeights;
|
||||||
@property(nonatomic, assign) NSUInteger itemCount;
|
@property(nonatomic, assign) NSUInteger itemCount;
|
||||||
@property(nonatomic, assign) NSUInteger batchCount;
|
@property(nonatomic, assign) NSUInteger batchCount;
|
||||||
@end
|
@end
|
||||||
@ -31,7 +39,8 @@ @interface DoricListNode () <UITableViewDataSource, UITableViewDelegate>
|
|||||||
@implementation DoricListNode
|
@implementation DoricListNode
|
||||||
- (instancetype)initWithContext:(DoricContext *)doricContext {
|
- (instancetype)initWithContext:(DoricContext *)doricContext {
|
||||||
if (self = [super initWithContext:doricContext]) {
|
if (self = [super initWithContext:doricContext]) {
|
||||||
_tempNodes = [NSMutableDictionary new];
|
_itemViewIds = [NSMutableDictionary new];
|
||||||
|
_itemHeights = [NSMutableDictionary new];
|
||||||
_batchCount = 15;
|
_batchCount = 15;
|
||||||
}
|
}
|
||||||
return self;
|
return self;
|
||||||
@ -41,6 +50,7 @@ - (UITableView *)build {
|
|||||||
return [[UITableView new] also:^(UITableView *it) {
|
return [[UITableView new] also:^(UITableView *it) {
|
||||||
it.dataSource = self;
|
it.dataSource = self;
|
||||||
it.delegate = self;
|
it.delegate = self;
|
||||||
|
it.separatorStyle = UITableViewCellSeparatorStyleNone;
|
||||||
}];
|
}];
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -48,7 +58,7 @@ - (void)blendView:(UITableView *)view forPropName:(NSString *)name propValue:(id
|
|||||||
if ([@"itemCount" isEqualToString:name]) {
|
if ([@"itemCount" isEqualToString:name]) {
|
||||||
self.itemCount = [prop unsignedIntegerValue];
|
self.itemCount = [prop unsignedIntegerValue];
|
||||||
} else if ([@"renderItem" isEqualToString:name]) {
|
} else if ([@"renderItem" isEqualToString:name]) {
|
||||||
[self.tempNodes removeAllObjects];
|
[self.itemViewIds removeAllObjects];
|
||||||
[self clearSubModel];
|
[self clearSubModel];
|
||||||
} else if ([@"batchCount" isEqualToString:name]) {
|
} else if ([@"batchCount" isEqualToString:name]) {
|
||||||
self.batchCount = [prop unsignedIntegerValue];
|
self.batchCount = [prop unsignedIntegerValue];
|
||||||
@ -59,7 +69,6 @@ - (void)blendView:(UITableView *)view forPropName:(NSString *)name propValue:(id
|
|||||||
|
|
||||||
- (void)blend:(NSDictionary *)props {
|
- (void)blend:(NSDictionary *)props {
|
||||||
[super blend:props];
|
[super blend:props];
|
||||||
[self.view reloadData];
|
|
||||||
}
|
}
|
||||||
|
|
||||||
- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section {
|
- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section {
|
||||||
@ -72,25 +81,35 @@ - (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(N
|
|||||||
NSDictionary *props = model[@"props"];
|
NSDictionary *props = model[@"props"];
|
||||||
NSString *reuseId = props[@"identifier"];
|
NSString *reuseId = props[@"identifier"];
|
||||||
|
|
||||||
UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:reuseId];
|
DoricTableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:reuseId];
|
||||||
if (!cell) {
|
if (!cell) {
|
||||||
cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:reuseId];
|
cell = [[DoricTableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:reuseId];
|
||||||
|
DoricListItemNode *listItemNode = [[DoricListItemNode alloc] initWithContext:self.doricContext];
|
||||||
|
[listItemNode initWithSuperNode:self];
|
||||||
|
cell.doricListItemNode = listItemNode;
|
||||||
|
[cell.contentView addSubview:listItemNode.view];
|
||||||
}
|
}
|
||||||
DoricListItemNode *node = self.tempNodes[@(position)];
|
DoricListItemNode *node = cell.doricListItemNode;
|
||||||
node.view = cell;
|
node.viewId = model[@"id"];
|
||||||
[node blend:props];
|
[node blend:props];
|
||||||
|
CGSize size = [node.view sizeThatFits:CGSizeMake(cell.width, cell.height)];
|
||||||
|
[self callItem:position height:size.height];
|
||||||
return cell;
|
return cell;
|
||||||
}
|
}
|
||||||
|
|
||||||
- (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath {
|
- (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath {
|
||||||
NSUInteger position = (NSUInteger) indexPath.row;
|
NSUInteger position = (NSUInteger) indexPath.row;
|
||||||
NSDictionary *model = [self itemModelAt:position];
|
NSNumber *heightNumber = self.itemHeights[@(position)];
|
||||||
|
if (heightNumber) {
|
||||||
|
return [heightNumber floatValue];
|
||||||
|
} else {
|
||||||
|
return 44.f;
|
||||||
|
}
|
||||||
|
|
||||||
return 60;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
- (NSDictionary *)itemModelAt:(NSUInteger)position {
|
- (NSDictionary *)itemModelAt:(NSUInteger)position {
|
||||||
NSString *viewId = self.tempNodes[@(position)].viewId;
|
NSString *viewId = self.itemViewIds[@(position)];
|
||||||
if (viewId && viewId.length > 0) {
|
if (viewId && viewId.length > 0) {
|
||||||
return [self subModelOf:viewId];
|
return [self subModelOf:viewId];
|
||||||
} else {
|
} else {
|
||||||
@ -101,13 +120,30 @@ - (NSDictionary *)itemModelAt:(NSUInteger)position {
|
|||||||
NSString *thisViewId = obj[@"id"];
|
NSString *thisViewId = obj[@"id"];
|
||||||
[self setSubModel:obj in:thisViewId];
|
[self setSubModel:obj in:thisViewId];
|
||||||
NSUInteger pos = position + idx;
|
NSUInteger pos = position + idx;
|
||||||
DoricListItemNode *node = [[DoricListItemNode alloc] initWithContext:self.doricContext];
|
self.itemViewIds[@(pos)] = thisViewId;
|
||||||
node.viewId = thisViewId;
|
|
||||||
[node initWithSuperNode:self];
|
|
||||||
self.tempNodes[@(pos)] = node;
|
|
||||||
}];
|
}];
|
||||||
return array[0];
|
return array[0];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
- (void)blendSubNode:(NSDictionary *)subModel {
|
||||||
|
NSString *viewId = subModel[@"id"];
|
||||||
|
[self.itemViewIds enumerateKeysAndObjectsUsingBlock:^(NSNumber *_Nonnull key, NSString *_Nonnull obj, BOOL *_Nonnull stop) {
|
||||||
|
if ([viewId isEqualToString:obj]) {
|
||||||
|
*stop = YES;
|
||||||
|
NSIndexPath *indexPath = [NSIndexPath indexPathForRow:[key integerValue] inSection:0];
|
||||||
|
[self.view reloadRowsAtIndexPaths:@[indexPath] withRowAnimation:UITableViewRowAnimationNone];
|
||||||
|
}
|
||||||
|
}];
|
||||||
|
}
|
||||||
|
|
||||||
|
- (void)callItem:(NSUInteger)position height:(CGFloat)height {
|
||||||
|
NSNumber *old = self.itemHeights[@(position)];
|
||||||
|
if (old && old.floatValue == height) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
self.itemHeights[@(position)] = @(height);
|
||||||
|
NSIndexPath *indexPath = [NSIndexPath indexPathForRow:position inSection:0];
|
||||||
|
[self.view reloadRowsAtIndexPaths:@[indexPath] withRowAnimation:UITableViewRowAnimationNone];
|
||||||
|
}
|
||||||
@end
|
@end
|
||||||
|
Reference in New Issue
Block a user