Patches added to HSS and VSS to include programatic support
Line Renderer updated with Bezier support --HG-- branch : develop_5.3release
parent
ca29aabed3
commit
2d3b348825
24
README.md
24
README.md
|
@ -4,7 +4,7 @@ This is an extension project for the new Unity UI system which can be found at:
|
||||||
[Unity UI Source](https://bitbucket.org/Unity-Technologies/ui)
|
[Unity UI Source](https://bitbucket.org/Unity-Technologies/ui)
|
||||||
|
|
||||||
#For Unity 5.2.2+ - Use the new 5.3 package!#
|
#For Unity 5.2.2+ - Use the new 5.3 package!#
|
||||||
|
#*Note, due to limited demand, this is the last release we will update the 4.x/5.1 asset package, we'll be focusing on 5.3/5.4 from now on.
|
||||||
|
|
||||||
##Intro##
|
##Intro##
|
||||||
For more info, here's a little introduction video for the project:
|
For more info, here's a little introduction video for the project:
|
||||||
|
@ -15,7 +15,7 @@ For more info, here's a little introduction video for the project:
|
||||||
|
|
||||||
In this repository is a collection of extension scripts to enhance your Unity UI experience. These scripts have been gathered from many sources and combined and improved over time.
|
In this repository is a collection of extension scripts to enhance your Unity UI experience. These scripts have been gathered from many sources and combined and improved over time.
|
||||||
(The majority of the scripts came from the Scripts thread on the [Unity UI forum here](http://bit.ly/UnityUIScriptsForumPost))
|
(The majority of the scripts came from the Scripts thread on the [Unity UI forum here](http://bit.ly/UnityUIScriptsForumPost))
|
||||||
You can either download / fork the project to access the scripts, or you can download this precompiled Unity Asset, chock full of goodness
|
You can either download / fork the project to access the scripts, or you can also download this precompiled Unity Asset, chock full of goodness
|
||||||
### [Unity UI Extensions Unity 4.x Asset](https://bitbucket.org/ddreaper/unity-ui-extensions/downloads/UnityUIExtensions-4.x.unitypackage)###
|
### [Unity UI Extensions Unity 4.x Asset](https://bitbucket.org/ddreaper/unity-ui-extensions/downloads/UnityUIExtensions-4.x.unitypackage)###
|
||||||
### [Unity UI Extensions Unity 5.1 Asset](https://bitbucket.org/ddreaper/unity-ui-extensions/downloads/UnityUIExtensions-5.1.unitypackage)###
|
### [Unity UI Extensions Unity 5.1 Asset](https://bitbucket.org/ddreaper/unity-ui-extensions/downloads/UnityUIExtensions-5.1.unitypackage)###
|
||||||
### [Unity UI Extensions Unity 5.2 Asset](https://bitbucket.org/ddreaper/unity-ui-extensions/downloads/UnityUIExtensions-5.2.unitypackage)### <- 5.2.0 - 5.2.1 base releases ONLY
|
### [Unity UI Extensions Unity 5.2 Asset](https://bitbucket.org/ddreaper/unity-ui-extensions/downloads/UnityUIExtensions-5.2.unitypackage)### <- 5.2.0 - 5.2.1 base releases ONLY
|
||||||
|
@ -28,23 +28,13 @@ To get started with the project, here's a little guide:
|
||||||
---
|
---
|
||||||
## Updates: ##
|
## Updates: ##
|
||||||
|
|
||||||
###Update 1.0.4###
|
###Update 1.1.0###
|
||||||
|
Coming soon
|
||||||
|
|
||||||
[](http://www.youtube.com/watch?v=oF48Qpaq3ls "Update 1.0.0.4 for the Unity UI Extensions Project")
|
###Release History###
|
||||||
---
|
|
||||||
###Update 1.0.5###
|
|
||||||
Few minor fixes and a couple of additional scripts. Predominately created the new 5.3 branch to maintain the UI API changes from the 5.2.1 Patch releases. 5.3 package is 100% compatible with 5.2.1 Patch releases.
|
|
||||||
|
|
||||||
###Update 1.0.6###
|
For the full release history, follow the below link to the full release notes page.
|
||||||
|
### [Release Notes](https://bitbucket.org/ddreaper/unity-ui-extensions/src/c9354eadf4b9287790eb636c21e4107760cd7b87/RELEASENOTES.md?at=develop_5.3)###
|
||||||
[](http://www.youtube.com/watch?v=jpyFiRvSmbg "Update 1.0.6 for the Unity UI Extensions Project")
|
|
||||||
|
|
||||||
* Added the awesome ReOrderable List control, plus some other minor bugfixes / changes.
|
|
||||||
* Added a new version of the Scroll Snap control as an alternative to the fixed versions.
|
|
||||||
* New set of controls including some shader enhanced solutions
|
|
||||||
* I've added a donate column to the lists. If you are getting great use out of a control, help out the dev who created it. Optional of course. Will update with links as I get them.
|
|
||||||
|
|
||||||
**1.0.6.1 - Minor update to enhance soft alpha mask and add cylinder text plus a fix to letter spacing**
|
|
||||||
|
|
||||||
---
|
---
|
||||||
## Controls and extensions listed in this project are: ##
|
## Controls and extensions listed in this project are: ##
|
||||||
|
|
|
@ -0,0 +1,45 @@
|
||||||
|
# Unity UI Extensions release notes #
|
||||||
|
|
||||||
|
**1.0.6.1 - Minor update to enhance soft alpha mask and add cylinder text plus a fix to letter spacing**
|
||||||
|
|
||||||
|
###Update 1.0.6###
|
||||||
|
[](http://www.youtube.com/watch?v=jpyFiRvSmbg "Update 1.0.6 for the Unity UI Extensions Project")
|
||||||
|
|
||||||
|
* Added the awesome ReOrderable List control, plus some other minor bugfixes / changes.
|
||||||
|
* Added a new version of the Scroll Snap control as an alternative to the fixed versions.
|
||||||
|
* New set of controls including some shader enhanced solutions
|
||||||
|
* I've added a donate column to the lists. If you are getting great use out of a control, help out the dev who created it. Optional of course. Will update with links as I get them.
|
||||||
|
|
||||||
|
###Update 1.0.5###
|
||||||
|
Few minor fixes and a couple of additional scripts. Predominately created the new 5.3 branch to maintain the UI API changes from the 5.2.1 Patch releases. 5.3 package is 100% compatible with 5.2.1 Patch releases.
|
||||||
|
|
||||||
|
###Update 1.0.4###
|
||||||
|
|
||||||
|
[](http://www.youtube.com/watch?v=oF48Qpaq3ls "Update 1.0.0.4 for the Unity UI Extensions Project")
|
||||||
|
---
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
#Additional Info#
|
||||||
|
=======================
|
||||||
|
### How do I get set up? ###
|
||||||
|
Either clone / download this repository to your machine and then copy the scripts in, or use the pre-packaged .UnityPackage for your version of Unity and import it as a custom package in to your project.
|
||||||
|
|
||||||
|
### Contribution guidelines ###
|
||||||
|
Got a script you want added, then just fork and submit a PR. All contributions accepted (including fixes)
|
||||||
|
Just ensure
|
||||||
|
* The header of the script matches the standard used in all scripts
|
||||||
|
* The script uses the **Unity.UI.Extensions** namespace so they do not affect any other developments
|
||||||
|
* (optional) Add Component and Editor options where possible (editor options are in the Editor\UIExtensionsMenuOptions.cs file)
|
||||||
|
|
||||||
|
### License ###
|
||||||
|
All scripts conform to the BSD license and are free to use / distribute. See the [LICENSE](https://bitbucket.org/ddreaper/unity-ui-extensions/src/6d03f25b0150994afa97c6a55854d6ae696cad13/LICENSE?at=default) file for more information
|
||||||
|
|
||||||
|
### Like what you see? ###
|
||||||
|
All these scripts were put together for my latest book Unity3D UI Essentials
|
||||||
|
Check out the [page on my blog](http://bit.ly/Unity3DUIEssentials) for more details and learn all about the inner workings of the new Unity UI System.
|
||||||
|
|
||||||
|
### The downloads ###
|
||||||
|
As this repo was created to support my new Unity UI Title ["Unity 3D UI Essentials"](http://bit.ly/Unity3DUIEssentials), in the downloads section you will find two custom assets (SpaceShip-DemoScene-Start.unitypackage and RollABallSample-Start.unitypackage). These are just here as starter scenes for doing UI tasks in the book.
|
||||||
|
|
||||||
|
I will add more sample scenes for the UI examples in this repository and detail them above over time.
|
|
@ -0,0 +1,8 @@
|
||||||
|
fileFormatVersion: 2
|
||||||
|
guid: 295bdc5c9df74e146bd5fa63aa5aeaf8
|
||||||
|
timeCreated: 1463598733
|
||||||
|
licenseType: Pro
|
||||||
|
DefaultImporter:
|
||||||
|
userData:
|
||||||
|
assetBundleName:
|
||||||
|
assetBundleVariant:
|
|
@ -7,7 +7,7 @@ using UnityEngine.EventSystems;
|
||||||
|
|
||||||
namespace UnityEngine.UI.Extensions
|
namespace UnityEngine.UI.Extensions
|
||||||
{
|
{
|
||||||
|
|
||||||
[RequireComponent(typeof(ScrollRect))]
|
[RequireComponent(typeof(ScrollRect))]
|
||||||
[AddComponentMenu("Layout/Extensions/Horizontal Scroll Snap")]
|
[AddComponentMenu("Layout/Extensions/Horizontal Scroll Snap")]
|
||||||
public class HorizontalScrollSnap : MonoBehaviour, IBeginDragHandler, IEndDragHandler, IDragHandler
|
public class HorizontalScrollSnap : MonoBehaviour, IBeginDragHandler, IEndDragHandler, IDragHandler
|
||||||
|
@ -41,13 +41,25 @@ namespace UnityEngine.UI.Extensions
|
||||||
|
|
||||||
private bool _startDrag = true;
|
private bool _startDrag = true;
|
||||||
private Vector3 _startPosition = new Vector3();
|
private Vector3 _startPosition = new Vector3();
|
||||||
|
|
||||||
|
[Tooltip("The currently active page")]
|
||||||
|
[SerializeField]
|
||||||
private int _currentScreen;
|
private int _currentScreen;
|
||||||
|
|
||||||
[Tooltip("The screen / page to start the control on")]
|
[Tooltip("The screen / page to start the control on")]
|
||||||
public int StartingScreen = 1;
|
public int StartingScreen = 1;
|
||||||
|
|
||||||
[Tooltip("The distance between two pages, by default 3 times the height of the control")]
|
[Tooltip("The distance between two pages, by default 3 times the height of the control")]
|
||||||
public int PageStep = 0;
|
public int PageStep = 0;
|
||||||
|
|
||||||
|
public int CurrentPage
|
||||||
|
{
|
||||||
|
get
|
||||||
|
{
|
||||||
|
return _currentScreen;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
// Use this for initialization
|
// Use this for initialization
|
||||||
|
@ -61,22 +73,10 @@ namespace UnityEngine.UI.Extensions
|
||||||
}
|
}
|
||||||
DistributePages();
|
DistributePages();
|
||||||
|
|
||||||
_screens = _screensContainer.childCount;
|
|
||||||
|
|
||||||
_lerp = false;
|
_lerp = false;
|
||||||
|
_currentScreen = StartingScreen;
|
||||||
|
|
||||||
_positions = new System.Collections.Generic.List<Vector3>();
|
_scroll_rect.horizontalNormalizedPosition = (float)(_currentScreen - 1) / (_screens - 1);
|
||||||
|
|
||||||
if (_screens > 0)
|
|
||||||
{
|
|
||||||
for (float i = 0; i < _screens; ++i)
|
|
||||||
{
|
|
||||||
_scroll_rect.horizontalNormalizedPosition = i / (_screens - 1);
|
|
||||||
_positions.Add(_screensContainer.localPosition);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
_scroll_rect.horizontalNormalizedPosition = (float)(StartingScreen - 1) / (_screens - 1);
|
|
||||||
|
|
||||||
ChangeBulletsInfo(_currentScreen);
|
ChangeBulletsInfo(_currentScreen);
|
||||||
|
|
||||||
|
@ -221,6 +221,19 @@ namespace UnityEngine.UI.Extensions
|
||||||
_dimension = currentXPosition + _offset * -1;
|
_dimension = currentXPosition + _offset * -1;
|
||||||
|
|
||||||
_screensContainer.GetComponent<RectTransform>().offsetMax = new Vector2(_dimension, 0f);
|
_screensContainer.GetComponent<RectTransform>().offsetMax = new Vector2(_dimension, 0f);
|
||||||
|
|
||||||
|
_screens = _screensContainer.childCount;
|
||||||
|
|
||||||
|
_positions = new System.Collections.Generic.List<Vector3>();
|
||||||
|
|
||||||
|
if (_screens > 0)
|
||||||
|
{
|
||||||
|
for (float i = 0; i < _screens; ++i)
|
||||||
|
{
|
||||||
|
_scroll_rect.horizontalNormalizedPosition = i / (_screens - 1);
|
||||||
|
_positions.Add(_screensContainer.localPosition);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
int GetPageforPosition(Vector3 pos)
|
int GetPageforPosition(Vector3 pos)
|
||||||
|
@ -247,13 +260,63 @@ namespace UnityEngine.UI.Extensions
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Add a new child to this Scroll Snap and recalculate it's children
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="GO">GameObject to add to the ScrollSnap</param>
|
||||||
|
public void AddChild(GameObject GO)
|
||||||
|
{
|
||||||
|
_scroll_rect.horizontalNormalizedPosition = 0;
|
||||||
|
GO.transform.SetParent(_screensContainer);
|
||||||
|
DistributePages();
|
||||||
|
|
||||||
|
_scroll_rect.horizontalNormalizedPosition = (float)(_currentScreen) / (_screens - 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Remove a new child to this Scroll Snap and recalculate it's children
|
||||||
|
/// *Note, this is an index address (0-x)
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="index"></param>
|
||||||
|
/// <param name="ChildRemoved"></param>
|
||||||
|
public void RemoveChild(int index, out GameObject ChildRemoved)
|
||||||
|
{
|
||||||
|
ChildRemoved = null;
|
||||||
|
if (index < 0 || index > _screensContainer.childCount)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
_scroll_rect.horizontalNormalizedPosition = 0;
|
||||||
|
var children = _screensContainer.transform;
|
||||||
|
int i = 0;
|
||||||
|
foreach (Transform child in children)
|
||||||
|
{
|
||||||
|
if (i == index)
|
||||||
|
{
|
||||||
|
child.SetParent(null);
|
||||||
|
ChildRemoved = child.gameObject;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
i++;
|
||||||
|
}
|
||||||
|
DistributePages();
|
||||||
|
if (_currentScreen > _screens - 1)
|
||||||
|
{
|
||||||
|
_currentScreen = _screens - 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
_scroll_rect.horizontalNormalizedPosition = (float)(_currentScreen) / (_screens - 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
#region Interfaces
|
#region Interfaces
|
||||||
public void OnBeginDrag(PointerEventData eventData)
|
public void OnBeginDrag(PointerEventData eventData)
|
||||||
{
|
{
|
||||||
_startPosition = _screensContainer.localPosition;
|
_startPosition = _screensContainer.localPosition;
|
||||||
_fastSwipeCounter = 0;
|
_fastSwipeCounter = 0;
|
||||||
_fastSwipeTimer = true;
|
_fastSwipeTimer = true;
|
||||||
_currentScreen = CurrentScreen();
|
_currentScreen = CurrentScreen();
|
||||||
}
|
}
|
||||||
|
|
||||||
public void OnEndDrag(PointerEventData eventData)
|
public void OnEndDrag(PointerEventData eventData)
|
||||||
|
|
|
@ -40,13 +40,25 @@ namespace UnityEngine.UI.Extensions
|
||||||
|
|
||||||
private bool _startDrag = true;
|
private bool _startDrag = true;
|
||||||
private Vector3 _startPosition = new Vector3();
|
private Vector3 _startPosition = new Vector3();
|
||||||
|
|
||||||
|
[Tooltip("The currently active page")]
|
||||||
|
[SerializeField]
|
||||||
private int _currentScreen;
|
private int _currentScreen;
|
||||||
|
|
||||||
[Tooltip("The screen / page to start the control on")]
|
[Tooltip("The screen / page to start the control on")]
|
||||||
public int StartingScreen = 1;
|
public int StartingScreen = 1;
|
||||||
|
|
||||||
[Tooltip("The distance between two pages, by default 3 times the width of the control")]
|
[Tooltip("The distance between two pages, by default 3 times the width of the control")]
|
||||||
public int PageStep = 0;
|
public int PageStep = 0;
|
||||||
|
|
||||||
|
public int CurrentPage
|
||||||
|
{
|
||||||
|
get
|
||||||
|
{
|
||||||
|
return _currentScreen;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Use this for initialization
|
// Use this for initialization
|
||||||
void Start()
|
void Start()
|
||||||
{
|
{
|
||||||
|
@ -58,22 +70,10 @@ namespace UnityEngine.UI.Extensions
|
||||||
}
|
}
|
||||||
DistributePages();
|
DistributePages();
|
||||||
|
|
||||||
_screens = _screensContainer.childCount;
|
|
||||||
|
|
||||||
_lerp = false;
|
_lerp = false;
|
||||||
|
_currentScreen = StartingScreen;
|
||||||
|
|
||||||
_positions = new System.Collections.Generic.List<Vector3>();
|
_scroll_rect.verticalNormalizedPosition = (float)(_currentScreen - 1) / (float)(_screens - 1);
|
||||||
|
|
||||||
if (_screens > 0)
|
|
||||||
{
|
|
||||||
for (int i = 0; i < _screens; ++i)
|
|
||||||
{
|
|
||||||
_scroll_rect.verticalNormalizedPosition = (float)i / (float)(_screens - 1);
|
|
||||||
_positions.Add(_screensContainer.localPosition);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
_scroll_rect.verticalNormalizedPosition = (float)(StartingScreen - 1) / (float)(_screens - 1);
|
|
||||||
|
|
||||||
ChangeBulletsInfo(_currentScreen);
|
ChangeBulletsInfo(_currentScreen);
|
||||||
|
|
||||||
|
@ -219,6 +219,19 @@ namespace UnityEngine.UI.Extensions
|
||||||
_dimension = currentYPosition + _offset * -1;
|
_dimension = currentYPosition + _offset * -1;
|
||||||
|
|
||||||
_screensContainer.GetComponent<RectTransform>().offsetMax = new Vector2(0f,_dimension);
|
_screensContainer.GetComponent<RectTransform>().offsetMax = new Vector2(0f,_dimension);
|
||||||
|
|
||||||
|
_screens = _screensContainer.childCount;
|
||||||
|
|
||||||
|
_positions = new System.Collections.Generic.List<Vector3>();
|
||||||
|
|
||||||
|
if (_screens > 0)
|
||||||
|
{
|
||||||
|
for (int i = 0; i < _screens; ++i)
|
||||||
|
{
|
||||||
|
_scroll_rect.verticalNormalizedPosition = (float)i / (float)(_screens - 1);
|
||||||
|
_positions.Add(_screensContainer.localPosition);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
int GetPageforPosition(Vector3 pos)
|
int GetPageforPosition(Vector3 pos)
|
||||||
|
@ -245,6 +258,54 @@ namespace UnityEngine.UI.Extensions
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Add a new child to this Scroll Snap and recalculate it's children
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="GO">GameObject to add to the ScrollSnap</param>
|
||||||
|
public void AddChild(GameObject GO)
|
||||||
|
{
|
||||||
|
_scroll_rect.verticalNormalizedPosition = 0;
|
||||||
|
GO.transform.SetParent(_screensContainer);
|
||||||
|
DistributePages();
|
||||||
|
|
||||||
|
_scroll_rect.verticalNormalizedPosition = (float)(_currentScreen) / (_screens - 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Remove a new child to this Scroll Snap and recalculate it's children
|
||||||
|
/// *Note, this is an index address (0-x)
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="index"></param>
|
||||||
|
/// <param name="ChildRemoved"></param>
|
||||||
|
public void RemoveChild(int index, out GameObject ChildRemoved)
|
||||||
|
{
|
||||||
|
ChildRemoved = null;
|
||||||
|
if (index < 0 || index > _screensContainer.childCount)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
_scroll_rect.verticalNormalizedPosition = 0;
|
||||||
|
var children = _screensContainer.transform;
|
||||||
|
int i = 0;
|
||||||
|
foreach (Transform child in children)
|
||||||
|
{
|
||||||
|
if (i == index)
|
||||||
|
{
|
||||||
|
child.SetParent(null);
|
||||||
|
ChildRemoved = child.gameObject;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
i++;
|
||||||
|
}
|
||||||
|
DistributePages();
|
||||||
|
if (_currentScreen > _screens - 1)
|
||||||
|
{
|
||||||
|
_currentScreen = _screens - 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
_scroll_rect.verticalNormalizedPosition = (float)(_currentScreen) / (_screens - 1);
|
||||||
|
}
|
||||||
|
|
||||||
#region Interfaces
|
#region Interfaces
|
||||||
public void OnBeginDrag(PointerEventData eventData)
|
public void OnBeginDrag(PointerEventData eventData)
|
||||||
{
|
{
|
||||||
|
|
|
@ -21,6 +21,13 @@ namespace UnityEngine.UI.Extensions
|
||||||
Bevel,
|
Bevel,
|
||||||
Miter
|
Miter
|
||||||
}
|
}
|
||||||
|
public enum BezierType
|
||||||
|
{
|
||||||
|
None,
|
||||||
|
Quick,
|
||||||
|
Basic,
|
||||||
|
Improved,
|
||||||
|
}
|
||||||
|
|
||||||
private const float MIN_MITER_JOIN = 15 * Mathf.Deg2Rad;
|
private const float MIN_MITER_JOIN = 15 * Mathf.Deg2Rad;
|
||||||
|
|
||||||
|
@ -58,6 +65,9 @@ namespace UnityEngine.UI.Extensions
|
||||||
public bool LineCaps = false;
|
public bool LineCaps = false;
|
||||||
public JoinType LineJoins = JoinType.Bevel;
|
public JoinType LineJoins = JoinType.Bevel;
|
||||||
|
|
||||||
|
public BezierType BezierMode = BezierType.None;
|
||||||
|
public int BezierSegmentsPerCurve = 10;
|
||||||
|
|
||||||
public override Texture mainTexture
|
public override Texture mainTexture
|
||||||
{
|
{
|
||||||
get
|
get
|
||||||
|
@ -126,6 +136,29 @@ namespace UnityEngine.UI.Extensions
|
||||||
{
|
{
|
||||||
if (m_points == null)
|
if (m_points == null)
|
||||||
return;
|
return;
|
||||||
|
Vector2[] pointsToDraw = m_points;
|
||||||
|
//If Bezier is desired, pick the implementation
|
||||||
|
if (BezierMode != BezierType.None && m_points.Length > 3)
|
||||||
|
{
|
||||||
|
BezierPath bezierPath = new BezierPath();
|
||||||
|
|
||||||
|
bezierPath.SetControlPoints(pointsToDraw);
|
||||||
|
bezierPath.SegmentsPerCurve = BezierSegmentsPerCurve;
|
||||||
|
List<Vector2> drawingPoints;
|
||||||
|
switch (BezierMode)
|
||||||
|
{
|
||||||
|
case BezierType.Basic:
|
||||||
|
drawingPoints = bezierPath.GetDrawingPoints0();
|
||||||
|
break;
|
||||||
|
case BezierType.Improved:
|
||||||
|
drawingPoints = bezierPath.GetDrawingPoints1();
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
drawingPoints = bezierPath.GetDrawingPoints2();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
pointsToDraw = drawingPoints.ToArray();
|
||||||
|
}
|
||||||
|
|
||||||
var sizeX = rectTransform.rect.width;
|
var sizeX = rectTransform.rect.width;
|
||||||
var sizeY = rectTransform.rect.height;
|
var sizeY = rectTransform.rect.height;
|
||||||
|
@ -153,10 +186,10 @@ namespace UnityEngine.UI.Extensions
|
||||||
var segments = new List<UIVertex[]>();
|
var segments = new List<UIVertex[]>();
|
||||||
if (LineList)
|
if (LineList)
|
||||||
{
|
{
|
||||||
for (var i = 1; i < m_points.Length; i += 2)
|
for (var i = 1; i < pointsToDraw.Length; i += 2)
|
||||||
{
|
{
|
||||||
var start = m_points[i - 1];
|
var start = pointsToDraw[i - 1];
|
||||||
var end = m_points[i];
|
var end = pointsToDraw[i];
|
||||||
start = new Vector2(start.x * sizeX + offsetX, start.y * sizeY + offsetY);
|
start = new Vector2(start.x * sizeX + offsetX, start.y * sizeY + offsetY);
|
||||||
end = new Vector2(end.x * sizeX + offsetX, end.y * sizeY + offsetY);
|
end = new Vector2(end.x * sizeX + offsetX, end.y * sizeY + offsetY);
|
||||||
|
|
||||||
|
@ -175,10 +208,10 @@ namespace UnityEngine.UI.Extensions
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
for (var i = 1; i < m_points.Length; i++)
|
for (var i = 1; i < pointsToDraw.Length; i++)
|
||||||
{
|
{
|
||||||
var start = m_points[i - 1];
|
var start = pointsToDraw[i - 1];
|
||||||
var end = m_points[i];
|
var end = pointsToDraw[i];
|
||||||
start = new Vector2(start.x * sizeX + offsetX, start.y * sizeY + offsetY);
|
start = new Vector2(start.x * sizeX + offsetX, start.y * sizeY + offsetY);
|
||||||
end = new Vector2(end.x * sizeX + offsetX, end.y * sizeY + offsetY);
|
end = new Vector2(end.x * sizeX + offsetX, end.y * sizeY + offsetY);
|
||||||
|
|
||||||
|
@ -189,7 +222,7 @@ namespace UnityEngine.UI.Extensions
|
||||||
|
|
||||||
segments.Add(CreateLineSegment(start, end, SegmentType.Middle));
|
segments.Add(CreateLineSegment(start, end, SegmentType.Middle));
|
||||||
|
|
||||||
if (LineCaps && i == m_points.Length - 1)
|
if (LineCaps && i == pointsToDraw.Length - 1)
|
||||||
{
|
{
|
||||||
segments.Add(CreateLineCap(start, end, SegmentType.End));
|
segments.Add(CreateLineCap(start, end, SegmentType.End));
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,350 @@
|
||||||
|
/**
|
||||||
|
|
||||||
|
This class demonstrates the code discussed in these two articles:
|
||||||
|
|
||||||
|
http://devmag.org.za/2011/04/05/bzier-curves-a-tutorial/
|
||||||
|
http://devmag.org.za/2011/06/23/bzier-path-algorithms/
|
||||||
|
|
||||||
|
Use this code as you wish, at your own risk. If it blows up
|
||||||
|
your computer, makes a plane crash, or otherwise cause damage,
|
||||||
|
injury, or death, it is not my fault.
|
||||||
|
|
||||||
|
@author Herman Tulleken, dev.mag.org.za
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
using System.Collections.Generic;
|
||||||
|
|
||||||
|
namespace UnityEngine.UI.Extensions
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
Class for representing a Bezier path, and methods for getting suitable points to
|
||||||
|
draw the curve with line segments.
|
||||||
|
*/
|
||||||
|
public class BezierPath
|
||||||
|
{
|
||||||
|
public int SegmentsPerCurve = 10;
|
||||||
|
public float MINIMUM_SQR_DISTANCE = 0.01f;
|
||||||
|
|
||||||
|
// This corresponds to about 172 degrees, 8 degrees from a straight line
|
||||||
|
public float DIVISION_THRESHOLD = -0.99f;
|
||||||
|
|
||||||
|
private List<Vector2> controlPoints;
|
||||||
|
|
||||||
|
private int curveCount; //how many bezier curves in this path?
|
||||||
|
|
||||||
|
/**
|
||||||
|
Constructs a new empty Bezier curve. Use one of these methods
|
||||||
|
to add points: SetControlPoints, Interpolate, SamplePoints.
|
||||||
|
*/
|
||||||
|
public BezierPath()
|
||||||
|
{
|
||||||
|
controlPoints = new List<Vector2>();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
Sets the control points of this Bezier path.
|
||||||
|
Points 0-3 forms the first Bezier curve, points
|
||||||
|
3-6 forms the second curve, etc.
|
||||||
|
*/
|
||||||
|
public void SetControlPoints(List<Vector2> newControlPoints)
|
||||||
|
{
|
||||||
|
controlPoints.Clear();
|
||||||
|
controlPoints.AddRange(newControlPoints);
|
||||||
|
curveCount = (controlPoints.Count - 1) / 3;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void SetControlPoints(Vector2[] newControlPoints)
|
||||||
|
{
|
||||||
|
controlPoints.Clear();
|
||||||
|
controlPoints.AddRange(newControlPoints);
|
||||||
|
curveCount = (controlPoints.Count - 1) / 3;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
Returns the control points for this Bezier curve.
|
||||||
|
*/
|
||||||
|
public List<Vector2> GetControlPoints()
|
||||||
|
{
|
||||||
|
return controlPoints;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
Calculates a Bezier interpolated path for the given points.
|
||||||
|
*/
|
||||||
|
public void Interpolate(List<Vector2> segmentPoints, float scale)
|
||||||
|
{
|
||||||
|
controlPoints.Clear();
|
||||||
|
|
||||||
|
if (segmentPoints.Count < 2)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (int i = 0; i < segmentPoints.Count; i++)
|
||||||
|
{
|
||||||
|
if (i == 0) // is first
|
||||||
|
{
|
||||||
|
Vector2 p1 = segmentPoints[i];
|
||||||
|
Vector2 p2 = segmentPoints[i + 1];
|
||||||
|
|
||||||
|
Vector2 tangent = (p2 - p1);
|
||||||
|
Vector2 q1 = p1 + scale * tangent;
|
||||||
|
|
||||||
|
controlPoints.Add(p1);
|
||||||
|
controlPoints.Add(q1);
|
||||||
|
}
|
||||||
|
else if (i == segmentPoints.Count - 1) //last
|
||||||
|
{
|
||||||
|
Vector2 p0 = segmentPoints[i - 1];
|
||||||
|
Vector2 p1 = segmentPoints[i];
|
||||||
|
Vector2 tangent = (p1 - p0);
|
||||||
|
Vector2 q0 = p1 - scale * tangent;
|
||||||
|
|
||||||
|
controlPoints.Add(q0);
|
||||||
|
controlPoints.Add(p1);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
Vector2 p0 = segmentPoints[i - 1];
|
||||||
|
Vector2 p1 = segmentPoints[i];
|
||||||
|
Vector2 p2 = segmentPoints[i + 1];
|
||||||
|
Vector2 tangent = (p2 - p0).normalized;
|
||||||
|
Vector2 q0 = p1 - scale * tangent * (p1 - p0).magnitude;
|
||||||
|
Vector2 q1 = p1 + scale * tangent * (p2 - p1).magnitude;
|
||||||
|
|
||||||
|
controlPoints.Add(q0);
|
||||||
|
controlPoints.Add(p1);
|
||||||
|
controlPoints.Add(q1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
curveCount = (controlPoints.Count - 1) / 3;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
Sample the given points as a Bezier path.
|
||||||
|
*/
|
||||||
|
public void SamplePoints(List<Vector2> sourcePoints, float minSqrDistance, float maxSqrDistance, float scale)
|
||||||
|
{
|
||||||
|
if (sourcePoints.Count < 2)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
Stack<Vector2> samplePoints = new Stack<Vector2>();
|
||||||
|
|
||||||
|
samplePoints.Push(sourcePoints[0]);
|
||||||
|
|
||||||
|
Vector2 potentialSamplePoint = sourcePoints[1];
|
||||||
|
|
||||||
|
int i = 2;
|
||||||
|
|
||||||
|
for (i = 2; i < sourcePoints.Count; i++)
|
||||||
|
{
|
||||||
|
if (
|
||||||
|
((potentialSamplePoint - sourcePoints[i]).sqrMagnitude > minSqrDistance) &&
|
||||||
|
((samplePoints.Peek() - sourcePoints[i]).sqrMagnitude > maxSqrDistance))
|
||||||
|
{
|
||||||
|
samplePoints.Push(potentialSamplePoint);
|
||||||
|
}
|
||||||
|
|
||||||
|
potentialSamplePoint = sourcePoints[i];
|
||||||
|
}
|
||||||
|
|
||||||
|
//now handle last bit of curve
|
||||||
|
Vector2 p1 = samplePoints.Pop(); //last sample point
|
||||||
|
Vector2 p0 = samplePoints.Peek(); //second last sample point
|
||||||
|
Vector2 tangent = (p0 - potentialSamplePoint).normalized;
|
||||||
|
float d2 = (potentialSamplePoint - p1).magnitude;
|
||||||
|
float d1 = (p1 - p0).magnitude;
|
||||||
|
p1 = p1 + tangent * ((d1 - d2) / 2);
|
||||||
|
|
||||||
|
samplePoints.Push(p1);
|
||||||
|
samplePoints.Push(potentialSamplePoint);
|
||||||
|
|
||||||
|
|
||||||
|
Interpolate(new List<Vector2>(samplePoints), scale);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
Caluclates a point on the path.
|
||||||
|
|
||||||
|
@param curveIndex The index of the curve that the point is on. For example,
|
||||||
|
the second curve (index 1) is the curve with controlpoints 3, 4, 5, and 6.
|
||||||
|
|
||||||
|
@param t The paramater indicating where on the curve the point is. 0 corresponds
|
||||||
|
to the "left" point, 1 corresponds to the "right" end point.
|
||||||
|
*/
|
||||||
|
public Vector2 CalculateBezierPoint(int curveIndex, float t)
|
||||||
|
{
|
||||||
|
int nodeIndex = curveIndex * 3;
|
||||||
|
|
||||||
|
Vector2 p0 = controlPoints[nodeIndex];
|
||||||
|
Vector2 p1 = controlPoints[nodeIndex + 1];
|
||||||
|
Vector2 p2 = controlPoints[nodeIndex + 2];
|
||||||
|
Vector2 p3 = controlPoints[nodeIndex + 3];
|
||||||
|
|
||||||
|
return CalculateBezierPoint(t, p0, p1, p2, p3);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
Gets the drawing points. This implementation simply calculates a certain number
|
||||||
|
of points per curve.
|
||||||
|
*/
|
||||||
|
public List<Vector2> GetDrawingPoints0()
|
||||||
|
{
|
||||||
|
List<Vector2> drawingPoints = new List<Vector2>();
|
||||||
|
|
||||||
|
for (int curveIndex = 0; curveIndex < curveCount; curveIndex++)
|
||||||
|
{
|
||||||
|
if (curveIndex == 0) //Only do this for the first end point.
|
||||||
|
//When i != 0, this coincides with the
|
||||||
|
//end point of the previous segment,
|
||||||
|
{
|
||||||
|
drawingPoints.Add(CalculateBezierPoint(curveIndex, 0));
|
||||||
|
}
|
||||||
|
|
||||||
|
for (int j = 1; j <= SegmentsPerCurve; j++)
|
||||||
|
{
|
||||||
|
float t = j / (float)SegmentsPerCurve;
|
||||||
|
drawingPoints.Add(CalculateBezierPoint(curveIndex, t));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return drawingPoints;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
Gets the drawing points. This implementation simply calculates a certain number
|
||||||
|
of points per curve.
|
||||||
|
|
||||||
|
This is a lsightly different inplementation from the one above.
|
||||||
|
*/
|
||||||
|
public List<Vector2> GetDrawingPoints1()
|
||||||
|
{
|
||||||
|
List<Vector2> drawingPoints = new List<Vector2>();
|
||||||
|
|
||||||
|
for (int i = 0; i < controlPoints.Count - 3; i += 3)
|
||||||
|
{
|
||||||
|
Vector2 p0 = controlPoints[i];
|
||||||
|
Vector2 p1 = controlPoints[i + 1];
|
||||||
|
Vector2 p2 = controlPoints[i + 2];
|
||||||
|
Vector2 p3 = controlPoints[i + 3];
|
||||||
|
|
||||||
|
if (i == 0) //only do this for the first end point. When i != 0, this coincides with the end point of the previous segment,
|
||||||
|
{
|
||||||
|
drawingPoints.Add(CalculateBezierPoint(0, p0, p1, p2, p3));
|
||||||
|
}
|
||||||
|
|
||||||
|
for (int j = 1; j <= SegmentsPerCurve; j++)
|
||||||
|
{
|
||||||
|
float t = j / (float)SegmentsPerCurve;
|
||||||
|
drawingPoints.Add(CalculateBezierPoint(t, p0, p1, p2, p3));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return drawingPoints;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
This gets the drawing points of a bezier curve, using recursive division,
|
||||||
|
which results in less points for the same accuracy as the above implementation.
|
||||||
|
*/
|
||||||
|
public List<Vector2> GetDrawingPoints2()
|
||||||
|
{
|
||||||
|
List<Vector2> drawingPoints = new List<Vector2>();
|
||||||
|
|
||||||
|
for (int curveIndex = 0; curveIndex < curveCount; curveIndex++)
|
||||||
|
{
|
||||||
|
List<Vector2> bezierCurveDrawingPoints = FindDrawingPoints(curveIndex);
|
||||||
|
|
||||||
|
if (curveIndex != 0)
|
||||||
|
{
|
||||||
|
//remove the fist point, as it coincides with the last point of the previous Bezier curve.
|
||||||
|
bezierCurveDrawingPoints.RemoveAt(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
drawingPoints.AddRange(bezierCurveDrawingPoints);
|
||||||
|
}
|
||||||
|
|
||||||
|
return drawingPoints;
|
||||||
|
}
|
||||||
|
|
||||||
|
List<Vector2> FindDrawingPoints(int curveIndex)
|
||||||
|
{
|
||||||
|
List<Vector2> pointList = new List<Vector2>();
|
||||||
|
|
||||||
|
Vector2 left = CalculateBezierPoint(curveIndex, 0);
|
||||||
|
Vector2 right = CalculateBezierPoint(curveIndex, 1);
|
||||||
|
|
||||||
|
pointList.Add(left);
|
||||||
|
pointList.Add(right);
|
||||||
|
|
||||||
|
FindDrawingPoints(curveIndex, 0, 1, pointList, 1);
|
||||||
|
|
||||||
|
return pointList;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
@returns the number of points added.
|
||||||
|
*/
|
||||||
|
int FindDrawingPoints(int curveIndex, float t0, float t1,
|
||||||
|
List<Vector2> pointList, int insertionIndex)
|
||||||
|
{
|
||||||
|
Vector2 left = CalculateBezierPoint(curveIndex, t0);
|
||||||
|
Vector2 right = CalculateBezierPoint(curveIndex, t1);
|
||||||
|
|
||||||
|
if ((left - right).sqrMagnitude < MINIMUM_SQR_DISTANCE)
|
||||||
|
{
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
float tMid = (t0 + t1) / 2;
|
||||||
|
Vector2 mid = CalculateBezierPoint(curveIndex, tMid);
|
||||||
|
|
||||||
|
Vector2 leftDirection = (left - mid).normalized;
|
||||||
|
Vector2 rightDirection = (right - mid).normalized;
|
||||||
|
|
||||||
|
if (Vector2.Dot(leftDirection, rightDirection) > DIVISION_THRESHOLD || Mathf.Abs(tMid - 0.5f) < 0.0001f)
|
||||||
|
{
|
||||||
|
int pointsAddedCount = 0;
|
||||||
|
|
||||||
|
pointsAddedCount += FindDrawingPoints(curveIndex, t0, tMid, pointList, insertionIndex);
|
||||||
|
pointList.Insert(insertionIndex + pointsAddedCount, mid);
|
||||||
|
pointsAddedCount++;
|
||||||
|
pointsAddedCount += FindDrawingPoints(curveIndex, tMid, t1, pointList, insertionIndex + pointsAddedCount);
|
||||||
|
|
||||||
|
return pointsAddedCount;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
Caluclates a point on the Bezier curve represented with the four controlpoints given.
|
||||||
|
*/
|
||||||
|
private Vector2 CalculateBezierPoint(float t, Vector2 p0, Vector2 p1, Vector2 p2, Vector2 p3)
|
||||||
|
{
|
||||||
|
float u = 1 - t;
|
||||||
|
float tt = t * t;
|
||||||
|
float uu = u * u;
|
||||||
|
float uuu = uu * u;
|
||||||
|
float ttt = tt * t;
|
||||||
|
|
||||||
|
Vector2 p = uuu * p0; //first term
|
||||||
|
|
||||||
|
p += 3 * uu * t * p1; //second term
|
||||||
|
p += 3 * u * tt * p2; //third term
|
||||||
|
p += ttt * p3; //fourth term
|
||||||
|
|
||||||
|
return p;
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,12 @@
|
||||||
|
fileFormatVersion: 2
|
||||||
|
guid: 89b61e0416e1256469165fca78d5ed21
|
||||||
|
timeCreated: 1463598733
|
||||||
|
licenseType: Pro
|
||||||
|
MonoImporter:
|
||||||
|
serializedVersion: 2
|
||||||
|
defaultReferences: []
|
||||||
|
executionOrder: 0
|
||||||
|
icon: {instanceID: 0}
|
||||||
|
userData:
|
||||||
|
assetBundleName:
|
||||||
|
assetBundleVariant:
|
Loading…
Reference in New Issue