mirror of
https://github.com/SoPat712/YTLitePlus.git
synced 2025-08-24 11:28:52 -04:00
483 lines
15 KiB
Objective-C
483 lines
15 KiB
Objective-C
//
|
|
// FLEXShortcutsSection.m
|
|
// FLEX
|
|
//
|
|
// Created by Tanner Bennett on 8/29/19.
|
|
// Copyright © 2020 FLEX Team. All rights reserved.
|
|
//
|
|
|
|
#import "FLEXShortcutsSection.h"
|
|
#import "FLEXTableView.h"
|
|
#import "FLEXTableViewCell.h"
|
|
#import "FLEXUtility.h"
|
|
#import "FLEXShortcut.h"
|
|
#import "FLEXProperty.h"
|
|
#import "FLEXPropertyAttributes.h"
|
|
#import "FLEXIvar.h"
|
|
#import "FLEXMethod.h"
|
|
#import "FLEXRuntime+UIKitHelpers.h"
|
|
#import "FLEXObjectExplorer.h"
|
|
|
|
#pragma mark Private
|
|
|
|
@interface FLEXShortcutsSection ()
|
|
@property (nonatomic, copy) NSArray<NSString *> *titles;
|
|
@property (nonatomic, copy) NSArray<NSString *> *subtitles;
|
|
|
|
@property (nonatomic, copy) NSArray<NSString *> *allTitles;
|
|
@property (nonatomic, copy) NSArray<NSString *> *allSubtitles;
|
|
|
|
// Shortcuts are not used if initialized with static titles and subtitles
|
|
@property (nonatomic, copy) NSArray<id<FLEXShortcut>> *shortcuts;
|
|
@property (nonatomic, readonly) NSArray<id<FLEXShortcut>> *allShortcuts;
|
|
@end
|
|
|
|
@implementation FLEXShortcutsSection
|
|
@synthesize isNewSection = _isNewSection;
|
|
|
|
#pragma mark Initialization
|
|
|
|
+ (instancetype)forObject:(id)objectOrClass rowTitles:(NSArray<NSString *> *)titles {
|
|
return [self forObject:objectOrClass rowTitles:titles rowSubtitles:nil];
|
|
}
|
|
|
|
+ (instancetype)forObject:(id)objectOrClass
|
|
rowTitles:(NSArray<NSString *> *)titles
|
|
rowSubtitles:(NSArray<NSString *> *)subtitles {
|
|
return [[self alloc] initWithObject:objectOrClass titles:titles subtitles:subtitles];
|
|
}
|
|
|
|
+ (instancetype)forObject:(id)objectOrClass rows:(NSArray *)rows {
|
|
return [[self alloc] initWithObject:objectOrClass rows:rows isNewSection:YES];
|
|
}
|
|
|
|
+ (instancetype)forObject:(id)objectOrClass additionalRows:(NSArray *)toPrepend {
|
|
NSArray *rows = [FLEXShortcutsFactory shortcutsForObjectOrClass:objectOrClass];
|
|
NSArray *allRows = [toPrepend arrayByAddingObjectsFromArray:rows] ?: rows;
|
|
return [[self alloc] initWithObject:objectOrClass rows:allRows isNewSection:NO];
|
|
}
|
|
|
|
+ (instancetype)forObject:(id)objectOrClass {
|
|
return [self forObject:objectOrClass additionalRows:nil];
|
|
}
|
|
|
|
- (id)initWithObject:(id)object
|
|
titles:(NSArray<NSString *> *)titles
|
|
subtitles:(NSArray<NSString *> *)subtitles {
|
|
|
|
NSParameterAssert(titles.count == subtitles.count || !subtitles);
|
|
NSParameterAssert(titles.count);
|
|
|
|
self = [super init];
|
|
if (self) {
|
|
_object = object;
|
|
_allTitles = titles.copy;
|
|
_allSubtitles = subtitles.copy;
|
|
_isNewSection = YES;
|
|
_numberOfLines = 1;
|
|
}
|
|
|
|
return self;
|
|
}
|
|
|
|
- (id)initWithObject:object rows:(NSArray *)rows isNewSection:(BOOL)newSection {
|
|
self = [super init];
|
|
if (self) {
|
|
_object = object;
|
|
_isNewSection = newSection;
|
|
|
|
_allShortcuts = [rows flex_mapped:^id(id obj, NSUInteger idx) {
|
|
return [FLEXShortcut shortcutFor:obj];
|
|
}];
|
|
_numberOfLines = 1;
|
|
|
|
// Populate titles and subtitles
|
|
[self reloadData];
|
|
}
|
|
|
|
return self;
|
|
}
|
|
|
|
|
|
#pragma mark - Public
|
|
|
|
- (void)setCacheSubtitles:(BOOL)cacheSubtitles {
|
|
if (_cacheSubtitles == cacheSubtitles) return;
|
|
|
|
// cacheSubtitles only applies if we have shortcut objects
|
|
if (self.allShortcuts) {
|
|
_cacheSubtitles = cacheSubtitles;
|
|
[self reloadData];
|
|
} else {
|
|
NSLog(@"Warning: setting 'cacheSubtitles' on a shortcut section with static subtitles");
|
|
}
|
|
}
|
|
|
|
|
|
#pragma mark - Overrides
|
|
|
|
- (UITableViewCellAccessoryType)accessoryTypeForRow:(NSInteger)row {
|
|
if (_allShortcuts) {
|
|
return [self.shortcuts[row] accessoryTypeWith:self.object];
|
|
}
|
|
|
|
return UITableViewCellAccessoryNone;
|
|
}
|
|
|
|
- (void)setFilterText:(NSString *)filterText {
|
|
super.filterText = filterText;
|
|
|
|
NSAssert(
|
|
self.allTitles.count == self.allSubtitles.count,
|
|
@"Each title needs a (possibly empty) subtitle"
|
|
);
|
|
|
|
if (filterText.length) {
|
|
// Tally up indexes of titles and subtitles matching the filter
|
|
NSMutableIndexSet *filterMatches = [NSMutableIndexSet new];
|
|
id filterBlock = ^BOOL(NSString *obj, NSUInteger idx) {
|
|
if ([obj localizedCaseInsensitiveContainsString:filterText]) {
|
|
[filterMatches addIndex:idx];
|
|
return YES;
|
|
}
|
|
|
|
return NO;
|
|
};
|
|
|
|
// Get all matching indexes, including subtitles
|
|
[self.allTitles flex_forEach:filterBlock];
|
|
[self.allSubtitles flex_forEach:filterBlock];
|
|
// Filter to matching indexes only
|
|
self.titles = [self.allTitles objectsAtIndexes:filterMatches];
|
|
self.subtitles = [self.allSubtitles objectsAtIndexes:filterMatches];
|
|
self.shortcuts = [self.allShortcuts objectsAtIndexes:filterMatches];
|
|
} else {
|
|
self.shortcuts = self.allShortcuts;
|
|
self.titles = self.allTitles;
|
|
self.subtitles = [self.allSubtitles flex_filtered:^BOOL(NSString *sub, NSUInteger idx) {
|
|
return sub.length > 0;
|
|
}];
|
|
}
|
|
}
|
|
|
|
- (void)reloadData {
|
|
[FLEXObjectExplorer configureDefaultsForItems:self.allShortcuts];
|
|
|
|
// Generate all (sub)titles from shortcuts
|
|
if (self.allShortcuts) {
|
|
self.allTitles = [self.allShortcuts flex_mapped:^id(id<FLEXShortcut> s, NSUInteger idx) {
|
|
return [s titleWith:self.object];
|
|
}];
|
|
self.allSubtitles = [self.allShortcuts flex_mapped:^id(id<FLEXShortcut> s, NSUInteger idx) {
|
|
return [s subtitleWith:self.object] ?: @"";
|
|
}];
|
|
}
|
|
|
|
// Re-generate filtered (sub)titles and shortcuts
|
|
self.filterText = self.filterText;
|
|
}
|
|
|
|
- (NSString *)title {
|
|
return @"Shortcuts";
|
|
}
|
|
|
|
- (NSInteger)numberOfRows {
|
|
return self.titles.count;
|
|
}
|
|
|
|
- (BOOL)canSelectRow:(NSInteger)row {
|
|
UITableViewCellAccessoryType type = [self.shortcuts[row] accessoryTypeWith:self.object];
|
|
BOOL hasDisclosure = NO;
|
|
hasDisclosure |= type == UITableViewCellAccessoryDisclosureIndicator;
|
|
hasDisclosure |= type == UITableViewCellAccessoryDetailDisclosureButton;
|
|
return hasDisclosure;
|
|
}
|
|
|
|
- (void (^)(__kindof UIViewController *))didSelectRowAction:(NSInteger)row {
|
|
return [self.shortcuts[row] didSelectActionWith:self.object];
|
|
}
|
|
|
|
- (UIViewController *)viewControllerToPushForRow:(NSInteger)row {
|
|
/// Nil if shortcuts is nil, i.e. if initialized with forObject:rowTitles:rowSubtitles:
|
|
return [self.shortcuts[row] viewerWith:self.object];
|
|
}
|
|
|
|
- (void (^)(__kindof UIViewController *))didPressInfoButtonAction:(NSInteger)row {
|
|
id<FLEXShortcut> shortcut = self.shortcuts[row];
|
|
if ([shortcut respondsToSelector:@selector(editorWith:forSection:)]) {
|
|
id object = self.object;
|
|
return ^(UIViewController *host) {
|
|
UIViewController *editor = [shortcut editorWith:object forSection:self];
|
|
[host.navigationController pushViewController:editor animated:YES];
|
|
};
|
|
}
|
|
|
|
return nil;
|
|
}
|
|
|
|
- (NSString *)reuseIdentifierForRow:(NSInteger)row {
|
|
FLEXTableViewCellReuseIdentifier defaultReuse = kFLEXDetailCell;
|
|
if (@available(iOS 11, *)) {
|
|
defaultReuse = kFLEXMultilineDetailCell;
|
|
}
|
|
|
|
return [self.shortcuts[row] customReuseIdentifierWith:self.object] ?: defaultReuse;
|
|
}
|
|
|
|
- (void)configureCell:(__kindof FLEXTableViewCell *)cell forRow:(NSInteger)row {
|
|
cell.titleLabel.text = [self titleForRow:row];
|
|
cell.titleLabel.numberOfLines = self.numberOfLines;
|
|
cell.subtitleLabel.text = [self subtitleForRow:row];
|
|
cell.subtitleLabel.numberOfLines = self.numberOfLines;
|
|
cell.accessoryType = [self accessoryTypeForRow:row];
|
|
}
|
|
|
|
- (NSString *)titleForRow:(NSInteger)row {
|
|
return self.titles[row];
|
|
}
|
|
|
|
- (NSString *)subtitleForRow:(NSInteger)row {
|
|
// Case: dynamic, uncached subtitles
|
|
if (!self.cacheSubtitles) {
|
|
NSString *subtitle = [self.shortcuts[row] subtitleWith:self.object];
|
|
return subtitle.length ? subtitle : nil;
|
|
}
|
|
|
|
// Case: static subtitles, or cached subtitles
|
|
return self.subtitles[row];
|
|
}
|
|
|
|
@end
|
|
|
|
|
|
#pragma mark - Global shortcut registration
|
|
|
|
@interface FLEXShortcutsFactory () {
|
|
BOOL _append, _prepend, _replace, _notInstance;
|
|
NSArray<NSString *> *_properties, *_ivars, *_methods;
|
|
}
|
|
@end
|
|
|
|
#define NewAndSet(ivar) ({ FLEXShortcutsFactory *r = [self sharedFactory]; r->ivar = YES; r; })
|
|
#define SetIvar(ivar) ({ self->ivar = YES; self; })
|
|
#define SetParamBlock(ivar) ^(NSArray *p) { self->ivar = p; return self; }
|
|
|
|
typedef NSMutableDictionary<Class, NSMutableArray<id<FLEXRuntimeMetadata>> *> RegistrationBuckets;
|
|
|
|
@implementation FLEXShortcutsFactory {
|
|
// Class buckets
|
|
RegistrationBuckets *cProperties;
|
|
RegistrationBuckets *cIvars;
|
|
RegistrationBuckets *cMethods;
|
|
// Metaclass buckets
|
|
RegistrationBuckets *mProperties;
|
|
RegistrationBuckets *mMethods;
|
|
}
|
|
|
|
+ (instancetype)sharedFactory {
|
|
static FLEXShortcutsFactory *shared = nil;
|
|
static dispatch_once_t onceToken;
|
|
dispatch_once(&onceToken, ^{
|
|
shared = [self new];
|
|
});
|
|
|
|
return shared;
|
|
}
|
|
|
|
- (id)init {
|
|
self = [super init];
|
|
if (self) {
|
|
cProperties = [NSMutableDictionary new];
|
|
cIvars = [NSMutableDictionary new];
|
|
cMethods = [NSMutableDictionary new];
|
|
|
|
mProperties = [NSMutableDictionary new];
|
|
mMethods = [NSMutableDictionary new];
|
|
}
|
|
|
|
return self;
|
|
}
|
|
|
|
+ (NSArray<id<FLEXRuntimeMetadata>> *)shortcutsForObjectOrClass:(id)objectOrClass {
|
|
return [[self sharedFactory] shortcutsForObjectOrClass:objectOrClass];
|
|
}
|
|
|
|
- (NSArray<id<FLEXRuntimeMetadata>> *)shortcutsForObjectOrClass:(id)objectOrClass {
|
|
NSParameterAssert(objectOrClass);
|
|
|
|
NSMutableArray<id<FLEXRuntimeMetadata>> *shortcuts = [NSMutableArray new];
|
|
BOOL isClass = object_isClass(objectOrClass);
|
|
// The -class does not give you a metaclass, and we want a metaclass
|
|
// if a class is passed in, or a class if an object is passed in
|
|
Class classKey = object_getClass(objectOrClass);
|
|
|
|
RegistrationBuckets *propertyBucket = isClass ? mProperties : cProperties;
|
|
RegistrationBuckets *methodBucket = isClass ? mMethods : cMethods;
|
|
RegistrationBuckets *ivarBucket = isClass ? nil : cIvars;
|
|
|
|
BOOL stop = NO;
|
|
while (!stop && classKey) {
|
|
NSArray *properties = propertyBucket[classKey];
|
|
NSArray *ivars = ivarBucket[classKey];
|
|
NSArray *methods = methodBucket[classKey];
|
|
|
|
// Stop if we found anything
|
|
stop = properties || ivars || methods;
|
|
if (stop) {
|
|
// Add things we found to the list
|
|
[shortcuts addObjectsFromArray:properties];
|
|
[shortcuts addObjectsFromArray:ivars];
|
|
[shortcuts addObjectsFromArray:methods];
|
|
} else {
|
|
classKey = class_getSuperclass(classKey);
|
|
}
|
|
}
|
|
|
|
[FLEXObjectExplorer configureDefaultsForItems:shortcuts];
|
|
return shortcuts;
|
|
}
|
|
|
|
+ (FLEXShortcutsFactory *)append {
|
|
return NewAndSet(_append);
|
|
}
|
|
|
|
+ (FLEXShortcutsFactory *)prepend {
|
|
return NewAndSet(_prepend);
|
|
}
|
|
|
|
+ (FLEXShortcutsFactory *)replace {
|
|
return NewAndSet(_replace);
|
|
}
|
|
|
|
- (void)_register:(NSArray<id<FLEXRuntimeMetadata>> *)items to:(RegistrationBuckets *)global class:(Class)key {
|
|
@synchronized (self) {
|
|
// Get (or initialize) the bucket for this class
|
|
NSMutableArray *bucket = ({
|
|
id bucket = global[key];
|
|
if (!bucket) {
|
|
bucket = [NSMutableArray new];
|
|
global[(id)key] = bucket;
|
|
}
|
|
bucket;
|
|
});
|
|
|
|
if (self->_append) { [bucket addObjectsFromArray:items]; }
|
|
if (self->_replace) { [bucket setArray:items]; }
|
|
if (self->_prepend) {
|
|
if (bucket.count) {
|
|
// Set new items as array, add old items behind them
|
|
id copy = bucket.copy;
|
|
[bucket setArray:items];
|
|
[bucket addObjectsFromArray:copy];
|
|
} else {
|
|
[bucket addObjectsFromArray:items];
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
- (void)reset {
|
|
_append = NO;
|
|
_prepend = NO;
|
|
_replace = NO;
|
|
_notInstance = NO;
|
|
|
|
_properties = nil;
|
|
_ivars = nil;
|
|
_methods = nil;
|
|
}
|
|
|
|
- (FLEXShortcutsFactory *)class {
|
|
return SetIvar(_notInstance);
|
|
}
|
|
|
|
- (FLEXShortcutsFactoryNames)properties {
|
|
NSAssert(!_notInstance, @"Do not try to set properties+classProperties at the same time");
|
|
return SetParamBlock(_properties);
|
|
}
|
|
|
|
- (FLEXShortcutsFactoryNames)classProperties {
|
|
_notInstance = YES;
|
|
return SetParamBlock(_properties);
|
|
}
|
|
|
|
- (FLEXShortcutsFactoryNames)ivars {
|
|
return SetParamBlock(_ivars);
|
|
}
|
|
|
|
- (FLEXShortcutsFactoryNames)methods {
|
|
NSAssert(!_notInstance, @"Do not try to set methods+classMethods at the same time");
|
|
return SetParamBlock(_methods);
|
|
}
|
|
|
|
- (FLEXShortcutsFactoryNames)classMethods {
|
|
_notInstance = YES;
|
|
return SetParamBlock(_methods);
|
|
}
|
|
|
|
- (FLEXShortcutsFactoryTarget)forClass {
|
|
return ^(Class cls) {
|
|
NSAssert(
|
|
( self->_append && !self->_prepend && !self->_replace) ||
|
|
(!self->_append && self->_prepend && !self->_replace) ||
|
|
(!self->_append && !self->_prepend && self->_replace),
|
|
@"You can only do one of [append, prepend, replace]"
|
|
);
|
|
|
|
|
|
/// Whether the metadata we're about to add is instance or
|
|
/// class metadata, i.e. class properties vs instance properties
|
|
BOOL instanceMetadata = !self->_notInstance;
|
|
/// Whether the given class is a metaclass or not; we need to switch to
|
|
/// the metaclass to add class metadata if we are given the normal class object
|
|
BOOL isMeta = class_isMetaClass(cls);
|
|
/// Whether the shortcuts we're about to add should appear for classes or instances
|
|
BOOL instanceShortcut = !isMeta;
|
|
|
|
if (instanceMetadata) {
|
|
NSAssert(!isMeta,
|
|
@"Instance metadata can only be added as an instance shortcut"
|
|
);
|
|
}
|
|
|
|
Class metaclass = isMeta ? cls : object_getClass(cls);
|
|
Class clsForMetadata = instanceMetadata ? cls : metaclass;
|
|
|
|
// The factory is a singleton so we don't need to worry about "leaking" it
|
|
#pragma clang diagnostic push
|
|
#pragma clang diagnostic ignored "-Wimplicit-retain-self"
|
|
|
|
RegistrationBuckets *propertyBucket = instanceShortcut ? cProperties : mProperties;
|
|
RegistrationBuckets *methodBucket = instanceShortcut ? cMethods : mMethods;
|
|
RegistrationBuckets *ivarBucket = instanceShortcut ? cIvars : nil;
|
|
|
|
#pragma clang diagnostic pop
|
|
|
|
if (self->_properties) {
|
|
NSArray *items = [self->_properties flex_mapped:^id(NSString *name, NSUInteger idx) {
|
|
return [FLEXProperty named:name onClass:clsForMetadata];
|
|
}];
|
|
[self _register:items to:propertyBucket class:cls];
|
|
}
|
|
|
|
if (self->_methods) {
|
|
NSArray *items = [self->_methods flex_mapped:^id(NSString *name, NSUInteger idx) {
|
|
return [FLEXMethod selector:NSSelectorFromString(name) class:clsForMetadata];
|
|
}];
|
|
[self _register:items to:methodBucket class:cls];
|
|
}
|
|
|
|
if (self->_ivars) {
|
|
NSAssert(instanceMetadata, @"Instance metadata can only be added as an instance shortcut (%@)", cls);
|
|
NSArray *items = [self->_ivars flex_mapped:^id(NSString *name, NSUInteger idx) {
|
|
return [FLEXIvar named:name onClass:clsForMetadata];
|
|
}];
|
|
[self _register:items to:ivarBucket class:cls];
|
|
}
|
|
|
|
[self reset];
|
|
};
|
|
}
|
|
|
|
@end
|