mirror of
https://github.com/faroukbmiled/RyukGram.git
synced 2026-04-23 04:36:03 +02:00
1235 lines
39 KiB
Objective-C
Executable File
1235 lines
39 KiB
Objective-C
Executable File
//
|
|
// JGProgressHUD.m
|
|
// JGProgressHUD
|
|
//
|
|
// Created by Jonas Gessner on 20.7.14.
|
|
// Copyright (c) 2014 Jonas Gessner. All rights reserved.
|
|
//
|
|
|
|
#import "JGProgressHUD.h"
|
|
#import <QuartzCore/QuartzCore.h>
|
|
#import "JGProgressHUDFadeAnimation.h"
|
|
#import "JGProgressHUDIndeterminateIndicatorView.h"
|
|
|
|
#if !__has_feature(objc_arc)
|
|
#error "JGProgressHUD requires ARC!"
|
|
#endif
|
|
|
|
static inline CGRect JGProgressHUD_CGRectIntegral(CGRect rect) {
|
|
CGFloat scale = [[UIScreen mainScreen] scale];
|
|
|
|
return (CGRect){{((CGFloat)floor(rect.origin.x*scale))/scale, ((CGFloat)floor(rect.origin.y*scale))/scale}, {((CGFloat)ceil(rect.size.width*scale))/scale, ((CGFloat)ceil(rect.size.height*scale))/scale}};
|
|
}
|
|
|
|
API_AVAILABLE(ios(12.0), tvos(10.0))
|
|
static inline JGProgressHUDStyle JGProgressHUDStyleFromUIUserInterfaceStyle(UIUserInterfaceStyle uiStyle) {
|
|
if (uiStyle == UIUserInterfaceStyleDark) {
|
|
return JGProgressHUDStyleDark;
|
|
}
|
|
else {
|
|
return JGProgressHUDStyleExtraLight;
|
|
}
|
|
}
|
|
|
|
@interface JGProgressHUD () {
|
|
BOOL _transitioning;
|
|
BOOL _updateAfterAppear;
|
|
|
|
BOOL _dismissAfterTransitionFinished;
|
|
BOOL _dismissAfterTransitionFinishedWithAnimation;
|
|
|
|
CFAbsoluteTime _displayTimestamp;
|
|
dispatch_block_t __nullable _displayBlock;
|
|
|
|
BOOL _effectiveIndicatorViewNeedsUpdate;
|
|
|
|
UIView *__nonnull _blurViewContainer;
|
|
UIView *__nonnull _shadowView;
|
|
CAShapeLayer *__nonnull _shadowMaskLayer;
|
|
|
|
NSMutableArray<void (^)(void)> *_dismissActions;
|
|
|
|
BOOL _automaticStyle;
|
|
}
|
|
|
|
// In 'beta'
|
|
/**
|
|
Setting this to @c YES makes the text and indicator views bigger.
|
|
@b Default: NO.
|
|
*/
|
|
@property (nonatomic, assign) BOOL thick;
|
|
|
|
@property (nonatomic, strong, readonly, nonnull) UIVisualEffectView *blurView;
|
|
@property (nonatomic, strong, readonly, nonnull) UIVisualEffectView *vibrancyView;
|
|
|
|
@property (nonatomic, strong, readonly, nonnull) JGProgressHUDIndicatorView *effectiveIndicatorView;
|
|
|
|
@end
|
|
|
|
@interface JGProgressHUDAnimation (Private)
|
|
|
|
@property (nonatomic, weak, nullable) JGProgressHUD *progressHUD;
|
|
|
|
@end
|
|
|
|
@implementation JGProgressHUD
|
|
|
|
@synthesize HUDView = _HUDView;
|
|
@synthesize blurView = _blurView;
|
|
@synthesize vibrancyView = _vibrancyView;
|
|
@synthesize textLabel = _textLabel;
|
|
@synthesize detailTextLabel = _detailTextLabel;
|
|
@synthesize indicatorView = _indicatorView;
|
|
@synthesize animation = _animation;
|
|
@synthesize contentView = _contentView;
|
|
|
|
@dynamic visible;
|
|
|
|
#pragma mark - Keyboard
|
|
|
|
static CGRect keyboardFrame = (CGRect){{0.0, 0.0}, {0.0, 0.0}};
|
|
|
|
#if TARGET_OS_IOS
|
|
+ (void)keyboardFrameWillChange:(NSNotification *)notification {
|
|
keyboardFrame = [notification.userInfo[UIKeyboardFrameEndUserInfoKey] CGRectValue];
|
|
if (CGRectIsEmpty(keyboardFrame)) {
|
|
keyboardFrame = [notification.userInfo[UIKeyboardFrameBeginUserInfoKey] CGRectValue];
|
|
}
|
|
}
|
|
|
|
+ (void)keyboardFrameDidChange:(NSNotification *)notification {
|
|
keyboardFrame = [notification.userInfo[UIKeyboardFrameEndUserInfoKey] CGRectValue];
|
|
}
|
|
|
|
+ (void)keyboardDidHide {
|
|
keyboardFrame = CGRectZero;
|
|
}
|
|
|
|
+ (void)load {
|
|
@autoreleasepool {
|
|
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(keyboardFrameWillChange:) name:UIKeyboardWillChangeFrameNotification object:nil];
|
|
|
|
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(keyboardFrameDidChange:) name:UIKeyboardDidChangeFrameNotification object:nil];
|
|
|
|
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(keyboardDidHide) name:UIKeyboardDidHideNotification object:nil];
|
|
}
|
|
}
|
|
#endif
|
|
|
|
+ (CGRect)currentKeyboardFrame {
|
|
return keyboardFrame;
|
|
}
|
|
|
|
#pragma mark - Initializers
|
|
|
|
- (instancetype)init {
|
|
return [self initWithAutomaticStyle];
|
|
}
|
|
|
|
- (instancetype)initWithFrame:(CGRect __unused)frame {
|
|
return [self initWithAutomaticStyle];
|
|
}
|
|
|
|
/*
|
|
Basic architecture:
|
|
|
|
* self covers the entire target view.
|
|
* self.HUDView is a subview of self and has the size and position of the visible HUD. The layer has rounded corners set with self.cornerRadius. The layer does not clip to bounds.
|
|
* _shadowView is a subview of self.HUDView and always covers the entire HUDView. The corners are also rounded in the same way as self.HUDView. It draws its shadow only on the outside by using a masking layer, so that the shadow does not interfere with the blur view.
|
|
* _blurViewContainer is a subview of self.HUDView and always covers the entire self.HUDView. The corners are also rounded in the same way as self.HUDView but it additionally clips to bounds.
|
|
* self.blurView is a subview of _blurViewContainer and provides the blur effect. The corners are not rounded and the view does not clip to bounds.
|
|
* self.vibrancyView is a subview of self.blurView.contentView and provides the vibrancy effect. The corners are not rounded and the view does not clip to bounds.
|
|
* self.contentView is a subview of self.vibrancyView.contentView. It does not always have the same frame as it's superview (during transitions).
|
|
* self.contentView contains the labels and the indicator view.
|
|
|
|
*/
|
|
- (instancetype)initWithStyle:(JGProgressHUDStyle)style {
|
|
self = [super initWithFrame:CGRectZero];
|
|
|
|
if (self) {
|
|
_style = style;
|
|
_voiceOverEnabled = YES;
|
|
_automaticStyle = NO;
|
|
|
|
_HUDView = [[UIView alloc] init];
|
|
self.HUDView.backgroundColor = [UIColor clearColor];
|
|
[self addSubview:self.HUDView];
|
|
|
|
_blurViewContainer = [[UIView alloc] init];
|
|
_blurViewContainer.backgroundColor = [UIColor clearColor];
|
|
_blurViewContainer.clipsToBounds = YES;
|
|
[self.HUDView addSubview:_blurViewContainer];
|
|
|
|
_shadowView = [[UIView alloc] init];
|
|
_shadowView.backgroundColor = [UIColor blackColor];
|
|
_shadowView.userInteractionEnabled = NO;
|
|
_shadowView.layer.shadowOpacity = 1.0;
|
|
_shadowView.alpha = 0.0;
|
|
|
|
_shadowMaskLayer = [CAShapeLayer layer];
|
|
_shadowMaskLayer.fillRule = kCAFillRuleEvenOdd;
|
|
_shadowMaskLayer.fillColor = [UIColor blackColor].CGColor;
|
|
_shadowMaskLayer.opacity = 1.0;
|
|
|
|
_shadowView.layer.mask = _shadowMaskLayer;
|
|
|
|
[self.HUDView addSubview:_shadowView];
|
|
|
|
_indicatorView = [[JGProgressHUDIndeterminateIndicatorView alloc] init];
|
|
_effectiveIndicatorView = _indicatorView;
|
|
[self.effectiveIndicatorView setUpForHUDStyle:self.style vibrancyEnabled:self.vibrancyEnabled];
|
|
|
|
self.hidden = YES;
|
|
self.backgroundColor = [UIColor clearColor];
|
|
|
|
self.contentInsets = UIEdgeInsetsMake(20.0, 20.0, 20.0, 20.0);
|
|
self.layoutMargins = UIEdgeInsetsMake(20.0, 20.0, 20.0, 20.0);
|
|
|
|
self.cornerRadius = 10.0;
|
|
|
|
_dismissActions = [[NSMutableArray alloc] init];
|
|
|
|
#if TARGET_OS_IOS
|
|
[self addGestureRecognizer:[[UITapGestureRecognizer alloc] initWithTarget:self action:@selector(tapped:)]];
|
|
#elif TARGET_OS_TV
|
|
_wantsFocus = YES;
|
|
#endif
|
|
}
|
|
|
|
return self;
|
|
}
|
|
|
|
+ (instancetype)progressHUDWithStyle:(JGProgressHUDStyle)style {
|
|
return [(JGProgressHUD *)[self alloc] initWithStyle:style];
|
|
}
|
|
|
|
- (instancetype)initWithAutomaticStyle {
|
|
if (@available(iOS 12.0, tvOS 10.0, *)) {
|
|
JGProgressHUDStyle initialStyle;
|
|
|
|
if (@available(iOS 13.0, tvOS 13.0, *)) {
|
|
initialStyle = JGProgressHUDStyleFromUIUserInterfaceStyle([[UITraitCollection currentTraitCollection] userInterfaceStyle]);
|
|
}
|
|
else {
|
|
initialStyle = JGProgressHUDStyleExtraLight;
|
|
}
|
|
|
|
self = [self initWithStyle:initialStyle];
|
|
|
|
if (self != nil) {
|
|
_automaticStyle = YES;
|
|
}
|
|
}
|
|
else {
|
|
self = [self initWithStyle:JGProgressHUDStyleExtraLight];
|
|
}
|
|
|
|
return self;
|
|
}
|
|
|
|
+ (instancetype)progressHUDWithAutomaticStyle {
|
|
return [[self alloc] initWithAutomaticStyle];
|
|
}
|
|
|
|
#pragma mark - Layout
|
|
|
|
- (void)setHUDViewFrameCenterWithSize:(CGSize)size insetViewFrame:(CGRect)viewFrame {
|
|
CGRect frame = (CGRect){CGPointZero, size};
|
|
|
|
switch (self.position) {
|
|
case JGProgressHUDPositionTopLeft:
|
|
frame.origin.x = CGRectGetMinX(viewFrame);
|
|
frame.origin.y = CGRectGetMinY(viewFrame);
|
|
break;
|
|
|
|
case JGProgressHUDPositionTopCenter:
|
|
frame.origin.x = CGRectGetMidX(viewFrame) - size.width/2.0;
|
|
frame.origin.y = CGRectGetMinY(viewFrame);
|
|
break;
|
|
|
|
case JGProgressHUDPositionTopRight:
|
|
frame.origin.x = CGRectGetMaxX(viewFrame) - size.width;
|
|
frame.origin.y = CGRectGetMinY(viewFrame);
|
|
break;
|
|
|
|
case JGProgressHUDPositionCenterLeft:
|
|
frame.origin.x = CGRectGetMinX(viewFrame);
|
|
frame.origin.y = CGRectGetMidY(viewFrame) - size.height/2.0;
|
|
break;
|
|
|
|
case JGProgressHUDPositionCenter:
|
|
frame.origin.x = CGRectGetMidX(viewFrame) - size.width/2.0;
|
|
frame.origin.y = CGRectGetMidY(viewFrame) - size.height/2.0;
|
|
break;
|
|
|
|
case JGProgressHUDPositionCenterRight:
|
|
frame.origin.x = CGRectGetMaxX(viewFrame) - frame.size.width;
|
|
frame.origin.y = CGRectGetMidY(viewFrame) - size.height/2.0;
|
|
break;
|
|
|
|
case JGProgressHUDPositionBottomLeft:
|
|
frame.origin.x = CGRectGetMinX(viewFrame);
|
|
frame.origin.y = CGRectGetMaxY(viewFrame) - size.height;
|
|
break;
|
|
|
|
case JGProgressHUDPositionBottomCenter:
|
|
frame.origin.x = CGRectGetMidX(viewFrame) - size.width/2.0;
|
|
frame.origin.y = CGRectGetMaxY(viewFrame) - frame.size.height;
|
|
break;
|
|
|
|
case JGProgressHUDPositionBottomRight:
|
|
frame.origin.x = CGRectGetMaxX(viewFrame) - size.width;
|
|
frame.origin.y = CGRectGetMaxY(viewFrame) - size.height;
|
|
break;
|
|
}
|
|
|
|
CGRect oldHUDFrame = self.HUDView.frame;
|
|
CGRect updatedHUDFrame = JGProgressHUD_CGRectIntegral(frame);
|
|
|
|
self.HUDView.frame = updatedHUDFrame;
|
|
_shadowView.frame = self.HUDView.bounds;
|
|
[self updateShadowViewMask];
|
|
|
|
_blurViewContainer.frame = self.HUDView.bounds;
|
|
self.blurView.frame = self.HUDView.bounds;
|
|
self.vibrancyView.frame = self.HUDView.bounds;
|
|
|
|
[UIView performWithoutAnimation:^{
|
|
self.contentView.frame = (CGRect){{(oldHUDFrame.size.width - updatedHUDFrame.size.width)/2.0, (oldHUDFrame.size.height - updatedHUDFrame.size.height)/2.0}, updatedHUDFrame.size};
|
|
}];
|
|
|
|
self.contentView.frame = self.HUDView.bounds;
|
|
}
|
|
|
|
- (void)updateShadowViewMask {
|
|
if (CGRectIsEmpty(_shadowView.layer.bounds)) {
|
|
return;
|
|
}
|
|
|
|
CGRect layerBounds = CGRectMake(0.0, 0.0, _shadowView.layer.bounds.size.width + self.shadow.radius*4.0, _shadowView.layer.bounds.size.height + self.shadow.radius*4.0);
|
|
|
|
UIBezierPath *path = [UIBezierPath bezierPathWithRect:layerBounds];
|
|
|
|
CGRect maskRect = CGRectInset(layerBounds, self.shadow.radius*2.0, self.shadow.radius*2.0);
|
|
|
|
UIBezierPath *roundedPath = [UIBezierPath bezierPathWithRoundedRect:maskRect cornerRadius:self.cornerRadius];
|
|
|
|
[path appendPath:roundedPath];
|
|
|
|
_shadowMaskLayer.frame = CGRectInset(_shadowView.layer.bounds, -self.shadow.radius*2.0, -self.shadow.radius*2.0);
|
|
|
|
CAAnimation *currentAnimation = [self.HUDView.layer animationForKey:@"position"];
|
|
if (currentAnimation != nil) {
|
|
[CATransaction begin];
|
|
|
|
[CATransaction setAnimationDuration:currentAnimation.duration];
|
|
[CATransaction setAnimationTimingFunction:currentAnimation.timingFunction];
|
|
|
|
CABasicAnimation *pathAnimation = [CABasicAnimation animationWithKeyPath:@"path"];
|
|
[_shadowMaskLayer addAnimation:pathAnimation forKey:@"path"];
|
|
|
|
_shadowMaskLayer.path = path.CGPath;
|
|
|
|
[CATransaction commit];
|
|
}
|
|
else {
|
|
_shadowMaskLayer.path = path.CGPath;
|
|
// Remove implicit CALayer animations:
|
|
[_shadowMaskLayer removeAllAnimations];
|
|
}
|
|
}
|
|
|
|
- (void)layoutHUD {
|
|
if (_transitioning) {
|
|
_updateAfterAppear = YES;
|
|
return;
|
|
}
|
|
|
|
if (_targetView == nil) {
|
|
return;
|
|
}
|
|
|
|
CGRect indicatorFrame = self.effectiveIndicatorView.frame;
|
|
indicatorFrame.origin.y = self.contentInsets.top;
|
|
|
|
CGRect insetFrame = [self insetFrameForView:self];
|
|
|
|
CGFloat maxContentWidth = insetFrame.size.width - self.contentInsets.left - self.contentInsets.right;
|
|
CGFloat maxContentHeight = insetFrame.size.height - self.contentInsets.top - self.contentInsets.bottom;
|
|
|
|
CGRect labelFrame = CGRectZero;
|
|
CGRect detailFrame = CGRectZero;
|
|
|
|
CGFloat currentY = CGRectGetMaxY(indicatorFrame);
|
|
if (!CGRectIsEmpty(indicatorFrame)) {
|
|
currentY += 10.0;
|
|
}
|
|
|
|
if (_textLabel.text.length > 0) {
|
|
_textLabel.preferredMaxLayoutWidth = maxContentWidth;
|
|
|
|
CGSize neededSize = _textLabel.intrinsicContentSize;
|
|
neededSize.height = MIN(neededSize.height, maxContentHeight);
|
|
|
|
labelFrame.size = neededSize;
|
|
labelFrame.origin.y = currentY;
|
|
currentY = CGRectGetMaxY(labelFrame) + 6.0;
|
|
}
|
|
|
|
if (_detailTextLabel.text.length > 0) {
|
|
_detailTextLabel.preferredMaxLayoutWidth = maxContentWidth;
|
|
|
|
CGSize neededSize = _detailTextLabel.intrinsicContentSize;
|
|
neededSize.height = MIN(neededSize.height, maxContentHeight);
|
|
|
|
detailFrame.size = neededSize;
|
|
detailFrame.origin.y = currentY;
|
|
}
|
|
|
|
CGSize size = CGSizeZero;
|
|
|
|
CGFloat width = MIN(self.contentInsets.left + MAX(indicatorFrame.size.width, MAX(labelFrame.size.width, detailFrame.size.width)) + self.contentInsets.right, insetFrame.size.width);
|
|
|
|
CGFloat height = MAX(CGRectGetMaxY(labelFrame), MAX(CGRectGetMaxY(detailFrame), CGRectGetMaxY(indicatorFrame))) + self.contentInsets.bottom;
|
|
|
|
if (self.square) {
|
|
CGFloat uniSize = MAX(width, height);
|
|
|
|
size.width = uniSize;
|
|
size.height = uniSize;
|
|
|
|
CGFloat heightDelta = (uniSize-height)/2.0;
|
|
|
|
labelFrame.origin.y += heightDelta;
|
|
detailFrame.origin.y += heightDelta;
|
|
indicatorFrame.origin.y += heightDelta;
|
|
}
|
|
else {
|
|
size.width = width;
|
|
size.height = height;
|
|
}
|
|
|
|
CGPoint center = CGPointMake(size.width/2.0, size.height/2.0);
|
|
|
|
indicatorFrame.origin.x = center.x - indicatorFrame.size.width/2.0;
|
|
labelFrame.origin.x = center.x - labelFrame.size.width/2.0;
|
|
detailFrame.origin.x = center.x - detailFrame.size.width/2.0;
|
|
|
|
[UIView performWithoutAnimation:^{
|
|
self.effectiveIndicatorView.frame = indicatorFrame;
|
|
self->_textLabel.frame = JGProgressHUD_CGRectIntegral(labelFrame);
|
|
self->_detailTextLabel.frame = JGProgressHUD_CGRectIntegral(detailFrame);
|
|
}];
|
|
|
|
[self setHUDViewFrameCenterWithSize:size insetViewFrame:insetFrame];
|
|
}
|
|
|
|
- (CGRect)insetFrameForView:(UIView *)view {
|
|
CGRect localKeyboardFrame = [view convertRect:[[self class] currentKeyboardFrame] fromView:nil];
|
|
CGRect frame = view.bounds;
|
|
|
|
if (!CGRectIsEmpty(localKeyboardFrame) && CGRectIntersectsRect(frame, localKeyboardFrame)) {
|
|
CGFloat keyboardMinY = CGRectGetMinY(localKeyboardFrame);
|
|
|
|
if (@available(iOS 11, tvOS 11, *)) {
|
|
if (self.insetsLayoutMarginsFromSafeArea) {
|
|
// This makes sure that the bottom safe area inset is only respected when that area is not covered by the keyboard. When the keyboard covers the bottom area outside of the safe area it is not necessary to keep the bottom safe area insets part of the insets for the HUD.
|
|
keyboardMinY += self.safeAreaInsets.bottom;
|
|
}
|
|
}
|
|
|
|
frame.size.height = MAX(MIN(frame.size.height, keyboardMinY), 0.0);
|
|
}
|
|
|
|
return UIEdgeInsetsInsetRect(frame, view.layoutMargins);
|
|
}
|
|
|
|
- (void)applyCornerRadius {
|
|
self.HUDView.layer.cornerRadius = self.cornerRadius;
|
|
_blurViewContainer.layer.cornerRadius = self.cornerRadius;
|
|
_shadowView.layer.cornerRadius = self.cornerRadius;
|
|
|
|
[self updateShadowViewMask];
|
|
}
|
|
|
|
#pragma mark - Showing
|
|
|
|
- (void)cleanUpAfterPresentation {
|
|
#if TARGET_OS_TV
|
|
if (self.wantsFocus) {
|
|
[self.targetView setNeedsFocusUpdate];
|
|
}
|
|
#endif
|
|
|
|
self.hidden = NO;
|
|
|
|
_transitioning = NO;
|
|
// Correct timestamp to the current time for animated presentations:
|
|
_displayTimestamp = CFAbsoluteTimeGetCurrent();
|
|
|
|
if (_effectiveIndicatorViewNeedsUpdate) {
|
|
self.effectiveIndicatorView = self.indicatorView;
|
|
_effectiveIndicatorViewNeedsUpdate = NO;
|
|
_updateAfterAppear = NO;
|
|
}
|
|
else if (_updateAfterAppear) {
|
|
[self layoutHUD];
|
|
_updateAfterAppear = NO;
|
|
}
|
|
|
|
if ([self.delegate respondsToSelector:@selector(progressHUD:didPresentInView:)]){
|
|
[self.delegate progressHUD:self didPresentInView:self.targetView];
|
|
}
|
|
|
|
if (_dismissAfterTransitionFinished) {
|
|
[self dismissAnimated:_dismissAfterTransitionFinishedWithAnimation];
|
|
_dismissAfterTransitionFinished = NO;
|
|
_dismissAfterTransitionFinishedWithAnimation = NO;
|
|
}
|
|
else if (self.voiceOverEnabled && UIAccessibilityIsVoiceOverRunning()) {
|
|
UIAccessibilityPostNotification(UIAccessibilityLayoutChangedNotification, self);
|
|
}
|
|
}
|
|
|
|
- (void)showInView:(UIView *)view {
|
|
[self showInView:view animated:YES];
|
|
}
|
|
|
|
- (void)layoutSubviews {
|
|
[super layoutSubviews];
|
|
|
|
[self layoutHUD];
|
|
}
|
|
|
|
- (void)showInView:(UIView *)view animated:(BOOL)animated {
|
|
if (_transitioning) {
|
|
return;
|
|
}
|
|
else if (self.targetView != nil) {
|
|
#if DEBUG
|
|
NSLog(@"[Warning] The HUD is already visible! Ignoring.");
|
|
#endif
|
|
return;
|
|
}
|
|
|
|
if ([self.delegate respondsToSelector:@selector(progressHUD:willPresentInView:)]) {
|
|
[self.delegate progressHUD:self willPresentInView:view];
|
|
}
|
|
|
|
_targetView = view;
|
|
|
|
self.frame = _targetView.bounds;
|
|
|
|
[_targetView addSubview:self];
|
|
|
|
self.translatesAutoresizingMaskIntoConstraints = NO;
|
|
|
|
[NSLayoutConstraint constraintWithItem:self attribute:NSLayoutAttributeWidth relatedBy:NSLayoutRelationEqual toItem:_targetView attribute:NSLayoutAttributeWidth multiplier:1.0 constant:0.0].active = YES;
|
|
[NSLayoutConstraint constraintWithItem:self attribute:NSLayoutAttributeHeight relatedBy:NSLayoutRelationEqual toItem:_targetView attribute:NSLayoutAttributeHeight multiplier:1.0 constant:0.0].active = YES;
|
|
[NSLayoutConstraint constraintWithItem:self attribute:NSLayoutAttributeCenterX relatedBy:NSLayoutRelationEqual toItem:_targetView attribute:NSLayoutAttributeCenterX multiplier:1.0 constant:0.0].active = YES;
|
|
[NSLayoutConstraint constraintWithItem:self attribute:NSLayoutAttributeCenterY relatedBy:NSLayoutRelationEqual toItem:_targetView attribute:NSLayoutAttributeCenterY multiplier:1.0 constant:0.0].active = YES;
|
|
|
|
[self setNeedsLayout];
|
|
[self layoutIfNeeded];
|
|
|
|
_transitioning = YES;
|
|
|
|
_displayTimestamp = CFAbsoluteTimeGetCurrent();
|
|
|
|
if (animated && self.animation != nil) {
|
|
[self.animation show];
|
|
}
|
|
else {
|
|
[self cleanUpAfterPresentation];
|
|
}
|
|
|
|
#if TARGET_OS_IOS
|
|
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(keyboardFrameChanged:) name:UIKeyboardWillChangeFrameNotification object:nil];
|
|
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(keyboardFrameChanged:) name:UIKeyboardDidChangeFrameNotification object:nil];
|
|
#endif
|
|
}
|
|
|
|
- (void)showInView:(UIView *__nonnull)view animated:(BOOL)animated afterDelay:(NSTimeInterval)delay {
|
|
__weak __typeof(self) weakSelf = self;
|
|
|
|
if (_displayBlock != NULL) {
|
|
dispatch_cancel(_displayBlock);
|
|
}
|
|
|
|
_displayBlock = dispatch_block_create(0, ^{
|
|
if (weakSelf) {
|
|
__strong __typeof(weakSelf) strongSelf = weakSelf;
|
|
strongSelf->_displayBlock = NULL;
|
|
|
|
if (!strongSelf.visible) {
|
|
[strongSelf showInView:view animated:animated];
|
|
}
|
|
}
|
|
});
|
|
|
|
dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(delay * NSEC_PER_SEC)), dispatch_get_main_queue(), _displayBlock);
|
|
}
|
|
|
|
#pragma mark - Dismissing
|
|
|
|
- (void)cleanUpAfterDismissal {
|
|
self.hidden = YES;
|
|
[self removeFromSuperview];
|
|
|
|
[self removeObservers];
|
|
|
|
_transitioning = NO;
|
|
_dismissAfterTransitionFinished = NO;
|
|
_dismissAfterTransitionFinishedWithAnimation = NO;
|
|
|
|
__typeof(self.targetView) targetView = self.targetView;
|
|
_targetView = nil;
|
|
|
|
for (void (^action)(void) in _dismissActions) {
|
|
action();
|
|
}
|
|
[_dismissActions removeAllObjects];
|
|
|
|
if ([self.delegate respondsToSelector:@selector(progressHUD:didDismissFromView:)]) {
|
|
[self.delegate progressHUD:self didDismissFromView:targetView];
|
|
}
|
|
}
|
|
|
|
- (void)dismiss {
|
|
[self dismissAnimated:YES];
|
|
}
|
|
|
|
- (void)dismissAnimated:(BOOL)animated {
|
|
if (_displayBlock != NULL) {
|
|
dispatch_cancel(_displayBlock);
|
|
_displayBlock = NULL;
|
|
}
|
|
|
|
if (_transitioning) {
|
|
_dismissAfterTransitionFinished = YES;
|
|
_dismissAfterTransitionFinishedWithAnimation = animated;
|
|
return;
|
|
}
|
|
|
|
if (self.targetView == nil) {
|
|
return;
|
|
}
|
|
|
|
if (self.minimumDisplayTime > 0.0 && _displayTimestamp > 0.0) {
|
|
CFAbsoluteTime displayedTime = CFAbsoluteTimeGetCurrent()-_displayTimestamp;
|
|
|
|
if (displayedTime < self.minimumDisplayTime) {
|
|
NSTimeInterval delta = self.minimumDisplayTime-displayedTime;
|
|
|
|
[self dismissAfterDelay:delta animated:animated];
|
|
|
|
return;
|
|
}
|
|
}
|
|
|
|
if ([self.delegate respondsToSelector:@selector(progressHUD:willDismissFromView:)]) {
|
|
[self.delegate progressHUD:self willDismissFromView:self.targetView];
|
|
}
|
|
|
|
_transitioning = YES;
|
|
|
|
if (animated && self.animation) {
|
|
[self.animation hide];
|
|
}
|
|
else {
|
|
[self cleanUpAfterDismissal];
|
|
}
|
|
}
|
|
|
|
- (void)dismissAfterDelay:(NSTimeInterval)delay {
|
|
[self dismissAfterDelay:delay animated:YES];
|
|
}
|
|
|
|
- (void)dismissAfterDelay:(NSTimeInterval)delay animated:(BOOL)animated {
|
|
[self dismissAfterDelay:delay animated:animated completion:nil];
|
|
}
|
|
|
|
- (void)dismissAfterDelay:(NSTimeInterval)delay animated:(BOOL)animated completion:(void (^_Nullable)(void))dismissCompletion {
|
|
__weak __typeof(self) weakSelf = self;
|
|
|
|
dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(delay * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
|
|
if (weakSelf) {
|
|
__strong __typeof(weakSelf) strongSelf = weakSelf;
|
|
if (strongSelf.visible) {
|
|
if (dismissCompletion != nil) {
|
|
[self performAfterDismiss:dismissCompletion];
|
|
}
|
|
[strongSelf dismissAnimated:animated];
|
|
}
|
|
}
|
|
});
|
|
}
|
|
|
|
- (void)performAfterDismiss:(void (^_Nonnull)(void))dismissCompletion {
|
|
if (self.visible) {
|
|
[_dismissActions addObject:dismissCompletion];
|
|
}
|
|
}
|
|
|
|
#pragma mark - Callbacks
|
|
|
|
#if TARGET_OS_IOS
|
|
- (void)tapped:(UITapGestureRecognizer *)t {
|
|
if (CGRectContainsPoint(self.contentView.bounds, [t locationInView:self.contentView])) {
|
|
if (self.tapOnHUDViewBlock != nil) {
|
|
self.tapOnHUDViewBlock(self);
|
|
}
|
|
}
|
|
else if (self.tapOutsideBlock != nil) {
|
|
self.tapOutsideBlock(self);
|
|
}
|
|
}
|
|
|
|
static inline UIViewAnimationOptions UIViewAnimationOptionsFromUIViewAnimationCurve(UIViewAnimationCurve curve) {
|
|
UIViewAnimationOptions testOptions = UIViewAnimationCurveLinear << 16;
|
|
|
|
if (testOptions != UIViewAnimationOptionCurveLinear) {
|
|
NSLog(@"Unexpected implementation of UIViewAnimationOptionCurveLinear");
|
|
}
|
|
|
|
return (UIViewAnimationOptions)(curve << 16);
|
|
}
|
|
|
|
- (void)keyboardFrameChanged:(NSNotification *)notification {
|
|
NSTimeInterval duration = [notification.userInfo[UIKeyboardAnimationDurationUserInfoKey] doubleValue];
|
|
|
|
UIViewAnimationCurve curve = (UIViewAnimationCurve)[notification.userInfo[UIKeyboardAnimationCurveUserInfoKey] unsignedIntegerValue];
|
|
|
|
[UIView animateWithDuration:duration delay:0.0 options:UIViewAnimationOptionBeginFromCurrentState | UIViewAnimationOptionsFromUIViewAnimationCurve(curve) animations:^{
|
|
[self layoutHUD];
|
|
} completion:nil];
|
|
}
|
|
#endif
|
|
|
|
- (void)updateMotionOnHUDView {
|
|
BOOL reduceMotionEnabled = UIAccessibilityIsReduceMotionEnabled();
|
|
|
|
BOOL wantsParallax = ((self.parallaxMode == JGProgressHUDParallaxModeDevice && !reduceMotionEnabled) || self.parallaxMode == JGProgressHUDParallaxModeAlwaysOn);
|
|
BOOL hasParallax = (self.HUDView.motionEffects.count > 0);
|
|
|
|
if (wantsParallax == hasParallax) {
|
|
return;
|
|
}
|
|
|
|
if (!wantsParallax) {
|
|
self.HUDView.motionEffects = @[];
|
|
}
|
|
else {
|
|
UIInterpolatingMotionEffect *x = [[UIInterpolatingMotionEffect alloc] initWithKeyPath:@"center.x" type:UIInterpolatingMotionEffectTypeTiltAlongHorizontalAxis];
|
|
|
|
CGFloat maxMovement = 20.0;
|
|
|
|
x.minimumRelativeValue = @(-maxMovement);
|
|
x.maximumRelativeValue = @(maxMovement);
|
|
|
|
UIInterpolatingMotionEffect *y = [[UIInterpolatingMotionEffect alloc] initWithKeyPath:@"center.y" type:UIInterpolatingMotionEffectTypeTiltAlongVerticalAxis];
|
|
|
|
y.minimumRelativeValue = @(-maxMovement);
|
|
y.maximumRelativeValue = @(maxMovement);
|
|
|
|
self.HUDView.motionEffects = @[x, y];
|
|
}
|
|
}
|
|
|
|
- (void)animationDidFinish:(BOOL)presenting {
|
|
if (presenting) {
|
|
[self cleanUpAfterPresentation];
|
|
}
|
|
else {
|
|
[self cleanUpAfterDismissal];
|
|
}
|
|
}
|
|
|
|
#pragma mark - Getters
|
|
|
|
- (BOOL)isVisible {
|
|
return (self.superview != nil);
|
|
}
|
|
|
|
- (UIVisualEffectView *)blurView {
|
|
if (!_blurView) {
|
|
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(updateMotionOnHUDView) name:UIAccessibilityReduceMotionStatusDidChangeNotification object:nil];
|
|
|
|
UIBlurEffectStyle effect;
|
|
|
|
if (self.style == JGProgressHUDStyleDark) {
|
|
effect = UIBlurEffectStyleDark;
|
|
}
|
|
else if (self.style == JGProgressHUDStyleLight) {
|
|
effect = UIBlurEffectStyleLight;
|
|
}
|
|
else {
|
|
effect = UIBlurEffectStyleExtraLight;
|
|
}
|
|
|
|
UIBlurEffect *blurEffect = [UIBlurEffect effectWithStyle:effect];
|
|
|
|
_blurView = [[UIVisualEffectView alloc] initWithEffect:blurEffect];
|
|
|
|
[self updateMotionOnHUDView];
|
|
|
|
[_blurViewContainer addSubview:_blurView];
|
|
|
|
#if TARGET_OS_IOS
|
|
[self.contentView addGestureRecognizer:[[UITapGestureRecognizer alloc] initWithTarget:self action:@selector(tapped:)]];
|
|
#endif
|
|
}
|
|
|
|
return _blurView;
|
|
}
|
|
|
|
- (UIVisualEffectView *)vibrancyView {
|
|
if (!_vibrancyView) {
|
|
UIVibrancyEffect *vibrancyEffect = (self.vibrancyEnabled ? [UIVibrancyEffect effectForBlurEffect:(UIBlurEffect *)self.blurView.effect] : nil);
|
|
|
|
_vibrancyView = [[UIVisualEffectView alloc] initWithEffect:vibrancyEffect];
|
|
|
|
[self.blurView.contentView addSubview:_vibrancyView];
|
|
}
|
|
|
|
return _vibrancyView;
|
|
}
|
|
|
|
- (UIView *)contentView {
|
|
if (_contentView == nil) {
|
|
_contentView = [[UIView alloc] init];
|
|
[self.vibrancyView.contentView addSubview:_contentView];
|
|
|
|
if (self.effectiveIndicatorView != nil) {
|
|
[self.contentView addSubview:self.effectiveIndicatorView];
|
|
}
|
|
}
|
|
|
|
return _contentView;
|
|
}
|
|
|
|
- (UILabel *)textLabel {
|
|
if (!_textLabel) {
|
|
_textLabel = [[UILabel alloc] init];
|
|
_textLabel.backgroundColor = [UIColor clearColor];
|
|
_textLabel.textColor = (self.style == JGProgressHUDStyleDark ? [UIColor whiteColor] : [UIColor blackColor]);
|
|
_textLabel.textAlignment = NSTextAlignmentCenter;
|
|
#if TARGET_OS_TV
|
|
CGFloat fontSize = 20.0;
|
|
#else
|
|
CGFloat fontSize = 17.0;
|
|
#endif
|
|
if (self.thick) {
|
|
fontSize *= 1.3;
|
|
}
|
|
|
|
_textLabel.font = [UIFont boldSystemFontOfSize:fontSize];
|
|
_textLabel.numberOfLines = 0;
|
|
[_textLabel addObserver:self forKeyPath:@"attributedText" options:(NSKeyValueObservingOptions)kNilOptions context:NULL];
|
|
[_textLabel addObserver:self forKeyPath:@"text" options:(NSKeyValueObservingOptions)kNilOptions context:NULL];
|
|
[_textLabel addObserver:self forKeyPath:@"font" options:(NSKeyValueObservingOptions)kNilOptions context:NULL];
|
|
_textLabel.isAccessibilityElement = YES;
|
|
|
|
[self.contentView addSubview:_textLabel];
|
|
}
|
|
|
|
return _textLabel;
|
|
}
|
|
|
|
- (UILabel *)detailTextLabel {
|
|
if (!_detailTextLabel) {
|
|
_detailTextLabel = [[UILabel alloc] init];
|
|
_detailTextLabel.backgroundColor = [UIColor clearColor];
|
|
_detailTextLabel.textColor = (self.style == JGProgressHUDStyleDark ? [UIColor whiteColor] : [UIColor blackColor]);
|
|
_detailTextLabel.textAlignment = NSTextAlignmentCenter;
|
|
#if TARGET_OS_TV
|
|
CGFloat fontSize = 17.0;
|
|
#else
|
|
CGFloat fontSize = 15.0;
|
|
#endif
|
|
if (self.thick) {
|
|
fontSize *= 1.3;
|
|
}
|
|
|
|
_detailTextLabel.font = [UIFont systemFontOfSize:fontSize];
|
|
_detailTextLabel.numberOfLines = 0;
|
|
[_detailTextLabel addObserver:self forKeyPath:@"attributedText" options:(NSKeyValueObservingOptions)kNilOptions context:NULL];
|
|
[_detailTextLabel addObserver:self forKeyPath:@"text" options:(NSKeyValueObservingOptions)kNilOptions context:NULL];
|
|
[_detailTextLabel addObserver:self forKeyPath:@"font" options:(NSKeyValueObservingOptions)kNilOptions context:NULL];
|
|
_detailTextLabel.isAccessibilityElement = YES;
|
|
|
|
[self.contentView addSubview:_detailTextLabel];
|
|
}
|
|
|
|
return _detailTextLabel;
|
|
}
|
|
|
|
- (JGProgressHUDAnimation *)animation {
|
|
if (!_animation) {
|
|
self.animation = [JGProgressHUDFadeAnimation animation];
|
|
}
|
|
|
|
return _animation;
|
|
}
|
|
|
|
#pragma mark - Setters
|
|
|
|
- (void)setStyle:(JGProgressHUDStyle)style {
|
|
if (self.style == style) {
|
|
return;
|
|
}
|
|
|
|
_style = style;
|
|
|
|
// Update indicator
|
|
[self.effectiveIndicatorView setUpForHUDStyle:self.style vibrancyEnabled:self.vibrancyEnabled];
|
|
|
|
// Update labels
|
|
self.textLabel.textColor = (self.style == JGProgressHUDStyleDark ? [UIColor whiteColor] : [UIColor blackColor]);
|
|
self.detailTextLabel.textColor = (self.style == JGProgressHUDStyleDark ? [UIColor whiteColor] : [UIColor blackColor]);
|
|
|
|
// Update blur effect
|
|
UIBlurEffectStyle effect;
|
|
|
|
if (self.style == JGProgressHUDStyleDark) {
|
|
effect = UIBlurEffectStyleDark;
|
|
}
|
|
else if (self.style == JGProgressHUDStyleLight) {
|
|
effect = UIBlurEffectStyleLight;
|
|
}
|
|
else {
|
|
effect = UIBlurEffectStyleExtraLight;
|
|
}
|
|
|
|
UIBlurEffect *blurEffect = [UIBlurEffect effectWithStyle:effect];
|
|
self.blurView.effect = blurEffect;
|
|
|
|
// Update vibrancy effect
|
|
UIVibrancyEffect *vibrancyEffect = (self.vibrancyEnabled ? [UIVibrancyEffect effectForBlurEffect:(UIBlurEffect *)self.blurView.effect] : nil);
|
|
self.vibrancyView.effect = vibrancyEffect;
|
|
}
|
|
|
|
#if TARGET_OS_TV
|
|
- (void)setWantsFocus:(BOOL)wantsFocus {
|
|
if (self.wantsFocus == wantsFocus) {
|
|
return;
|
|
}
|
|
|
|
_wantsFocus = wantsFocus;
|
|
|
|
self.userInteractionEnabled = self.wantsFocus;
|
|
|
|
[self.targetView setNeedsFocusUpdate];
|
|
}
|
|
#endif
|
|
|
|
- (void)setCornerRadius:(CGFloat)cornerRadius {
|
|
if (fequal(self.cornerRadius, cornerRadius)) {
|
|
return;
|
|
}
|
|
|
|
_cornerRadius = cornerRadius;
|
|
|
|
[self applyCornerRadius];
|
|
}
|
|
|
|
- (void)setShadow:(JGProgressHUDShadow *)shadow {
|
|
if (shadow == self.shadow) {
|
|
return;
|
|
}
|
|
|
|
_shadow = shadow;
|
|
|
|
[self updateShadowViewMask];
|
|
|
|
if (_shadow != nil) {
|
|
_shadowView.layer.shadowColor = _shadow.color.CGColor;
|
|
_shadowView.layer.shadowOffset = _shadow.offset;
|
|
_shadowView.layer.shadowRadius = _shadow.radius;
|
|
|
|
_shadowView.alpha = _shadow.opacity;
|
|
}
|
|
else {
|
|
_shadowView.layer.shadowOffset = CGSizeZero;
|
|
_shadowView.layer.shadowRadius = 0.0;
|
|
|
|
_shadowView.alpha = 0.0;
|
|
}
|
|
}
|
|
|
|
- (void)setAnimation:(JGProgressHUDAnimation *)animation {
|
|
if (_animation == animation) {
|
|
return;
|
|
}
|
|
|
|
_animation.progressHUD = nil;
|
|
|
|
_animation = animation;
|
|
|
|
_animation.progressHUD = self;
|
|
}
|
|
|
|
- (void)setParallaxMode:(JGProgressHUDParallaxMode)parallaxMode {
|
|
if (self.parallaxMode == parallaxMode) {
|
|
return;
|
|
}
|
|
|
|
_parallaxMode = parallaxMode;
|
|
|
|
[self updateMotionOnHUDView];
|
|
}
|
|
|
|
- (void)setPosition:(JGProgressHUDPosition)position {
|
|
if (self.position == position) {
|
|
return;
|
|
}
|
|
|
|
_position = position;
|
|
[self layoutHUD];
|
|
}
|
|
|
|
- (void)setSquare:(BOOL)square {
|
|
if (self.square == square) {
|
|
return;
|
|
}
|
|
|
|
_square = square;
|
|
|
|
[self layoutHUD];
|
|
}
|
|
|
|
- (void)setThick:(BOOL)thick {
|
|
if (self.thick == thick) {
|
|
return;
|
|
}
|
|
|
|
_thick = thick;
|
|
|
|
if (self.thick) {
|
|
self.indicatorView.transform = CGAffineTransformMakeScale(1.5, 1.5);
|
|
}
|
|
else {
|
|
self.indicatorView.transform = CGAffineTransformIdentity;
|
|
}
|
|
|
|
#if TARGET_OS_TV
|
|
CGFloat fontSize = 20.0;
|
|
#else
|
|
CGFloat fontSize = 17.0;
|
|
#endif
|
|
if (self.thick) {
|
|
fontSize *= 1.3;
|
|
}
|
|
|
|
_textLabel.font = [UIFont boldSystemFontOfSize:fontSize];
|
|
#if TARGET_OS_TV
|
|
fontSize = 17.0;
|
|
#else
|
|
fontSize = 15.0;
|
|
#endif
|
|
if (self.thick) {
|
|
fontSize *= 1.3;
|
|
}
|
|
|
|
_detailTextLabel.font = [UIFont systemFontOfSize:fontSize];
|
|
|
|
[self layoutHUD];
|
|
}
|
|
|
|
- (void)setVibrancyEnabled:(BOOL)vibrancyEnabled {
|
|
if (vibrancyEnabled == self.vibrancyEnabled) {
|
|
return;
|
|
}
|
|
|
|
_vibrancyEnabled = vibrancyEnabled;
|
|
|
|
UIVibrancyEffect *vibrancyEffect = (self.vibrancyEnabled ? [UIVibrancyEffect effectForBlurEffect:(UIBlurEffect *)self.blurView.effect] : nil);
|
|
|
|
self.vibrancyView.effect = vibrancyEffect;
|
|
|
|
[self.effectiveIndicatorView setUpForHUDStyle:self.style vibrancyEnabled:self.vibrancyEnabled];
|
|
}
|
|
|
|
- (void)setIndicatorView:(JGProgressHUDIndicatorView *)indicatorView {
|
|
if (self.indicatorView == indicatorView) {
|
|
return;
|
|
}
|
|
|
|
_indicatorView = indicatorView;
|
|
|
|
if (_transitioning) {
|
|
_effectiveIndicatorViewNeedsUpdate = YES;
|
|
return;
|
|
}
|
|
else {
|
|
self.effectiveIndicatorView = self.indicatorView;
|
|
}
|
|
}
|
|
|
|
- (void)setEffectiveIndicatorView:(JGProgressHUDIndicatorView * _Nonnull)effectiveIndicatorView {
|
|
if (self.effectiveIndicatorView == effectiveIndicatorView) {
|
|
return;
|
|
}
|
|
|
|
[UIView performWithoutAnimation:^{
|
|
[self->_effectiveIndicatorView removeFromSuperview];
|
|
self->_effectiveIndicatorView = effectiveIndicatorView;
|
|
|
|
if (self.indicatorView != nil) {
|
|
[self.effectiveIndicatorView setUpForHUDStyle:self.style vibrancyEnabled:self.vibrancyEnabled];
|
|
|
|
if (self.thick) {
|
|
self.effectiveIndicatorView.transform = CGAffineTransformMakeScale(1.5, 1.5);
|
|
}
|
|
else {
|
|
self.effectiveIndicatorView.transform = CGAffineTransformIdentity;
|
|
}
|
|
|
|
[self.contentView addSubview:self.effectiveIndicatorView];
|
|
}
|
|
}];
|
|
|
|
[self layoutHUD];
|
|
}
|
|
|
|
- (void)layoutMarginsDidChange {
|
|
[super layoutMarginsDidChange];
|
|
|
|
[self layoutHUD];
|
|
}
|
|
|
|
- (void)setContentInsets:(UIEdgeInsets)contentInsets {
|
|
if (UIEdgeInsetsEqualToEdgeInsets(self.contentInsets, contentInsets)) {
|
|
return;
|
|
}
|
|
|
|
_contentInsets = contentInsets;
|
|
|
|
[self layoutHUD];
|
|
}
|
|
|
|
- (void)setProgress:(float)progress {
|
|
[self setProgress:progress animated:NO];
|
|
}
|
|
|
|
- (void)setProgress:(float)progress animated:(BOOL)animated {
|
|
if (fequal(self.progress, progress)) {
|
|
return;
|
|
}
|
|
|
|
_progress = progress;
|
|
|
|
[self.effectiveIndicatorView setProgress:progress animated:animated];
|
|
}
|
|
|
|
#pragma mark - Overrides
|
|
|
|
- (void)traitCollectionDidChange:(UITraitCollection *)previousTraitCollection {
|
|
[super traitCollectionDidChange:previousTraitCollection];
|
|
|
|
if (@available(iOS 12.0, tvOS 10.0, *)) {
|
|
if (_automaticStyle) {
|
|
self.style = JGProgressHUDStyleFromUIUserInterfaceStyle(self.traitCollection.userInterfaceStyle);
|
|
}
|
|
}
|
|
}
|
|
|
|
#if TARGET_OS_IOS
|
|
- (UIView *)hitTest:(CGPoint)point withEvent:(UIEvent *)event {
|
|
if (self.interactionType == JGProgressHUDInteractionTypeBlockNoTouches) {
|
|
return nil;
|
|
}
|
|
else {
|
|
UIView *view = [super hitTest:point withEvent:event];
|
|
|
|
if (self.interactionType == JGProgressHUDInteractionTypeBlockAllTouches) {
|
|
return view;
|
|
}
|
|
else if (self.interactionType == JGProgressHUDInteractionTypeBlockTouchesOnHUDView && view != self) {
|
|
return view;
|
|
}
|
|
|
|
return nil;
|
|
}
|
|
}
|
|
#elif TARGET_OS_TV
|
|
- (NSArray<id<UIFocusEnvironment>> *)preferredFocusEnvironments {
|
|
return @[self];
|
|
}
|
|
|
|
- (UIView *)preferredFocusedView {
|
|
return nil;
|
|
}
|
|
|
|
- (BOOL)canBecomeFocused {
|
|
return self.wantsFocus;
|
|
}
|
|
#endif
|
|
|
|
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context {
|
|
if (object == _textLabel || object == _detailTextLabel) {
|
|
[self layoutHUD];
|
|
}
|
|
else {
|
|
[super observeValueForKeyPath:keyPath ofObject:object change:change context:context];
|
|
}
|
|
}
|
|
|
|
- (void)dealloc {
|
|
[self removeObservers];
|
|
|
|
[_textLabel removeObserver:self forKeyPath:@"attributedText"];
|
|
[_textLabel removeObserver:self forKeyPath:@"text"];
|
|
[_textLabel removeObserver:self forKeyPath:@"font"];
|
|
|
|
[_detailTextLabel removeObserver:self forKeyPath:@"attributedText"];
|
|
[_detailTextLabel removeObserver:self forKeyPath:@"text"];
|
|
[_detailTextLabel removeObserver:self forKeyPath:@"font"];
|
|
}
|
|
|
|
- (void)removeObservers {
|
|
[[NSNotificationCenter defaultCenter] removeObserver:self name:UIAccessibilityReduceMotionStatusDidChangeNotification object:nil];
|
|
|
|
#if TARGET_OS_IOS
|
|
[[NSNotificationCenter defaultCenter] removeObserver:self name:UIKeyboardWillChangeFrameNotification object:nil];
|
|
[[NSNotificationCenter defaultCenter] removeObserver:self name:UIKeyboardDidChangeFrameNotification object:nil];
|
|
#endif
|
|
}
|
|
|
|
@end
|
|
|
|
@implementation JGProgressHUD (HUDManagement)
|
|
|
|
+ (NSArray *)allProgressHUDsInView:(UIView *)view {
|
|
NSMutableArray *HUDs = [NSMutableArray array];
|
|
|
|
for (UIView *v in view.subviews) {
|
|
if ([v isKindOfClass:[JGProgressHUD class]]) {
|
|
[HUDs addObject:v];
|
|
}
|
|
}
|
|
|
|
return HUDs.copy;
|
|
}
|
|
|
|
+ (NSMutableArray *)_allProgressHUDsInViewHierarchy:(UIView *)view {
|
|
NSMutableArray *HUDs = [NSMutableArray array];
|
|
|
|
for (UIView *v in view.subviews) {
|
|
if ([v isKindOfClass:[JGProgressHUD class]]) {
|
|
[HUDs addObject:v];
|
|
}
|
|
else {
|
|
[HUDs addObjectsFromArray:[self _allProgressHUDsInViewHierarchy:v]];
|
|
}
|
|
}
|
|
|
|
return HUDs;
|
|
}
|
|
|
|
+ (NSArray *)allProgressHUDsInViewHierarchy:(UIView *)view {
|
|
return [self _allProgressHUDsInViewHierarchy:view].copy;
|
|
}
|
|
|
|
@end
|