-
Notifications
You must be signed in to change notification settings - Fork 1.4k
/
Copy pathNSObject+QMUI.m
531 lines (434 loc) · 20.4 KB
/
NSObject+QMUI.m
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
/**
* Tencent is pleased to support the open source community by making QMUI_iOS available.
* Copyright (C) 2016-2021 THL A29 Limited, a Tencent company. All rights reserved.
* Licensed under the MIT License (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at
* http://opensource.org/licenses/MIT
* 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.
*/
//
// NSObject+QMUI.m
// qmui
//
// Created by QMUI Team on 2016/11/1.
//
#import "NSObject+QMUI.h"
#import "QMUIWeakObjectContainer.h"
#import "QMUICore.h"
#import "NSString+QMUI.h"
#import <objc/message.h>
@implementation NSObject (QMUI)
- (BOOL)qmui_hasOverrideMethod:(SEL)selector ofSuperclass:(Class)superclass {
return [NSObject qmui_hasOverrideMethod:selector forClass:self.class ofSuperclass:superclass];
}
+ (BOOL)qmui_hasOverrideMethod:(SEL)selector forClass:(Class)aClass ofSuperclass:(Class)superclass {
if (![aClass isSubclassOfClass:superclass]) {
return NO;
}
if (![superclass instancesRespondToSelector:selector]) {
return NO;
}
Method superclassMethod = class_getInstanceMethod(superclass, selector);
Method instanceMethod = class_getInstanceMethod(aClass, selector);
if (!instanceMethod || instanceMethod == superclassMethod) {
return NO;
}
return YES;
}
- (id)qmui_performSelectorToSuperclass:(SEL)aSelector {
struct objc_super mySuper;
mySuper.receiver = self;
mySuper.super_class = class_getSuperclass(object_getClass(self));
id (*objc_superAllocTyped)(struct objc_super *, SEL) = (void *)&objc_msgSendSuper;
return (*objc_superAllocTyped)(&mySuper, aSelector);
}
- (id)qmui_performSelectorToSuperclass:(SEL)aSelector withObject:(id)object {
struct objc_super mySuper;
mySuper.receiver = self;
mySuper.super_class = class_getSuperclass(object_getClass(self));
id (*objc_superAllocTyped)(struct objc_super *, SEL, ...) = (void *)&objc_msgSendSuper;
return (*objc_superAllocTyped)(&mySuper, aSelector, object);
}
- (id)qmui_performSelector:(SEL)selector withArguments:(void *)firstArgument, ... {
NSInvocation *invocation = [NSInvocation invocationWithMethodSignature:[self methodSignatureForSelector:selector]];
[invocation setTarget:self];
[invocation setSelector:selector];
if (firstArgument) {
va_list valist;
va_start(valist, firstArgument);
[invocation setArgument:firstArgument atIndex:2];// 0->self, 1->_cmd
void *currentArgument;
NSInteger index = 3;
while ((currentArgument = va_arg(valist, void *))) {
[invocation setArgument:currentArgument atIndex:index];
index++;
}
va_end(valist);
}
[invocation invoke];
const char *typeEncoding = method_getTypeEncoding(class_getInstanceMethod(object_getClass(self), selector));
if (isObjectTypeEncoding(typeEncoding)) {
__unsafe_unretained id returnValue;
[invocation getReturnValue:&returnValue];
return returnValue;
}
return nil;
}
- (void)qmui_performSelector:(SEL)selector withPrimitiveReturnValue:(void *)returnValue {
[self qmui_performSelector:selector withPrimitiveReturnValue:returnValue arguments:nil];
}
- (void)qmui_performSelector:(SEL)selector withPrimitiveReturnValue:(void *)returnValue arguments:(void *)firstArgument, ... {
NSMethodSignature *methodSignature = [self methodSignatureForSelector:selector];
QMUIAssert(methodSignature, @"NSObject (QMUI)", @"- [%@ qmui_performSelector:@selector(%@)] 失败,方法不存在。", NSStringFromClass(self.class), NSStringFromSelector(selector));
NSInvocation *invocation = [NSInvocation invocationWithMethodSignature:methodSignature];
[invocation setTarget:self];
[invocation setSelector:selector];
if (firstArgument) {
va_list valist;
va_start(valist, firstArgument);
[invocation setArgument:firstArgument atIndex:2];// 0->self, 1->_cmd
void *currentArgument;
NSInteger index = 3;
while ((currentArgument = va_arg(valist, void *))) {
[invocation setArgument:currentArgument atIndex:index];
index++;
}
va_end(valist);
}
[invocation invoke];
if (returnValue) {
[invocation getReturnValue:returnValue];
}
}
- (void)qmui_enumrateIvarsUsingBlock:(void (^)(Ivar ivar, NSString *ivarDescription))block {
[self qmui_enumrateIvarsIncludingInherited:NO usingBlock:block];
}
- (void)qmui_enumrateIvarsIncludingInherited:(BOOL)includingInherited usingBlock:(void (^)(Ivar ivar, NSString *ivarDescription))block {
NSMutableArray<NSString *> *ivarDescriptions = [NSMutableArray new];
BeginIgnorePerformSelectorLeaksWarning
NSString *ivarList = [self performSelector:NSSelectorFromString(@"_ivarDescription")];
EndIgnorePerformSelectorLeaksWarning
NSError *error;
NSRegularExpression *reg = [NSRegularExpression regularExpressionWithPattern:[NSString stringWithFormat:@"in %@:(.*?)((?=in \\w+:)|$)", NSStringFromClass(self.class)] options:NSRegularExpressionDotMatchesLineSeparators error:&error];
if (!error) {
NSArray<NSTextCheckingResult *> *result = [reg matchesInString:ivarList options:NSMatchingReportCompletion range:NSMakeRange(0, ivarList.length)];
[result enumerateObjectsUsingBlock:^(NSTextCheckingResult * _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
NSString *ivars = [ivarList substringWithRange:[obj rangeAtIndex:1]];
[ivars enumerateLinesUsingBlock:^(NSString * _Nonnull line, BOOL * _Nonnull stop) {
if (![line hasPrefix:@"\t\t"]) {// 有些 struct 类型的变量,会把 struct 的成员也缩进打出来,所以用这种方式过滤掉
line = line.qmui_trim;
if (line.length > 2) {// 过滤掉空行或者 struct 结尾的"}"
NSRange range = [line rangeOfString:@":"];
if (range.location != NSNotFound)// 有些"unknow type"的变量不会显示指针地址(例如 UIView->_viewFlags)
line = [line substringToIndex:range.location];// 去掉指针地址
NSUInteger typeStart = [line rangeOfString:@" ("].location;
line = [NSString stringWithFormat:@"%@ %@", [line substringWithRange:NSMakeRange(typeStart + 2, line.length - 1 - (typeStart + 2))], [line substringToIndex:typeStart]];// 交换变量类型和变量名的位置,变量类型在前,变量名在后,空格隔开
[ivarDescriptions addObject:line];
}
}
}];
}];
}
unsigned int outCount = 0;
Ivar *ivars = class_copyIvarList(self.class, &outCount);
for (unsigned int i = 0; i < outCount; i ++) {
Ivar ivar = ivars[i];
NSString *ivarName = [NSString stringWithFormat:@"%s", ivar_getName(ivar)];
for (NSString *desc in ivarDescriptions) {
if ([desc hasSuffix:ivarName]) {
block(ivar, desc);
break;
}
}
}
free(ivars);
if (includingInherited) {
Class superclass = self.superclass;
if (superclass) {
[NSObject qmui_enumrateIvarsOfClass:superclass includingInherited:includingInherited usingBlock:block];
}
}
}
+ (void)qmui_enumrateIvarsOfClass:(Class)aClass includingInherited:(BOOL)includingInherited usingBlock:(void (^)(Ivar, NSString *))block {
if (!block) return;
NSObject *obj = nil;
if ([aClass isSubclassOfClass:[UICollectionView class]]) {
obj = [[aClass alloc] initWithFrame:CGRectZero collectionViewLayout:UICollectionViewFlowLayout.new];
} else if ([aClass isSubclassOfClass:[UIApplication class]]) {
obj = UIApplication.sharedApplication;
} else {
obj = [aClass new];
}
[obj qmui_enumrateIvarsIncludingInherited:includingInherited usingBlock:block];
}
- (void)qmui_enumratePropertiesUsingBlock:(void (^)(objc_property_t property, NSString *propertyName))block {
[NSObject qmui_enumratePropertiesOfClass:self.class includingInherited:NO usingBlock:block];
}
+ (void)qmui_enumratePropertiesOfClass:(Class)aClass includingInherited:(BOOL)includingInherited usingBlock:(void (^)(objc_property_t, NSString *))block {
if (!block) return;
unsigned int propertiesCount = 0;
objc_property_t *properties = class_copyPropertyList(aClass, &propertiesCount);
for (unsigned int i = 0; i < propertiesCount; i++) {
objc_property_t property = properties[i];
if (block) block(property, [NSString stringWithFormat:@"%s", property_getName(property)]);
}
free(properties);
if (includingInherited) {
Class superclass = class_getSuperclass(aClass);
if (superclass) {
[NSObject qmui_enumratePropertiesOfClass:superclass includingInherited:includingInherited usingBlock:block];
}
}
}
- (void)qmui_enumrateInstanceMethodsUsingBlock:(void (^)(Method, SEL))block {
[NSObject qmui_enumrateInstanceMethodsOfClass:self.class includingInherited:NO usingBlock:block];
}
+ (void)qmui_enumrateInstanceMethodsOfClass:(Class)aClass includingInherited:(BOOL)includingInherited usingBlock:(void (^)(Method, SEL))block {
if (!block) return;
unsigned int methodCount = 0;
Method *methods = class_copyMethodList(aClass, &methodCount);
for (unsigned int i = 0; i < methodCount; i++) {
Method method = methods[i];
SEL selector = method_getName(method);
if (block) block(method, selector);
}
free(methods);
if (includingInherited) {
Class superclass = class_getSuperclass(aClass);
if (superclass) {
[NSObject qmui_enumrateInstanceMethodsOfClass:superclass includingInherited:includingInherited usingBlock:block];
}
}
}
+ (void)qmui_enumerateProtocolMethods:(Protocol *)protocol usingBlock:(void (^)(SEL))block {
if (!block) return;
unsigned int methodCount = 0;
struct objc_method_description *methods = protocol_copyMethodDescriptionList(protocol, NO, YES, &methodCount);
for (int i = 0; i < methodCount; i++) {
struct objc_method_description methodDescription = methods[i];
if (block) {
block(methodDescription.name);
}
}
free(methods);
}
@end
@implementation NSObject (QMUI_KeyValueCoding)
- (id)qmui_valueForKey:(NSString *)key {
if ([self isKindOfClass:[UIView class]] && QMUICMIActivated && !IgnoreKVCAccessProhibited) {
BeginIgnoreUIKVCAccessProhibited
id value = [self valueForKey:key];
EndIgnoreUIKVCAccessProhibited
return value;
}
return [self valueForKey:key];
}
- (void)qmui_setValue:(id)value forKey:(NSString *)key {
if ([self isKindOfClass:[UIView class]] && QMUICMIActivated && !IgnoreKVCAccessProhibited) {
BeginIgnoreUIKVCAccessProhibited
[self setValue:value forKey:key];
EndIgnoreUIKVCAccessProhibited
return;
}
[self setValue:value forKey:key];
}
- (BOOL)qmui_canGetValueForKey:(NSString *)key {
NSArray<NSString *> *getters = @[
[NSString stringWithFormat:@"get%@", key.qmui_capitalizedString], // get<Key>
key,
[NSString stringWithFormat:@"is%@", key.qmui_capitalizedString], // is<Key>
[NSString stringWithFormat:@"_%@", key] // _<key>
];
for (NSString *selectorString in getters) {
if ([self respondsToSelector:NSSelectorFromString(selectorString)]) return YES;
}
if (![self.class accessInstanceVariablesDirectly]) return NO;
return [self _qmui_hasSpecifiedIvarWithKey:key];
}
- (BOOL)qmui_canSetValueForKey:(NSString *)key {
NSArray<NSString *> *setter = @[
[NSString stringWithFormat:@"set%@:", key.qmui_capitalizedString], // set<Key>:
[NSString stringWithFormat:@"_set%@", key.qmui_capitalizedString] // _set<Key>
];
for (NSString *selectorString in setter) {
if ([self respondsToSelector:NSSelectorFromString(selectorString)]) return YES;
}
if (![self.class accessInstanceVariablesDirectly]) return NO;
return [self _qmui_hasSpecifiedIvarWithKey:key];
}
- (BOOL)_qmui_hasSpecifiedIvarWithKey:(NSString *)key {
__block BOOL result = NO;
NSArray<NSString *> *ivars = @[
[NSString stringWithFormat:@"_%@", key],
[NSString stringWithFormat:@"_is%@", key.qmui_capitalizedString],
key,
[NSString stringWithFormat:@"is%@", key.qmui_capitalizedString]
];
[NSObject qmui_enumrateIvarsOfClass:self.class includingInherited:YES usingBlock:^(Ivar _Nonnull ivar, NSString * _Nonnull ivarDescription) {
if (!result) {
NSString *ivarName = [NSString stringWithFormat:@"%s", ivar_getName(ivar)];
if ([ivars containsObject:ivarName]) {
result = YES;
}
}
}];
return result;
}
@end
@implementation NSObject (QMUI_DataBind)
static char kAssociatedObjectKey_QMUIAllBoundObjects;
- (NSMutableDictionary<id, id> *)qmui_allBoundObjects {
NSMutableDictionary<id, id> *dict = objc_getAssociatedObject(self, &kAssociatedObjectKey_QMUIAllBoundObjects);
if (!dict) {
dict = [NSMutableDictionary dictionary];
objc_setAssociatedObject(self, &kAssociatedObjectKey_QMUIAllBoundObjects, dict, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}
return dict;
}
- (void)qmui_bindObject:(id)object forKey:(NSString *)key {
if (!key.length) {
return;
}
if (object) {
[[self qmui_allBoundObjects] setObject:object forKey:key];
} else {
[[self qmui_allBoundObjects] removeObjectForKey:key];
}
}
- (void)qmui_bindObjectWeakly:(id)object forKey:(NSString *)key {
if (!key.length) {
return;
}
if (object) {
QMUIWeakObjectContainer *container = [[QMUIWeakObjectContainer alloc] initWithObject:object];
[self qmui_bindObject:container forKey:key];
} else {
[[self qmui_allBoundObjects] removeObjectForKey:key];
}
}
- (id)qmui_getBoundObjectForKey:(NSString *)key {
if (!key.length) {
return nil;
}
id storedObj = [[self qmui_allBoundObjects] objectForKey:key];
if ([storedObj respondsToSelector:@selector(isQMUIWeakObjectContainer)] && ((QMUIWeakObjectContainer *)storedObj).isQMUIWeakObjectContainer) {
storedObj = [(QMUIWeakObjectContainer *)storedObj object];
}
return storedObj;
}
- (void)qmui_bindDouble:(double)doubleValue forKey:(NSString *)key {
[self qmui_bindObject:@(doubleValue) forKey:key];
}
- (double)qmui_getBoundDoubleForKey:(NSString *)key {
id object = [self qmui_getBoundObjectForKey:key];
if ([object isKindOfClass:[NSNumber class]]) {
double doubleValue = [(NSNumber *)object doubleValue];
return doubleValue;
} else {
return 0.0;
}
}
- (void)qmui_bindBOOL:(BOOL)boolValue forKey:(NSString *)key {
[self qmui_bindObject:@(boolValue) forKey:key];
}
- (BOOL)qmui_getBoundBOOLForKey:(NSString *)key {
id object = [self qmui_getBoundObjectForKey:key];
if ([object isKindOfClass:[NSNumber class]]) {
BOOL boolValue = [(NSNumber *)object boolValue];
return boolValue;
} else {
return NO;
}
}
- (void)qmui_bindLong:(long)longValue forKey:(NSString *)key {
[self qmui_bindObject:@(longValue) forKey:key];
}
- (long)qmui_getBoundLongForKey:(NSString *)key {
id object = [self qmui_getBoundObjectForKey:key];
if ([object isKindOfClass:[NSNumber class]]) {
long longValue = [(NSNumber *)object longValue];
return longValue;
} else {
return 0;
}
}
- (void)qmui_clearBindingForKey:(NSString *)key {
[self qmui_bindObject:nil forKey:key];
}
- (void)qmui_clearAllBinding {
[[self qmui_allBoundObjects] removeAllObjects];
}
- (NSArray<NSString *> *)qmui_allBindingKeys {
NSArray<NSString *> *allKeys = [[self qmui_allBoundObjects] allKeys];
return allKeys;
}
- (BOOL)qmui_hasBindingKey:(NSString *)key {
return [[self qmui_allBindingKeys] containsObject:key];
}
@end
@implementation NSObject (QMUI_Debug)
BeginIgnorePerformSelectorLeaksWarning
- (NSString *)qmui_methodList {
return [self performSelector:NSSelectorFromString(@"_methodDescription")];
}
- (NSString *)qmui_shortMethodList {
return [self performSelector:NSSelectorFromString(@"_shortMethodDescription")];
}
- (NSString *)qmui_ivarList {
NSString *systemResult = [self performSelector:NSSelectorFromString(@"_ivarDescription")];
NSRegularExpression *regx = [NSRegularExpression regularExpressionWithPattern:@"^(\\s+)(\\S+)" options:NSRegularExpressionCaseInsensitive error:nil];
NSMutableArray<NSString *> *lines = [systemResult componentsSeparatedByString:@"\n"].mutableCopy;
[lines enumerateObjectsUsingBlock:^(NSString *line, NSUInteger idx, BOOL * _Nonnull stop) {
// 过滤掉空行或者 struct 结尾的"}"
if (line.qmui_trim.length <= 2) return;
// 有些 struct 类型的变量,会把 struct 的成员也缩进打出来,所以用这种方式过滤掉
if ([line hasPrefix:@"\t\t"]) return;
NSTextCheckingResult *regxResult = [regx firstMatchInString:line options:NSMatchingReportCompletion range:NSMakeRange(0, line.length)];
if (regxResult.numberOfRanges < 3) return;
NSRange indentRange = [regxResult rangeAtIndex:1];
NSRange offsetRange = NSMakeRange(NSMaxRange(indentRange), 0);
NSRange ivarNameRange = [regxResult rangeAtIndex:2];
NSString *ivarName = [line substringWithRange:ivarNameRange];
Ivar ivar = class_getInstanceVariable(self.class, ivarName.UTF8String);
ptrdiff_t ivarOffset = ivar_getOffset(ivar);
NSString *lineWithOffset = [line stringByReplacingCharactersInRange:offsetRange withString:[NSString stringWithFormat:@"[%@|0x%@]", @(ivarOffset), [NSString stringWithFormat:@"%lx", (NSInteger)ivarOffset].uppercaseString]];
[lines setObject:lineWithOffset atIndexedSubscript:idx];
}];
NSString *result = [lines componentsJoinedByString:@"\n"];
return result;
}
- (NSString *)qmui_viewInfo {
if ([self isKindOfClass:UIView.class]) {
return [self performSelector:NSSelectorFromString(@"recursiveDescription")];
}
return @"仅支持 UIView";
}
EndIgnorePerformSelectorLeaksWarning
@end
@implementation NSThread (QMUI_KVC)
QMUISynthesizeBOOLProperty(qmui_shouldIgnoreUIKVCAccessProhibited, setQmui_shouldIgnoreUIKVCAccessProhibited)
@end
@interface NSException (QMUI_KVC)
@end
@implementation NSException (QMUI_KVC)
+ (void)load {
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
OverrideImplementation(object_getClass([NSException class]), @selector(raise:format:), ^id(__unsafe_unretained Class originClass, SEL originCMD, IMP (^originalIMPProvider)(void)) {
return ^(NSObject *selfObject, NSExceptionName raise, NSString *format, ...) {
if (raise == NSGenericException && [format isEqualToString:@"Access to %@'s %@ ivar is prohibited. This is an application bug"]) {
BOOL shouldIgnoreUIKVCAccessProhibited = ((QMUICMIActivated && IgnoreKVCAccessProhibited) || NSThread.currentThread.qmui_shouldIgnoreUIKVCAccessProhibited);
if (shouldIgnoreUIKVCAccessProhibited) return;
QMUILogWarn(@"NSObject (QMUI)", @"使用 KVC 访问了 UIKit 的私有属性,会触发系统的 NSException,建议尽量避免此类操作,仍需访问可使用 BeginIgnoreUIKVCAccessProhibited 和 EndIgnoreUIKVCAccessProhibited 把相关代码包裹起来,或者直接使用 qmui_valueForKey: 、qmui_setValue:forKey:");
}
id (*originSelectorIMP)(id, SEL, NSExceptionName name, NSString *, ...);
originSelectorIMP = (id (*)(id, SEL, NSExceptionName name, NSString *, ...))originalIMPProvider();
va_list args;
va_start(args, format);
NSString *reason = [[NSString alloc] initWithFormat:format arguments:args];
originSelectorIMP(selfObject, originCMD, raise, reason);
va_end(args);
};
});
});
}
@end