PaniniProjection.shader 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136
  1. Shader "Hidden/Universal Render Pipeline/PaniniProjection"
  2. {
  3. Properties
  4. {
  5. _MainTex("Source", 2D) = "white" {}
  6. }
  7. HLSLINCLUDE
  8. #pragma multi_compile_local _GENERIC _UNIT_DISTANCE
  9. #include "Packages/com.unity.render-pipelines.universal/ShaderLibrary/Core.hlsl"
  10. #include "Packages/com.unity.render-pipelines.universal/Shaders/PostProcessing/Common.hlsl"
  11. TEXTURE2D_X(_MainTex);
  12. float4 _Params;
  13. // Back-ported & adapted from the work of the Stockholm demo team - thanks Lasse
  14. float2 Panini_UnitDistance(float2 view_pos)
  15. {
  16. // Given
  17. // S----------- E--X-------
  18. // | ` . /,´
  19. // |-- --- Q
  20. // 1 | ,´/ `
  21. // | ,´ / ´
  22. // | ,´ / `
  23. // | ,´ / .
  24. // O` / .
  25. // | / `
  26. // | / ´
  27. // 1 | / ´
  28. // | / ´
  29. // |/_ . ´
  30. // P
  31. //
  32. // Have E
  33. // Want to find X
  34. //
  35. // First apply tangent-secant theorem to find Q
  36. // PE*QE = SE*SE
  37. // QE = PE-PQ
  38. // PQ = PE-(SE*SE)/PE
  39. // Q = E*(PQ/PE)
  40. // Then project Q to find X
  41. const float d = 1.0;
  42. const float view_dist = 2.0;
  43. const float view_dist_sq = 4.0;
  44. float view_hyp = sqrt(view_pos.x * view_pos.x + view_dist_sq);
  45. float cyl_hyp = view_hyp - (view_pos.x * view_pos.x) / view_hyp;
  46. float cyl_hyp_frac = cyl_hyp / view_hyp;
  47. float cyl_dist = view_dist * cyl_hyp_frac;
  48. float2 cyl_pos = view_pos * cyl_hyp_frac;
  49. return cyl_pos / (cyl_dist - d);
  50. }
  51. float2 Panini_Generic(float2 view_pos, float d)
  52. {
  53. // Given
  54. // S----------- E--X-------
  55. // | ` ~. /,´
  56. // |-- --- Q
  57. // | ,/ `
  58. // 1 | ,´/ `
  59. // | ,´ / ´
  60. // | ,´ / ´
  61. // |,` / ,
  62. // O /
  63. // | / ,
  64. // d | /
  65. // | / ,
  66. // |/ .
  67. // P
  68. // | ´
  69. // | , ´
  70. // +- ´
  71. //
  72. // Have E
  73. // Want to find X
  74. //
  75. // First compute line-circle intersection to find Q
  76. // Then project Q to find X
  77. float view_dist = 1.0 + d;
  78. float view_hyp_sq = view_pos.x * view_pos.x + view_dist * view_dist;
  79. float isect_D = view_pos.x * d;
  80. float isect_discrim = view_hyp_sq - isect_D * isect_D;
  81. float cyl_dist_minus_d = (-isect_D * view_pos.x + view_dist * sqrt(isect_discrim)) / view_hyp_sq;
  82. float cyl_dist = cyl_dist_minus_d + d;
  83. float2 cyl_pos = view_pos * (cyl_dist / view_dist);
  84. return cyl_pos / (cyl_dist - d);
  85. }
  86. half4 Frag(Varyings input) : SV_Target
  87. {
  88. UNITY_SETUP_STEREO_EYE_INDEX_POST_VERTEX(input);
  89. #if _GENERIC
  90. float2 proj_pos = Panini_Generic((2.0 * input.uv - 1.0) * _Params.xy * _Params.w, _Params.z);
  91. #else // _UNIT_DISTANCE
  92. float2 proj_pos = Panini_UnitDistance((2.0 * input.uv - 1.0) * _Params.xy * _Params.w);
  93. #endif
  94. float2 proj_ndc = proj_pos / _Params.xy;
  95. float2 coords = proj_ndc * 0.5 + 0.5;
  96. return SAMPLE_TEXTURE2D_X(_MainTex, sampler_LinearClamp, coords);
  97. }
  98. ENDHLSL
  99. SubShader
  100. {
  101. Tags { "RenderType" = "Opaque" "RenderPipeline" = "UniversalPipeline"}
  102. LOD 100
  103. ZTest Always ZWrite Off Cull Off
  104. Pass
  105. {
  106. Name "Panini Projection"
  107. HLSLPROGRAM
  108. #pragma vertex Vert
  109. #pragma fragment Frag
  110. ENDHLSL
  111. }
  112. }
  113. }