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区域限制的做法(矩形,弧形)的更多相关文章

  1. 设置一个DIV块固定在屏幕中央(两种方法)

    设置一个DIV块固定在屏幕中央(两种方法) 方法一: 对一个div进行以下设置即可实现居中. <style> #a{ position: fixed; top: 0px; left: 0p ...

  2. jqGrid中实现radiobutton的两种做法

    http://blog.sina.com.cn/s/blog_4f925fc30102e27j.html   jqGrid中实现radiobutton的两种做法 ------------------- ...

  3. SqlServer保留几位小数的两种做法

    SqlServer保留几位小数的两种做法   数据库里的 float momey 类型,都会精确到多位小数.但有时候 我们不需要那么精确,例如,只精确到两位有效数字. 解决: 1. 使用 Round( ...

  4. HTML5工具做屏幕自适应的两种方法

    近一两年,HTML5在中国很火,也出了不少HTML5工具和模板.别的先不说,对于不同的H5工具,解决屏幕自适应问题的区别是什么? 简单来说,感应式设计是当用不同设备访问时,能够根据设备的宽度和高度对设 ...

  5. 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上 ...

  6. Android 高级UI设计笔记23:Android 夜间模式之 两种常用方法(降低屏幕亮度+替换theme)

    1. 夜间模式 所谓的夜间模式,就是能够根据不同的设定,呈现不同风格的界面给用户,而且晚上看着不伤眼睛.特别是一些新闻类App实现夜间模式是非常人性化的,增强用户体验. 2. 我根据网上的资料 以及自 ...

  7. [正经分析] DAG上dp两种做法的区别——拓扑序与SPFA

    在下最近刷了几道DAG图上dp的题目. 要提到的第一道是NOIP原题<最优贸易>.这是一个缩点后带点权的DAG上dp,它同时规定了起点和终点. 第二道是洛谷上的NOI导刊题目<最长路 ...

  8. CSS实现自适应不同大小屏幕的背景大图的两种方法(转自简书)

    CSS实现自适应不同大小屏幕的背景大图的两种方法 一张清晰漂亮的背景图片能给网页加分不少,设计师也经常会给页面的背景使用大图,我们既不想图片因为不同分辨率图片变形,也不希望当在大屏的情况下,背景有一块 ...

  9. Android 截取手机屏幕两种实现方案解析

    近期在开发的过程中,遇到了一个须要截取屏幕保存为图片的需求,详细为截取webview的视图保存图片. 方法1:首先想到的思路是利用SDK提供的View.getDrawingCache()方法: pub ...

随机推荐

  1. psql 查询表大小

    select schemaname,tablename,pg_relation_size(schemaname||'.'||tablename) as tabsize from pg_tables o ...

  2. java 文件复制

      java实现文件复制 CreateTime--2017年9月7日15:04:48 Author:Marydon 1.需求 根据原文件复制一份到指定位置 2.代码实现 需要导入: import ja ...

  3. 查询后n条记录

    查询后n条记录 SELECT * FROM tb_stu ORDER BY id ASC LIMIT n

  4. excel文件批量重命名

    1.创建bat文件 2.在文件内输入以下格式的内容并保存,注意期间有空格 ren 1.txt 0011.txt     ren 2.txt 0021.txt     ren 3.txt 0031.tx ...

  5. excel快速访问工具栏和自定义选项卡

    自定义选项卡: excel命令选项--自定义功能区--

  6. python模块之JSON

    # -*- coding: utf-8 -*- #python 27 #xiaodeng #python模块之JSON #1.JSON #JSON表示的对象就是标准的JavaScript语言的对象 # ...

  7. 二维码Data Matrix简单介绍及在VS2010中的编译

    Data Matrix 二维条码原名Datacode,由美国国际资料公司(International Data Matrix, 简称ID Matrix)于1989年发明.Data-Matrix二维条码 ...

  8. Android中使用DialogFragment来取代popopwindow

    DialogFragment +fragment 来取代popopwindow +fragment 先留个标题,这几天过来写,重大发现

  9. keras中的模型保存和加载

    tensorflow中的模型常常是protobuf格式,这种格式既可以是二进制也可以是文本.keras模型保存和加载与tensorflow不同,keras中的模型保存和加载往往是保存成hdf5格式. ...

  10. Tensorflow高级封装

    Tensorflow比较灵活,但是它提供的操作比较低级,于是许多封装库应运而生. slim 导入方式 import tensorflow as tf import tensorflow.contrib ...