Unity3d:在屏幕边缘显示其他玩家/敌人/物体的方位

系列传送门

IOS:
IOS:使用shell命令打包并上传Itunes
Unity3d:
Unity3d:Canvas适配屏幕分辨率与锚点(Anchors与Pivot)
Unity3d:在屏幕边缘显示其他玩家方位
Unity3d:命令行打包Android
Unity3d:命令行编译IOS
Unity3d:使用Jenkins自动编译打包IOS(只能打包Development)
Unity3d:使用Jenkins自动编译打包IOS(打包Ad-hoc,上传itunes)

最近需要我完成一个小功能:当有玩家在屏幕外时,就在屏幕边缘显示该玩家所在的方位。
特此记下解决方法,予以大家借鉴。
感谢帮助过的朋友们🙏。

1、先复习点数学小知识

1.1、已知两个点:P1(x1, y1)、P2(x2, y2),在直线方程为:a * x + b = y 的直线上,求a,b

解:
由 :a * x1 + b=y1,a * x2 + b=y2
=> a * x2 + y1 - a * x1 = y2
=> a = (y2 - y1) / (x2 - x1)
把a代入: a * x1 + b = y1
=> (y2 - y1) / (x2 - x1) * x1 + b=y1
=> b = y1 - (y2 - y1) / (x2 - x1) * x1
 最终:
a = (y2 - y1) / (x2 - x1)
b = y1 - (y2 - y1) / (x2 - x1) * x1

需要注意两个点x坐标相等的情况:
如果两个点x坐标相等,则该直线与y轴平行;
如果两个点y坐标相等,则该直线与x轴平行;
如果两个点x,y坐标都相等,则不是线,是同一个点。

1.2、已知两条直线的方程:a1 * x + b1 = y 和 a2 * x + b2 = y,求这两条直线的交点

解:
由 :a1 * x + b1=y,a2 * x + b2=y
=> a1 * x + b1 = a2 * x + b2
=> x = (b2 - b1) / (a2 - a1)
把x代入: a1 * x + b1=y
=> y = a1 * (b2 - b1) / (a2 - a1) + b1
 最终:
x = (b2 - b1) / (a2 - a1)
y = a1 * (b2 - b1) / (a2 - a1) + b1

需要注意a1和a2相等的情况:
如果a1和a2相等,b1和b2不相等,则两条直线平行,无交点;
如果a1和a2相等,b1和b2相等,则两条直线是同一条线,无交点;
如果a1和a2不相等,b1和b2相等,则两条直线不平行,有交点,相交于y轴b1点或b2点;
如果a1和a2不相等,b1和b2不相等,则两条直线不平行,有交点。

1.3、已知线段1和线段2的端点,求这两条线段的交点

注意是线段,不是直线。
通过(1)求得端点所在的直线的方程;
通过(2)求得这两条直线的交点。

这里需要判断下交点是落在线段内,还是线段外(虚交点):
如果交点x坐标小于线段1的左侧端点的x坐标,或者交点x坐标大于线段1的右侧端点的x坐标,则该交点落在线段1外,即线段的延长线上,是个虚交点。
反之,是线段1的正常交点。

2、再学习点Unity小知识

传送门

Unity3d:Canvas适配屏幕分辨率与锚点(Anchors与Pivot)

3、进入正题

场景准备:当前玩家(我)是playerA;另一个玩家是playerB。
方位指示器的UI我们放在Root Canvas下。
Root Canvas的CanvasScaler组件配置如下:

  • UI Scale Mode:Scale With Screen Size
  • Reference Resolution:X 1080 ,Y 1920
  • Screen Match Mode:Expand

3.1、理清思路

首先,playerA是一定在屏幕内的,这里需要判断playerB是不是在屏幕内。
如果playerB是在屏幕内,则无需后面的处理。
如果playerB是在屏幕外,让我们继续。

