iOS中仿QQ侧滑菜单功能

发布时间 - 2026-01-11 02:25:43    点击率:

UITabBarController做QQ侧滑菜单效果:

首先要了解UITabBarController的层级结构:

UITabBarController加载的其它UIViewController的View都是被添加在UITransitionView上(这是一个私有API),UITransitionView在self.view的0层,UITabBar在的第一层。

所以我的思路是这样的:

UITransitionView与UITabBar转移到一个新的View1上去,作为滑动的部分;

在View1与self.view之间再添加一个View2,作为菜单的容器;

给View1添加相应的手势响应;

代码:

#import <UIKit/UIKit.h>
@protocol SlideTab_VC_TCVDelegate <NSObject>
@optional
-(void)didOpenMenu:(UIView*)menu;
-(void)didCloseMenu:(UIView*)menu;
@end
@interface SlideTab_VC : UITabBarController
@property(strong, nonatomic)UIView *mMenuV;
@property(weak, nonatomic)id <SlideTab_VC_TCVDelegate> mDelegate;
-(void)openMenu;
-(void)closeMenu;
@end
#import "SlideTab_VC.h"
#define DEVICE_W [UIScreen mainScreen].bounds.size.width
@interface SlideTab_VC ()<UITabBarDelegate>{
 CGFloat _centerMaxX;
}
@property(strong, nonatomic)UIView *mTransitionView;
@property(strong, nonatomic)UITapGestureRecognizer *mTapGester;
@property(assign, nonatomic)BOOL mMenuIsOpen;
@property(strong, nonatomic)UITabBar *mTabBar;
@end
@implementation SlideTab_VC
- (void)viewDidLoad {
 [super viewDidLoad];
 // Do any additional setup after loading the view.
 self.view.backgroundColor = [UIColor purpleColor];
 _mMenuIsOpen = NO;
 _centerMaxX = DEVICE_W*3/2-80.0f;
 [self initMTransitionView];
 [self addGestureForMTransitionView];
}
#pragma mark 菜单懒加载
-(void)setMMenuV:(UIView *)mMenuV{
 if (mMenuV!=nil) {
  _mMenuV = mMenuV;
  [self.view insertSubview:_mMenuV atIndex:0];
 }
}
#pragma mark 打开菜单
-(void)openMenu{
 CGPoint center = self.mTransitionView.center;
 center.x = _centerMaxX;
 [UIView animateWithDuration:0.15f animations:^{
  self.mTransitionView.center = center;
  if (_mDelegate != nil&& [_mDelegate respondsToSelector:@selector(didOpenMenu:)]) {
  [_mDelegate didOpenMenu:_mMenuV];
 }
 }];
 [self mTransitionSubViewsEnable:NO];
}
#pragma mark 关闭菜单
-(void)closeMenu{
 CGPoint center = self.mTransitionView.center;
 center.x = DEVICE_W/2;
 [UIView animateWithDuration:0.15f animations:^{
  self.mTransitionView.center = center;
 } completion:^(BOOL finished) {
  [self mTransitionSubViewsEnable:YES];
  if (_mDelegate != nil&& [_mDelegate respondsToSelector:@selector(didCloseMenu:)]) {
  [_mDelegate didCloseMenu:_mMenuV];
 }
 }];
}
#pragma mark Transition用户交互使能
-(void)mTransitionSubViewsEnable:(BOOL)enable{
 for (UIView *tmp in self.mTransitionView.subviews) {
  tmp.userInteractionEnabled = enable;
 }
 if (enable) {
  [self.mTransitionView removeGestureRecognizer:_mTapGester];
 }else{
  [self.mTransitionView addGestureRecognizer:_mTapGester];
 }
}
#pragma mark 配置mTransitionView
-(void)initMTransitionView{
 for (UIView *tmp in self.view.subviews) {
  [tmp removeFromSuperview];
  [self.mTransitionView addSubview:tmp];
 }
 [self.view addSubview:self.mTransitionView];
}
#pragma mark 拖动手势动作
-(void)panAction:(UIPanGestureRecognizer*)pan{
 CGPoint location = [pan translationInView:pan.view.superview];
 CGPoint center = self.mTransitionView.center;
 if (pan.state==UIGestureRecognizerStateEnded) {
  if (center.x<_centerMaxX*0.5+DEVICE_W*0.25){
   [self closeMenu];
  }else{
   [self openMenu];
  }
 }else if(pan.state==UIGestureRecognizerStateChanged){
  if (location.x<0) {//向左滑
   center.x = center.x+location.x<=DEVICE_W/2? DEVICE_W/2 : center.x+location.x;
  }else{
   center.x = center.x+location.x>=_centerMaxX? _centerMaxX : center.x+location.x;
  }
  self.mTransitionView.center = center;
  [pan setTranslation:CGPointMake(0, 0) inView:pan.view.superview];
 }
}
#pragma mark 添加手势
-(void)addGestureForMTransitionView{
 UIPanGestureRecognizer *pan = [[UIPanGestureRecognizer alloc]initWithTarget:self action:@selector(panAction:)];
 [_mTransitionView addGestureRecognizer:pan];
 _mTapGester = [[UITapGestureRecognizer alloc]initWithTarget:self action:@selector(tapAction)];
}
#pragma mark--懒加载
-(UIView *)mTransitionView{
 if (_mTransitionView==nil) {
  _mTransitionView = [[UIView alloc]initWithFrame:self.view.bounds];
 }
 return _mTransitionView;
}
-(void)tapAction{
 CGFloat x = _mTransitionView.center.x;
 if (x>=_centerMaxX) {
  [self closeMenu];
 }
}
@end

