This repository has been archived on 2024-07-22. You can view files and clone it, but cannot push or open issues or pull requests.
Doric/doric-iOS/Devkit/Classes/DoricDevPerfVC.m

345 lines
13 KiB
Mathematica
Raw Permalink Normal View History

2021-07-21 15:05:41 +08:00
/*
* 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;
}
}
2021-07-21 15:05:41 +08:00
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;
2021-07-21 15:05:41 +08:00
@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];
}];
2021-07-21 15:05:41 +08:00
}
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;
2021-07-21 15:05:41 +08:00
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 stringByReplacingOccurrencesOfString:@"\n" withString:@"\t"];
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;
}
2021-07-21 15:05:41 +08:00
} else {
self.expandedView.height = 5 + 16;
2021-07-21 15:05:41 +08:00
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;
2021-07-21 15:05:41 +08:00
}
}
@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;
2021-07-21 15:05:41 +08:00
[self.headerView addSubview:it];
UITapGestureRecognizer *tapGestureRecognizer = [[UITapGestureRecognizer alloc]
initWithTarget:self
action:@selector(onRightButtonClicked)];
[it addGestureRecognizer:tapGestureRecognizer];
2021-07-21 15:05:41 +08:00
}];
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));
2021-07-21 15:05:41 +08:00
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));
2021-07-21 15:05:41 +08:00
[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];
}
2021-07-21 15:05:41 +08:00
@end