具体细节可以参考另外一篇随笔!

以下提供的算法完成的事:

1.自适应1280x720分辨率以下的屏幕

2.自适应1280x720分辨率以上的屏幕

在我设定的要求内包括的分辨率大部分都测过了,背景图、全屏透明Sprite(主要用于九宫格区域的控件摆放)自适应都没问题(不会变形),其他的控件当然是由UIRoot组件搞定的!下面的算法主要就是说明背景、全屏透明Sprite的自适应!

之前的随笔已经说过实际屏幕比设定屏幕大的自适应,现在加入自适应比设定屏幕小的算法!

detail:

使用NGUI版本为3.6

设置:UIRoot中Manual Height=768(由于我们游戏UI定的大小是1280x768),Minimum=480(最小支持高为480的屏,更小的我没测过), Maximum=1536

脚本(将其挂在游戏对象下,每一个需要自适应的UI,都调用这个脚本里面的函数,传进去对应的参数,如AdaptiveUI函数、背景Sprite函数、全屏Sprite函数)

 //注:此脚本所在UIRoot的状态必须是活动的
     public class Adjust : MonoBehaviour
     {
         public UIRoot mRoot = null;

         private static Adjust mInstance = null;
         public static Adjust Instance { get { return mInstance; } }

         /// <summary>
         /// 自适应相关变量声明
         /// </summary>
         ;//实际屏幕宽
         ;//实际屏幕高
         ;
         ;
         private readonly float mWidthScale = Convert.ToSingle(Screen.width) / cDesignWidth;
         private readonly float mHeightScale = Convert.ToSingle(Screen.height) / cDesignHeight;
         private float mScreenSizeIsUnchanged = 0f;//屏幕大小不变
         private readonly bool mIsMinScreen = (Screen.height < cDesignHeight && Screen.width < cDesignWidth);//是否小屏幕(比设定屏幕1280x768小)

         void Awake()
         {
             if (mInstance == null)
             {
                 mInstance = this;
             }

             CalculateScreenWidthHeight();

             Output.Log(string.Format("Adjust.Awake(), mRealScreenWidth = {0}, mRealScreenHeight = {1}", mRealScreenWidth, mRealScreenHeight));
         }

         private void CalculateScreenWidthHeight()
         {
             if (mRoot != null)
             {
                 float scale = 1.0f;
                 if (mRoot.activeHeight < Screen.height)
                 {
                     scale = (float)mRoot.activeHeight / Screen.height;
                 }
                 mRealScreenWidth = Mathf.CeilToInt(Screen.width * scale);
                 mRealScreenHeight = Mathf.CeilToInt(Screen.height * scale);

                 mScreenSizeIsUnchanged = Mathf.Abs(Convert.ToSingle(mRealScreenWidth) / mRealScreenHeight - Convert.ToSingle(cDesignWidth) / cDesignHeight);

                 return;
             }
             Output.Error("Adjust.CalculateScreenWidthHeight(), mRoot is null");
         }

         public void AdaptiveUI(UIRoot root)
         {
             //宽高比不变
             if (mScreenSizeIsUnchanged < 0.0001f)
                 return;

             //实际屏幕宽高比设定宽高小
             if (mIsMinScreen)
                 return;

             if (Convert.ToSingle(Screen.height) / Screen.width > Convert.ToSingle(cDesignHeight) / cDesignWidth)
                 root.manualHeight = Mathf.RoundToInt(Convert.ToSingle(cDesignWidth) / Screen.width * Screen.height);
             else
                 root.manualHeight = cDesignHeight;

             Output.Log(string.Format("Adjust.AdaptiveUI(), Screen.height={0} Screen.width={1}", Screen.height, Screen.width));
         }

         //自适应背景
         public void AdaptiveBackground(UISprite backgroundSprite)
         {
             if (mScreenSizeIsUnchanged < 0.0001f)
                 return;

             if (mIsMinScreen)
             {
                 if (Convert.ToSingle(mRealScreenWidth) / mRealScreenHeight < (float)cDesignWidth / cDesignHeight)
                 {
                     //实际屏幕宽高比 比 设定的屏幕宽高比小,不需做适配
                     return;
                 }
                 else
                 {
                     float scale = cDesignWidth * mHeightScale / mRealScreenWidth;
                     int minScreenW = Convert.ToInt32(Convert.ToSingle(cDesignWidth) / scale);
                     int minScreenH = Convert.ToInt32(Convert.ToSingle(cDesignHeight) / scale);

                     backgroundSprite.SetDimensions(minScreenW, minScreenH);

                     return;
                 }
             }

             //实际宽高比设定宽高大 做适配
             int maxScreenW = Mathf.RoundToInt(cDesignWidth * mHeightScale);
             int maxScreenH = Mathf.RoundToInt(cDesignHeight * mHeightScale);
             if (mHeightScale < mWidthScale)
             {
                 maxScreenW = Mathf.RoundToInt(cDesignWidth * mWidthScale);
                 maxScreenH = Mathf.RoundToInt(cDesignHeight * mWidthScale);
             }
             Output.Log(string.Format("maxScreenW = {0}, maxScreenH = {1}", maxScreenW, maxScreenH));
             backgroundSprite.SetDimensions(maxScreenW, maxScreenH);
         }

         //自适应全屏(透明)Sprite
         public void AdaptiveFullScreenSprite(UISprite fullScreenSprite)
         {
             if (mScreenSizeIsUnchanged < 0.0001f)
                 return;

             if (mIsMinScreen)
             {
                 //小屏的height肯定小于设定height,因此只对width做适配
                 int w = Convert.ToInt32(Convert.ToSingle(cDesignWidth) / ((cDesignWidth * mHeightScale) / mRealScreenWidth));
                 int h = fullScreenSprite.height;
                 fullScreenSprite.SetDimensions(w, h);

                 return;
             }

             fullScreenSprite.SetDimensions(mRealScreenWidth, mRealScreenHeight);
         }
     }