总结

以上所述是小编给大家介绍的iOS中仿QQ侧滑菜单功能,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对网站的支持!


# ios侧滑菜单  # iOS 纯代码写个侧滑栏功能  # Android仿iOS侧滑退出当前界面功能  # Android仿iOS实现侧滑返回功能(类似微信)  # IOS中Swift仿QQ最新版抽屉侧滑和弹框视图  # iOS中给UITableView的侧滑删除增加多个按钮的实现方法  # LRecyclerView侧滑iOS阻塞效果不完整的解决办法  # iOS实现侧滑栏效果  # iOS禁用侧滑返回手势要点解析  # 加载  # 小编  # 都是  # 在此  # 是这样  # 这是一个  # 给大家  # 拖动  # 要了  # 所述  # 给我留言  # 感谢大家  # 再添  # 第一层  # 疑问请  # 有任何  # 转移到  # optional  # mMenuV  # nonatomic 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: 谷歌Google入口永久地址_Google搜索引擎官网首页永久入口  Laravel怎么集成Vue.js_Laravel Mix配置Vue开发环境  如何在IIS服务器上快速部署高效网站?  html5怎么画眼睛_HT5用Canvas或SVG画眼球瞳孔加JS控制动态【绘制】  胶州企业网站制作公司,青岛石头网络科技有限公司怎么样?  Bootstrap CSS布局之列表  Linux后台任务运行方法_nohup与&使用技巧【技巧】  如何快速搭建高效简练网站?  Laravel如何使用Gate和Policy进行授权?(权限控制)  Laravel如何配置Horizon来管理队列?(安装和使用)  node.js报错:Cannot find module &#39;ejs&#39;的解决办法  Laravel如何记录日志_Laravel Logging系统配置与自定义日志通道  浅谈Javascript中的Label语句  Android GridView 滑动条设置一直显示状态(推荐)  python中快速进行多个字符替换的方法小结  Laravel如何处理文件下载请求?(Response示例)  如何快速搭建个人网站并优化SEO?  如何用IIS7快速搭建并优化网站站点?  深圳网站制作培训,深圳哪些招聘网站比较好?  网站制作公司哪里好做,成都网站制作公司哪家做得比较好,更正规?  零服务器AI建站解决方案:快速部署与云端平台低成本实践  Laravel如何安装使用Debugbar工具栏_Laravel性能调试与SQL监控插件【步骤】  logo在线制作免费网站在线制作好吗,DW网页制作时,如何在网页标题前加上logo?  Windows驱动无法加载错误解决方法_驱动签名验证失败处理步骤  大型企业网站制作流程,做网站需要注册公司吗?  ChatGPT常用指令模板大全 新手快速上手的万能Prompt合集  网站制作报价单模板图片,小松挖机官方网站报价?  Laravel如何设置自定义的日志文件名_Laravel根据日期或用户ID生成动态日志【技巧】  Laravel怎么生成二维码图片_Laravel集成Simple-QrCode扩展包与参数设置【实战】  BootStrap整体框架之基础布局组件  Laravel API资源(Resource)怎么用_格式化Laravel API响应的最佳实践  如何快速搭建高效WAP手机网站?  Laravel怎么实现API接口鉴权_Laravel Sanctum令牌生成与请求验证【教程】  详解jQuery停止动画——stop()方法的使用  如何制作一个表白网站视频,关于勇敢表白的小标题?  Laravel如何实现全文搜索功能?(Scout和Algolia示例)  JavaScript如何实现路由_前端路由原理是什么  如何续费美橙建站之星域名及服务?  高端建站如何打造兼具美学与转化的品牌官网?  Laravel如何配置中间件Middleware_Laravel自定义中间件拦截请求与权限校验【步骤】  如何挑选最适合建站的高性能VPS主机?  linux写shell需要注意的问题(必看)  如何用y主机助手快速搭建网站?  Laravel如何处理CORS跨域请求?(配置示例)  Laravel如何使用Blade组件和插槽?(Component代码示例)  HTML5建模怎么导出为FBX格式_FBX格式兼容性及导出步骤【指南】  Laravel API资源类怎么用_Laravel API Resource数据转换  Laravel如何理解并使用服务容器(Service Container)_Laravel依赖注入与容器绑定说明  成都品牌网站制作公司,成都营业执照年报网上怎么办理?  谷歌浏览器如何更改浏览器主题 Google Chrome主题设置教程