#import #import /* SMCalloutView ------------- Created by Nick Farina (nfarina@gmail.com) Version 2.1.5 */ /// options for which directions the callout is allowed to "point" in. typedef NS_OPTIONS(NSUInteger, MGLSMCalloutArrowDirection) { MGLSMCalloutArrowDirectionUp = 1 << 0, MGLSMCalloutArrowDirectionDown = 1 << 1, MGLSMCalloutArrowDirectionAny = MGLSMCalloutArrowDirectionUp | MGLSMCalloutArrowDirectionDown }; /// options for the callout present/dismiss animation typedef NS_ENUM(NSInteger, MGLSMCalloutAnimation) { /// the "bounce" animation we all know and love from @c UIAlertView MGLSMCalloutAnimationBounce, /// a simple fade in or out MGLSMCalloutAnimationFade, /// grow or shrink linearly, like in the iPad Calendar app MGLSMCalloutAnimationStretch }; NS_ASSUME_NONNULL_BEGIN /// when delaying our popup in order to scroll content into view, you can use this amount to match the /// animation duration of UIScrollView when using @c -setContentOffset:animated. extern NSTimeInterval const kMGLSMCalloutViewRepositionDelayForUIScrollView; @protocol MGLSMCalloutViewDelegate; @class MGLSMCalloutBackgroundView; // // Callout view. // // iOS 10+ expects CAAnimationDelegate to be set explicitly. #if __IPHONE_OS_VERSION_MAX_ALLOWED < 100000 @interface MGLSMCalloutView : UIView #else @interface MGLSMCalloutView : UIView #endif @property (nonatomic, weak, nullable) id delegate; /// title/titleView relationship mimics UINavigationBar. @property (nonatomic, copy, nullable) NSString *title; @property (nonatomic, copy, nullable) NSString *subtitle; /// Left accessory view for the call out @property (nonatomic, strong, nullable) UIView *leftAccessoryView; /// Right accessoty view for the call out @property (nonatomic, strong, nullable) UIView *rightAccessoryView; /// Default @c SMCalloutArrowDirectionDown @property (nonatomic, assign) MGLSMCalloutArrowDirection permittedArrowDirection; /// The current arrow direction @property (nonatomic, readonly) MGLSMCalloutArrowDirection currentArrowDirection; /// if the @c UIView you're constraining to has portions that are overlapped by nav bar, tab bar, etc. you'll need to tell us those insets. @property (nonatomic, assign) UIEdgeInsets constrainedInsets; /// default is @c SMCalloutMaskedBackgroundView, or @c SMCalloutDrawnBackgroundView when using @c SMClassicCalloutView @property (nonatomic, strong) MGLSMCalloutBackgroundView *backgroundView; /** @brief Custom title view. @disucssion Keep in mind that @c SMCalloutView calls @c -sizeThatFits on titleView/subtitleView if defined, so your view may be resized as a result of that (especially if you're using @c UILabel/UITextField). You may want to subclass and override @c -sizeThatFits, or just wrap your view in a "generic" @c UIView if you do not want it to be auto-sized. @warning If this is set, the respective @c title property will be ignored. */ @property (nonatomic, strong, nullable) UIView *titleView; /** @brief Custom subtitle view. @discussion Keep in mind that @c SMCalloutView calls @c -sizeThatFits on subtitleView if defined, so your view may be resized as a result of that (especially if you're using @c UILabel/UITextField). You may want to subclass and override @c -sizeThatFits, or just wrap your view in a "generic" @c UIView if you do not want it to be auto-sized. @warning If this is set, the respective @c subtitle property will be ignored. */ @property (nonatomic, strong, nullable) UIView *subtitleView; /// Custom "content" view that can be any width/height. If this is set, title/subtitle/titleView/subtitleView are all ignored. @property (nonatomic, retain, nullable) UIView *contentView; /// Custom content view margin @property (nonatomic, assign) UIEdgeInsets contentViewInset; /// calloutOffset is the offset in screen points from the top-middle of the target view, where the anchor of the callout should be shown. @property (nonatomic, assign) CGPoint calloutOffset; /// default SMCalloutAnimationBounce, SMCalloutAnimationFade respectively @property (nonatomic, assign) MGLSMCalloutAnimation presentAnimation, dismissAnimation; /// Returns a new instance of SMCalloutView if running on iOS 7 or better, otherwise a new instance of SMClassicCalloutView if available. + (MGLSMCalloutView *)platformCalloutView; /** @brief Presents a callout view by adding it to "inView" and pointing at the given rect of inView's bounds. @discussion Constrains the callout to the bounds of the given view. Optionally scrolls the given rect into view (plus margins) if @c -delegate is set and responds to @c -delayForRepositionWithSize. @param rect @c CGRect to present the view from @param view view to 'constrain' the @c constrainedView to @param constrainedView @c UIView to be constrainted in @c view @param animated @c BOOL if presentation should be animated */ - (void)presentCalloutFromRect:(CGRect)rect inView:(UIView *)view constrainedToView:(UIView *)constrainedView animated:(BOOL)animated; /** @brief Present a callout layer in the `layer` and pointing at the given rect of the `layer` bounds @discussion Same as the view-based presentation, but inserts the callout into a CALayer hierarchy instead. @note Be aware that you'll have to direct your own touches to any accessory views, since CALayer doesn't relay touch events. @param rect @c CGRect to present the view from @param layer layer to 'constrain' the @c constrainedLayer to @param constrainedLayer @c CALayer to be constrained in @c layer @param animated @c BOOL if presentation should be animated */ - (void)presentCalloutFromRect:(CGRect)rect inLayer:(CALayer *)layer constrainedToLayer:(CALayer *)constrainedLayer animated:(BOOL)animated; /** Dismiss the callout view @param animated @c BOOL if dismissal should be animated */ - (void)dismissCalloutAnimated:(BOOL)animated; /// For subclassers. You can override this method to provide your own custom animation for presenting/dismissing the callout. - (CAAnimation *)animationWithType:(MGLSMCalloutAnimation)type presenting:(BOOL)presenting; @end // // Background view - default draws the iOS 7 system background style (translucent white with rounded arrow). // /// Abstract base class @interface MGLSMCalloutBackgroundView : UIView /// indicates where the tip of the arrow should be drawn, as a pixel offset @property (nonatomic, assign) CGPoint arrowPoint; /// will be set by the callout when the callout is in a highlighted state @property (nonatomic, assign) BOOL highlighted; /// returns an optional layer whose contents should mask the callout view's contents (not honored by @c SMClassicCalloutView ) @property (nonatomic, assign) CALayer *contentMask; /// height of the callout "arrow" @property (nonatomic, assign) CGFloat anchorHeight; /// the smallest possible distance from the edge of our control to the "tip" of the anchor, from either left or right @property (nonatomic, assign) CGFloat anchorMargin; @end /// Default for iOS 7, this reproduces the "masked" behavior of the iOS 7-style callout view. /// Accessories are masked by the shape of the callout (including the arrow itself). @interface MGLSMCalloutMaskedBackgroundView : MGLSMCalloutBackgroundView @end // // Delegate methods // @protocol MGLSMCalloutViewDelegate @optional /// Controls whether the callout "highlights" when pressed. default YES. You must also respond to @c -calloutViewClicked below. /// Not honored by @c SMClassicCalloutView. - (BOOL)calloutViewShouldHighlight:(MGLSMCalloutView *)calloutView; /// Called when the callout view is clicked. Not honored by @c SMClassicCalloutView. - (void)calloutViewClicked:(MGLSMCalloutView *)calloutView; /** Called when the callout view detects that it will be outside the constrained view when it appears, or if the target rect was already outside the constrained view. You can implement this selector to respond to this situation by repositioning your content first in order to make everything visible. The @c CGSize passed is the calculated offset necessary to make everything visible (plus a nice margin). It expects you to return the amount of time you need to reposition things so the popup can be delayed. Typically you would return @c kSMCalloutViewRepositionDelayForUIScrollView if you're repositioning by calling @c [UIScrollView @c setContentOffset:animated:]. @param calloutView the @c SMCalloutView to reposition @param offset caluclated offset necessary to make everything visible @returns @c NSTimeInterval to delay the repositioning */ - (NSTimeInterval)calloutView:(MGLSMCalloutView *)calloutView delayForRepositionWithSize:(CGSize)offset; /// Called before the callout view appears on screen, or before the appearance animation will start. - (void)calloutViewWillAppear:(MGLSMCalloutView *)calloutView; /// Called after the callout view appears on screen, or after the appearance animation is complete. - (void)calloutViewDidAppear:(MGLSMCalloutView *)calloutView; /// Called before the callout view is removed from the screen, or before the disappearance animation is complete. - (void)calloutViewWillDisappear:(MGLSMCalloutView *)calloutView; /// Called after the callout view is removed from the screen, or after the disappearance animation is complete. - (void)calloutViewDidDisappear:(MGLSMCalloutView *)calloutView; NS_ASSUME_NONNULL_END @end