两种屏幕HUD区域限制的做法(矩形,弧形)
HUD区域限制算是比较常用到的功能,方形的HUD区域限制多见于小地图,弧形或者椭圆多见于屏幕范围约束。


我没有研究倒角矩形做法,而是将椭圆和矩形进行插值得到一个弧度比较高的形状:

当插值为0时限制范围是椭圆,插值为1限制范围是矩形
矩形的判断使用了线段相交函数,函数来自网络收集
线段相交: http://www.cnblogs.com/hont/p/6106043.html
代码如下
using UnityEngine;
using System;
using System.Collections; public class HUDTest2 : MonoBehaviour
{
[Serializable]
public class EllipseSetting
{
public float radius = 2f;
public Vector2 scale = new Vector2(, );
} public Transform[] hudObjectArray;
public EllipseSetting ellipseSetting = new EllipseSetting();
[Range(, )]
public float lerp; void OnGUI()
{
for (int i = ; i < hudObjectArray.Length; i++)
{
var item = hudObjectArray[i]; var screenPosition = Camera.main.WorldToScreenPoint(item.transform.position);
screenPosition.z = ;
screenPosition.y = Screen.height - screenPosition.y;
screenPosition = HUDFix(screenPosition); GUI.Box(new Rect(screenPosition, Vector3.one * ), "");
}
} Vector3 HUDFix(Vector3 position)
{
var center = new Vector3(Screen.width, Screen.height, ) * 0.5f; var ellipsePoint = GetEllipsePoint(position);
var quadPoint = GetRectPoint(position); var finalPoint = Vector3.Lerp(ellipsePoint, quadPoint, lerp); if (Vector3.Distance(center, finalPoint) >= Vector3.Distance(center, position))
return position;
else
return finalPoint;
} Vector3 GetEllipsePoint(Vector3 currentPoint)
{
var center = new Vector3(Screen.width, Screen.height, ) * 0.5f;
var current = center + (currentPoint - center).normalized * ellipseSetting.radius; current.x *= ellipseSetting.scale.x;
current.y *= ellipseSetting.scale.y; return current;
} Vector3 GetRectPoint(Vector3 currentPoint)
{
var center = new Vector3(Screen.width, Screen.height, ) * 0.5f; const float INNER = ;
const float RAY = * ;
var p0 = new Vector3(INNER, INNER, );
var p1 = new Vector3(Screen.width - INNER, INNER, );
var p2 = new Vector3(INNER, Screen.height - INNER, );
var p3 = new Vector3(Screen.width - INNER, Screen.height - INNER, ); var dir = (currentPoint - center).normalized * RAY; var contractPoint = default(Vector3);
var r = GetIntersection(p0, p1, center, center + dir, out contractPoint);
if (r == )
return contractPoint; r = GetIntersection(p1, p3, center, center + dir, out contractPoint);
if (r == )
return contractPoint; r = GetIntersection(p0, p2, center, center + dir, out contractPoint);
if (r == )
return contractPoint; GetIntersection(p2, p3, center, center + dir, out contractPoint);
return contractPoint;
} int GetIntersection(Vector3 a, Vector3 b, Vector3 c, Vector3 d, out Vector3 contractPoint)
{
contractPoint = new Vector3(, ); if (Mathf.Abs(b.y - a.y) + Mathf.Abs(b.x - a.x) + Mathf.Abs(d.y - c.y)
+ Mathf.Abs(d.x - c.x) == )
{
if ((c.x - a.x) + (c.y - a.y) == )
{
//Debug.Log("ABCD是同一个点!");
}
else
{
//Debug.Log("AB是一个点,CD是一个点,且AC不同!");
}
return ;
} if (Mathf.Abs(b.y - a.y) + Mathf.Abs(b.x - a.x) == )
{
if ((a.x - d.x) * (c.y - d.y) - (a.y - d.y) * (c.x - d.x) == )
{
//Debug.Log("A、B是一个点,且在CD线段上!");
}
else
{
//Debug.Log("A、B是一个点,且不在CD线段上!");
}
return ;
}
if (Mathf.Abs(d.y - c.y) + Mathf.Abs(d.x - c.x) == )
{
if ((d.x - b.x) * (a.y - b.y) - (d.y - b.y) * (a.x - b.x) == )
{
//Debug.Log("C、D是一个点,且在AB线段上!");
}
else
{
//Debug.Log("C、D是一个点,且不在AB线段上!");
}
return ;
} if ((b.y - a.y) * (c.x - d.x) - (b.x - a.x) * (c.y - d.y) == )
{
//Debug.Log("线段平行,无交点!");
return ;
} contractPoint.x = ((b.x - a.x) * (c.x - d.x) * (c.y - a.y) -
c.x * (b.x - a.x) * (c.y - d.y) + a.x * (b.y - a.y) * (c.x - d.x)) /
((b.y - a.y) * (c.x - d.x) - (b.x - a.x) * (c.y - d.y));
contractPoint.y = ((b.y - a.y) * (c.y - d.y) * (c.x - a.x) - c.y
* (b.y - a.y) * (c.x - d.x) + a.y * (b.x - a.x) * (c.y - d.y))
/ ((b.x - a.x) * (c.y - d.y) - (b.y - a.y) * (c.x - d.x)); if ((contractPoint.x - a.x) * (contractPoint.x - b.x) <=
&& (contractPoint.x - c.x) * (contractPoint.x - d.x) <=
&& (contractPoint.y - a.y) * (contractPoint.y - b.y) <=
&& (contractPoint.y - c.y) * (contractPoint.y - d.y) <= )
{ //Debug.Log("线段相交于点(" + contractPoint.x + "," + contractPoint.z + ")!");
return ; // '相交
}
else
{
//Debug.Log("线段相交于虚交点(" + contractPoint.x + "," + contractPoint.z + ")!");
return -; // '相交但不在线段上
}
}
}
两种屏幕HUD区域限制的做法(矩形,弧形)的更多相关文章
- 设置一个DIV块固定在屏幕中央(两种方法)
设置一个DIV块固定在屏幕中央(两种方法) 方法一: 对一个div进行以下设置即可实现居中. <style> #a{ position: fixed; top: 0px; left: 0p ...
- jqGrid中实现radiobutton的两种做法
http://blog.sina.com.cn/s/blog_4f925fc30102e27j.html jqGrid中实现radiobutton的两种做法 ------------------- ...
- SqlServer保留几位小数的两种做法
SqlServer保留几位小数的两种做法 数据库里的 float momey 类型,都会精确到多位小数.但有时候 我们不需要那么精确,例如,只精确到两位有效数字. 解决: 1. 使用 Round( ...
- HTML5工具做屏幕自适应的两种方法
近一两年,HTML5在中国很火,也出了不少HTML5工具和模板.别的先不说,对于不同的H5工具,解决屏幕自适应问题的区别是什么? 简单来说,感应式设计是当用不同设备访问时,能够根据设备的宽度和高度对设 ...
- LVS+keepalived 的DR模式的两种做法
LVS DR模式搭建 准备工作 三台机器: dr:192.168.13.15 rs1:192.168.13.16 rs2: 192.168.13.17 vip:192.168.13.100 修改DR上 ...
- Android 高级UI设计笔记23:Android 夜间模式之 两种常用方法(降低屏幕亮度+替换theme)
1. 夜间模式 所谓的夜间模式,就是能够根据不同的设定,呈现不同风格的界面给用户,而且晚上看着不伤眼睛.特别是一些新闻类App实现夜间模式是非常人性化的,增强用户体验. 2. 我根据网上的资料 以及自 ...
- [正经分析] DAG上dp两种做法的区别——拓扑序与SPFA
在下最近刷了几道DAG图上dp的题目. 要提到的第一道是NOIP原题<最优贸易>.这是一个缩点后带点权的DAG上dp,它同时规定了起点和终点. 第二道是洛谷上的NOI导刊题目<最长路 ...
- CSS实现自适应不同大小屏幕的背景大图的两种方法(转自简书)
CSS实现自适应不同大小屏幕的背景大图的两种方法 一张清晰漂亮的背景图片能给网页加分不少,设计师也经常会给页面的背景使用大图,我们既不想图片因为不同分辨率图片变形,也不希望当在大屏的情况下,背景有一块 ...
- Android 截取手机屏幕两种实现方案解析
近期在开发的过程中,遇到了一个须要截取屏幕保存为图片的需求,详细为截取webview的视图保存图片. 方法1:首先想到的思路是利用SDK提供的View.getDrawingCache()方法: pub ...
随机推荐
- .NET 工具生成引擎概述
Mark Michaelis 微软中国MSDN 过去几年大家一直都在使用 .NET Core(有这么久吗?)并且都知道“生成系统”经历了重大改变,不论是终止对 Gulp 的内置支持,还是放弃 Proj ...
- mysql数据导出为excel文件
select * from xi_table into outfile ’d:test.xls’
- Centos6.5安装ansible2.6.3
需求描述: 管理具有特征性的集群服务器,50台左右,服务都是规划好的!为了更加有效地管理服务器,需要引入协助管理员关系的工具!ansible基于ssh通信不需要安装agent(agentless),使 ...
- 使用nmap查看web服务支持的http methods
安装nmap yum install nmap 查看web server支持的http methods u02 ~]$ nmap -p --script http-methods www.somewh ...
- iOS 瀑布流之栅格布局
代码地址如下:http://www.demodashi.com/demo/14760.html 一 .效果预览 二.确定需求 由下面的需求示意图可知模块的最小单位是正方形,边长是屏幕宽除去边距间隔后的 ...
- ios中NSObject分类(2)
#import <Foundation/Foundation.h> UIColor * rgb(int r, int g, int b); UIColor * rgbA(int r, in ...
- hihocoder 1638:多级并查集
题目链接 并查集可以用于聚类. import java.io.FileInputStream; import java.io.FileNotFoundException; import java.ut ...
- Librec的AoBPR算法实现
Librec的AoBPR算法实现:(基于1.3版本) 要用AoBPR,但是没有找到相应的配置文件,应该怎么办呢? ——因为用的是1.3版本,所以没有,2.0版本有的.[跟BPR参数一样,就 ...
- 树莓派进阶之路 (016) - 通过595驱动4位LED显示系统时间
模块图片,4位共阳极数码管. 我们使用树莓派wiringPi的库来通过74HC595驱动4位数码管: C 代码如下: #include <wiringPi.h> #include < ...
- Ubuntu中安装配置和卸载FTP(转)
转载:http://zyjustin9.iteye.com/blog/2178943 一.安装1.用apt-get工具安装vsftpd: $ sudo apt-get install vsftpd 2 ...