345 lines
13 KiB
Objective-C
345 lines
13 KiB
Objective-C
/*
|
|
* Copyright [2021] [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.
|
|
*/
|
|
//
|
|
// Created by pengfei.zhou on 2021/7/20.
|
|
//
|
|
|
|
#import "DoricDevPerfVC.h"
|
|
#import <DoricCore/Doric.h>
|
|
#import <DoricCore/DoricContextManager.h>
|
|
#import "DoricDevPerformanceAnchorHook.h"
|
|
|
|
@interface DoricDevAnchorItem : NSObject
|
|
@property(nonatomic, copy) NSString *name;
|
|
@property(nonatomic, assign) long position;
|
|
@property(nonatomic, assign) long prepared;
|
|
@property(nonatomic, assign) long worked;
|
|
@property(nonatomic, assign) BOOL expanded;
|
|
@end
|
|
|
|
@implementation DoricDevAnchorItem
|
|
- (CGFloat)cellHeight {
|
|
if (self.expanded) {
|
|
if ([self.name hasPrefix:@"Call"]) {
|
|
if ([self.name componentsSeparatedByString:@","].count > 1) {
|
|
return 40 + 5 + 16 + 5 + 16 + 5 + 16 + 10;
|
|
} else {
|
|
return 40 + 5 + 16 + 5 + 16 + 10;
|
|
}
|
|
} else {
|
|
return 40 + 5 + 16 + 10;
|
|
}
|
|
}
|
|
return 40;
|
|
}
|
|
@end
|
|
|
|
@interface DoricDevAnchorCell : UITableViewCell
|
|
@property(nonatomic, strong) UILabel *labelName;
|
|
@property(nonatomic, strong) UILabel *funcName;
|
|
@property(nonatomic, strong) UILabel *argumentName;
|
|
@property(nonatomic, strong) UILabel *costTime;
|
|
@property(nonatomic, strong) UIView *expandedView;
|
|
@property(nonatomic, strong) UIView *waterfallPrepared;
|
|
@property(nonatomic, strong) UIView *waterfallWorked;
|
|
@property(nonatomic, assign) long duration;
|
|
@end
|
|
|
|
@implementation DoricDevAnchorCell
|
|
- (instancetype)initWithStyle:(UITableViewCellStyle)style
|
|
reuseIdentifier:(NSString *)reuseIdentifier {
|
|
if (self = [super initWithStyle:style reuseIdentifier:reuseIdentifier]) {
|
|
self.selectionStyle = UITableViewCellSelectionStyleNone;
|
|
self.labelName = [[UILabel new] also:^(UILabel *it) {
|
|
it.font = [UIFont systemFontOfSize:16];
|
|
it.width = 80;
|
|
it.height = 25;
|
|
it.backgroundColor = DoricColor(@(0xff3498db));
|
|
it.textColor = [UIColor whiteColor];
|
|
it.text = @"Destroy";
|
|
it.textAlignment = NSTextAlignmentCenter;
|
|
[self.contentView addSubview:it];
|
|
}];
|
|
self.waterfallPrepared = [[UIView new] also:^(UIView *it) {
|
|
it.backgroundColor = DoricColor(@(0xff1abc9c));
|
|
it.height = 20;
|
|
[self.contentView addSubview:it];
|
|
|
|
}];
|
|
self.waterfallWorked = [[UIView new] also:^(UIView *it) {
|
|
it.backgroundColor = DoricColor(@(0xfff1c40f));
|
|
it.height = 20;
|
|
[self.contentView addSubview:it];
|
|
}];
|
|
self.expandedView = [[UIView new] also:^(UIView *it) {
|
|
it.backgroundColor = DoricColor(@(0x0f000000));
|
|
[self.contentView addSubview:it];
|
|
}];
|
|
self.funcName = [[UILabel new] also:^(UILabel *it) {
|
|
it.font = [UIFont systemFontOfSize:12];
|
|
it.textColor = [UIColor blackColor];
|
|
[self.expandedView addSubview:it];
|
|
}];
|
|
self.argumentName = [[UILabel new] also:^(UILabel *it) {
|
|
it.font = [UIFont systemFontOfSize:12];
|
|
it.textColor = [UIColor blackColor];
|
|
[self.expandedView addSubview:it];
|
|
}];
|
|
self.costTime = [[UILabel new] also:^(UILabel *it) {
|
|
it.font = [UIFont systemFontOfSize:12];
|
|
it.textColor = [UIColor blackColor];
|
|
[self.expandedView addSubview:it];
|
|
}];
|
|
}
|
|
return self;
|
|
}
|
|
|
|
- (void)refreshUI:(DoricDevAnchorItem *)anchorItem {
|
|
self.contentView.width = self.width;
|
|
self.contentView.height = [anchorItem cellHeight];
|
|
self.labelName.left = 15;
|
|
self.labelName.centerY = 20;
|
|
|
|
self.waterfallPrepared.centerY = 20;
|
|
self.waterfallWorked.centerY = 20;
|
|
CGFloat all = self.width - 15 - self.labelName.width - 15 - 15;
|
|
self.waterfallPrepared.left = self.labelName.right + 15 + ((CGFloat) anchorItem.position / (CGFloat) self.duration) * all;
|
|
self.waterfallPrepared.width = ((CGFloat) anchorItem.prepared / (CGFloat) self.duration) * all;
|
|
self.waterfallWorked.left = self.waterfallPrepared.right;
|
|
self.waterfallWorked.width = ((CGFloat) MAX(anchorItem.worked, 1) / (CGFloat) self.duration) * all;
|
|
|
|
self.expandedView.hidden = anchorItem.expanded ? NO : YES;
|
|
self.expandedView.width = self.width - 15;
|
|
self.expandedView.left = 15;
|
|
self.expandedView.top = 40;
|
|
self.costTime.text = [NSString stringWithFormat:@"%@ ms", @(anchorItem.prepared + anchorItem.worked)];
|
|
self.costTime.width = self.expandedView.width - 15;
|
|
self.costTime.left = 15;
|
|
self.costTime.height = 16;
|
|
|
|
if ([anchorItem.name hasPrefix:@"Call"]) {
|
|
self.labelName.text = @"Call";
|
|
self.labelName.backgroundColor = DoricColor(@(0xff3498db));
|
|
NSString *extraInfo = [anchorItem.name substringFromIndex:@"Call:".length];
|
|
NSRange range = [extraInfo rangeOfString:@","];
|
|
if (range.location != NSNotFound) {
|
|
NSUInteger location = range.location;
|
|
NSString *method = [extraInfo substringToIndex:location];
|
|
NSString *params = [extraInfo substringFromIndex:location + 1];
|
|
|
|
self.expandedView.height = 5 + 16 + 5 + 16 + 5 + 16;
|
|
[self.funcName also:^(UILabel *it) {
|
|
it.hidden = NO;
|
|
it.text = method;
|
|
it.width = self.expandedView.width - 15;
|
|
it.height = 16;
|
|
it.left = 15;
|
|
}];
|
|
|
|
[self.argumentName also:^(UILabel *it) {
|
|
it.hidden = NO;
|
|
it.text = params;
|
|
it.width = self.expandedView.width - 15;
|
|
it.height = 16;
|
|
it.left = 15;
|
|
}];
|
|
|
|
self.funcName.top = 5;
|
|
self.argumentName.top = self.funcName.bottom + 5;
|
|
self.costTime.top = self.argumentName.bottom + 5;
|
|
} else {
|
|
self.expandedView.height = 5 + 16 + 5 + 16;
|
|
self.argumentName.hidden = YES;
|
|
[self.funcName also:^(UILabel *it) {
|
|
it.hidden = NO;
|
|
it.text = extraInfo;
|
|
it.width = self.expandedView.width - 15;
|
|
it.height = 16;
|
|
it.left = 15;
|
|
}];
|
|
|
|
self.funcName.top = 5;
|
|
self.costTime.top = self.funcName.bottom + 5;
|
|
}
|
|
} else {
|
|
self.expandedView.height = 5 + 16;
|
|
if ([anchorItem.name isEqualToString:@"Render"]) {
|
|
self.labelName.backgroundColor = DoricColor(@(0xffe74c3c));
|
|
} else {
|
|
self.labelName.backgroundColor = DoricColor(@(0xff2ecc71));
|
|
}
|
|
self.labelName.text = anchorItem.name;
|
|
self.funcName.hidden = YES;
|
|
self.argumentName.hidden = YES;
|
|
self.costTime.top = 5;
|
|
}
|
|
}
|
|
@end
|
|
|
|
@interface DoricDevPerfVC () <UITableViewDelegate, UITableViewDataSource>
|
|
@property(nonatomic, weak) DoricContext *doricContext;
|
|
@property(nonatomic, strong) UILabel *rightButton;
|
|
@property(nonatomic, strong) UIView *headerView;
|
|
@property(nonatomic, strong) UITableView *tableView;
|
|
@property(nonatomic, strong) NSMutableArray<DoricDevAnchorItem *> *anchorItems;
|
|
@property(nonatomic, assign) long duration;
|
|
@end
|
|
|
|
@implementation DoricDevPerfVC
|
|
- (instancetype)initWithContextId:(NSString *)contextId {
|
|
if (self = [super init]) {
|
|
_doricContext = [DoricContextManager.instance getContext:contextId];
|
|
_anchorItems = [NSMutableArray new];
|
|
_duration = 1;
|
|
}
|
|
return self;
|
|
}
|
|
|
|
- (void)viewDidLoad {
|
|
[super viewDidLoad];
|
|
self.title = @"Performance";
|
|
self.view.backgroundColor = [UIColor whiteColor];
|
|
self.headerView = [[UIView new] also:^(UIView *it) {
|
|
it.width = self.view.width;
|
|
it.height = 50.f;
|
|
it.backgroundColor = DoricColor(@(0xff7f8c8d));
|
|
[self.view addSubview:it];
|
|
}];
|
|
UILabel *label = [[UILabel new] also:^(UILabel *it) {
|
|
it.textColor = [UIColor whiteColor];
|
|
it.font = [UIFont systemFontOfSize:16];
|
|
it.text = [NSString stringWithFormat:@"%@ <%@>",
|
|
self.doricContext.source,
|
|
self.doricContext.contextId];
|
|
[it sizeToFit];
|
|
[self.headerView addSubview:it];
|
|
}];
|
|
label.left = 15.f;
|
|
label.centerY = self.headerView.centerY;
|
|
self.rightButton = [[UILabel new] also:^(UILabel *it) {
|
|
it.textColor = [UIColor whiteColor];
|
|
it.font = [UIFont systemFontOfSize:16];
|
|
it.text = @"Expand[+]";
|
|
[it sizeToFit];
|
|
it.userInteractionEnabled = YES;
|
|
[self.headerView addSubview:it];
|
|
UITapGestureRecognizer *tapGestureRecognizer = [[UITapGestureRecognizer alloc]
|
|
initWithTarget:self
|
|
action:@selector(onRightButtonClicked)];
|
|
[it addGestureRecognizer:tapGestureRecognizer];
|
|
}];
|
|
self.rightButton.right = self.headerView.right - 15;
|
|
self.rightButton.centerY = self.headerView.centerY;
|
|
self.tableView = [[UITableView new] also:^(UITableView *it) {
|
|
it.width = self.view.width;
|
|
it.height = self.view.height - self.headerView.height;
|
|
it.delegate = self;
|
|
it.dataSource = self;
|
|
it.separatorStyle = UITableViewCellSeparatorStyleNone;
|
|
[self.view addSubview:it];
|
|
}];
|
|
self.tableView.top = self.headerView.bottom;
|
|
self.tableView.backgroundColor = DoricColor(@(0xffeeeeee));
|
|
if ([self.doricContext.driver.registry.globalPerformanceAnchorHook
|
|
isKindOfClass:DoricDevPerformanceAnchorHook.class]) {
|
|
NSArray <DoricDevAnchorNode *> *nodes = [((DoricDevPerformanceAnchorHook *) self.doricContext.driver.registry.globalPerformanceAnchorHook)
|
|
getAnchorNodeList:self.doricContext.contextId];
|
|
DoricDevAnchorNode *prevNode = nil;
|
|
long position = 0;
|
|
for (DoricDevAnchorNode *node in nodes) {
|
|
DoricDevAnchorItem *item = [DoricDevAnchorItem new];
|
|
item.name = node.name;
|
|
long gap = 0;
|
|
if (prevNode) {
|
|
gap = MIN(16, node.prepare - prevNode.end);
|
|
}
|
|
position += gap;
|
|
item.position = position;
|
|
item.prepared = node.start - node.prepare;
|
|
item.worked = node.end - node.start;
|
|
[self.anchorItems addObject:item];
|
|
position += (node.end - node.prepare);
|
|
prevNode = node;
|
|
}
|
|
self.duration = position;
|
|
}
|
|
[self.tableView reloadData];
|
|
}
|
|
|
|
- (void)viewWillAppear:(BOOL)animated {
|
|
[super viewWillAppear:animated];
|
|
self.tableView.height = self.view.height - self.headerView.height;
|
|
}
|
|
|
|
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
|
|
DoricDevAnchorCell *cell = [tableView dequeueReusableCellWithIdentifier:@"DoricDevAnchorCell"];
|
|
if (!cell) {
|
|
cell = [[DoricDevAnchorCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:@"DoricDevAnchorCell"];
|
|
}
|
|
cell.width = tableView.width;
|
|
cell.duration = self.duration;
|
|
NSUInteger position = (NSUInteger) indexPath.row;
|
|
cell.backgroundColor = position % 2 == 0 ? DoricColor(@(0x2ff1c40f)) : DoricColor(@(0x2f2ecc71));
|
|
[cell refreshUI:self.anchorItems[position]];
|
|
return cell;
|
|
}
|
|
|
|
- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section {
|
|
return self.anchorItems.count;
|
|
}
|
|
|
|
- (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath {
|
|
NSUInteger position = (NSUInteger) indexPath.row;
|
|
return [self.anchorItems[position] cellHeight];
|
|
}
|
|
|
|
- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath {
|
|
NSUInteger position = (NSUInteger) indexPath.row;
|
|
self.anchorItems[position].expanded = !self.anchorItems[position].expanded;
|
|
[tableView reloadData];
|
|
[self updateButton];
|
|
}
|
|
|
|
- (BOOL)allExpanded {
|
|
for (DoricDevAnchorItem *item in self.anchorItems) {
|
|
if (!item.expanded) {
|
|
return NO;
|
|
}
|
|
}
|
|
return YES;
|
|
}
|
|
|
|
- (void)updateButton {
|
|
self.rightButton.text = self.allExpanded ? @"Collapse[-]" : @"Expand[+]";
|
|
[self.rightButton sizeToFit];
|
|
self.rightButton.right = self.view.width - 15;
|
|
}
|
|
|
|
- (void)onRightButtonClicked {
|
|
if (self.allExpanded) {
|
|
[self.anchorItems forEach:^(DoricDevAnchorItem *obj) {
|
|
obj.expanded = NO;
|
|
}];
|
|
} else {
|
|
[self.anchorItems forEach:^(DoricDevAnchorItem *obj) {
|
|
obj.expanded = YES;
|
|
}];
|
|
}
|
|
[self updateButton];
|
|
[self.tableView reloadData];
|
|
}
|
|
@end
|