创建如Unity3D中Scene场景中的编辑轴

问题分析:

最近在搞软件底层开发,将一些工具或者底层脚本打成dll导入unity使用,有这样一需求,就是编辑功能,需要像Scene场景一样,实现那种编辑轴

实现方式:

创建Mesh,构建编辑轴,这个地方这么几步:

1.线(轴)

2.圆(旋转线)

3.正方形(轴面)

4.圆锥(轴方向)

具体步骤:

1.创建线Mesh:

代码:

 1   /// <summary>
 2         /// 创建线Mesh
 3         /// </summary>
 4         /// <param name="start">线起点</param>
 5         /// <param name="end">线终点</param>
 6         /// <returns>Mesh对象</returns>
 7         private Mesh CreateLineMesh(Vector3 start, Vector3 end)
 8         {
 9             var vertices = new List<Vector3> { start, end };
10             var indices = new List<int> { 0, 1 };
11
12             Mesh mesh = new Mesh();
13             mesh.SetVertices(vertices);
14             mesh.SetIndices(indices.ToArray(), MeshTopology.Lines, 0);
15
16             return mesh;
17         }

这就创建一条起点为start,终点为end的线,是这样,这里在创建是通过Mesh的拓扑结构MeshTopology实现的,MeshTopology是一个枚举,

使用SetIndices去赋值索引,参数分别是索引数组,选择的拓扑结构,要修改的子网格,还有两种重载自己去查。

2.创建圆Mesh:

代码:

 1         /// <summary>
 2         /// 创建(旋转)圆圈Mesh
 3         /// </summary>
 4         /// <param name="radius">圆圈半径</param>
 5         /// <returns></returns>
 6         private Mesh CreateCircleMesh(float radius)
 7         {
 8             List<Vector3> vertexList = new List<Vector3>();
 9             List<int> indexList = new List<int>();
10             for (float i = 0; i < 360.0f; i += 5.0f)
11             {
12                 float rad = Mathf.Deg2Rad * i;
13                 float cosA = Mathf.Cos(rad);
14                 float sinA = Mathf.Sin(rad);
15                 vertexList.Add(new Vector3(radius * cosA, radius * sinA, 0));
16                 if (i != 0)
17                 {
18                     vertexList.Add(new Vector3(radius * cosA, radius * sinA, 0));
19                 }
20             }
21             vertexList.Add(new Vector3(radius * Mathf.Cos(Mathf.Deg2Rad * 0), radius * Mathf.Sin(Mathf.Deg2Rad * 0), 0));
22             for (int i = 0; i < 144; i++)
23             {
24                 indexList.Add(i);
25             }
26             Mesh mesh = new Mesh();
27             mesh.SetVertices(vertexList);
28             mesh.SetIndices(indexList.ToArray(), MeshTopology.Lines, 0);
29             return mesh;
30         }

代码中144=(360/5)*2(端点相连问题)

当时这里的实现思路我想了三种:

1.使用Mesh,自己创建圆面(俩圆面创建出圆圈)

2.LineRender画圆

3.使用Mesh,创建拓扑结构线画圆圈

说一下我为什么选择最后一种:

首先我选的第一种方式,画出来没问题,但是有一种情况,因为圆是在旋转时用的,旋转线需要旋转,当圆旋转到与你成90度时,你就看不到线了。因为这是一面啊,垂直于你指定接近看不着了。

其次呢,我在想让在任何角度看到他都是一根线一个线圆,所以我想到了LineRender画线,画圆,这次画的很好,实现了想要的效果,但是又出现了问题,就是这都是轴,我需要加碰撞器,我需要拾取处理相应操作,但是我查阅了一下,反正有说可以的,但是我试了一圈不行(我没加上碰撞器,加上有问题,自己踩吧你们),所以最后使用的是拓扑结构线插值出一个圆。我成功了完美实现。

这里加上我的另两种尝试代码:

 1     //private LineRenderer line;
 2     //private int r = 20;
 3     //private int n = 360;
 4     ///lineRender画圆
 5     //void Start()
 6     //{
 7     //    line = this.GetComponent<LineRenderer>();
 8     //    line.positionCount = 360 + 1;
 9     //    for (int i = 0; i < n + 1; i++)
10     //    {
11     //        //划线的话2D坐标就行了,这里我们计算x和z坐标轴上的坐标,而y永远是0
12     //        //计算x和z的长度,乘以半径r来得到最终长度
13     //        float x = Mathf.Cos((360 * (i + 1) / n) * Mathf.Deg2Rad) * r;
14     //        float z = Mathf.Sin((360 * (i + 1) / n) * Mathf.Deg2Rad) * r;
15     //        //设置坐标画线
16     //        line.SetPosition(i, new Vector3(0, x, z));
17     //    }
18     //}

 1   /// <summary>
 2     /// 使用Mesh画两个圆面组成的圆环
 3     /// </summary>
 4     /// <param name="radius"></param>
 5     /// <param name="innerradius"></param>
 6     /// <param name="angledegree"></param>
 7     /// <param name="segments"></param>
 8     /// <returns></returns>
 9     Mesh CreateMesh(float radius, float innerradius, float angledegree, int segments)
