added files via upload

This commit is contained in:
Balackburn
2023-06-27 09:54:41 +02:00
commit 2ff6aac218
1420 changed files with 88898 additions and 0 deletions

View File

@@ -0,0 +1,19 @@
//
// FLEXHierarchyTableViewCell.h
// Flipboard
//
// Created by Ryan Olson on 2014-05-02.
// Copyright (c) 2020 FLEX Team. All rights reserved.
//
#import <UIKit/UIKit.h>
@interface FLEXHierarchyTableViewCell : UITableViewCell
- (id)initWithReuseIdentifier:(NSString *)reuseIdentifier;
@property (nonatomic) NSInteger viewDepth;
@property (nonatomic) UIColor *randomColorTag;
@property (nonatomic) UIColor *indicatedViewColor;
@end

View File

@@ -0,0 +1,169 @@
//
// FLEXHierarchyTableViewCell.m
// Flipboard
//
// Created by Ryan Olson on 2014-05-02.
// Copyright (c) 2020 FLEX Team. All rights reserved.
//
#import "FLEXHierarchyTableViewCell.h"
#import "FLEXUtility.h"
#import "FLEXResources.h"
#import "FLEXColor.h"
@interface FLEXHierarchyTableViewCell ()
/// Indicates how deep the view is in the hierarchy
@property (nonatomic) UIView *depthIndicatorView;
/// Holds the color that visually distinguishes views from one another
@property (nonatomic) UIImageView *colorCircleImageView;
/// A checker-patterned view, used to help show the color of a view, like a photoshop canvas
@property (nonatomic) UIView *backgroundColorCheckerPatternView;
/// The subview of the checker pattern view which holds the actual color of the view
@property (nonatomic) UIView *viewBackgroundColorView;
@end
@implementation FLEXHierarchyTableViewCell
- (id)initWithReuseIdentifier:(NSString *)reuseIdentifier {
return [self initWithStyle:UITableViewCellStyleSubtitle reuseIdentifier:reuseIdentifier];
}
- (id)initWithStyle:(UITableViewCellStyle)style reuseIdentifier:(NSString *)reuseIdentifier {
self = [super initWithStyle:style reuseIdentifier:reuseIdentifier];
if (self) {
self.depthIndicatorView = [UIView new];
self.depthIndicatorView.backgroundColor = FLEXUtility.hierarchyIndentPatternColor;
[self.contentView addSubview:self.depthIndicatorView];
UIImage *defaultCircleImage = [FLEXUtility circularImageWithColor:UIColor.blackColor radius:5];
self.colorCircleImageView = [[UIImageView alloc] initWithImage:defaultCircleImage];
[self.contentView addSubview:self.colorCircleImageView];
self.textLabel.font = UIFont.flex_defaultTableCellFont;
self.detailTextLabel.font = UIFont.flex_defaultTableCellFont;
self.accessoryType = UITableViewCellAccessoryDetailButton;
// Use a pattern-based color to simplify application of the checker pattern
static UIColor *checkerPatternColor = nil;
static dispatch_once_t once;
dispatch_once(&once, ^{
checkerPatternColor = [UIColor colorWithPatternImage:FLEXResources.checkerPattern];
});
self.backgroundColorCheckerPatternView = [UIView new];
self.backgroundColorCheckerPatternView.clipsToBounds = YES;
self.backgroundColorCheckerPatternView.layer.borderColor = FLEXColor.tertiaryBackgroundColor.CGColor;
self.backgroundColorCheckerPatternView.layer.borderWidth = 2.f / UIScreen.mainScreen.scale;
self.backgroundColorCheckerPatternView.backgroundColor = checkerPatternColor;
[self.contentView addSubview:self.backgroundColorCheckerPatternView];
self.viewBackgroundColorView = [UIView new];
[self.backgroundColorCheckerPatternView addSubview:self.viewBackgroundColorView];
}
return self;
}
- (void)setHighlighted:(BOOL)highlighted animated:(BOOL)animated {
UIColor *originalColour = self.viewBackgroundColorView.backgroundColor;
[super setHighlighted:highlighted animated:animated];
// UITableViewCell changes all subviews in the contentView to backgroundColor = clearColor.
// We want to preserve the hierarchy background color when highlighted.
self.depthIndicatorView.backgroundColor = FLEXUtility.hierarchyIndentPatternColor;
self.viewBackgroundColorView.backgroundColor = originalColour;
}
- (void)setSelected:(BOOL)selected animated:(BOOL)animated {
UIColor *originalColour = self.viewBackgroundColorView.backgroundColor;
[super setSelected:selected animated:animated];
// See setHighlighted above.
self.depthIndicatorView.backgroundColor = FLEXUtility.hierarchyIndentPatternColor;
self.viewBackgroundColorView.backgroundColor = originalColour;
}
- (void)layoutSubviews {
[super layoutSubviews];
const CGFloat kContentPadding = 6;
const CGFloat kDepthIndicatorWidthMultiplier = 4;
const CGFloat kViewColorIndicatorSize = 22;
const CGRect bounds = self.contentView.bounds;
const CGFloat centerY = CGRectGetMidY(bounds);
const CGFloat textLabelCenterY = CGRectGetMidY(self.textLabel.frame);
BOOL hideCheckerView = self.backgroundColorCheckerPatternView.hidden;
CGFloat maxWidth = CGRectGetMaxX(bounds);
maxWidth -= (hideCheckerView ? kContentPadding : (kViewColorIndicatorSize + kContentPadding * 2));
CGRect depthIndicatorFrame = self.depthIndicatorView.frame = CGRectMake(
kContentPadding, 0, self.viewDepth * kDepthIndicatorWidthMultiplier, CGRectGetHeight(bounds)
);
// Circle goes after depth, and its center Y = textLabel's center Y
CGRect circleFrame = self.colorCircleImageView.frame;
circleFrame.origin.x = CGRectGetMaxX(depthIndicatorFrame) + kContentPadding;
circleFrame.origin.y = FLEXFloor(textLabelCenterY - CGRectGetHeight(circleFrame) / 2.f);
self.colorCircleImageView.frame = circleFrame;
// Text label goes after random color circle, width extends to the edge
// of the contentView or to the padding before the color indicator view
CGRect textLabelFrame = self.textLabel.frame;
CGFloat textOriginX = CGRectGetMaxX(circleFrame) + kContentPadding;
textLabelFrame.origin.x = textOriginX;
textLabelFrame.size.width = maxWidth - textOriginX;
self.textLabel.frame = textLabelFrame;
// detailTextLabel leading edge lines up with the circle, and the
// width extends to the same max X as the same max X as the textLabel
CGRect detailTextLabelFrame = self.detailTextLabel.frame;
CGFloat detailOriginX = circleFrame.origin.x;
detailTextLabelFrame.origin.x = detailOriginX;
detailTextLabelFrame.size.width = maxWidth - detailOriginX;
self.detailTextLabel.frame = detailTextLabelFrame;
// Checker pattern view starts after the padding after the max X of textLabel,
// and is centered vertically within the entire contentView
self.backgroundColorCheckerPatternView.frame = CGRectMake(
CGRectGetMaxX(self.textLabel.frame) + kContentPadding,
centerY - kViewColorIndicatorSize / 2.f,
kViewColorIndicatorSize,
kViewColorIndicatorSize
);
// Background color view fills it's superview
self.viewBackgroundColorView.frame = self.backgroundColorCheckerPatternView.bounds;
self.backgroundColorCheckerPatternView.layer.cornerRadius = kViewColorIndicatorSize / 2.f;
}
- (void)setRandomColorTag:(UIColor *)randomColorTag {
if (![_randomColorTag isEqual:randomColorTag]) {
_randomColorTag = randomColorTag;
self.colorCircleImageView.image = [FLEXUtility circularImageWithColor:randomColorTag radius:6];
}
}
- (void)setViewDepth:(NSInteger)viewDepth {
if (_viewDepth != viewDepth) {
_viewDepth = viewDepth;
[self setNeedsLayout];
}
}
- (UIColor *)indicatedViewColor {
return self.viewBackgroundColorView.backgroundColor;
}
- (void)setIndicatedViewColor:(UIColor *)color {
self.viewBackgroundColorView.backgroundColor = color;
// Hide the checker pattern view if there is no background color
self.backgroundColorCheckerPatternView.hidden = color == nil;
[self setNeedsLayout];
}
@end

