Patched Particle System with fixes from author
Added 3 new Utility controls --HG-- branch : develop_5.3pull/413/head
parent
4024e6a786
commit
4547716f3e
|
@ -1,6 +1,19 @@
|
|||
# Unity UI Extensions release notes #
|
||||
This file contains the up to date release notes for each release of the UI Extensions project including release videos where required.
|
||||
|
||||
----------------
|
||||
##Update 1.1.1##
|
||||
|
||||
Incremental release with a few new features and fixes
|
||||
|
||||
###New / updated features###
|
||||
* New UI ScrollRect Occlusion utility
|
||||
* New UI Tween Scale utility
|
||||
* New UI Infinite ScrollRect
|
||||
|
||||
###Fixes###
|
||||
* UI Particles effect updated with minor fixes
|
||||
|
||||
----------------
|
||||
##Update 1.1##
|
||||
|
||||
|
|
|
@ -660,7 +660,7 @@ namespace UnityEditor.UI
|
|||
inputFieldRT.anchorMin = Vector2.zero;
|
||||
inputFieldRT.anchorMax = Vector2.one;
|
||||
inputFieldRT.sizeDelta = Vector2.zero;
|
||||
Events.UnityEventTools.AddPersistentListener<string>(inputField.GetComponent<InputField>().onValueChange, new UnityEngine.Events.UnityAction<string>(autoCompleteComboBox.OnValueChanged));
|
||||
Events.UnityEventTools.AddPersistentListener<string>(inputField.GetComponent<InputField>().onValueChanged, new UnityEngine.Events.UnityAction<string>(autoCompleteComboBox.OnValueChanged));
|
||||
|
||||
//Setup Overlay
|
||||
var overlayRT = overlay.GetComponent<RectTransform>();
|
||||
|
@ -756,7 +756,7 @@ namespace UnityEditor.UI
|
|||
inputFieldRT.anchorMin = Vector2.zero;
|
||||
inputFieldRT.anchorMax = Vector2.one;
|
||||
inputFieldRT.sizeDelta = Vector2.zero;
|
||||
Events.UnityEventTools.AddPersistentListener<string>(inputField.GetComponent<InputField>().onValueChange, new UnityEngine.Events.UnityAction<string>(comboBox.OnValueChanged));
|
||||
Events.UnityEventTools.AddPersistentListener<string>(inputField.GetComponent<InputField>().onValueChanged, new UnityEngine.Events.UnityAction<string>(comboBox.OnValueChanged));
|
||||
|
||||
//Setup Overlay
|
||||
var overlayRT = overlay.GetComponent<RectTransform>();
|
||||
|
|
|
@ -6,230 +6,288 @@ namespace UnityEngine.UI.Extensions
|
|||
{
|
||||
#if UNITY_5_3_OR_NEWER
|
||||
[ExecuteInEditMode]
|
||||
[RequireComponent(typeof(CanvasRenderer))]
|
||||
[RequireComponent(typeof(ParticleSystem))]
|
||||
[AddComponentMenu("UI/Effects/Extensions/UI Particle System")]
|
||||
public class UIParticleSystem : MaskableGraphic {
|
||||
|
||||
public Texture particleTexture;
|
||||
public Sprite particleSprite;
|
||||
|
||||
private Transform _transform;
|
||||
private ParticleSystem _particleSystem;
|
||||
private ParticleSystem.Particle[] _particles;
|
||||
private UIVertex[] _quad = new UIVertex[4];
|
||||
private Vector4 _uv = Vector4.zero;
|
||||
private ParticleSystem.TextureSheetAnimationModule _textureSheetAnimation;
|
||||
private int _textureSheetAnimationFrames;
|
||||
private Vector2 _textureSheedAnimationFrameSize;
|
||||
|
||||
public override Texture mainTexture {
|
||||
get {
|
||||
if (particleTexture) {
|
||||
return particleTexture;
|
||||
}
|
||||
|
||||
if (particleSprite) {
|
||||
return particleSprite.texture;
|
||||
}
|
||||
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
protected bool Initialize() {
|
||||
// initialize members
|
||||
if (_transform == null) {
|
||||
_transform = transform;
|
||||
}
|
||||
if (_particleSystem == null) {
|
||||
_particleSystem = GetComponent<ParticleSystem>();
|
||||
|
||||
if (_particleSystem == null) {
|
||||
return false;
|
||||
}
|
||||
|
||||
// automatically set material to UI/Particles/Hidden shader, and get previous texture
|
||||
ParticleSystemRenderer renderer = _particleSystem.GetComponent<ParticleSystemRenderer>();
|
||||
if (renderer == null) {
|
||||
renderer = _particleSystem.gameObject.AddComponent<ParticleSystemRenderer>();
|
||||
}
|
||||
Material currentMaterial = renderer.sharedMaterial;
|
||||
if (currentMaterial && currentMaterial.HasProperty("_MainTex")) {
|
||||
particleTexture = currentMaterial.mainTexture;
|
||||
}
|
||||
Material material = new Material(Shader.Find("UI/Particles/Hidden")); // TODO - You should create this discard shader
|
||||
if (Application.isPlaying) {
|
||||
renderer.material = material;
|
||||
}
|
||||
[RequireComponent(typeof(CanvasRenderer))]
|
||||
[RequireComponent(typeof(ParticleSystem))]
|
||||
public class UIParticleSystem : MaskableGraphic
|
||||
{
|
||||
|
||||
public Texture particleTexture;
|
||||
public Sprite particleSprite;
|
||||
|
||||
private Transform _transform;
|
||||
private ParticleSystem _particleSystem;
|
||||
private ParticleSystem.Particle[] _particles;
|
||||
private UIVertex[] _quad = new UIVertex[4];
|
||||
private Vector4 _uv = Vector4.zero;
|
||||
private ParticleSystem.TextureSheetAnimationModule _textureSheetAnimation;
|
||||
private int _textureSheetAnimationFrames;
|
||||
private Vector2 _textureSheedAnimationFrameSize;
|
||||
|
||||
public override Texture mainTexture
|
||||
{
|
||||
get
|
||||
{
|
||||
if (particleTexture)
|
||||
{
|
||||
return particleTexture;
|
||||
}
|
||||
|
||||
if (particleSprite)
|
||||
{
|
||||
return particleSprite.texture;
|
||||
}
|
||||
|
||||
return null;
|
||||
}
|
||||
}
|
||||
|
||||
protected bool Initialize()
|
||||
{
|
||||
// initialize members
|
||||
if (_transform == null)
|
||||
{
|
||||
_transform = transform;
|
||||
}
|
||||
|
||||
// prepare particle system
|
||||
ParticleSystemRenderer renderer = GetComponent<ParticleSystemRenderer>();
|
||||
bool setParticleSystemMaterial = false;
|
||||
|
||||
if (_particleSystem == null)
|
||||
{
|
||||
_particleSystem = GetComponent<ParticleSystem>();
|
||||
|
||||
if (_particleSystem == null)
|
||||
{
|
||||
return false;
|
||||
}
|
||||
|
||||
// get current particle texture
|
||||
if (renderer == null)
|
||||
{
|
||||
renderer = _particleSystem.gameObject.AddComponent<ParticleSystemRenderer>();
|
||||
}
|
||||
Material currentMaterial = renderer.sharedMaterial;
|
||||
if (currentMaterial && currentMaterial.HasProperty("_MainTex"))
|
||||
{
|
||||
particleTexture = currentMaterial.mainTexture;
|
||||
}
|
||||
|
||||
// automatically set scaling
|
||||
_particleSystem.scalingMode = ParticleSystemScalingMode.Local;
|
||||
|
||||
_particles = null;
|
||||
setParticleSystemMaterial = true;
|
||||
}
|
||||
else
|
||||
{
|
||||
if (Application.isPlaying)
|
||||
{
|
||||
setParticleSystemMaterial = (renderer.material == null);
|
||||
}
|
||||
#if UNITY_EDITOR
|
||||
else {
|
||||
material.hideFlags = HideFlags.DontSave;
|
||||
renderer.sharedMaterial = material;
|
||||
}
|
||||
else
|
||||
{
|
||||
setParticleSystemMaterial = (renderer.sharedMaterial == null);
|
||||
}
|
||||
#endif
|
||||
|
||||
// automatically set scaling
|
||||
_particleSystem.scalingMode = ParticleSystemScalingMode.Hierarchy;
|
||||
|
||||
_particles = null;
|
||||
}
|
||||
if (_particles == null) {
|
||||
_particles = new ParticleSystem.Particle[_particleSystem.maxParticles];
|
||||
}
|
||||
|
||||
// prepare uvs
|
||||
if (particleTexture) {
|
||||
_uv = new Vector4(0, 0, 1, 1);
|
||||
} else if (particleSprite) {
|
||||
_uv = UnityEngine.Sprites.DataUtility.GetOuterUV(particleSprite);
|
||||
}
|
||||
|
||||
// prepare texture sheet animation
|
||||
_textureSheetAnimation = _particleSystem.textureSheetAnimation;
|
||||
_textureSheetAnimationFrames = 0;
|
||||
_textureSheedAnimationFrameSize = Vector2.zero;
|
||||
if (_textureSheetAnimation.enabled) {
|
||||
_textureSheetAnimationFrames = _textureSheetAnimation.numTilesX * _textureSheetAnimation.numTilesY;
|
||||
_textureSheedAnimationFrameSize = new Vector2(1f / _textureSheetAnimation.numTilesX, 1f / _textureSheetAnimation.numTilesY);
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
protected override void Awake() {
|
||||
base.Awake();
|
||||
|
||||
if (!Initialize()) {
|
||||
enabled = false;
|
||||
}
|
||||
}
|
||||
|
||||
protected override void OnPopulateMesh(VertexHelper vh) {
|
||||
}
|
||||
|
||||
// automatically set material to UI/Particles/Hidden shader, and get previous texture
|
||||
if (setParticleSystemMaterial)
|
||||
{
|
||||
Material material = new Material(Shader.Find("UI/Particles/Hidden"));
|
||||
if (Application.isPlaying)
|
||||
{
|
||||
renderer.material = material;
|
||||
}
|
||||
#if UNITY_EDITOR
|
||||
if (!Application.isPlaying) {
|
||||
if (!Initialize()) {
|
||||
return;
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
material.hideFlags = HideFlags.DontSave;
|
||||
renderer.sharedMaterial = material;
|
||||
}
|
||||
#endif
|
||||
|
||||
// prepare vertices
|
||||
vh.Clear();
|
||||
|
||||
if (!gameObject.activeInHierarchy) {
|
||||
return;
|
||||
}
|
||||
|
||||
// iterate through current particles
|
||||
int count = _particleSystem.GetParticles(_particles);
|
||||
|
||||
for (int i = 0; i < count; ++i) {
|
||||
ParticleSystem.Particle particle = _particles[i];
|
||||
|
||||
// get particle properties
|
||||
Vector2 position = (_particleSystem.simulationSpace == ParticleSystemSimulationSpace.Local ? particle.position : _transform.InverseTransformPoint(particle.position));
|
||||
float rotation = -particle.rotation * Mathf.Deg2Rad;
|
||||
float rotation90 = rotation + Mathf.PI / 2;
|
||||
Color32 color = particle.GetCurrentColor(_particleSystem);
|
||||
float size = particle.GetCurrentSize(_particleSystem) * 0.5f;
|
||||
|
||||
// apply scale
|
||||
if (_particleSystem.scalingMode == ParticleSystemScalingMode.Shape) {
|
||||
position /= canvas.scaleFactor;
|
||||
}
|
||||
|
||||
// apply texture sheet animation
|
||||
Vector4 particleUV = _uv;
|
||||
if (_textureSheetAnimation.enabled) {
|
||||
float frameProgress = 1 - (particle.lifetime / particle.startLifetime);
|
||||
// float frameProgress = textureSheetAnimation.frameOverTime.curveMin.Evaluate(1 - (particle.lifetime / particle.startLifetime)); // TODO - once Unity allows MinMaxCurve reading
|
||||
frameProgress = Mathf.Repeat(frameProgress * _textureSheetAnimation.cycleCount, 1);
|
||||
int frame = 0;
|
||||
|
||||
switch (_textureSheetAnimation.animation) {
|
||||
|
||||
case ParticleSystemAnimationType.WholeSheet:
|
||||
frame = Mathf.FloorToInt(frameProgress * _textureSheetAnimationFrames);
|
||||
break;
|
||||
|
||||
case ParticleSystemAnimationType.SingleRow:
|
||||
frame = Mathf.FloorToInt(frameProgress * _textureSheetAnimation.numTilesX);
|
||||
|
||||
int row = _textureSheetAnimation.rowIndex;
|
||||
// if (textureSheetAnimation.useRandomRow) { // FIXME - is this handled internally by rowIndex?
|
||||
// row = Random.Range(0, textureSheetAnimation.numTilesY, using: particle.randomSeed);
|
||||
// }
|
||||
frame += row * _textureSheetAnimation.numTilesX;
|
||||
break;
|
||||
|
||||
}
|
||||
|
||||
frame %= _textureSheetAnimationFrames;
|
||||
|
||||
particleUV.x = (frame % _textureSheetAnimation.numTilesX) * _textureSheedAnimationFrameSize.x;
|
||||
particleUV.y = Mathf.FloorToInt(frame / _textureSheetAnimation.numTilesX) * _textureSheedAnimationFrameSize.y;
|
||||
particleUV.z = particleUV.x + _textureSheedAnimationFrameSize.x;
|
||||
particleUV.w = particleUV.y + _textureSheedAnimationFrameSize.y;
|
||||
}
|
||||
|
||||
_quad[0] = UIVertex.simpleVert;
|
||||
_quad[0].color = color;
|
||||
_quad[0].uv0 = new Vector2(particleUV.x, particleUV.y);
|
||||
|
||||
_quad[1] = UIVertex.simpleVert;
|
||||
_quad[1].color = color;
|
||||
_quad[1].uv0 = new Vector2(particleUV.x, particleUV.w);
|
||||
|
||||
_quad[2] = UIVertex.simpleVert;
|
||||
_quad[2].color = color;
|
||||
_quad[2].uv0 = new Vector2(particleUV.z, particleUV.w);
|
||||
|
||||
_quad[3] = UIVertex.simpleVert;
|
||||
_quad[3].color = color;
|
||||
_quad[3].uv0 = new Vector2(particleUV.z, particleUV.y);
|
||||
|
||||
if (rotation == 0) {
|
||||
// no rotation
|
||||
Vector2 corner1 = new Vector2(position.x - size, position.y - size);
|
||||
Vector2 corner2 = new Vector2(position.x + size, position.y + size);
|
||||
|
||||
_quad[0].position = new Vector2(corner1.x, corner1.y);
|
||||
_quad[1].position = new Vector2(corner1.x, corner2.y);
|
||||
_quad[2].position = new Vector2(corner2.x, corner2.y);
|
||||
_quad[3].position = new Vector2(corner2.x, corner1.y);
|
||||
} else {
|
||||
// apply rotation
|
||||
Vector2 right = new Vector2(Mathf.Cos(rotation), Mathf.Sin(rotation)) * size;
|
||||
Vector2 up = new Vector2(Mathf.Cos(rotation90), Mathf.Sin(rotation90)) * size;
|
||||
|
||||
_quad[0].position = position - right - up;
|
||||
_quad[1].position = position - right + up;
|
||||
_quad[2].position = position + right + up;
|
||||
_quad[3].position = position + right - up;
|
||||
}
|
||||
|
||||
vh.AddUIVertexQuad(_quad);
|
||||
}
|
||||
}
|
||||
|
||||
void Update() {
|
||||
if (Application.isPlaying) {
|
||||
// unscaled animation within UI
|
||||
_particleSystem.Simulate(Time.unscaledDeltaTime, false, false);
|
||||
|
||||
SetAllDirty();
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
// prepare particles array
|
||||
if (_particles == null)
|
||||
{
|
||||
_particles = new ParticleSystem.Particle[_particleSystem.maxParticles];
|
||||
}
|
||||
|
||||
// prepare uvs
|
||||
if (particleTexture)
|
||||
{
|
||||
_uv = new Vector4(0, 0, 1, 1);
|
||||
}
|
||||
else if (particleSprite)
|
||||
{
|
||||
_uv = UnityEngine.Sprites.DataUtility.GetOuterUV(particleSprite);
|
||||
}
|
||||
|
||||
// prepare texture sheet animation
|
||||
_textureSheetAnimation = _particleSystem.textureSheetAnimation;
|
||||
_textureSheetAnimationFrames = 0;
|
||||
_textureSheedAnimationFrameSize = Vector2.zero;
|
||||
if (_textureSheetAnimation.enabled)
|
||||
{
|
||||
_textureSheetAnimationFrames = _textureSheetAnimation.numTilesX * _textureSheetAnimation.numTilesY;
|
||||
_textureSheedAnimationFrameSize = new Vector2(1f / _textureSheetAnimation.numTilesX, 1f / _textureSheetAnimation.numTilesY);
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
protected override void Awake()
|
||||
{
|
||||
base.Awake();
|
||||
|
||||
if (!Initialize())
|
||||
{
|
||||
enabled = false;
|
||||
}
|
||||
}
|
||||
|
||||
protected override void OnPopulateMesh(VertexHelper vh)
|
||||
{
|
||||
#if UNITY_EDITOR
|
||||
void LateUpdate() {
|
||||
if (!Application.isPlaying) {
|
||||
SetAllDirty();
|
||||
}
|
||||
}
|
||||
if (!Application.isPlaying)
|
||||
{
|
||||
if (!Initialize())
|
||||
{
|
||||
return;
|
||||
}
|
||||
}
|
||||
#endif
|
||||
|
||||
}
|
||||
|
||||
// prepare vertices
|
||||
vh.Clear();
|
||||
|
||||
if (!gameObject.activeInHierarchy)
|
||||
{
|
||||
return;
|
||||
}
|
||||
|
||||
// iterate through current particles
|
||||
int count = _particleSystem.GetParticles(_particles);
|
||||
|
||||
for (int i = 0; i < count; ++i)
|
||||
{
|
||||
ParticleSystem.Particle particle = _particles[i];
|
||||
|
||||
// get particle properties
|
||||
Vector2 position = (_particleSystem.simulationSpace == ParticleSystemSimulationSpace.Local ? particle.position : _transform.InverseTransformPoint(particle.position));
|
||||
float rotation = -particle.rotation * Mathf.Deg2Rad;
|
||||
float rotation90 = rotation + Mathf.PI / 2;
|
||||
Color32 color = particle.GetCurrentColor(_particleSystem);
|
||||
float size = particle.GetCurrentSize(_particleSystem) * 0.5f;
|
||||
|
||||
// apply scale
|
||||
if (_particleSystem.scalingMode == ParticleSystemScalingMode.Shape)
|
||||
{
|
||||
position /= canvas.scaleFactor;
|
||||
}
|
||||
|
||||
// apply texture sheet animation
|
||||
Vector4 particleUV = _uv;
|
||||
if (_textureSheetAnimation.enabled)
|
||||
{
|
||||
float frameProgress = 1 - (particle.lifetime / particle.startLifetime);
|
||||
// float frameProgress = textureSheetAnimation.frameOverTime.curveMin.Evaluate(1 - (particle.lifetime / particle.startLifetime)); // TODO - once Unity allows MinMaxCurve reading
|
||||
frameProgress = Mathf.Repeat(frameProgress * _textureSheetAnimation.cycleCount, 1);
|
||||
int frame = 0;
|
||||
|
||||
switch (_textureSheetAnimation.animation)
|
||||
{
|
||||
|
||||
case ParticleSystemAnimationType.WholeSheet:
|
||||
frame = Mathf.FloorToInt(frameProgress * _textureSheetAnimationFrames);
|
||||
break;
|
||||
|
||||
case ParticleSystemAnimationType.SingleRow:
|
||||
frame = Mathf.FloorToInt(frameProgress * _textureSheetAnimation.numTilesX);
|
||||
|
||||
int row = _textureSheetAnimation.rowIndex;
|
||||
// if (textureSheetAnimation.useRandomRow) { // FIXME - is this handled internally by rowIndex?
|
||||
// row = Random.Range(0, textureSheetAnimation.numTilesY, using: particle.randomSeed);
|
||||
// }
|
||||
frame += row * _textureSheetAnimation.numTilesX;
|
||||
break;
|
||||
|
||||
}
|
||||
|
||||
frame %= _textureSheetAnimationFrames;
|
||||
|
||||
particleUV.x = (frame % _textureSheetAnimation.numTilesX) * _textureSheedAnimationFrameSize.x;
|
||||
particleUV.y = Mathf.FloorToInt(frame / _textureSheetAnimation.numTilesX) * _textureSheedAnimationFrameSize.y;
|
||||
particleUV.z = particleUV.x + _textureSheedAnimationFrameSize.x;
|
||||
particleUV.w = particleUV.y + _textureSheedAnimationFrameSize.y;
|
||||
}
|
||||
|
||||
_quad[0] = UIVertex.simpleVert;
|
||||
_quad[0].color = color;
|
||||
_quad[0].uv0 = new Vector2(particleUV.x, particleUV.y);
|
||||
|
||||
_quad[1] = UIVertex.simpleVert;
|
||||
_quad[1].color = color;
|
||||
_quad[1].uv0 = new Vector2(particleUV.x, particleUV.w);
|
||||
|
||||
_quad[2] = UIVertex.simpleVert;
|
||||
_quad[2].color = color;
|
||||
_quad[2].uv0 = new Vector2(particleUV.z, particleUV.w);
|
||||
|
||||
_quad[3] = UIVertex.simpleVert;
|
||||
_quad[3].color = color;
|
||||
_quad[3].uv0 = new Vector2(particleUV.z, particleUV.y);
|
||||
|
||||
if (rotation == 0)
|
||||
{
|
||||
// no rotation
|
||||
Vector2 corner1 = new Vector2(position.x - size, position.y - size);
|
||||
Vector2 corner2 = new Vector2(position.x + size, position.y + size);
|
||||
|
||||
_quad[0].position = new Vector2(corner1.x, corner1.y);
|
||||
_quad[1].position = new Vector2(corner1.x, corner2.y);
|
||||
_quad[2].position = new Vector2(corner2.x, corner2.y);
|
||||
_quad[3].position = new Vector2(corner2.x, corner1.y);
|
||||
}
|
||||
else
|
||||
{
|
||||
// apply rotation
|
||||
Vector2 right = new Vector2(Mathf.Cos(rotation), Mathf.Sin(rotation)) * size;
|
||||
Vector2 up = new Vector2(Mathf.Cos(rotation90), Mathf.Sin(rotation90)) * size;
|
||||
|
||||
_quad[0].position = position - right - up;
|
||||
_quad[1].position = position - right + up;
|
||||
_quad[2].position = position + right + up;
|
||||
_quad[3].position = position + right - up;
|
||||
}
|
||||
|
||||
vh.AddUIVertexQuad(_quad);
|
||||
}
|
||||
}
|
||||
|
||||
void Update()
|
||||
{
|
||||
if (Application.isPlaying)
|
||||
{
|
||||
// unscaled animation within UI
|
||||
_particleSystem.Simulate(Time.unscaledDeltaTime, false, false);
|
||||
|
||||
SetAllDirty();
|
||||
}
|
||||
}
|
||||
|
||||
#if UNITY_EDITOR
|
||||
void LateUpdate()
|
||||
{
|
||||
if (!Application.isPlaying)
|
||||
{
|
||||
SetAllDirty();
|
||||
}
|
||||
}
|
||||
#endif
|
||||
}
|
||||
#endif
|
||||
}
|
|
@ -0,0 +1,173 @@
|
|||
/// Credit Tomasz Schelenz
|
||||
/// Sourced from - https://bitbucket.org/ddreaper/unity-ui-extensions/issues/81/infinite-scrollrect
|
||||
/// Demo - https://www.youtube.com/watch?v=uVTV7Udx78k - configures automatically. - works in both vertical and horizontal (but not both at the same time) - drag and drop - can be initialized by code (in case you populate your scrollview content from code)
|
||||
|
||||
using System.Collections.Generic;
|
||||
|
||||
namespace UnityEngine.UI.Extensions
|
||||
{
|
||||
/// <summary>
|
||||
/// Infinite scroll view with automatic configuration
|
||||
///
|
||||
/// Fields
|
||||
/// - InitByUSer - in case your scrollrect is populated from code, you can explicitly Initialize the infinite scroll after your scroll is ready
|
||||
/// by callin Init() method
|
||||
///
|
||||
/// Notes
|
||||
/// - doesn't work in both vertical and horizontal orientation at the same time.
|
||||
/// - in order to work it disables layout components and size fitter if present(automatically)
|
||||
///
|
||||
/// </summary>
|
||||
[AddComponentMenu("UI/Extensions/UI Infinite Scroll")]
|
||||
public class UI_InfiniteScroll : MonoBehaviour
|
||||
{
|
||||
//if true user will need to call Init() method manually (in case the contend of the scrollview is generated from code or requires special initialization)
|
||||
[Tooltip("If false, will Init automatically, otherwise you need to call Init() method")]
|
||||
public bool InitByUser = false;
|
||||
private ScrollRect _scrollRect;
|
||||
private ContentSizeFitter _contentSizeFitter;
|
||||
private VerticalLayoutGroup _verticalLayoutGroup;
|
||||
private HorizontalLayoutGroup _horizontalLayoutGroup;
|
||||
private GridLayoutGroup _gridLayoutGroup;
|
||||
private bool _isVertical = false;
|
||||
private bool _isHorizontal = false;
|
||||
private float _disableMarginX = 0;
|
||||
private float _disableMarginY = 0;
|
||||
private bool _hasDisabledGridComponents = false;
|
||||
private List<RectTransform> items = new List<RectTransform>();
|
||||
private Vector2 _newAnchoredPosition = Vector2.zero;
|
||||
//TO DISABLE FLICKERING OBJECT WHEN SCROLL VIEW IS IDLE IN BETWEEN OBJECTS
|
||||
private float _treshold = 100f;
|
||||
private int _itemCount = 0;
|
||||
private float _recordOffsetX = 0;
|
||||
private float _recordOffsetY = 0;
|
||||
|
||||
void Awake()
|
||||
{
|
||||
if (!InitByUser)
|
||||
Init();
|
||||
}
|
||||
|
||||
public void Init()
|
||||
{
|
||||
if (GetComponent<ScrollRect>() != null)
|
||||
{
|
||||
_scrollRect = GetComponent<ScrollRect>();
|
||||
_scrollRect.onValueChanged.AddListener(OnScroll);
|
||||
_scrollRect.movementType = ScrollRect.MovementType.Unrestricted;
|
||||
|
||||
for (int i = 0; i < _scrollRect.content.childCount; i++)
|
||||
{
|
||||
items.Add(_scrollRect.content.GetChild(i).GetComponent<RectTransform>());
|
||||
}
|
||||
if (_scrollRect.content.GetComponent<VerticalLayoutGroup>() != null)
|
||||
{
|
||||
_verticalLayoutGroup = _scrollRect.content.GetComponent<VerticalLayoutGroup>();
|
||||
}
|
||||
if (_scrollRect.content.GetComponent<HorizontalLayoutGroup>() != null)
|
||||
{
|
||||
_horizontalLayoutGroup = _scrollRect.content.GetComponent<HorizontalLayoutGroup>();
|
||||
}
|
||||
if (_scrollRect.content.GetComponent<GridLayoutGroup>() != null)
|
||||
{
|
||||
_gridLayoutGroup = _scrollRect.content.GetComponent<GridLayoutGroup>();
|
||||
}
|
||||
if (_scrollRect.content.GetComponent<ContentSizeFitter>() != null)
|
||||
{
|
||||
_contentSizeFitter = _scrollRect.content.GetComponent<ContentSizeFitter>();
|
||||
}
|
||||
|
||||
_isHorizontal = _scrollRect.horizontal;
|
||||
_isVertical = _scrollRect.vertical;
|
||||
|
||||
if (_isHorizontal && _isVertical)
|
||||
{
|
||||
Debug.LogError("UI_InfiniteScroll doesn't support scrolling in both directions, plase choose one direction (horizontal or vertical)");
|
||||
}
|
||||
|
||||
_itemCount = _scrollRect.content.childCount;
|
||||
}
|
||||
else
|
||||
{
|
||||
Debug.LogError("UI_InfiniteScroll => No ScrollRect component found");
|
||||
}
|
||||
}
|
||||
|
||||
void DisableGridComponents()
|
||||
{
|
||||
if (_isVertical)
|
||||
{
|
||||
_recordOffsetY = items[0].GetComponent<RectTransform>().anchoredPosition.y - items[1].GetComponent<RectTransform>().anchoredPosition.y;
|
||||
_disableMarginY = _recordOffsetY * _itemCount / 2;// _scrollRect.GetComponent<RectTransform>().rect.height/2 + items[0].sizeDelta.y;
|
||||
}
|
||||
if (_isHorizontal)
|
||||
{
|
||||
_recordOffsetX = items[1].GetComponent<RectTransform>().anchoredPosition.x - items[0].GetComponent<RectTransform>().anchoredPosition.x;
|
||||
_disableMarginX = _recordOffsetX * _itemCount / 2;//_scrollRect.GetComponent<RectTransform>().rect.width/2 + items[0].sizeDelta.x;
|
||||
}
|
||||
|
||||
if (_verticalLayoutGroup)
|
||||
{
|
||||
_verticalLayoutGroup.enabled = false;
|
||||
}
|
||||
if (_horizontalLayoutGroup)
|
||||
{
|
||||
_horizontalLayoutGroup.enabled = false;
|
||||
}
|
||||
if (_contentSizeFitter)
|
||||
{
|
||||
_contentSizeFitter.enabled = false;
|
||||
}
|
||||
if (_gridLayoutGroup)
|
||||
{
|
||||
_gridLayoutGroup.enabled = false;
|
||||
}
|
||||
_hasDisabledGridComponents = true;
|
||||
}
|
||||
|
||||
public void OnScroll(Vector2 pos)
|
||||
{
|
||||
if (!_hasDisabledGridComponents)
|
||||
DisableGridComponents();
|
||||
|
||||
for (int i = 0; i < items.Count; i++)
|
||||
{
|
||||
if (_isHorizontal)
|
||||
{
|
||||
if (_scrollRect.transform.InverseTransformPoint(items[i].gameObject.transform.position).x > _disableMarginX + _treshold)
|
||||
{
|
||||
_newAnchoredPosition = items[i].anchoredPosition;
|
||||
_newAnchoredPosition.x -= _itemCount * _recordOffsetX;
|
||||
items[i].anchoredPosition = _newAnchoredPosition;
|
||||
_scrollRect.content.GetChild(_itemCount - 1).transform.SetAsFirstSibling();
|
||||
}
|
||||
else if (_scrollRect.transform.InverseTransformPoint(items[i].gameObject.transform.position).x < -_disableMarginX)
|
||||
{
|
||||
_newAnchoredPosition = items[i].anchoredPosition;
|
||||
_newAnchoredPosition.x += _itemCount * _recordOffsetX;
|
||||
items[i].anchoredPosition = _newAnchoredPosition;
|
||||
_scrollRect.content.GetChild(0).transform.SetAsLastSibling();
|
||||
}
|
||||
}
|
||||
|
||||
if (_isVertical)
|
||||
{
|
||||
if (_scrollRect.transform.InverseTransformPoint(items[i].gameObject.transform.position).y > _disableMarginY + _treshold)
|
||||
{
|
||||
_newAnchoredPosition = items[i].anchoredPosition;
|
||||
_newAnchoredPosition.y -= _itemCount * _recordOffsetY;
|
||||
items[i].anchoredPosition = _newAnchoredPosition;
|
||||
_scrollRect.content.GetChild(_itemCount - 1).transform.SetAsFirstSibling();
|
||||
}
|
||||
else if (_scrollRect.transform.InverseTransformPoint(items[i].gameObject.transform.position).y < -_disableMarginY)
|
||||
{
|
||||
_newAnchoredPosition = items[i].anchoredPosition;
|
||||
_newAnchoredPosition.y += _itemCount * _recordOffsetY;
|
||||
items[i].anchoredPosition = _newAnchoredPosition;
|
||||
_scrollRect.content.GetChild(0).transform.SetAsLastSibling();
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
|
@ -0,0 +1,12 @@
|
|||
fileFormatVersion: 2
|
||||
guid: 61fe0f6272fa94d4eb3dafc6aecd50ff
|
||||
timeCreated: 1467462282
|
||||
licenseType: Pro
|
||||
MonoImporter:
|
||||
serializedVersion: 2
|
||||
defaultReferences: []
|
||||
executionOrder: 0
|
||||
icon: {instanceID: 0}
|
||||
userData:
|
||||
assetBundleName:
|
||||
assetBundleVariant:
|
|
@ -0,0 +1,165 @@
|
|||
/// Credit Tomasz Schelenz
|
||||
/// Sourced from - https://bitbucket.org/ddreaper/unity-ui-extensions/issues/82/scrollrectocclusion
|
||||
/// Demo - https://youtu.be/uVTV7Udx78k?t=39s ScrollRectOcclusion - disables the objects outside of the scrollrect viewport. Usefull for scrolls with lots of content, reduces geometry and drawcalls (if content is not batched) In some cases it might create a bit of spikes, especially if you have lots of UI.Text objects in the childs. In that case consider to Add CanvasGroup to your childs and instead of calling setActive on game object change CanvasGroup.alpha value. At 0 it is not being rendered hence will also optimize the prformance.
|
||||
|
||||
using System.Collections.Generic;
|
||||
|
||||
namespace UnityEngine.UI.Extensions
|
||||
{
|
||||
/// <summary>
|
||||
/// ScrollRectOcclusion - disables the objects outside of the scrollrect viewport.
|
||||
/// Useful for scrolls with lots of content, reduces geometry and drawcalls (if content is not batched)
|
||||
///
|
||||
/// Fields
|
||||
/// - InitByUSer - in case your scrollrect is populated from code, you can explicitly Initialize the infinite scroll after your scroll is ready
|
||||
/// by calling Init() method
|
||||
///
|
||||
/// Notes
|
||||
/// - In some cases it might create a bit of spikes, especially if you have lots of UI.Text objects in the child's. In that case consider to Add
|
||||
/// CanvasGroup to your child's and instead of calling setActive on game object change CanvasGroup.alpha value. At 0 it is not being rendered hence will
|
||||
/// also optimize the performance.
|
||||
/// - works for both vertical and horizontal scrolls, even at the same time (grid layout)
|
||||
/// - in order to work it disables layout components and size fitter if present (automatically)
|
||||
/// </summary>
|
||||
|
||||
[AddComponentMenu("UI/Extensions/UI Scrollrect Occlusion")]
|
||||
public class UI_ScrollRectOcclusion : MonoBehaviour
|
||||
{
|
||||
//if true user will need to call Init() method manually (in case the contend of the scrollview is generated from code or requires special initialization)
|
||||
public bool InitByUser = false;
|
||||
private ScrollRect _scrollRect;
|
||||
private ContentSizeFitter _contentSizeFitter;
|
||||
private VerticalLayoutGroup _verticalLayoutGroup;
|
||||
private HorizontalLayoutGroup _horizontalLayoutGroup;
|
||||
private GridLayoutGroup _gridLayoutGroup;
|
||||
private bool _isVertical = false;
|
||||
private bool _isHorizontal = false;
|
||||
private float _disableMarginX = 0;
|
||||
private float _disableMarginY = 0;
|
||||
private bool hasDisabledGridComponents = false;
|
||||
private List<RectTransform> items = new List<RectTransform>();
|
||||
|
||||
void Awake()
|
||||
{
|
||||
if (InitByUser)
|
||||
return;
|
||||
|
||||
Init();
|
||||
|
||||
}
|
||||
|
||||
public void Init()
|
||||
{
|
||||
if (GetComponent<ScrollRect>() != null)
|
||||
{
|
||||
_scrollRect = GetComponent<ScrollRect>();
|
||||
_scrollRect.onValueChanged.AddListener(OnScroll);
|
||||
|
||||
_isHorizontal = _scrollRect.horizontal;
|
||||
_isVertical = _scrollRect.vertical;
|
||||
|
||||
for (int i = 0; i < _scrollRect.content.childCount; i++)
|
||||
{
|
||||
items.Add(_scrollRect.content.GetChild(i).GetComponent<RectTransform>());
|
||||
}
|
||||
if (_scrollRect.content.GetComponent<VerticalLayoutGroup>() != null)
|
||||
{
|
||||
_verticalLayoutGroup = _scrollRect.content.GetComponent<VerticalLayoutGroup>();
|
||||
}
|
||||
if (_scrollRect.content.GetComponent<HorizontalLayoutGroup>() != null)
|
||||
{
|
||||
_horizontalLayoutGroup = _scrollRect.content.GetComponent<HorizontalLayoutGroup>();
|
||||
}
|
||||
if (_scrollRect.content.GetComponent<GridLayoutGroup>() != null)
|
||||
{
|
||||
_gridLayoutGroup = _scrollRect.content.GetComponent<GridLayoutGroup>();
|
||||
}
|
||||
if (_scrollRect.content.GetComponent<ContentSizeFitter>() != null)
|
||||
{
|
||||
_contentSizeFitter = _scrollRect.content.GetComponent<ContentSizeFitter>();
|
||||
}
|
||||
|
||||
}
|
||||
else
|
||||
{
|
||||
Debug.LogError("UI_ScrollRectOcclusion => No ScrollRect component found");
|
||||
}
|
||||
}
|
||||
|
||||
void DisableGridComponents()
|
||||
{
|
||||
if (_isVertical)
|
||||
_disableMarginY = _scrollRect.GetComponent<RectTransform>().rect.height / 2 + items[0].sizeDelta.y;
|
||||
|
||||
if (_isHorizontal)
|
||||
_disableMarginX = _scrollRect.GetComponent<RectTransform>().rect.width / 2 + items[0].sizeDelta.x;
|
||||
|
||||
if (_verticalLayoutGroup)
|
||||
{
|
||||
_verticalLayoutGroup.enabled = false;
|
||||
}
|
||||
if (_horizontalLayoutGroup)
|
||||
{
|
||||
_horizontalLayoutGroup.enabled = false;
|
||||
}
|
||||
if (_contentSizeFitter)
|
||||
{
|
||||
_contentSizeFitter.enabled = false;
|
||||
}
|
||||
if (_gridLayoutGroup)
|
||||
{
|
||||
_gridLayoutGroup.enabled = false;
|
||||
}
|
||||
hasDisabledGridComponents = true;
|
||||
}
|
||||
|
||||
public void OnScroll(Vector2 pos)
|
||||
{
|
||||
|
||||
if (!hasDisabledGridComponents)
|
||||
DisableGridComponents();
|
||||
|
||||
for (int i = 0; i < items.Count; i++)
|
||||
{
|
||||
if (_isVertical && _isHorizontal)
|
||||
{
|
||||
if (_scrollRect.transform.InverseTransformPoint(items[i].position).y < -_disableMarginY || _scrollRect.transform.InverseTransformPoint(items[i].position).y > _disableMarginY
|
||||
|| _scrollRect.transform.InverseTransformPoint(items[i].position).x < -_disableMarginX || _scrollRect.transform.InverseTransformPoint(items[i].position).x > _disableMarginX)
|
||||
{
|
||||
items[i].gameObject.SetActive(false);
|
||||
}
|
||||
else
|
||||
{
|
||||
items[i].gameObject.SetActive(true);
|
||||
}
|
||||
}
|
||||
else
|
||||
{
|
||||
if (_isVertical)
|
||||
{
|
||||
if (_scrollRect.transform.InverseTransformPoint(items[i].position).y < -_disableMarginY || _scrollRect.transform.InverseTransformPoint(items[i].position).y > _disableMarginY)
|
||||
{
|
||||
items[i].gameObject.SetActive(false);
|
||||
}
|
||||
else
|
||||
{
|
||||
items[i].gameObject.SetActive(true);
|
||||
}
|
||||
}
|
||||
|
||||
if (_isHorizontal)
|
||||
{
|
||||
if (_scrollRect.transform.InverseTransformPoint(items[i].position).x < -_disableMarginX || _scrollRect.transform.InverseTransformPoint(items[i].position).x > _disableMarginX)
|
||||
{
|
||||
items[i].gameObject.SetActive(false);
|
||||
}
|
||||
else
|
||||
{
|
||||
items[i].gameObject.SetActive(true);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
|
@ -0,0 +1,12 @@
|
|||
fileFormatVersion: 2
|
||||
guid: f1bb7fb3155d62546a876bd63cdcd81f
|
||||
timeCreated: 1467462085
|
||||
licenseType: Pro
|
||||
MonoImporter:
|
||||
serializedVersion: 2
|
||||
defaultReferences: []
|
||||
executionOrder: 0
|
||||
icon: {instanceID: 0}
|
||||
userData:
|
||||
assetBundleName:
|
||||
assetBundleVariant:
|
|
@ -0,0 +1,104 @@
|
|||
/// Credit Tomasz Schelenz
|
||||
/// Sourced from - https://bitbucket.org/ddreaper/unity-ui-extensions/issues/83/ui_tweenscale
|
||||
/// Demo - https://youtu.be/uVTV7Udx78k?t=1m33s Dynamic scalling of text or image (including button) based on curves. works on scrollrect scale so you can pretty much use it for any ui type.
|
||||
/// Notes In some cases it can create spikes due to redrawing on change, it is recommended to use it on simple objects in separated canvases to avoid redrawing full canvas.
|
||||
|
||||
|
||||
using System.Collections;
|
||||
|
||||
namespace UnityEngine.UI.Extensions
|
||||
{
|
||||
/// <summary>
|
||||
/// Dynamic scalling of text or image (including button) based on curves
|
||||
///
|
||||
/// Fields
|
||||
/// - animCurve - animation curve for scale (if isUniform set to false, will apply only to X scale)
|
||||
/// - speed - animation speed
|
||||
/// - isLoop - animation will play infinitely (in order to make it work set your animation curve to loop)
|
||||
/// - playAtAwake - starts automatically with script becoming active. Otherwise you need to call Play() method.
|
||||
/// - isUniform - if false animCurve will modify object X scale and animCurveY - Y scale.
|
||||
///
|
||||
///
|
||||
/// Notes
|
||||
/// - If you want to stop the animation call the ResetTween() method.
|
||||
/// - In some cases it can create spikes due to redrawing on change, it is recommended to use it on simple objects in separated canvases to
|
||||
/// avoid redrawing full canvas.
|
||||
/// - If you want to scale object only in 1 axis select non unifor and use linear curve from 1 to 1 to lock the scale.
|
||||
///
|
||||
/// </summary>
|
||||
[AddComponentMenu("UI/Extensions/UI Tween Scale")]
|
||||
public class UI_TweenScale : MonoBehaviour
|
||||
{
|
||||
//ANIMATION FOR X AND Y, OR X IF isUniform set to false
|
||||
public AnimationCurve animCurve;
|
||||
[Tooltip("Animation speed multiplier")]
|
||||
public float speed = 1;
|
||||
|
||||
[Tooltip("If true animation will loop, for best effect set animation curve to loop on start and end point")]
|
||||
|
||||
public bool isLoop = false;
|
||||
//IF TRUE ANIMATION STARTS AUTOMATICALLY
|
||||
[Tooltip("If true animation will start automatically, otherwise you need to call Play() method to start the animation")]
|
||||
|
||||
public bool playAtAwake = false;
|
||||
|
||||
[Space(10)]
|
||||
//if true both x and y axis will be using animCurve;
|
||||
[Header("Non uniform scale")]
|
||||
[Tooltip("If true component will scale by the same amount in X and Y axis, otherwise use animCurve for X scale and animCurveY for Y scale")]
|
||||
public bool isUniform = true;
|
||||
//IF isUniform set to false use this for Y axis
|
||||
public AnimationCurve animCurveY;
|
||||
private Vector3 initScale;
|
||||
private Transform myTransform;
|
||||
|
||||
void Awake()
|
||||
{
|
||||
myTransform = GetComponent<Transform>();
|
||||
initScale = myTransform.localScale;
|
||||
if (playAtAwake)
|
||||
{
|
||||
Play();
|
||||
}
|
||||
}
|
||||
|
||||
public void Play()
|
||||
{
|
||||
StartCoroutine("Tween");
|
||||
}
|
||||
|
||||
Vector3 newScale = Vector3.one;
|
||||
|
||||
IEnumerator Tween()
|
||||
{
|
||||
myTransform.localScale = initScale;
|
||||
float t = 0;
|
||||
float maxT = animCurve.keys[animCurve.length - 1].time;
|
||||
|
||||
while (t < maxT || isLoop)
|
||||
{
|
||||
t += speed * Time.deltaTime;
|
||||
|
||||
if (!isUniform)
|
||||
{
|
||||
newScale.x = 1 * animCurve.Evaluate(t);
|
||||
newScale.y = 1 * animCurveY.Evaluate(t);
|
||||
|
||||
myTransform.localScale = newScale;
|
||||
}
|
||||
else
|
||||
{
|
||||
myTransform.localScale = Vector3.one * animCurve.Evaluate(t);
|
||||
}
|
||||
|
||||
yield return null;
|
||||
}
|
||||
}
|
||||
|
||||
public void ResetTween()
|
||||
{
|
||||
StopCoroutine("Tween");
|
||||
myTransform.localScale = initScale;
|
||||
}
|
||||
}
|
||||
}
|
|
@ -0,0 +1,12 @@
|
|||
fileFormatVersion: 2
|
||||
guid: 61bf9d6fa29ed164e8a1df46e0d40835
|
||||
timeCreated: 1467462200
|
||||
licenseType: Pro
|
||||
MonoImporter:
|
||||
serializedVersion: 2
|
||||
defaultReferences: []
|
||||
executionOrder: 0
|
||||
icon: {instanceID: 0}
|
||||
userData:
|
||||
assetBundleName:
|
||||
assetBundleVariant:
|
Loading…
Reference in New Issue