10     {
11         //vertices(顶点):
12         int vertices_count = segments * 2 + 2;              //因为vertices(顶点)的个数与triangles(索引三角形顶点数)必须匹配
13         Vector3[] vertices = new Vector3[vertices_count];
14         float angleRad = Mathf.Deg2Rad * angledegree;
15         float angleCur = angleRad;
16         float angledelta = angleRad / segments;
17         for (int i = 0; i < vertices_count; i += 2)
18         {
19             float cosA = Mathf.Cos(angleCur);
20             float sinA = Mathf.Sin(angleCur);
21             vertices[i] = new Vector3(radius * cosA, 0, radius * sinA);
22             vertices[i + 1] = new Vector3(innerradius * cosA, 0, innerradius * sinA);
23             angleCur -= angledelta;
24         }
25         //triangles:
26         int triangle_count = segments * 6;
27         int[] triangles = new int[triangle_count];
28         for (int i = 0, vi = 0; i < triangle_count; i += 6, vi += 2)
29         {
30             triangles[i] = vi;
31             triangles[i + 1] = vi + 3;
32             triangles[i + 2] = vi + 1;
33             triangles[i + 3] = vi + 2;
34             triangles[i + 4] = vi + 3;
35             triangles[i + 5] = vi;
36         }
37         //负载属性与mesh
38         Mesh mesh = new Mesh();
39         mesh.vertices = vertices;
40         mesh.triangles = triangles;
41         //mesh.uv = uvs;
42         mesh.RecalculateNormals();
43         return mesh;
44     }

3.创建正方形Mesh:

代码:

 1  /// <summary>
 2         /// 创建正方形面Mesh
 3         /// </summary>
 4         /// <param name="size">面尺寸</param>
 5         /// <returns>Mesh对象</returns>
 6         private Mesh CreatePlaneMesh(Vector2 size)
 7         {
 8             var vertices = new List<Vector3>();
 9             var indices = new List<int>();
10
11             var x = size.x * 0.5f;
12             var z = size.y * 0.5f;
13
14             vertices.Add(new Vector3(x, 0.0f, z));
15             vertices.Add(new Vector3(-x, 0.0f, z));
16             vertices.Add(new Vector3(-x, 0.0f, -z));
17             vertices.Add(new Vector3(x, 0.0f, -z));
18
19             indices.Add(0);
20             indices.Add(1);
21             indices.Add(2);
22             indices.Add(0);
23             indices.Add(2);
24             indices.Add(3);
25
26             Mesh mesh = new Mesh();
27             mesh.SetVertices(vertices);
28             mesh.SetTriangles(indices, 0);
29             mesh.RecalculateNormals();
30
31             return mesh;
32         }

这个就不说啥了,前面说过创建立方体了。

4.创建圆锥Mesh

代码:

 1 /// <summary>
 2     /// 创建圆锥Mesh
 3     /// </summary>
 4     /// <param name="radius">圆锥底面半径</param>
 5     /// <param name="height">圆锥高度</param>
 6     /// <returns>Mesh对象</returns>
 7     private Mesh CreateConeMesh(float radius, float height)
 8     {
 9         var vertices = new List<Vector3>();
10         var indices = new List<int>();
11
12         vertices.Add(Vector3.zero);
13         vertices.Add(Vector3.up * height);
14
15         var temp = new List<Vector3>();
16         //底圆面
17         for (var i = 0.0f; i < 360.0f; i += 30)
18         {
19             var rad = Mathf.Deg2Rad * i;
20             var x = radius * Mathf.Cos(rad);
21             var z = radius * Mathf.Sin(rad);
22
23             temp.Add(new Vector3(x, 0.0f, z));
24         }
25
26         vertices.AddRange(temp);
27         vertices.AddRange(temp);
28
29         for (var i = 2; i <= 13; i++)
30         {
31             indices.Add(i);
32             if (i < 13)
33             {
34                 indices.Add(i + 1);
35             }
36             else
37             {
38                 indices.Add(2);
39             }
40             indices.Add(0);
41         }
42
43         for (var i = 14; i <= 25; i++)
44         {
45             indices.Add(i);
46             indices.Add(1);
47             if (i < 25)
48             {
49                 indices.Add(i + 1);
50             }
51             else
52             {
53                 indices.Add(14);
54             }
55         }
56
57         Mesh mesh = new Mesh();
58         mesh.SetVertices(vertices);
59         mesh.SetTriangles(indices, 0);
60         mesh.RecalculateNormals();
61
62         return mesh;
63     }

到这就需要的编辑轴物件都创建完成了。

效果图:

原文地址:https://www.cnblogs.com/answer-yj/p/11473947.html

时间: 2024-11-09 02:00:12

创建如Unity3D中Scene场景中的编辑轴的相关文章

解析OBJ模型并将其加载到Unity3D场景中