View File

@@ -0,0 +1,20 @@
//
// FLEXHierarchyTableViewController.h
// Flipboard
//
// Created by Ryan Olson on 2014-05-01.
// Copyright (c) 2020 FLEX Team. All rights reserved.
//
#import "FLEXTableViewController.h"
@interface FLEXHierarchyTableViewController : FLEXTableViewController
+ (instancetype)windows:(NSArray<UIWindow *> *)allWindows
viewsAtTap:(NSArray<UIView *> *)viewsAtTap
selectedView:(UIView *)selectedView;
@property (nonatomic) UIView *selectedView;
@property (nonatomic) void(^didSelectRowAction)(UIView *selectedView);
@end

View File

@@ -0,0 +1,253 @@
//
// FLEXHierarchyTableViewController.m
// Flipboard
//
// Created by Ryan Olson on 2014-05-01.
// Copyright (c) 2020 FLEX Team. All rights reserved.
//
#import "FLEXColor.h"
#import "FLEXHierarchyTableViewController.h"
#import "NSMapTable+FLEX_Subscripting.h"
#import "FLEXUtility.h"
#import "FLEXHierarchyTableViewCell.h"
#import "FLEXObjectExplorerViewController.h"
#import "FLEXObjectExplorerFactory.h"
#import "FLEXResources.h"
#import "FLEXWindow.h"
typedef NS_ENUM(NSUInteger, FLEXHierarchyScope) {
FLEXHierarchyScopeFullHierarchy,
FLEXHierarchyScopeViewsAtTap
};
@interface FLEXHierarchyTableViewController ()
@property (nonatomic) NSArray<UIView *> *allViews;
@property (nonatomic) NSMapTable<UIView *, NSNumber *> *depthsForViews;
@property (nonatomic) NSArray<UIView *> *viewsAtTap;
@property (nonatomic) NSArray<UIView *> *displayedViews;
@property (nonatomic, readonly) BOOL showScopeBar;
@end
@implementation FLEXHierarchyTableViewController
+ (instancetype)windows:(NSArray<UIWindow *> *)allWindows
viewsAtTap:(NSArray<UIView *> *)viewsAtTap
selectedView:(UIView *)selected {
NSParameterAssert(allWindows.count);
NSArray *allViews = [self allViewsInHierarchy:allWindows];
NSMapTable *depths = [self hierarchyDepthsForViews:allViews];
return [[self alloc] initWithViews:allViews viewsAtTap:viewsAtTap selectedView:selected depths:depths];
}
- (instancetype)initWithViews:(NSArray<UIView *> *)allViews
viewsAtTap:(NSArray<UIView *> *)viewsAtTap
selectedView:(UIView *)selectedView
depths:(NSMapTable<UIView *, NSNumber *> *)depthsForViews {
NSParameterAssert(allViews);
NSParameterAssert(depthsForViews.count == allViews.count);
self = [super initWithStyle:UITableViewStylePlain];
if (self) {
self.allViews = allViews;
self.depthsForViews = depthsForViews;
self.viewsAtTap = viewsAtTap;
self.selectedView = selectedView;
self.title = @"View Hierarchy Tree";
}
return self;
}
- (void)viewDidLoad {
[super viewDidLoad];
// Preserve selection between presentations
self.clearsSelectionOnViewWillAppear = NO;
// A little more breathing room
self.tableView.rowHeight = 50.0;
self.tableView.separatorStyle = UITableViewCellSeparatorStyleNone;
// Separator inset clashes with persistent cell selection
[self.tableView setSeparatorInset:UIEdgeInsetsZero];
self.showsSearchBar = YES;
self.showSearchBarInitially = YES;
// Using pinSearchBar on this screen causes a weird visual
// thing on the next view controller that gets pushed.
//
// self.pinSearchBar = YES;
self.searchBarDebounceInterval = kFLEXDebounceInstant;
self.automaticallyShowsSearchBarCancelButton = NO;
if (self.showScopeBar) {
self.searchController.searchBar.showsScopeBar = YES;
self.searchController.searchBar.scopeButtonTitles = @[@"Full Hierarchy", @"Views at Tap"];
self.selectedScope = FLEXHierarchyScopeViewsAtTap;
}
[self updateDisplayedViews];
}
- (void)viewWillAppear:(BOOL)animated {
[super viewWillAppear:animated];
[self disableToolbar];
}
- (void)viewDidAppear:(BOOL)animated {
[super viewDidAppear:animated];
[self trySelectCellForSelectedView];
}
#pragma mark - Hierarchy helpers
+ (NSArray<UIView *> *)allViewsInHierarchy:(NSArray<UIWindow *> *)windows {
return [windows flex_flatmapped:^id(UIWindow *window, NSUInteger idx) {
if (![window isKindOfClass:[FLEXWindow class]]) {
return [self viewWithRecursiveSubviews:window];
}
return nil;
}];
}
+ (NSArray<UIView *> *)viewWithRecursiveSubviews:(UIView *)view {
NSMutableArray<UIView *> *subviews = [NSMutableArray arrayWithObject:view];
for (UIView *subview in view.subviews) {
[subviews addObjectsFromArray:[self viewWithRecursiveSubviews:subview]];
}
return subviews;
}
+ (NSMapTable<UIView *, NSNumber *> *)hierarchyDepthsForViews:(NSArray<UIView *> *)views {
NSMapTable<UIView *, NSNumber *> *depths = [NSMapTable strongToStrongObjectsMapTable];
for (UIView *view in views) {
NSInteger depth = 0;
UIView *tryView = view;
while (tryView.superview) {
tryView = tryView.superview;
depth++;
}
depths[(id)view] = @(depth);
}
return depths;
}
#pragma mark Selection and Filtering Helpers
- (void)trySelectCellForSelectedView {
NSUInteger selectedViewIndex = [self.displayedViews indexOfObject:self.selectedView];
if (selectedViewIndex != NSNotFound) {
UITableViewScrollPosition scrollPosition = UITableViewScrollPositionMiddle;
NSIndexPath *selectedViewIndexPath = [NSIndexPath indexPathForRow:selectedViewIndex inSection:0];
[self.tableView selectRowAtIndexPath:selectedViewIndexPath animated:YES scrollPosition:scrollPosition];
}
}
- (void)updateDisplayedViews {
NSArray<UIView *> *candidateViews = nil;
if (self.showScopeBar) {
if (self.selectedScope == FLEXHierarchyScopeViewsAtTap) {
candidateViews = self.viewsAtTap;
} else if (self.selectedScope == FLEXHierarchyScopeFullHierarchy) {
candidateViews = self.allViews;
}
} else {
candidateViews = self.allViews;
}
if (self.searchText.length) {
self.displayedViews = [candidateViews filteredArrayUsingPredicate:[NSPredicate predicateWithBlock:^BOOL(UIView *candidateView, NSDictionary<NSString *, id> *bindings) {
NSString *title = [FLEXUtility descriptionForView:candidateView includingFrame:NO];
NSString *candidateViewPointerAddress = [NSString stringWithFormat:@"%p", candidateView];
BOOL matchedViewPointerAddress = [candidateViewPointerAddress rangeOfString:self.searchText options:NSCaseInsensitiveSearch].location != NSNotFound;
BOOL matchedViewTitle = [title rangeOfString:self.searchText options:NSCaseInsensitiveSearch].location != NSNotFound;
return matchedViewPointerAddress || matchedViewTitle;
}]];
} else {
self.displayedViews = candidateViews;
}
[self.tableView reloadData];
}
- (void)setSelectedView:(UIView *)selectedView {
_selectedView = selectedView;
if (self.isViewLoaded) {
[self trySelectCellForSelectedView];
}
}
#pragma mark - Search Bar / Scope Bar
- (BOOL)showScopeBar {
return self.viewsAtTap.count > 0;
}
- (void)updateSearchResults:(NSString *)newText {
[self updateDisplayedViews];
// If the search bar text field is active, don't scroll on selection because we may want
// to continue typing. Otherwise, scroll so that the selected cell is visible.
if (!self.searchController.searchBar.isFirstResponder) {
[self trySelectCellForSelectedView];
}
}
#pragma mark - Table View Data Source
- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section {
return self.displayedViews.count;
}
- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
static NSString *CellIdentifier = @"Cell";
FLEXHierarchyTableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:CellIdentifier];
if (!cell) {
cell = [[FLEXHierarchyTableViewCell alloc] initWithReuseIdentifier:CellIdentifier];
}
UIView *view = self.displayedViews[indexPath.row];
cell.textLabel.text = [FLEXUtility descriptionForView:view includingFrame:NO];
cell.detailTextLabel.text = [FLEXUtility detailDescriptionForView:view];
cell.randomColorTag = [FLEXUtility consistentRandomColorForObject:view];
cell.viewDepth = self.depthsForViews[view].integerValue;
cell.indicatedViewColor = view.backgroundColor;
if (view.isHidden || view.alpha < 0.01) {
cell.textLabel.textColor = FLEXColor.deemphasizedTextColor;
cell.detailTextLabel.textColor = FLEXColor.deemphasizedTextColor;
} else {
cell.textLabel.textColor = FLEXColor.primaryTextColor;
cell.detailTextLabel.textColor = FLEXColor.primaryTextColor;
}
return cell;
}
- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath {
_selectedView = self.displayedViews[indexPath.row]; // Don't scroll, avoid setter
if (self.didSelectRowAction) {
self.didSelectRowAction(_selectedView);
}
}
- (void)tableView:(UITableView *)tableView accessoryButtonTappedForRowWithIndexPath:(NSIndexPath *)indexPath {
UIView *drillInView = self.displayedViews[indexPath.row];
FLEXObjectExplorerViewController *viewExplorer = [FLEXObjectExplorerFactory explorerViewControllerForObject:drillInView];
[self.navigationController pushViewController:viewExplorer animated:YES];
}
@end