如果帮到你了,给个赞顶一下吧!转载请注明出处,谢谢!

NGUI 屏幕自适应大屏与小屏(初始设定宽高为1280x720,能适应比其小或者更大的屏)的更多相关文章

  1. NGUI 屏幕自适应(初始设定宽高800x480只支持比其大的屏幕)

    自适应讲解部分可以参考以下网址:http://www.xuanyusong.com/archives/2536,下面代码中提到的AdaptiveManualHeight()函数就是参考该文章的. 下面 ...

  2. NGUI 屏幕自适应

    雨松MOMO 2014年05月04日 于 雨松MOMO程序研究院 发表  现在用unity做项目 90%都是用NGUI,并且我个人觉得NGUI应该算是比较成熟的UI插件,虽然他也存在很多问题,但是至少 ...

  3. NGUI屏幕自适应(转)

      屏幕自适应 NGUI可以比较方便的实现屏幕自适应,但是它的官方教程里面针对这个问题没有详细的教程,所以可能在实现的时候会走比较多的弯路.以下是我在开发过程中找到的一个比较方便的实现方法. 主要组件 ...

  4. NGUI屏幕自适应

    NGUI确实是非常棒的一个做界面的插件,比起U3D自带的GUI要好很多,当然也有一些不好之处,毕竟什么都不可能那么完美. 最近在用Unity写游戏使用NGUI遇到了一个很多人都在遇到的问题,就是关于屏 ...

  5. 【Unity3D插件】NGUI屏幕自适应(转)

    屏幕自适应 NGUI可以比较方便的实现屏幕自适应,但是它的官方教程里面针对这个问题没有详细的教程,所以可能在实现的时候会走比较多的弯路.以下是我在开发过程中找到的一个比较方便的实现方法. 主要组件 1 ...

  6. NGUI屏幕自适应解决方案

    NGUI研究院之自适应屏幕 http://www.xuanyusong.com/archives/2536 Unity3D研究院之使用Android的硬件缩放技术优化执行效率 http://www.x ...

  7. [Unity3D]NGUI用Sprite动画和屏幕自适应做游戏开始场景

    我们在玩任何一款手游产品时,都是先上来个logo界面,游戏欢迎界面等,这就意味着我们要做一款游戏需要多个场景,场景之间来回切换实现游戏逻辑,unity也不例外,所以从本篇开始将会介绍如何搭建多个场景, ...

  8. 在一个没有设置宽高的容器中,为什么设置position:absolute后就可以全屏显示了?

    此场景适用于移动端百分比布局,背景全屏显示. 在一个没有设置宽高的容器中设置背景,想要背景全屏显示,设置bcakground-size:100%;后还需设置position:absolut; 原因: ...

  9. [Leetcode]下一个更大元素II

    题目 给定一个循环数组(最后一个元素的下一个元素是数组的第一个元素),输出每个元素的下一个更大元素.数字 x 的下一个更大的元素是按数组遍历顺序,这个数字之后的第一个比它更大的数,这意味着你应该循环地 ...

随机推荐

  1. 图形学基础教程02--顶点数据和SHADER寄存器 (带演示程序)

    本文系原创,欢迎转载,请标明链接 http://www.cnblogs.com/luming1979 有问题欢迎加qq群讨论:366239605

  2. tomcat7 ajax请求服务中文乱码

    在确保请求服务编码格式为utf-8的情况下,确保tomcat的配置 <Connector connectionTimeout="20000" port="8080& ...

  3. Retina屏实现1px边框

    问题描述 通常我们实现边框的方法都是设置1px的边框,但是在retina屏上因为设备像素比的不同,边框在移动设备上的表现也不相同,例如在devicePixelRatio = 2的retina屏下会显示 ...

  4. iOS 三种定时器

    http://www.cocoachina.com/ios/20160905/17482.html

  5. Hibernate一对一外键双向关联(Annotation配置)

    如上图所示:一个学生有一个学生证号,一个学生证号对应一名学生.在Hibernate中怎么用Annotation来实现呢? 学生类,主键是id:学生证的主键也是Id: Student.java pack ...

  6. lnmp编译安装

    lnmp超详细编译安装教程 linux采用的是centos,软件包统一放在/usr/local/src目录下.由于 centos源比较老,而且更新起来也比较慢,此处建议你换成163源.提示:如果你真打 ...

  7. chroot directory

    给 /A/B/C的C目录做chroot,要对C能读写,所以C目录不能做ROOT目录,对B做chroot. 设置C目录所有者为sftp 账户a,组也改为sftp组(这里a和sftp组都是之前建立好的sf ...

  8. JS function

    JS函数的定义 函数是由事件驱动的或者当它被调用时执行的可重复使用的代码块. JS函数的声明 function functionname(){语句} 注意:function必须是小写,否则会报错. J ...

  9. Sql Server中不常用的表运算符之UNPIVOT

    在Sql Server中不常用的表运算符之PIVOT中,介绍了PIVOT表运算符,现在来说说与之相对应的另一个表运算符UNPIVOT. 从名字可以看出,这个运算符的作用与PIVOT刚好相反,是将一行的 ...

  10. uitableview性能优化(转)

    这个感觉写的很好 收藏一下 以备后用 转自 http://www.cnblogs.com/pengyingh/articles/2354714.html 在iOS应用中,UITableView应该是使 ...