??各位朋友,大家好,欢迎大家关注我的博客,我是秦元培,我的博客地址是http://qinyuanpei.com.今天想和大家交流的是解析obj模型并将其加载到Unity3D场景中,虽然我们知道Unity3D是可以直接导入OBJ模型的,可是有时候我们并不能保证我们目标客户知道如何使用Unity3D的这套制作流程,可能对方最终提供给我们的就是一个模型文件而已,所以这个在这里做这个尝试想想还是蛮有趣的呢,既然如此,我们就选择在所有3D模型格式中最为简单的OBJ模型来一起探讨这个问题吧! 关于OBJ模

Unity3D NGUI从背包中拖出并在场景中生成物体

http://www.cnblogs.com/zhanghaipeng-Unity3D/p/4732592.html 由于游戏需要从背包中拖出武器并在场景中相应的位置生成出来,所以研究了一下这个. 一般来说,在Unity3D开发中如果使用NGUI为游戏做UI,我们的场景和UI并不是使用一个相机进行渲染的,所以从背包中拖出物体并在场景相应的位置生成物体,就会涉及到UICamera和场景摄像机的转换.我在做这个时主要通过打射线来实现坐标转换. 如何创建一个可拖动的UI组件可以参考NGUI中的Exam

unity3d中的物体,在Scene窗口中可以看到,而在Game窗口中看不到的原因

unity3d中的物体,在Scene窗口中可以看到,而在Game窗口中看不到的原因: 多半是因为物体所属Layer与照相机的culling mask不一致导致的,或者超出照相机的可视范围. 如果游戏中有多个相机,每个相机都有自己的可视范围和culling mask,物体在移动的过程中,进入不同的相机,其可见性可能是变化的,取决与物体所属Layer与当前相机是否一致

《Unity3D》通过对象池模式,管理场景中的元素

池管理类有啥用? 在游戏场景中,我们有时候会需要复用一些游戏物体,比如常见的子弹.子弹碰撞类,某些情况下,怪物也可以使用池管理,UI部分比如:血条.文字等等 这些元素共同的特性是:存在固定生命周期,使用比较频繁,场景中大量使用. 所以,我们就通过池管理思路,在游戏初始化的时候,生成一个初始的池,存放我们要复用的元素, 当要用到时,从池中取出:生命周期结束,放回到池中. 代码 这个池的参数有两个:1池中存放的元素 2 池的初始容量(如果池不够了,则会按照这个容量进行扩展) 代码如下 using S

Unity3D中关于场景销毁时事件调用顺序的一点记录

先说一下我遇到的问题,我弄了一个对象池管理多个对象,对象池绑定在一个GameObject上,每个对象在OnBecameInvisible时会进行回收(即移出屏幕就回收),但是当场景切换或停止运行程序时场景中如果还有待回收的对象,就会报错,报错显示的信息为,我的对象池GameObject已经被销毁了云云,因为回收的对象我会把他们作为绑定了对象池的GameObject的子级来方便管理. 所以唯一的可能就是脚本方法调用顺序不可控,即不同GameObject的OnBecameInvisible在其它Ga

搭建LoadRunner中的场景(一) 创建场景

一.创建场景 1. 使用场景创建设置对话框 场景分类: 1. 人工场景:相比面向目标场景,人工场景在实际工作中的应用更为广泛. 2. 面向目标场景:预先定义了一个测试目标,LoadRunner将根据这个目标自动构建场景,有点类似向导模式. 手动场景的两种方式: 用户组方式:虚拟用户分组,测试工程师可以自由地分配各组用户数量(LoadRunner默认用户组方式): 分布百分比方式:需要测试工程师指定某些用户所占的百分比和用户总数,系统再根据这些数据计算产生出具体某类用户的数量(通过勾选"Use t

unity3d 图形吧 之 场景中画圆

?? 孙广东:2015-2-6/2:28  转载请注明出处:http://blog.csdn.net/u010019717 更全的内容请看我的游戏蛮牛地址:http://www.unitymanual.com/space-uid-18602.html 先看一下效果:     区别就是一个2d一个3d. 2d就不介绍了,相对简单一些,对于3d的内容,我们先来看一看数学中的一个题和答案,这样就很容易理解程序了. 这样就好办了!    直接看下面几个脚本吧. using UnityEngine; us

LoadRunner性能测试中Controller场景创建需注意的几点

在LR工具做性能测试中,最关键的一步是Controller场景的设计,因为场景的设计与测试用例的设计相关联,而测试用例的执行,直接影响最终的测试结果是怎么的,因此,我们每设计一种场景,就有可能是一个测试用例的执行(一个场景设计里面可以有多个脚本,场景计划方式可以按组方式,也可以按场景方式),如果场景的设计不正确或不合理,那也无谓在Analysis中结果分析了,对吧? 下面分享一下,在Controller设计场景时需要注意和理解的问题: 1.  在场景中持续时间设置将覆盖Vuser迭代设置.这意味

在editor模式下遍历unity3d builtsetting中的场景

foreach (UnityEditor.EditorBuildSettingsScene S in UnityEditor.EditorBuildSettings.scenes) { //在built setting中是否已经开启 if(S.enabled) { //得到场景的名称 string name = S.path; //打开这个场景 EditorApplication.OpenScene(name); //遍历场景中的GameObject foreach (GameObject ob