-
Notifications
You must be signed in to change notification settings - Fork 0
/
MMDrawerController.h
executable file
·455 lines (331 loc) · 24.2 KB
/
MMDrawerController.h
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
// Copyright (c) 2013 Mutual Mobile (http://mutualmobile.com/)
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in
// all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
// THE SOFTWARE.
#import <UIKit/UIKit.h>
/**
`MMDrawerController` is a side drawer navigation container view controller designed to support the growing number of applications that leverage the side drawer paradigm. This library is designed to exclusively support side drawer navigation in light-weight, focused approach.
## Creating a MMDrawerController
`MMDrawerController` is a container view controller, similiar to `UINavigationController` or `UITabBarController`, with up to three child view controllers - Center, LeftDrawer, and RightDrawer. To create a `MMDrawerController`, you must first instantiate the drawer view controllers and the initial center controller, then call one of the init methods listed in this class.
## Handling a UINavigationController as the centerViewController
`MMDrawerController` automatically supports handling a `UINavigationController` as the `centerViewController`, and will correctly handle the proper gestures on each view (the navigation bar view as well as the content view for the visible view controller). Note that while this library does support other container view controllers, the open/close gestures are not customized to support them.
## Accessing MMDrawerController from the Child View Controller
You can leverage the category class (UIViewController+MMDrawerController) included with this library to access information about the parent `MMDrawerController`. Note that if you are contained within a UINavigationController, the `drawerContainerViewController` will still return the proper reference to the `drawerContainerViewController` parent, even though it is not the direct parent. Refer to the documentation included with the category for more information.
## How MMDrawerOpenCenterInteractionMode is handled
`MMDrawerOpenCenterInteractionMode` controls how the user should be able to interact with the center view controller when either drawer is open. By default, this is set to `MMDrawerOpenCenterInteractionModeNavigationBarOnly`, which allows the user to interact with UINavigationBarItems while either drawer is open (typicaly used to click the menu button to close). If you set the interaction mode to `MMDrawerOpenCenterInteractionModeNone`, no items within the center view will be able to be interacted with while a drawer is open. Note that this setting has no effect at all on the `MMCloseDrawerGestureMode`.
## How Open/Close Gestures are handled
Two gestures are added to every instance of a drawer controller, one for pan and one for touch. `MMDrawerController` is the delegate for each of the gesture recoginzers, and determines if a touch should be sent to the appropriate gesture when a touch is detected compared with the masks set for open and close gestures and the state of the drawer controller.
## Integrating with State Restoration
In order to opt in to state restoration for `MMDrawerController`, you must set the `restorationIdentifier` of your drawer controller. Instances of your centerViewController, leftDrawerViewController and rightDrawerViewController must also be configured with their own `restorationIdentifier` (and optionally a restorationClass) if you intend for those to be restored as well. If your MMDrawerController had an open drawer when your app was sent to the background, that state will also be restored.
## What this library doesn't do.
This library is not meant for:
- Top or bottom drawer views
- Displaying both drawers at one time
- Displaying a minimum drawer width
- Support container view controllers other than `UINavigationController` as the center view controller.
*/
typedef NS_ENUM(NSInteger,MMDrawerSide){
MMDrawerSideNone = 0,
MMDrawerSideLeft,
MMDrawerSideRight,
};
typedef NS_OPTIONS(NSInteger, MMOpenDrawerGestureMode) {
MMOpenDrawerGestureModeNone = 0,
MMOpenDrawerGestureModePanningNavigationBar = 1 << 1,
MMOpenDrawerGestureModePanningCenterView = 1 << 2,
MMOpenDrawerGestureModeBezelPanningCenterView = 1 << 3,
MMOpenDrawerGestureModeCustom = 1 << 4,
MMOpenDrawerGestureModeAll = MMOpenDrawerGestureModePanningNavigationBar |
MMOpenDrawerGestureModePanningCenterView |
MMOpenDrawerGestureModeBezelPanningCenterView |
MMOpenDrawerGestureModeCustom,
};
typedef NS_OPTIONS(NSInteger, MMCloseDrawerGestureMode) {
MMCloseDrawerGestureModeNone = 0,
MMCloseDrawerGestureModePanningNavigationBar = 1 << 1,
MMCloseDrawerGestureModePanningCenterView = 1 << 2,
MMCloseDrawerGestureModeBezelPanningCenterView = 1 << 3,
MMCloseDrawerGestureModeTapNavigationBar = 1 << 4,
MMCloseDrawerGestureModeTapCenterView = 1 << 5,
MMCloseDrawerGestureModePanningDrawerView = 1 << 6,
MMCloseDrawerGestureModeCustom = 1 << 7,
MMCloseDrawerGestureModeAll = MMCloseDrawerGestureModePanningNavigationBar |
MMCloseDrawerGestureModePanningCenterView |
MMCloseDrawerGestureModeBezelPanningCenterView |
MMCloseDrawerGestureModeTapNavigationBar |
MMCloseDrawerGestureModeTapCenterView |
MMCloseDrawerGestureModePanningDrawerView |
MMCloseDrawerGestureModeCustom,
};
typedef NS_ENUM(NSInteger, MMDrawerOpenCenterInteractionMode) {
MMDrawerOpenCenterInteractionModeNone,
MMDrawerOpenCenterInteractionModeFull,
MMDrawerOpenCenterInteractionModeNavigationBarOnly,
};
@class MMDrawerController;
typedef void (^MMDrawerControllerDrawerVisualStateBlock)(MMDrawerController * drawerController, MMDrawerSide drawerSide, CGFloat percentVisible);
@interface MMDrawerController : UIViewController
///---------------------------------------
/// @name Accessing Drawer Container View Controller Properties
///---------------------------------------
/**
The center view controller.
This can only be set via the init methods, as well as the `setNewCenterViewController:...` methods. The size of this view controller will automatically be set to the size of the drawer container view controller, and it's position is modified from within this class. Do not modify the frame externally.
*/
@property (nonatomic, strong) UIViewController * centerViewController;
/**
The left drawer view controller.
The size of this view controller is managed within this class, and is automatically set to the appropriate size based on the `maximumLeftDrawerWidth`. Do not modify the frame externally.
*/
@property (nonatomic, strong) UIViewController * leftDrawerViewController;
/**
The right drawer view controller.
The size of this view controller is managed within this class, and is automatically set to the appropriate size based on the `maximumRightDrawerWidth`. Do not modify the frame externally.
*/
@property (nonatomic, strong) UIViewController * rightDrawerViewController;
/**
The maximum width of the `leftDrawerViewController`.
By default, this is set to 280. If the `leftDrawerViewController` is nil, this property will return 0.0;
*/
@property (nonatomic, assign) CGFloat maximumLeftDrawerWidth;
/**
The maximum width of the `rightDrawerViewController`.
By default, this is set to 280. If the `rightDrawerViewController` is nil, this property will return 0.0;
*/
@property (nonatomic, assign) CGFloat maximumRightDrawerWidth;
/**
The visible width of the `leftDrawerViewController`.
Note this value can be greater than `maximumLeftDrawerWidth` during the full close animation when setting a new center view controller;
*/
@property (nonatomic, assign, readonly) CGFloat visibleLeftDrawerWidth;
/**
The visible width of the `rightDrawerViewController`.
Note this value can be greater than `maximumRightDrawerWidth` during the full close animation when setting a new center view controller;
*/
@property (nonatomic, assign, readonly) CGFloat visibleRightDrawerWidth;
/**
The animation velocity of the open and close methods, measured in points per second.
By default, this is set to 840 points per second (three times the default drawer width), meaning it takes 1/3 of a second for the `centerViewController` to open/close across the default drawer width. Note that there is a minimum .1 second duration for built in animations, to account for small distance animations.
*/
@property (nonatomic, assign) CGFloat animationVelocity;
/**
A boolean that determines whether or not the panning gesture will "hard-stop" at the maximum width for a given drawer side.
By default, this value is set to YES. Enabling `shouldStretchDrawer` will give the pan a gradual asymptotic stopping point much like `UIScrollView` behaves. Note that if this value is set to YES, the `drawerVisualStateBlock` can be passed a `percentVisible` greater than 1.0, so be sure to handle that case appropriately.
*/
@property (nonatomic, assign) BOOL shouldStretchDrawer;
/**
The current open side of the drawer.
Note this value will change as soon as a pan gesture opens a drawer, or when a open/close animation is finished.
*/
@property (nonatomic, assign, readonly) MMDrawerSide openSide;
/**
How a user is allowed to open a drawer using gestures.
By default, this is set to `MMOpenDrawerGestureModeNone`. Note these gestures may affect user interaction with the `centerViewController`, so be sure to use appropriately.
*/
@property (nonatomic, assign) MMOpenDrawerGestureMode openDrawerGestureModeMask;
/**
How a user is allowed to close a drawer.
By default, this is set to `MMCloseDrawerGestureModeNone`. Note these gestures may affect user interaction with the `centerViewController`, so be sure to use appropriately.
*/
@property (nonatomic, assign) MMCloseDrawerGestureMode closeDrawerGestureModeMask;
/**
The value determining if the user can interact with the `centerViewController` when a side drawer is open.
By default, it is `MMDrawerOpenCenterInteractionModeNavigationBarOnly`, meaning that the user can only interact with the buttons on the `UINavigationBar`, if the center view controller is a `UINavigationController`. Otherwise, the user cannot interact with any other center view controller elements.
*/
@property (nonatomic, assign) MMDrawerOpenCenterInteractionMode centerHiddenInteractionMode;
/**
The flag determining if a shadow should be drawn off of `centerViewController` when a drawer is open.
By default, this is set to YES.
*/
@property (nonatomic, assign) BOOL showsShadow;
/**
The shadow radius of `centerViewController` when a drawer is open.
By default, this is set to 10.0f;
*/
@property (nonatomic, assign) CGFloat shadowRadius;
/**
The shadow opacity of `centerViewController` when a drawer is open.
By default, this is set to 0.8f;
*/
@property (nonatomic, assign) CGFloat shadowOpacity;
/**
The shadow offset of `centerViewController` when a drawer is open.
By default, this is set to (0, -3);
*/
@property (nonatomic, assign) CGSize shadowOffset;
/**
The color of the shadow drawn off of 'centerViewController` when a drawer is open.
By default, this is set to the systme default (opaque black).
*/
@property (nonatomic, strong) UIColor * shadowColor;
/**
The flag determining if a custom background view should appear beneath the status bar, forcing the child content to be drawn lower than the status bar.
By default, this is set to NO.
*/
@property (nonatomic, assign) BOOL showsStatusBarBackgroundView;
/**
The color of the status bar background view if `showsStatusBarBackgroundView` is set to YES.
By default, this is set `[UIColor blackColor]`.
*/
@property (nonatomic, strong) UIColor * statusBarViewBackgroundColor;
/**
The value determining panning range of centerView's bezel if the user can open drawer with 'MMOpenDrawerGestureModeBezelPanningCenterView' or close drawer with 'MMCloseDrawerGestureModeBezelPanningCenterView' .
By default, this is set 20.0f.
*/
@property (nonatomic, assign) CGFloat bezelPanningCenterViewRange;
/**
The value determining if the user can open or close drawer with panGesture velocity.
By default, this is set 200.0f.
*/
@property (nonatomic, assign) CGFloat panVelocityXAnimationThreshold;
///---------------------------------------
/// @name Initializing a `MMDrawerController`
///---------------------------------------
/**
Creates and initializes an `MMDrawerController` object with the specified center view controller, left drawer view controller, and right drawer view controller.
@param centerViewController The center view controller. This argument must not be `nil`.
@param leftDrawerViewController The left drawer view controller.
@param rightDrawerViewController The right drawer controller.
@return The newly-initialized drawer container view controller.
*/
-(instancetype)initWithCenterViewController:(UIViewController *)centerViewController leftDrawerViewController:(UIViewController *)leftDrawerViewController rightDrawerViewController:(UIViewController *)rightDrawerViewController;
/**
Creates and initializes an `MMDrawerController` object with the specified center view controller, left drawer view controller.
@param centerViewController The center view controller. This argument must not be `nil`.
@param leftDrawerViewController The left drawer view controller.
@return The newly-initialized drawer container view controller.
*/
-(instancetype)initWithCenterViewController:(UIViewController *)centerViewController leftDrawerViewController:(UIViewController *)leftDrawerViewController;
/**
Creates and initializes an `MMDrawerController` object with the specified center view controller, right drawer view controller.
@param centerViewController The center view controller. This argument must not be `nil`.
@param rightDrawerViewController The right drawer controller.
@return The newly-initialized drawer container view controller.
*/
-(instancetype)initWithCenterViewController:(UIViewController *)centerViewController rightDrawerViewController:(UIViewController *)rightDrawerViewController;
///---------------------------------------
/// @name Opening and Closing a Drawer
///---------------------------------------
/**
Toggles the drawer open/closed based on the `drawer` passed in.
Note that if you attempt to toggle a drawer closed while the other is open, nothing will happen. For example, if you pass in MMDrawerSideLeft, but the right drawer is open, nothing will happen. In addition, the completion block will be called with the finished flag set to NO.
@param drawerSide The `MMDrawerSide` to toggle. This value cannot be `MMDrawerSideNone`.
@param animated Determines whether the `drawer` should be toggle animated.
@param completion The block that is called when the toggle is complete, or if no toggle took place at all.
*/
-(void)toggleDrawerSide:(MMDrawerSide)drawerSide animated:(BOOL)animated completion:(void(^)(BOOL finished))completion;
/**
Closes the open drawer.
@param animated Determines whether the drawer side should be closed animated
@param completion The block that is called when the close is complete
*/
-(void)closeDrawerAnimated:(BOOL)animated completion:(void(^)(BOOL finished))completion;
/**
Opens the `drawer` passed in.
@param drawerSide The `MMDrawerSide` to open. This value cannot be `MMDrawerSideNone`.
@param animated Determines whether the `drawer` should be open animated.
@param completion The block that is called when the toggle is open.
*/
-(void)openDrawerSide:(MMDrawerSide)drawerSide animated:(BOOL)animated completion:(void(^)(BOOL finished))completion;
///---------------------------------------
/// @name Setting a new Center View Controller
///---------------------------------------
/**
Sets the new `centerViewController`.
This sets the view controller and will automatically adjust the frame based on the current state of the drawer controller. If `closeAnimated` is YES, it will immediately change the center view controller, and close the drawer from its current position.
@param centerViewController The new `centerViewController`.
@param closeAnimated Determines whether the drawer should be closed with an animation.
@param completion The block called when the animation is finsihed.
*/
-(void)setCenterViewController:(UIViewController *)centerViewController withCloseAnimation:(BOOL)closeAnimated completion:(void(^)(BOOL finished))completion;
/**
Sets the new `centerViewController`.
This sets the view controller and will automatically adjust the frame based on the current state of the drawer controller. If `closeFullAnimated` is YES, the current center view controller will animate off the screen, the new center view controller will then be set, followed by the drawer closing across the full width of the screen.
@param newCenterViewController The new `centerViewController`.
@param fullCloseAnimated Determines whether the drawer should be closed with an animation.
@param completion The block called when the animation is finsihed.
*/
-(void)setCenterViewController:(UIViewController *)newCenterViewController withFullCloseAnimation:(BOOL)fullCloseAnimated completion:(void(^)(BOOL finished))completion;
///---------------------------------------
/// @name Animating the Width of a Drawer
///---------------------------------------
/**
Sets the maximum width of the left drawer view controller.
If the drawer is open, and `animated` is YES, it will animate the drawer frame as well as adjust the center view controller. If the drawer is not open, this change will take place immediately.
@param width The new width of left drawer view controller. This must be greater than zero.
@param animated Determines whether the drawer should be adjusted with an animation.
@param completion The block called when the animation is finished.
*/
-(void)setMaximumLeftDrawerWidth:(CGFloat)width animated:(BOOL)animated completion:(void(^)(BOOL finished))completion;
/**
Sets the maximum width of the right drawer view controller.
If the drawer is open, and `animated` is YES, it will animate the drawer frame as well as adjust the center view controller. If the drawer is not open, this change will take place immediately.
@param width The new width of right drawer view controller. This must be greater than zero.
@param animated Determines whether the drawer should be adjusted with an animation.
@param completion The block called when the animation is finished.
*/
-(void)setMaximumRightDrawerWidth:(CGFloat)width animated:(BOOL)animated completion:(void(^)(BOOL finished))completion;
///---------------------------------------
/// @name Previewing a Drawer
///---------------------------------------
/**
Bounce preview for the specified `drawerSide` a distance of 40 points.
@param drawerSide The drawer to preview. This value cannot be `MMDrawerSideNone`.
@param completion The block called when the animation is finsihed.
*/
-(void)bouncePreviewForDrawerSide:(MMDrawerSide)drawerSide completion:(void(^)(BOOL finished))completion;
/**
Bounce preview for the specified `drawerSide`.
@param drawerSide The drawer side to preview. This value cannot be `MMDrawerSideNone`.
@param distance The distance to bounce.
@param completion The block called when the animation is finsihed.
*/
-(void)bouncePreviewForDrawerSide:(MMDrawerSide)drawerSide distance:(CGFloat)distance completion:(void(^)(BOOL finished))completion;
///---------------------------------------
/// @name Custom Drawer Animations
///---------------------------------------
/**
Sets a callback to be called when a drawer visual state needs to be updated.
This block is responsible for updating the drawer's view state, and the drawer controller will handle animating to that state from the current state. This block will be called when the drawer is opened or closed, as well when the user is panning the drawer. This block is not responsible for doing animations directly, but instead just updating the state of the properies (such as alpha, anchor point, transform, etc). Note that if `shouldStretchDrawer` is set to YES, it is possible for `percentVisible` to be greater than 1.0. If `shouldStretchDrawer` is set to NO, `percentVisible` will never be greater than 1.0.
Note that when the drawer is finished opening or closing, the side drawer controller view will be reset with the following properies:
- alpha: 1.0
- transform: CATransform3DIdentity
- anchorPoint: (0.5,0.5)
@param drawerVisualStateBlock A block object to be called that allows the implementer to update visual state properties on the drawer. `percentVisible` represents the amount of the drawer space that is current visible, with drawer space being defined as the edge of the screen to the maxmimum drawer width. Note that you do have access to the drawerController, which will allow you to update things like the anchor point of the side drawer layer.
*/
-(void)setDrawerVisualStateBlock:(void(^)(MMDrawerController * drawerController, MMDrawerSide drawerSide, CGFloat percentVisible))drawerVisualStateBlock;
///---------------------------------------
/// @name Gesture Completion Handling
///---------------------------------------
/**
Sets a callback to be called when a gesture has been completed.
This block is called when a gesture action has been completed. You can query the `openSide` of the `drawerController` to determine what the new state of the drawer is.
@param gestureCompletionBlock A block object to be called that allows the implementer be notified when a gesture action has been completed.
*/
-(void)setGestureCompletionBlock:(void(^)(MMDrawerController * drawerController, UIGestureRecognizer * gesture))gestureCompletionBlock;
///---------------------------------------
/// @name Custom Gesture Handler
///---------------------------------------
/**
Sets a callback to be called to determine if a UIGestureRecognizer should recieve the given UITouch.
This block provides a way to allow a gesture to be recognized with custom logic. For example, you may have a certain part of your view that should accept a pan gesture recognizer to open the drawer, but not another a part. If you return YES, the gesture is recognized and the appropriate action is taken. This provides similar support to how Facebook allows you to pan on the background view of the main table view, but not the content itself. You can inspect the `openSide` property of the `drawerController` to determine the current state of the drawer, and apply the appropriate logic within your block.
Note that either `openDrawerGestureModeMask` must contain `MMOpenDrawerGestureModeCustom`, or `closeDrawerGestureModeMask` must contain `MMCloseDrawerGestureModeCustom` for this block to be consulted.
@param gestureShouldRecognizeTouchBlock A block object to be called to determine if the given `touch` should be recognized by the given gesture.
*/
-(void)setGestureShouldRecognizeTouchBlock:(BOOL(^)(MMDrawerController * drawerController, UIGestureRecognizer * gesture, UITouch * touch))gestureShouldRecognizeTouchBlock;
@end