然后,把playerA和playerB的世界坐标转换成屏幕坐标系里的坐标,分别记作A和B。
注意:这里有个容易钻进去的误区,就是很容易想到去计算方向向量,即(BA)。计算完之后又会自然而然的想着用这个向量去处理上面的问题,最后会发现,自己已经进入了一个深坑里,深深的把自己的思维给圈住了,即使偶尔换个方向思考,比如求交点的方式,还是容易被这个向量给迷惑,总想着去用这个向量去解决,尽管这个向量在这个方式里没啥卵用。

这时,因为A点在屏幕内,B点在屏幕外,所以线段AB一定与屏幕的某一个边框有一个交点(实交点,非虚交点)。
这个交点的位置就是我们需要放置方位指示器UI的位置了。
:实交点,即交点位于线段上;虚交点,即交点位于线段的延长线上。

3.2、着手解决

屏幕有4个边框,所以我们要先找出这四个边框线段各自拥有的端点坐标。

这里需要注意,A和B两个点是在屏幕坐标系上,线段AB与屏幕边框的交点也是位于屏幕坐标系上。
而方位指示器UI是放在Canvas上的,位于Canvas坐标系内。
所以在计算之前要先做坐标系转换。

这里有两个转换方式:

  • 方式1:把A和B两个点转换到Canvas的坐标系内,用转换之后的线段AB计算出与Canvas的四个边框相交的交点。
  • 方式2:先在屏幕坐标系上,计算出线段AB与屏幕的四个边框相交的交点,然后把交点转换到Canvas的坐标系内。

注意:后面用到的知识在文章Unity3d:Canvas适配屏幕分辨率与锚点(Anchors与Pivot)中有详细说明,遇到问题可自行查阅。

3.2.1、按方式1解决

先把A点和B点转换到Canvas的坐标系内:
cA = (A.x * Canvas.Width / Screen.Width, A.y * Canvas.Height / Screen.Height)
cB = (B.x * Canvas.Width / Screen.Width, B.y * Canvas.Height / Screen.Height)

再找出Canvas的四个边框线段对应的端点(左下角是原点):
左侧边框端点:lbP(0, 0) 、ltP(0, Canvas.Height) ;
底部边框端点:lbP(0, 0) 、rbP(Canvas.Width, 0) ;
右侧边框端点:rbP(Canvas.Width, 0) 、rtP(Canvas.Width, Canvas.Height) ;
顶部边框端点:ltP(0, Canvas.Height) 、rtP(Canvas.Width, Canvas.Height) ;

有了线段cAcB和Canvas的四个边框线段,根据上面的数学知识,可以轻松求得线段cAcB与Canvas某一个边框的交点,我们记作点P。此时点P位于Canvas的坐标系内。
最后把方位指示器UI在P点显示出来即可。在文章Unity3d:Canvas适配屏幕分辨率与锚点(Anchors与Pivot)的最后有相关知识,请自行查阅_

3.2.2、按方式2解决

现在线段AB的两个端点有了(即A、B点),下面再找出屏幕的四个边框线段对应的端点(左下角是原点):
左侧边框端点:lbP(0, 0) 、ltP(0, Screen.Height) ;
底部边框端点:lbP(0, 0) 、rbP(Screen.Width, 0) ;
右侧边框端点:rbP(Screen.Width, 0) 、rtP(Screen.Width, Screen.Height) ;
顶部边框端点:ltP(0, Screen.Height) 、rtP(Screen.Width, Screen.Height) ;

有了线段AB和屏幕的四个边框线段,根据上面的数学知识,可以轻松求得线段AB与屏幕某一个边框的交点,我们记作点P。此时点P位于屏幕的坐标系内。

然后把交点P转换到Canvas的坐标系内:
cP = (P.x * Canvas.Width / Screen.Width, P.y * Canvas.Height / Screen.Height)
最后把方位指示器UI在cP点显示出来即可。在文章Unity3d:Canvas适配屏幕分辨率与锚点(Anchors与Pivot)的最后有相关知识,请自行查阅_

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,658评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,482评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,213评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,395评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,487评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,523评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,525评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,300评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,753评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,048评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,223评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,905评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,541评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,168评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,417评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,094评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,088评论 2 352