SpriteManager 2
 All Classes Functions Variables Enumerations Enumerator Properties
EZScreenPlacement.cs
1 //-----------------------------------------------------------------
2 // Copyright 2011 Brady Wright and Above and Beyond Software
3 // All rights reserved
4 //-----------------------------------------------------------------
5 
6 
7 using UnityEngine;
8 using System.Collections;
9 using System.Collections.Generic;
10 
16 [ExecuteInEditMode]
17 [System.Serializable]
18 [AddComponentMenu("EZ GUI/Utility/EZ Screen Placement")]
19 public class EZScreenPlacement : MonoBehaviour, IUseCamera
20 {
24  public enum HORIZONTAL_ALIGN
25  {
29  NONE,
30 
35 
40 
45 
49  OBJECT
50  }
51 
55  public enum VERTICAL_ALIGN
56  {
60  NONE,
61 
65  SCREEN_TOP,
66 
71 
76 
80  OBJECT
81  }
82 
83  [System.Serializable]
84  public class RelativeTo
85  {
86  public HORIZONTAL_ALIGN horizontal = HORIZONTAL_ALIGN.SCREEN_LEFT;
87  public VERTICAL_ALIGN vertical = VERTICAL_ALIGN.SCREEN_TOP;
88 
89  // The script that contains this object
90  protected EZScreenPlacement script;
91 
92  public EZScreenPlacement Script
93  {
94  get { return script; }
95  set { Script = value; }
96  }
97 
98  public bool Equals(RelativeTo rt)
99  {
100  if (rt == null)
101  return false;
102  return (horizontal == rt.horizontal && vertical == rt.vertical);
103  }
104 
105  public void Copy(RelativeTo rt)
106  {
107  if (rt == null)
108  return;
109  horizontal = rt.horizontal;
110  vertical = rt.vertical;
111  }
112 
113  // Copy constructor
114  public RelativeTo(EZScreenPlacement sp, RelativeTo rt)
115  {
116  script = sp;
117  Copy(rt);
118  }
119 
120  public RelativeTo(EZScreenPlacement sp)
121  {
122  script = sp;
123  }
124  }
125 
129  public Camera renderCamera;
130 
134  public Vector3 screenPos = Vector3.forward;
135 
143  public bool ignoreZ = false;
144 
149  public RelativeTo relativeTo;
150 
156  public Transform relativeObject;
157 
164  public bool alwaysRecursive = true;
165 
176  public bool allowTransformDrag = false;
177 
178  protected Vector2 screenSize;
179 
180 #if (UNITY_3_0 || UNITY_3_1 || UNITY_3_2 || UNITY_3_3 || UNITY_3_4 || UNITY_3_5 || UNITY_3_6 || UNITY_3_7 || UNITY_3_8 || UNITY_3_9 || UNITY_4_0 || UNITY_4_1 || UNITY_4_2 || UNITY_4_3 || UNITY_4_4 || UNITY_4_5 || UNITY_4_6 || UNITY_4_7 || UNITY_4_8 || UNITY_4_9) && UNITY_EDITOR
181  [System.NonSerialized]
182  protected bool justEnabled = true; // Helps us get around a silly issue that, sometimes when an object has OnEnabled() called, it may call SetCamera() on us in edit mode, and this happens in response to an OnGUI() call in-editor, meaning we'll get invalid camera information as a result, thereby positioning the object in the wrong place, so we need to detect this and just ignore any such SetCamera() call.
183  [System.NonSerialized]
184  protected EZScreenPlacementMirror mirror = new EZScreenPlacementMirror();
185 #else
186  [System.NonSerialized]
187  protected bool justEnabled = true; // Helps us get around a silly issue that, sometimes when an object has OnEnabled() called, it may call SetCamera() on us in edit mode, and this happens in response to an OnGUI() call in-editor, meaning we'll get invalid camera information as a result, thereby positioning the object in the wrong place, so we need to detect this and just ignore any such SetCamera() call.
188  [System.NonSerialized]
189  protected EZScreenPlacementMirror mirror = new EZScreenPlacementMirror();
190 #endif
191 
192  protected bool m_awake = false;
193  protected bool m_started = false;
194 
195 
196  void Awake()
197  {
198  if (m_awake)
199  return;
200  m_awake = true;
201 
202  IUseCamera uc = (IUseCamera)GetComponent("IUseCamera");
203  if (uc != null)
204  renderCamera = uc.RenderCamera;
205 
206  if (renderCamera == null)
207  renderCamera = Camera.main;
208 
209  if (relativeTo == null)
210  relativeTo = new RelativeTo(this);
211  else if (relativeTo.Script != this)
212  {
213  // This appears to be a duplicate object,
214  // so create our own copy of the relativeTo:
215  RelativeTo newRT = new RelativeTo(this, relativeTo);
216  relativeTo = newRT;
217  }
218  }
219 
220  public void Start()
221  {
222  if (m_started)
223  return;
224  m_started = true;
225 
226  if (renderCamera != null)
227  {
228  screenSize.x = renderCamera.pixelWidth;
229  screenSize.y = renderCamera.pixelHeight;
230  }
231 
232 #if UNITY_EDITOR
233  if (relativeObject != null)
234  {
235  EZScreenPlacement c = relativeObject.GetComponent<EZScreenPlacement>();
236  if (c != null)
237  c.AddDependent(this);
238  }
239 
240  justEnabled = false;
241 #endif
242 
244  }
245 
246 #if UNITY_EDITOR
247  public void OnEnable()
248  {
249  justEnabled = true;
250  }
251 
252  public void OnDestroy()
253  {
254  if(relativeObject != null)
255  {
256  EZScreenPlacement c = relativeObject.GetComponent<EZScreenPlacement>();
257  if (c != null)
258  c.RemoveDependent(this);
259  }
260  }
261 #endif
262 
269  {
270  if (!m_started)
271  Start();
272 
273  if (relativeObject != null)
274  {
275  EZScreenPlacement sp = relativeObject.GetComponent(typeof(EZScreenPlacement)) as EZScreenPlacement;
276  if (sp != null)
277  sp.PositionOnScreenRecursively();
278  }
279 
281  }
282 
289  public Vector3 ScreenPosToLocalPos(Vector3 screenPos)
290  {
291  return transform.InverseTransformPoint(ScreenPosToWorldPos(screenPos));
292  }
293 
300  public Vector3 ScreenPosToParentPos(Vector3 screenPos)
301  {
302  return ScreenPosToLocalPos(screenPos) + transform.localPosition;
303  }
304 
311  public Vector3 ScreenPosToWorldPos(Vector3 screenPos)
312  {
313  if (!m_started)
314  Start();
315 
316  if (renderCamera == null)
317  {
318  Debug.LogError("Render camera not yet assigned to EZScreenPlacement component of \"" + name + "\" when attempting to call PositionOnScreen()");
319  return transform.position;
320  }
321 
322  Vector3 curPos = renderCamera.WorldToScreenPoint(transform.position);
323  Vector3 pos = screenPos;
324 
325  switch (relativeTo.horizontal)
326  {
327  case HORIZONTAL_ALIGN.SCREEN_RIGHT:
328  pos.x = screenSize.x + pos.x;
329  break;
330  case HORIZONTAL_ALIGN.SCREEN_CENTER:
331  pos.x = screenSize.x * 0.5f + pos.x;
332  break;
333  case HORIZONTAL_ALIGN.OBJECT:
334  if (relativeObject != null)
335  {
336  pos.x = renderCamera.WorldToScreenPoint(relativeObject.position).x + pos.x;
337  }
338  else
339  pos.x = curPos.x;
340  break;
341  case HORIZONTAL_ALIGN.NONE:
342  pos.x = curPos.x;
343  break;
344  }
345 
346  switch (relativeTo.vertical)
347  {
348  case VERTICAL_ALIGN.SCREEN_TOP:
349  pos.y = screenSize.y + pos.y;
350  break;
351  case VERTICAL_ALIGN.SCREEN_CENTER:
352  pos.y = screenSize.y * 0.5f + pos.y;
353  break;
354  case VERTICAL_ALIGN.OBJECT:
355  if (relativeObject != null)
356  {
357  pos.y = renderCamera.WorldToScreenPoint(relativeObject.position).y + pos.y;
358  }
359  else
360  pos.y = curPos.y;
361  break;
362  case VERTICAL_ALIGN.NONE:
363  pos.y = curPos.y;
364  break;
365  }
366 
367  return renderCamera.ScreenToWorldPoint(pos);
368  }
369 
370 
374  public void PositionOnScreen()
375  {
376  if (!m_awake)
377  return;
378 
379  // Keep the 'z' updated in the inspector
380  if (ignoreZ)
381  {
382  Plane plane = new Plane(renderCamera.transform.forward, renderCamera.transform.position);
383  screenPos.z = plane.GetDistanceToPoint(transform.position);
384  }
385 
386  if (ignoreZ)
387  {
388  Vector3 pos = ScreenPosToWorldPos(screenPos);
389  pos.z = transform.position.z;
390  transform.position = pos;
391  }
392  else
393  transform.position = ScreenPosToWorldPos(screenPos);
394 
395 #if UNITY_EDITOR
396  if(!Application.isPlaying)
397  UpdateDependents();
398 #endif
399 
400  // Notify the object that it was repositioned:
401  SendMessage("OnReposition", SendMessageOptions.DontRequireReceiver);
402  }
403 
411  public void PositionOnScreen(int x, int y, float depth)
412  {
413  PositionOnScreen(new Vector3((float)x, (float)y, depth));
414  }
415 
421  public void PositionOnScreen(Vector3 pos)
422  {
423  screenPos = pos;
425  }
426 
427 
433  public Camera RenderCamera
434  {
435  get { return renderCamera; }
436  set { SetCamera(value); }
437  }
438 
443  public void UpdateCamera()
444  {
446  }
447 
453  public void SetCamera()
454  {
456  }
457 
458 
463  public void SetCamera(Camera c)
464  {
465  if (c == null || !enabled)
466  return;
467 #if UNITY_EDITOR
468  if (!Application.isPlaying && justEnabled)
469  return;
470 #endif
471 
472  renderCamera = c;
473  screenSize.x = renderCamera.pixelWidth;
474  screenSize.y = renderCamera.pixelHeight;
475 
476  if (alwaysRecursive || (Application.isEditor && !Application.isPlaying))
478  else
480  }
481 
482 
483  public void WorldToScreenPos(Vector3 worldPos)
484  {
485  if (renderCamera == null)
486  return;
487 
488  Vector3 newPos = renderCamera.WorldToScreenPoint(worldPos);
489 
490  switch (relativeTo.horizontal)
491  {
492  case EZScreenPlacement.HORIZONTAL_ALIGN.SCREEN_CENTER:
493  screenPos.x = newPos.x - (renderCamera.pixelWidth / 2f);
494  break;
495  case EZScreenPlacement.HORIZONTAL_ALIGN.SCREEN_LEFT:
496  screenPos.x = newPos.x;
497  break;
498  case EZScreenPlacement.HORIZONTAL_ALIGN.SCREEN_RIGHT:
499  screenPos.x = newPos.x - renderCamera.pixelWidth;
500  break;
501  case EZScreenPlacement.HORIZONTAL_ALIGN.OBJECT:
502  if (relativeObject != null)
503  {
504  Vector3 objPos = renderCamera.WorldToScreenPoint(relativeObject.transform.position);
505  screenPos.x = newPos.x - objPos.x;
506  }
507  break;
508  }
509 
510  switch (relativeTo.vertical)
511  {
512  case EZScreenPlacement.VERTICAL_ALIGN.SCREEN_CENTER:
513  screenPos.y = newPos.y - (renderCamera.pixelHeight / 2f);
514  break;
515  case EZScreenPlacement.VERTICAL_ALIGN.SCREEN_TOP:
516  screenPos.y = newPos.y - renderCamera.pixelHeight;
517  break;
518  case EZScreenPlacement.VERTICAL_ALIGN.SCREEN_BOTTOM:
519  screenPos.y = newPos.y;
520  break;
521  case EZScreenPlacement.VERTICAL_ALIGN.OBJECT:
522  if (relativeObject != null)
523  {
524  Vector3 objPos = renderCamera.WorldToScreenPoint(relativeObject.transform.position);
525  screenPos.y = newPos.y - objPos.y;
526  }
527  break;
528  }
529 
530  screenPos.z = newPos.z;
531 
532  if (alwaysRecursive)
534  else
536  }
537 
538 
542  public Vector3 ScreenCoord
543  {
544  get { return renderCamera.WorldToScreenPoint(transform.position); }
545  }
546 
547 
548  // Tests dependencies for circular dependency.
549  // Returns true if safe, false if circular.
550  static public bool TestDepenency(EZScreenPlacement sp)
551  {
552  if (sp.relativeObject == null)
553  return true;
554 
555  // Table of all objects in the chain of dependency:
556  List<EZScreenPlacement> objs = new List<EZScreenPlacement>();
557 
558  objs.Add(sp);
559 
560  EZScreenPlacement curObj = sp.relativeObject.GetComponent(typeof(EZScreenPlacement)) as EZScreenPlacement;
561 
562  // Walk the chain:
563  while (curObj != null)
564  {
565  if (objs.Contains(curObj))
566  return false; // Circular!
567 
568  // Add this one to the list and keep walkin'
569  objs.Add(curObj);
570 
571  // See if we're at the end of the chain:
572  if (curObj.relativeObject == null)
573  return true;
574 
575  // Get the next one:
576  curObj = curObj.relativeObject.GetComponent(typeof(EZScreenPlacement)) as EZScreenPlacement;
577  }
578 
579  return true;
580  }
581 
582 
583  // List of dependent objects.
584  [HideInInspector]
585  public EZScreenPlacement[] dependents = new EZScreenPlacement[0];
586 
587 #if UNITY_EDITOR
588  // Notify this object that it has a dependent object.
589  public void AddDependent(EZScreenPlacement sp)
590  {
591  // Ensure the object isn't already on our list:
592  foreach (EZScreenPlacement d in dependents)
593  if(d == sp)
594  return;
595 
596  List<EZScreenPlacement> temp = new List<EZScreenPlacement>();
597  temp.AddRange(dependents);
598 
599  if(!temp.Contains(sp))
600  {
601  temp.Add(sp);
602  dependents = temp.ToArray();
603  }
604  }
605 
606  // Notify the object that it has one fewer dependent.
607  public void RemoveDependent(EZScreenPlacement sp)
608  {
609  List<EZScreenPlacement> temp = new List<EZScreenPlacement>();
610  temp.AddRange(dependents);
611  temp.Remove(sp);
612  dependents = temp.ToArray();
613  }
614 
615  // Cleans dependents from the list which are no longer valid:
616  public void CleanDependents()
617  {
618  if(dependents == null)
619  dependents = new EZScreenPlacement[0];
620 
621  List<EZScreenPlacement> temp = new List<EZScreenPlacement>();
622 
623  foreach (EZScreenPlacement sp in dependents)
624  {
625  if (sp != null)
626  {
627  temp.Add(sp);
628  }
629  }
630 
631  dependents = temp.ToArray();
632 
633  if(dependents == null)
634  dependents = new EZScreenPlacement[0];
635  }
636 
637  // Updates the positions of all dependent objects
638  public void UpdateDependents()
639  {
640  CleanDependents();
641 
642  foreach (EZScreenPlacement sp in dependents)
643  if(sp != null)
644  sp.PositionOnScreen();
645  }
646 #endif
647 
648 
649  public virtual void DoMirror()
650  {
651  // Only run if we're not playing:
652  if (Application.isPlaying)
653  return;
654 
655  if (mirror == null)
656  {
657  mirror = new EZScreenPlacementMirror();
658  mirror.Mirror(this);
659  }
660 
661  mirror.Validate(this);
662 
663  // Compare our mirrored settings to the current settings
664  // to see if something was changed:
665  if (mirror.DidChange(this))
666  {
668  mirror.Mirror(this); // Update the mirror
669  }
670  }
671 
672 #if (UNITY_3_0 || UNITY_3_1 || UNITY_3_2 || UNITY_3_3 || UNITY_3_4 || UNITY_3_5 || UNITY_3_6 || UNITY_3_7 || UNITY_3_8 || UNITY_3_9 || UNITY_4_0 || UNITY_4_1 || UNITY_4_2 || UNITY_4_3 || UNITY_4_4 || UNITY_4_5 || UNITY_4_6 || UNITY_4_7 || UNITY_4_8 || UNITY_4_9) && UNITY_EDITOR
673  void Update()
674  {
675  DoMirror();
676  }
677 #else
678  // Ensures that the text is updated in the scene view
679  // while editing:
680  public virtual void OnDrawGizmosSelected()
681  {
682  DoMirror();
683  }
684 
685  // Ensures that the text is updated in the scene view
686  // while editing:
687  public virtual void OnDrawGizmos()
688  {
689  DoMirror();
690  }
691 #endif
692 
693 }
694 
695 
696 
697 // Used to automatically update an EZScreenPlacement object
698 // when its settings are modified in-editor.
699 public class EZScreenPlacementMirror
700 {
701  public Vector3 worldPos;
702  public Vector3 screenPos;
703  public EZScreenPlacement.RelativeTo relativeTo;
704  public Transform relativeObject;
705  public Camera renderCamera;
706  public Vector2 screenSize;
707 
708  public EZScreenPlacementMirror()
709  {
710  relativeTo = new EZScreenPlacement.RelativeTo(null);
711  }
712 
713  public virtual void Mirror(EZScreenPlacement sp)
714  {
715  worldPos = sp.transform.position;
716  screenPos = sp.screenPos;
717  relativeTo.Copy(sp.relativeTo);
718  relativeObject = sp.relativeObject;
719  renderCamera = sp.renderCamera;
720  screenSize = new Vector2(sp.renderCamera.pixelWidth, sp.renderCamera.pixelHeight);
721  }
722 
723  public virtual bool Validate(EZScreenPlacement sp)
724  {
725  // Only allow assignment of a relative object IF
726  // we intend to use it:
727  if (sp.relativeTo.horizontal != EZScreenPlacement.HORIZONTAL_ALIGN.OBJECT &&
728  sp.relativeTo.vertical != EZScreenPlacement.VERTICAL_ALIGN.OBJECT)
729  sp.relativeObject = null;
730 
731  // See if our dependency is circular:
732  if (sp.relativeObject != null)
733  {
734  if (!EZScreenPlacement.TestDepenency(sp))
735  {
736  Debug.LogError("ERROR: The Relative Object you recently assigned on \"" + sp.name + "\" which points to \"" + sp.relativeObject.name + "\" would create a circular dependency. Please check your placement dependencies to resolve this.");
737  sp.relativeObject = null;
738  }
739  }
740 
741  return true;
742  }
743 
744  public virtual bool DidChange(EZScreenPlacement sp)
745  {
746  if (worldPos != sp.transform.position)
747  {
748  if (sp.allowTransformDrag)
749  {
750  // Calculate new screen position:
751  sp.WorldToScreenPos(sp.transform.position);
752  }
753  else
754  sp.PositionOnScreen();
755  return true;
756  }
757  if (screenPos != sp.screenPos)
758  return true;
759  if (renderCamera != null && (screenSize.x != sp.renderCamera.pixelWidth || screenSize.y != sp.renderCamera.pixelHeight))
760  return true;
761  if (!relativeTo.Equals(sp.relativeTo))
762  return true;
763  if (renderCamera != sp.renderCamera)
764  return true;
765  if (relativeObject != sp.relativeObject)
766  {
767 #if UNITY_EDITOR
768  // Remove ourselves as a dependent on the previous object:
769  if(relativeObject != null)
770  {
771  EZScreenPlacement c = relativeObject.GetComponent<EZScreenPlacement>();
772  if (c != null)
773  c.RemoveDependent(sp);
774  }
775 
776  // Add ourselves as a dependent to the new object:
777  if(sp.relativeObject != null)
778  {
779  EZScreenPlacement c = sp.relativeObject.GetComponent<EZScreenPlacement>();
780  if (c != null)
781  c.AddDependent(sp);
782  }
783 #endif
784  return true;
785  }
786 
787  return false;
788  }
789 }
Vector3 ScreenPosToLocalPos(Vector3 screenPos)
Calculates a local position from a given screen or object-relative position, according to the current...
void UpdateCamera()
Updates the object's position based on the currently selected renderCamera.
Vector3 ScreenPosToWorldPos(Vector3 screenPos)
void SetCamera()
A no-argument version of SetCamera() that simply re-assigns the same camera to the object...
The X coordinate of screenPos will be interpreted as the number of pixels from the left edge of the s...
bool allowTransformDrag
When checked, you can simply use the transform handles in the scene view to roughly position your obj...
HORIZONTAL_ALIGN
Specifies what the object will be aligned relative to on the horizontal axis.
bool alwaysRecursive
When true, positioning of the object is always done in a recursive fashion. That is, if this object is relative to any other objects, those objects, should they also hold an EZScreenPlacement component, will be positioned before this one.
Transform relativeObject
The object to which this object is relative. NOTE: Only used if either the vertical or horizontal ele...
The Y coordinate of screenPos will be interpreted as the number of pixels from the top edge of the sc...
void PositionOnScreen()
Repositions the object using the existing screen-space settings.
void PositionOnScreen(int x, int y, float depth)
Positions the object using screen coordinates, according to the relativity settings stored in relativ...
void PositionOnScreenRecursively()
Positions the object, taking into account any object-relative dependencies, making sure the objects t...
bool ignoreZ
When set to true, the Z component of Screen Pos will be calculated based upon the distance of the obj...
void PositionOnScreen(Vector3 pos)
Positions the object using screen coordinates, according to the relativity settings stored in relativ...
Camera RenderCamera
Accessor for the camera that will be used to render this object. Use this to ensure the object is pro...
The X coordinate of screenPos will be interpreted as the number of pixels from the center of the scre...
Camera renderCamera
The camera with which this object should be positioned.
The object will not be repositioned along the X axis.
RelativeTo relativeTo
Settings object that describes what this object is positioned relative to.
Vector3 screenPos
The position of the object, relative to the screen or other object.
The Y coordinate of screenPos will be interpreted as the number of pixels from the bottom edge of the...
The X coordinate of screenPos will be interpreted as the number of pixels from the object assigned to...
void SetCamera(Camera c)
Sets the camera to be used for calculating positions.
Vector3 ScreenPosToParentPos(Vector3 screenPos)
Calculates a in the parent's local space from a given screen or object-relative position, according to the current screen-space settings.
Vector3 ScreenCoord
Retrieves the screen coordinate of the object's current position.
The X coordinate of screenPos will be interpreted as the number of pixels from the right edge of the ...
VERTICAL_ALIGN
Specifies what the object will be aligned relative to on the vertical axis.