RenderManager.java 39 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025
  1. /*
  2. * Copyright (c) 2009-2012 jMonkeyEngine
  3. * All rights reserved.
  4. *
  5. * Redistribution and use in source and binary forms, with or without
  6. * modification, are permitted provided that the following conditions are
  7. * met:
  8. *
  9. * * Redistributions of source code must retain the above copyright
  10. * notice, this list of conditions and the following disclaimer.
  11. *
  12. * * Redistributions in binary form must reproduce the above copyright
  13. * notice, this list of conditions and the following disclaimer in the
  14. * documentation and/or other materials provided with the distribution.
  15. *
  16. * * Neither the name of 'jMonkeyEngine' nor the names of its contributors
  17. * may be used to endorse or promote products derived from this software
  18. * without specific prior written permission.
  19. *
  20. * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
  21. * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
  22. * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
  23. * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR
  24. * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
  25. * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
  26. * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
  27. * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
  28. * LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
  29. * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
  30. * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  31. */
  32. package com.jme3.renderer;
  33. import com.jme3.material.Material;
  34. import com.jme3.material.MaterialDef;
  35. import com.jme3.material.RenderState;
  36. import com.jme3.material.Technique;
  37. import com.jme3.math.*;
  38. import com.jme3.post.SceneProcessor;
  39. import com.jme3.renderer.queue.GeometryList;
  40. import com.jme3.renderer.queue.RenderQueue;
  41. import com.jme3.renderer.queue.RenderQueue.Bucket;
  42. import com.jme3.renderer.queue.RenderQueue.ShadowMode;
  43. import com.jme3.scene.*;
  44. import com.jme3.shader.Uniform;
  45. import com.jme3.shader.UniformBinding;
  46. import com.jme3.shader.UniformBindingManager;
  47. import com.jme3.shader.VarType;
  48. import com.jme3.system.NullRenderer;
  49. import com.jme3.system.Timer;
  50. import com.jme3.util.TempVars;
  51. import java.util.ArrayList;
  52. import java.util.Collections;
  53. import java.util.List;
  54. import java.util.logging.Logger;
  55. /**
  56. * <code>RenderManager</code> is a high-level rendering interface that is
  57. * above the Renderer implementation. RenderManager takes care
  58. * of rendering the scene graphs attached to each viewport and
  59. * handling SceneProcessors.
  60. *
  61. * @see SceneProcessor
  62. * @see ViewPort
  63. * @see Spatial
  64. */
  65. public class RenderManager {
  66. private static final Logger logger = Logger.getLogger(RenderManager.class.getName());
  67. private Renderer renderer;
  68. private UniformBindingManager uniformBindingManager = new UniformBindingManager();
  69. private ArrayList<ViewPort> preViewPorts = new ArrayList<ViewPort>();
  70. private ArrayList<ViewPort> viewPorts = new ArrayList<ViewPort>();
  71. private ArrayList<ViewPort> postViewPorts = new ArrayList<ViewPort>();
  72. private Camera prevCam = null;
  73. private Material forcedMaterial = null;
  74. private String forcedTechnique = null;
  75. private RenderState forcedRenderState = null;
  76. private boolean shader;
  77. private int viewX, viewY, viewWidth, viewHeight;
  78. private Matrix4f orthoMatrix = new Matrix4f();
  79. private String tmpTech;
  80. private boolean handleTranlucentBucket = true;
  81. /**
  82. * Create a high-level rendering interface over the
  83. * low-level rendering interface.
  84. * @param renderer
  85. */
  86. public RenderManager(Renderer renderer) {
  87. this.renderer = renderer;
  88. }
  89. /**
  90. * Returns the pre ViewPort with the given name.
  91. *
  92. * @param viewName The name of the pre ViewPort to look up
  93. * @return The ViewPort, or null if not found.
  94. *
  95. * @see #createPreView(java.lang.String, com.jme3.renderer.Camera)
  96. */
  97. public ViewPort getPreView(String viewName) {
  98. for (int i = 0; i < preViewPorts.size(); i++) {
  99. if (preViewPorts.get(i).getName().equals(viewName)) {
  100. return preViewPorts.get(i);
  101. }
  102. }
  103. return null;
  104. }
  105. /**
  106. * Removes the specified pre ViewPort.
  107. *
  108. * @param view The pre ViewPort to remove
  109. * @return True if the ViewPort was removed successfully.
  110. *
  111. * @see #createPreView(java.lang.String, com.jme3.renderer.Camera)
  112. */
  113. public boolean removePreView(ViewPort view) {
  114. return preViewPorts.remove(view);
  115. }
  116. /**
  117. * Returns the main ViewPort with the given name.
  118. *
  119. * @param viewName The name of the main ViewPort to look up
  120. * @return The ViewPort, or null if not found.
  121. *
  122. * @see #createMainView(java.lang.String, com.jme3.renderer.Camera)
  123. */
  124. public ViewPort getMainView(String viewName) {
  125. for (int i = 0; i < viewPorts.size(); i++) {
  126. if (viewPorts.get(i).getName().equals(viewName)) {
  127. return viewPorts.get(i);
  128. }
  129. }
  130. return null;
  131. }
  132. /**
  133. * Removes the main ViewPort with the specified name.
  134. *
  135. * @param viewName The main ViewPort name to remove
  136. * @return True if the ViewPort was removed successfully.
  137. *
  138. * @see #createMainView(java.lang.String, com.jme3.renderer.Camera)
  139. */
  140. public boolean removeMainView(String viewName) {
  141. for (int i = 0; i < viewPorts.size(); i++) {
  142. if (viewPorts.get(i).getName().equals(viewName)) {
  143. viewPorts.remove(i);
  144. return true;
  145. }
  146. }
  147. return false;
  148. }
  149. /**
  150. * Removes the specified main ViewPort.
  151. *
  152. * @param view The main ViewPort to remove
  153. * @return True if the ViewPort was removed successfully.
  154. *
  155. * @see #createMainView(java.lang.String, com.jme3.renderer.Camera)
  156. */
  157. public boolean removeMainView(ViewPort view) {
  158. return viewPorts.remove(view);
  159. }
  160. /**
  161. * Returns the post ViewPort with the given name.
  162. *
  163. * @param viewName The name of the post ViewPort to look up
  164. * @return The ViewPort, or null if not found.
  165. *
  166. * @see #createPostView(java.lang.String, com.jme3.renderer.Camera)
  167. */
  168. public ViewPort getPostView(String viewName) {
  169. for (int i = 0; i < postViewPorts.size(); i++) {
  170. if (postViewPorts.get(i).getName().equals(viewName)) {
  171. return postViewPorts.get(i);
  172. }
  173. }
  174. return null;
  175. }
  176. /**
  177. * Removes the post ViewPort with the specified name.
  178. *
  179. * @param viewName The post ViewPort name to remove
  180. * @return True if the ViewPort was removed successfully.
  181. *
  182. * @see #createPostView(java.lang.String, com.jme3.renderer.Camera)
  183. */
  184. public boolean removePostView(String viewName) {
  185. for (int i = 0; i < postViewPorts.size(); i++) {
  186. if (postViewPorts.get(i).getName().equals(viewName)) {
  187. postViewPorts.remove(i);
  188. return true;
  189. }
  190. }
  191. return false;
  192. }
  193. /**
  194. * Removes the specified post ViewPort.
  195. *
  196. * @param view The post ViewPort to remove
  197. * @return True if the ViewPort was removed successfully.
  198. *
  199. * @see #createPostView(java.lang.String, com.jme3.renderer.Camera)
  200. */
  201. public boolean removePostView(ViewPort view) {
  202. return postViewPorts.remove(view);
  203. }
  204. /**
  205. * Returns a read-only list of all pre ViewPorts
  206. * @return a read-only list of all pre ViewPorts
  207. * @see #createPreView(java.lang.String, com.jme3.renderer.Camera)
  208. */
  209. public List<ViewPort> getPreViews() {
  210. return Collections.unmodifiableList(preViewPorts);
  211. }
  212. /**
  213. * Returns a read-only list of all main ViewPorts
  214. * @return a read-only list of all main ViewPorts
  215. * @see #createMainView(java.lang.String, com.jme3.renderer.Camera)
  216. */
  217. public List<ViewPort> getMainViews() {
  218. return Collections.unmodifiableList(viewPorts);
  219. }
  220. /**
  221. * Returns a read-only list of all post ViewPorts
  222. * @return a read-only list of all post ViewPorts
  223. * @see #createPostView(java.lang.String, com.jme3.renderer.Camera)
  224. */
  225. public List<ViewPort> getPostViews() {
  226. return Collections.unmodifiableList(postViewPorts);
  227. }
  228. /**
  229. * Creates a new pre ViewPort, to display the given camera's content.
  230. * <p>
  231. * The view will be processed before the main and post viewports.
  232. */
  233. public ViewPort createPreView(String viewName, Camera cam) {
  234. ViewPort vp = new ViewPort(viewName, cam);
  235. preViewPorts.add(vp);
  236. return vp;
  237. }
  238. /**
  239. * Creates a new main ViewPort, to display the given camera's content.
  240. * <p>
  241. * The view will be processed before the post viewports but after
  242. * the pre viewports.
  243. */
  244. public ViewPort createMainView(String viewName, Camera cam) {
  245. ViewPort vp = new ViewPort(viewName, cam);
  246. viewPorts.add(vp);
  247. return vp;
  248. }
  249. /**
  250. * Creates a new post ViewPort, to display the given camera's content.
  251. * <p>
  252. * The view will be processed after the pre and main viewports.
  253. */
  254. public ViewPort createPostView(String viewName, Camera cam) {
  255. ViewPort vp = new ViewPort(viewName, cam);
  256. postViewPorts.add(vp);
  257. return vp;
  258. }
  259. private void notifyReshape(ViewPort vp, int w, int h) {
  260. List<SceneProcessor> processors = vp.getProcessors();
  261. for (SceneProcessor proc : processors) {
  262. if (!proc.isInitialized()) {
  263. proc.initialize(this, vp);
  264. } else {
  265. proc.reshape(vp, w, h);
  266. }
  267. }
  268. }
  269. /**
  270. * Internal use only.
  271. * Updates the resolution of all on-screen cameras to match
  272. * the given width and height.
  273. */
  274. public void notifyReshape(int w, int h) {
  275. for (ViewPort vp : preViewPorts) {
  276. if (vp.getOutputFrameBuffer() == null) {
  277. Camera cam = vp.getCamera();
  278. cam.resize(w, h, true);
  279. }
  280. notifyReshape(vp, w, h);
  281. }
  282. for (ViewPort vp : viewPorts) {
  283. if (vp.getOutputFrameBuffer() == null) {
  284. Camera cam = vp.getCamera();
  285. cam.resize(w, h, true);
  286. }
  287. notifyReshape(vp, w, h);
  288. }
  289. for (ViewPort vp : postViewPorts) {
  290. if (vp.getOutputFrameBuffer() == null) {
  291. Camera cam = vp.getCamera();
  292. cam.resize(w, h, true);
  293. }
  294. notifyReshape(vp, w, h);
  295. }
  296. }
  297. /**
  298. * Set the material to use to render all future objects.
  299. * This overrides the material set on the geometry and renders
  300. * with the provided material instead.
  301. * Use null to clear the material and return renderer to normal
  302. * functionality.
  303. * @param mat The forced material to set, or null to return to normal
  304. */
  305. public void setForcedMaterial(Material mat) {
  306. forcedMaterial = mat;
  307. }
  308. /**
  309. * Returns the forced render state previously set with
  310. * {@link #setForcedRenderState(com.jme3.material.RenderState) }.
  311. * @return the forced render state
  312. */
  313. public RenderState getForcedRenderState() {
  314. return forcedRenderState;
  315. }
  316. /**
  317. * Set the render state to use for all future objects.
  318. * This overrides the render state set on the material and instead
  319. * forces this render state to be applied for all future materials
  320. * rendered. Set to null to return to normal functionality.
  321. *
  322. * @param forcedRenderState The forced render state to set, or null
  323. * to return to normal
  324. */
  325. public void setForcedRenderState(RenderState forcedRenderState) {
  326. this.forcedRenderState = forcedRenderState;
  327. }
  328. /**
  329. * Set the timer that should be used to query the time based
  330. * {@link UniformBinding}s for material world parameters.
  331. *
  332. * @param timer The timer to query time world parameters
  333. */
  334. public void setTimer(Timer timer) {
  335. uniformBindingManager.setTimer(timer);
  336. }
  337. /**
  338. * Returns the forced technique name set.
  339. *
  340. * @return the forced technique name set.
  341. *
  342. * @see #setForcedTechnique(java.lang.String)
  343. */
  344. public String getForcedTechnique() {
  345. return forcedTechnique;
  346. }
  347. /**
  348. * Sets the forced technique to use when rendering geometries.
  349. * <p>
  350. * If the specified technique name is available on the geometry's
  351. * material, then it is used, otherwise, the
  352. * {@link #setForcedMaterial(com.jme3.material.Material) forced material} is used.
  353. * If a forced material is not set and the forced technique name cannot
  354. * be found on the material, the geometry will <em>not</em> be rendered.
  355. *
  356. * @param forcedTechnique The forced technique name to use, set to null
  357. * to return to normal functionality.
  358. *
  359. * @see #renderGeometry(com.jme3.scene.Geometry)
  360. */
  361. public void setForcedTechnique(String forcedTechnique) {
  362. this.forcedTechnique = forcedTechnique;
  363. }
  364. /**
  365. * Enable or disable alpha-to-coverage.
  366. * <p>
  367. * When alpha to coverage is enabled and the renderer implementation
  368. * supports it, then alpha blending will be replaced with alpha dissolve
  369. * if multi-sampling is also set on the renderer.
  370. * This feature allows avoiding of alpha blending artifacts due to
  371. * lack of triangle-level back-to-front sorting.
  372. *
  373. * @param value True to enable alpha-to-coverage, false otherwise.
  374. */
  375. public void setAlphaToCoverage(boolean value) {
  376. renderer.setAlphaToCoverage(value);
  377. }
  378. /**
  379. * True if the translucent bucket should automatically be rendered
  380. * by the RenderManager.
  381. *
  382. * @return Whether or not the translucent bucket is rendered.
  383. *
  384. * @see #setHandleTranslucentBucket(boolean)
  385. */
  386. public boolean isHandleTranslucentBucket() {
  387. return handleTranlucentBucket;
  388. }
  389. /**
  390. * Enable or disable rendering of the
  391. * {@link Bucket#Translucent translucent bucket}
  392. * by the RenderManager. The default is enabled.
  393. *
  394. * @param handleTranslucentBucket Whether or not the translucent bucket should
  395. * be rendered.
  396. */
  397. public void setHandleTranslucentBucket(boolean handleTranslucentBucket) {
  398. this.handleTranlucentBucket = handleTranslucentBucket;
  399. }
  400. /**
  401. * Internal use only. Sets the world matrix to use for future
  402. * rendering. This has no effect unless objects are rendered manually
  403. * using {@link Material#render(com.jme3.scene.Geometry, com.jme3.renderer.RenderManager) }.
  404. * Using {@link #renderGeometry(com.jme3.scene.Geometry) } will
  405. * override this value.
  406. *
  407. * @param mat The world matrix to set
  408. */
  409. public void setWorldMatrix(Matrix4f mat) {
  410. if (shader) {
  411. uniformBindingManager.setWorldMatrix(mat);
  412. } else {
  413. renderer.setWorldMatrix(mat);
  414. }
  415. }
  416. /**
  417. * Internal use only.
  418. * Updates the given list of uniforms with {@link UniformBinding uniform bindings}
  419. * based on the current world state.
  420. */
  421. public void updateUniformBindings(List<Uniform> params) {
  422. uniformBindingManager.updateUniformBindings(params);
  423. }
  424. /**
  425. * Renders the given geometry.
  426. * <p>
  427. * First the proper world matrix is set, if
  428. * the geometry's {@link Geometry#setIgnoreTransform(boolean) ignore transform}
  429. * feature is enabled, the identity world matrix is used, otherwise, the
  430. * geometry's {@link Geometry#getWorldMatrix() world transform matrix} is used.
  431. * <p>
  432. * Once the world matrix is applied, the proper material is chosen for rendering.
  433. * If a {@link #setForcedMaterial(com.jme3.material.Material) forced material} is
  434. * set on this RenderManager, then it is used for rendering the geometry,
  435. * otherwise, the {@link Geometry#getMaterial() geometry's material} is used.
  436. * <p>
  437. * If a {@link #setForcedTechnique(java.lang.String) forced technique} is
  438. * set on this RenderManager, then it is selected automatically
  439. * on the geometry's material and is used for rendering. Otherwise, one
  440. * of the {@link MaterialDef#getDefaultTechniques() default techniques} is
  441. * used.
  442. * <p>
  443. * If a {@link #setForcedRenderState(com.jme3.material.RenderState) forced
  444. * render state} is set on this RenderManager, then it is used
  445. * for rendering the material, and the material's own render state is ignored.
  446. * Otherwise, the material's render state is used as intended.
  447. *
  448. * @param g The geometry to render
  449. *
  450. * @see Technique
  451. * @see RenderState
  452. * @see Material#selectTechnique(java.lang.String, com.jme3.renderer.RenderManager)
  453. * @see Material#render(com.jme3.scene.Geometry, com.jme3.renderer.RenderManager)
  454. */
  455. public void renderGeometry(Geometry g) {
  456. if (g.isIgnoreTransform()) {
  457. setWorldMatrix(Matrix4f.IDENTITY);
  458. } else {
  459. setWorldMatrix(g.getWorldMatrix());
  460. }
  461. //if forcedTechnique we try to force it for render,
  462. //if it does not exists in the mat def, we check for forcedMaterial and render the geom if not null
  463. //else the geom is not rendered
  464. if (forcedTechnique != null) {
  465. if (g.getMaterial().getMaterialDef().getTechniqueDef(forcedTechnique) != null) {
  466. tmpTech = g.getMaterial().getActiveTechnique() != null ? g.getMaterial().getActiveTechnique().getDef().getName() : "Default";
  467. g.getMaterial().selectTechnique(forcedTechnique, this);
  468. // use geometry's material
  469. g.getMaterial().render(g, this);
  470. g.getMaterial().selectTechnique(tmpTech, this);
  471. //Reverted this part from revision 6197
  472. //If forcedTechnique does not exists, and frocedMaterial is not set, the geom MUST NOT be rendered
  473. } else if (forcedMaterial != null) {
  474. // use forced material
  475. forcedMaterial.render(g, this);
  476. }
  477. } else if (forcedMaterial != null) {
  478. // use forced material
  479. forcedMaterial.render(g, this);
  480. } else {
  481. g.getMaterial().render(g, this);
  482. }
  483. }
  484. /**
  485. * Renders the given GeometryList.
  486. * <p>
  487. * For every geometry in the list, the
  488. * {@link #renderGeometry(com.jme3.scene.Geometry) } method is called.
  489. *
  490. * @param gl The geometry list to render.
  491. *
  492. * @see GeometryList
  493. * @see #renderGeometry(com.jme3.scene.Geometry)
  494. */
  495. public void renderGeometryList(GeometryList gl) {
  496. for (int i = 0; i < gl.size(); i++) {
  497. renderGeometry(gl.get(i));
  498. }
  499. }
  500. /**
  501. * If a spatial is not inside the eye frustum, it
  502. * is still rendered in the shadow frustum (shadow casting queue)
  503. * through this recursive method.
  504. */
  505. private void renderShadow(Spatial s, RenderQueue rq) {
  506. if (s instanceof Node) {
  507. Node n = (Node) s;
  508. List<Spatial> children = n.getChildren();
  509. for (int i = 0; i < children.size(); i++) {
  510. renderShadow(children.get(i), rq);
  511. }
  512. } else if (s instanceof Geometry) {
  513. Geometry gm = (Geometry) s;
  514. RenderQueue.ShadowMode shadowMode = s.getShadowMode();
  515. if (shadowMode != RenderQueue.ShadowMode.Off && shadowMode != RenderQueue.ShadowMode.Receive) {
  516. //forcing adding to shadow cast mode, culled objects doesn't have to be in the receiver queue
  517. rq.addToShadowQueue(gm, RenderQueue.ShadowMode.Cast);
  518. }
  519. }
  520. }
  521. /**
  522. * Preloads a scene for rendering.
  523. * <p>
  524. * After invocation of this method, the underlying
  525. * renderer would have uploaded any textures, shaders and meshes
  526. * used by the given scene to the video driver.
  527. * Using this method is useful when wishing to avoid the initial pause
  528. * when rendering a scene for the first time. Note that it is not
  529. * guaranteed that the underlying renderer will actually choose to upload
  530. * the data to the GPU so some pause is still to be expected.
  531. *
  532. * @param scene The scene to preload
  533. */
  534. public void preloadScene(Spatial scene) {
  535. if (scene instanceof Node) {
  536. // recurse for all children
  537. Node n = (Node) scene;
  538. List<Spatial> children = n.getChildren();
  539. for (int i = 0; i < children.size(); i++) {
  540. preloadScene(children.get(i));
  541. }
  542. } else if (scene instanceof Geometry) {
  543. // add to the render queue
  544. Geometry gm = (Geometry) scene;
  545. if (gm.getMaterial() == null) {
  546. throw new IllegalStateException("No material is set for Geometry: " + gm.getName());
  547. }
  548. gm.getMaterial().preload(this);
  549. Mesh mesh = gm.getMesh();
  550. if (mesh != null) {
  551. for (VertexBuffer vb : mesh.getBufferList().getArray()) {
  552. if (vb.getData() != null) {
  553. renderer.updateBufferData(vb);
  554. }
  555. }
  556. }
  557. }
  558. }
  559. /**
  560. * Flattens the given scene graph into the ViewPort's RenderQueue,
  561. * checking for culling as the call goes down the graph recursively.
  562. * <p>
  563. * First, the scene is checked for culling based on the <code>Spatial</code>s
  564. * {@link Spatial#setCullHint(com.jme3.scene.Spatial.CullHint) cull hint},
  565. * if the camera frustum contains the scene, then this method is recursively
  566. * called on its children.
  567. * <p>
  568. * When the scene's leaves or {@link Geometry geometries} are reached,
  569. * they are each enqueued into the
  570. * {@link ViewPort#getQueue() ViewPort's render queue}.
  571. * <p>
  572. * In addition to enqueuing the visible geometries, this method
  573. * also scenes which cast or receive shadows, by putting them into the
  574. * RenderQueue's
  575. * {@link RenderQueue#addToShadowQueue(com.jme3.scene.Geometry, com.jme3.renderer.queue.RenderQueue.ShadowMode)
  576. * shadow queue}. Each Spatial which has its
  577. * {@link Spatial#setShadowMode(com.jme3.renderer.queue.RenderQueue.ShadowMode) shadow mode}
  578. * set to not off, will be put into the appropriate shadow queue, note that
  579. * this process does not check for frustum culling on any
  580. * {@link ShadowMode#Cast shadow casters}, as they don't have to be
  581. * in the eye camera frustum to cast shadows on objects that are inside it.
  582. *
  583. * @param scene The scene to flatten into the queue
  584. * @param vp The ViewPort provides the {@link ViewPort#getCamera() camera}
  585. * used for culling and the {@link ViewPort#getQueue() queue} used to
  586. * contain the flattened scene graph.
  587. */
  588. public void renderScene(Spatial scene, ViewPort vp) {
  589. if (scene.getParent() == null) {
  590. vp.getCamera().setPlaneState(0);
  591. }
  592. // check culling first.
  593. if (!scene.checkCulling(vp.getCamera())) {
  594. // move on to shadow-only render
  595. if ((scene.getShadowMode() != RenderQueue.ShadowMode.Off || scene instanceof Node) && scene.getCullHint()!=Spatial.CullHint.Always) {
  596. renderShadow(scene, vp.getQueue());
  597. }
  598. return;
  599. }
  600. scene.runControlRender(this, vp);
  601. if (scene instanceof Node) {
  602. // recurse for all children
  603. Node n = (Node) scene;
  604. List<Spatial> children = n.getChildren();
  605. //saving cam state for culling
  606. int camState = vp.getCamera().getPlaneState();
  607. for (int i = 0; i < children.size(); i++) {
  608. //restoring cam state before proceeding children recusively
  609. vp.getCamera().setPlaneState(camState);
  610. renderScene(children.get(i), vp);
  611. }
  612. } else if (scene instanceof Geometry) {
  613. // add to the render queue
  614. Geometry gm = (Geometry) scene;
  615. if (gm.getMaterial() == null) {
  616. throw new IllegalStateException("No material is set for Geometry: " + gm.getName());
  617. }
  618. vp.getQueue().addToQueue(gm, scene.getQueueBucket());
  619. // add to shadow queue if needed
  620. RenderQueue.ShadowMode shadowMode = scene.getShadowMode();
  621. if (shadowMode != RenderQueue.ShadowMode.Off) {
  622. vp.getQueue().addToShadowQueue(gm, shadowMode);
  623. }
  624. }
  625. }
  626. /**
  627. * Returns the camera currently used for rendering.
  628. * <p>
  629. * The camera can be set with {@link #setCamera(com.jme3.renderer.Camera, boolean) }.
  630. *
  631. * @return the camera currently used for rendering.
  632. */
  633. public Camera getCurrentCamera() {
  634. return prevCam;
  635. }
  636. /**
  637. * The renderer implementation used for rendering operations.
  638. *
  639. * @return The renderer implementation
  640. *
  641. * @see #RenderManager(com.jme3.renderer.Renderer)
  642. * @see Renderer
  643. */
  644. public Renderer getRenderer() {
  645. return renderer;
  646. }
  647. /**
  648. * Flushes the ViewPort's {@link ViewPort#getQueue() render queue}
  649. * by rendering each of its visible buckets.
  650. * By default the queues will automatically be cleared after rendering,
  651. * so there's no need to clear them manually.
  652. *
  653. * @param vp The ViewPort of which the queue will be flushed
  654. *
  655. * @see RenderQueue#renderQueue(com.jme3.renderer.queue.RenderQueue.Bucket, com.jme3.renderer.RenderManager, com.jme3.renderer.Camera)
  656. * @see #renderGeometryList(com.jme3.renderer.queue.GeometryList)
  657. */
  658. public void flushQueue(ViewPort vp) {
  659. renderViewPortQueues(vp, true);
  660. }
  661. /**
  662. * Clears the queue of the given ViewPort.
  663. * Simply calls {@link RenderQueue#clear() } on the ViewPort's
  664. * {@link ViewPort#getQueue() render queue}.
  665. *
  666. * @param vp The ViewPort of which the queue will be cleared.
  667. *
  668. * @see RenderQueue#clear()
  669. * @see ViewPort#getQueue()
  670. */
  671. public void clearQueue(ViewPort vp) {
  672. vp.getQueue().clear();
  673. }
  674. /**
  675. * Render the given viewport queues.
  676. * <p>
  677. * Changes the {@link Renderer#setDepthRange(float, float) depth range}
  678. * appropriately as expected by each queue and then calls
  679. * {@link RenderQueue#renderQueue(com.jme3.renderer.queue.RenderQueue.Bucket, com.jme3.renderer.RenderManager, com.jme3.renderer.Camera, boolean) }
  680. * on the queue. Makes sure to restore the depth range to [0, 1]
  681. * at the end of the call.
  682. * Note that the {@link Bucket#Translucent translucent bucket} is NOT
  683. * rendered by this method. Instead the user should call
  684. * {@link #renderTranslucentQueue(com.jme3.renderer.ViewPort) }
  685. * after this call.
  686. *
  687. * @param vp the viewport of which queue should be rendered
  688. * @param flush If true, the queues will be cleared after
  689. * rendering.
  690. *
  691. * @see RenderQueue
  692. * @see #renderTranslucentQueue(com.jme3.renderer.ViewPort)
  693. */
  694. public void renderViewPortQueues(ViewPort vp, boolean flush) {
  695. RenderQueue rq = vp.getQueue();
  696. Camera cam = vp.getCamera();
  697. boolean depthRangeChanged = false;
  698. // render opaque objects with default depth range
  699. // opaque objects are sorted front-to-back, reducing overdraw
  700. rq.renderQueue(Bucket.Opaque, this, cam, flush);
  701. // render the sky, with depth range set to the farthest
  702. if (!rq.isQueueEmpty(Bucket.Sky)) {
  703. renderer.setDepthRange(1, 1);
  704. rq.renderQueue(Bucket.Sky, this, cam, flush);
  705. depthRangeChanged = true;
  706. }
  707. // transparent objects are last because they require blending with the
  708. // rest of the scene's objects. Consequently, they are sorted
  709. // back-to-front.
  710. if (!rq.isQueueEmpty(Bucket.Transparent)) {
  711. if (depthRangeChanged) {
  712. renderer.setDepthRange(0, 1);
  713. depthRangeChanged = false;
  714. }
  715. rq.renderQueue(Bucket.Transparent, this, cam, flush);
  716. }
  717. if (!rq.isQueueEmpty(Bucket.Gui)) {
  718. renderer.setDepthRange(0, 0);
  719. setCamera(cam, true);
  720. rq.renderQueue(Bucket.Gui, this, cam, flush);
  721. setCamera(cam, false);
  722. depthRangeChanged = true;
  723. }
  724. // restore range to default
  725. if (depthRangeChanged) {
  726. renderer.setDepthRange(0, 1);
  727. }
  728. }
  729. /**
  730. * Renders the {@link Bucket#Translucent translucent queue} on the viewPort.
  731. * <p>
  732. * This call does nothing unless {@link #setHandleTranslucentBucket(boolean) }
  733. * is set to true. This method clears the translucent queue after rendering
  734. * it.
  735. *
  736. * @param vp The viewport of which the translucent queue should be rendered.
  737. *
  738. * @see #renderViewPortQueues(com.jme3.renderer.ViewPort, boolean)
  739. * @see #setHandleTranslucentBucket(boolean)
  740. */
  741. public void renderTranslucentQueue(ViewPort vp) {
  742. RenderQueue rq = vp.getQueue();
  743. if (!rq.isQueueEmpty(Bucket.Translucent) && handleTranlucentBucket) {
  744. rq.renderQueue(Bucket.Translucent, this, vp.getCamera(), true);
  745. }
  746. }
  747. private void setViewPort(Camera cam) {
  748. // this will make sure to update viewport only if needed
  749. if (cam != prevCam || cam.isViewportChanged()) {
  750. viewX = (int) (cam.getViewPortLeft() * cam.getWidth());
  751. viewY = (int) (cam.getViewPortBottom() * cam.getHeight());
  752. viewWidth = (int) ((cam.getViewPortRight() - cam.getViewPortLeft()) * cam.getWidth());
  753. viewHeight = (int) ((cam.getViewPortTop() - cam.getViewPortBottom()) * cam.getHeight());
  754. renderer.setViewPort(viewX, viewY, viewWidth, viewHeight);
  755. renderer.setClipRect(viewX, viewY, viewWidth, viewHeight);
  756. cam.clearViewportChanged();
  757. prevCam = cam;
  758. // float translateX = viewWidth == viewX ? 0 : -(viewWidth + viewX) / (viewWidth - viewX);
  759. // float translateY = viewHeight == viewY ? 0 : -(viewHeight + viewY) / (viewHeight - viewY);
  760. // float scaleX = viewWidth == viewX ? 1f : 2f / (viewWidth - viewX);
  761. // float scaleY = viewHeight == viewY ? 1f : 2f / (viewHeight - viewY);
  762. //
  763. // orthoMatrix.loadIdentity();
  764. // orthoMatrix.setTranslation(translateX, translateY, 0);
  765. // orthoMatrix.setScale(scaleX, scaleY, 0);
  766. orthoMatrix.loadIdentity();
  767. orthoMatrix.setTranslation(-1f, -1f, 0f);
  768. orthoMatrix.setScale(2f / cam.getWidth(), 2f / cam.getHeight(), 0f);
  769. }
  770. }
  771. private void setViewProjection(Camera cam, boolean ortho) {
  772. if (shader) {
  773. if (ortho) {
  774. uniformBindingManager.setCamera(cam, Matrix4f.IDENTITY, orthoMatrix, orthoMatrix);
  775. } else {
  776. uniformBindingManager.setCamera(cam, cam.getViewMatrix(), cam.getProjectionMatrix(), cam.getViewProjectionMatrix());
  777. }
  778. } else {
  779. if (ortho) {
  780. renderer.setViewProjectionMatrices(Matrix4f.IDENTITY, orthoMatrix);
  781. } else {
  782. renderer.setViewProjectionMatrices(cam.getViewMatrix(),
  783. cam.getProjectionMatrix());
  784. }
  785. }
  786. }
  787. /**
  788. * Set the camera to use for rendering.
  789. * <p>
  790. * First, the camera's
  791. * {@link Camera#setViewPort(float, float, float, float) view port parameters}
  792. * are applied. Then, the camera's {@link Camera#getViewMatrix() view} and
  793. * {@link Camera#getProjectionMatrix() projection} matrices are set
  794. * on the renderer. If <code>ortho</code> is <code>true</code>, then
  795. * instead of using the camera's view and projection matrices, an ortho
  796. * matrix is computed and used instead of the view projection matrix.
  797. * The ortho matrix converts from the range (0 ~ Width, 0 ~ Height, -1 ~ +1)
  798. * to the clip range (-1 ~ +1, -1 ~ +1, -1 ~ +1).
  799. *
  800. * @param cam The camera to set
  801. * @param ortho True if to use orthographic projection (for GUI rendering),
  802. * false if to use the camera's view and projection matrices.
  803. */
  804. public void setCamera(Camera cam, boolean ortho) {
  805. setViewPort(cam);
  806. setViewProjection(cam, ortho);
  807. }
  808. /**
  809. * Draws the viewport but without notifying {@link SceneProcessor scene
  810. * processors} of any rendering events.
  811. *
  812. * @param vp The ViewPort to render
  813. *
  814. * @see #renderViewPort(com.jme3.renderer.ViewPort, float)
  815. */
  816. public void renderViewPortRaw(ViewPort vp) {
  817. setCamera(vp.getCamera(), false);
  818. List<Spatial> scenes = vp.getScenes();
  819. for (int i = scenes.size() - 1; i >= 0; i--) {
  820. renderScene(scenes.get(i), vp);
  821. }
  822. flushQueue(vp);
  823. }
  824. /**
  825. * Renders the {@link ViewPort}.
  826. * <p>
  827. * If the ViewPort is {@link ViewPort#isEnabled() disabled}, this method
  828. * returns immediately. Otherwise, the ViewPort is rendered by
  829. * the following process:<br>
  830. * <ul>
  831. * <li>All {@link SceneProcessor scene processors} that are attached
  832. * to the ViewPort are {@link SceneProcessor#initialize(com.jme3.renderer.RenderManager, com.jme3.renderer.ViewPort) initialized}.
  833. * </li>
  834. * <li>The SceneProcessors' {@link SceneProcessor#preFrame(float) } method
  835. * is called.</li>
  836. * <li>The ViewPort's {@link ViewPort#getOutputFrameBuffer() output framebuffer}
  837. * is set on the Renderer</li>
  838. * <li>The camera is set on the renderer, including its view port parameters.
  839. * (see {@link #setCamera(com.jme3.renderer.Camera, boolean) })</li>
  840. * <li>Any buffers that the ViewPort requests to be cleared are cleared
  841. * and the {@link ViewPort#getBackgroundColor() background color} is set</li>
  842. * <li>Every scene that is attached to the ViewPort is flattened into
  843. * the ViewPort's render queue
  844. * (see {@link #renderViewPortQueues(com.jme3.renderer.ViewPort, boolean) })
  845. * </li>
  846. * <li>The SceneProcessors' {@link SceneProcessor#postQueue(com.jme3.renderer.queue.RenderQueue) }
  847. * method is called.</li>
  848. * <li>The render queue is sorted and then flushed, sending
  849. * rendering commands to the underlying Renderer implementation.
  850. * (see {@link #flushQueue(com.jme3.renderer.ViewPort) })</li>
  851. * <li>The SceneProcessors' {@link SceneProcessor#postFrame(com.jme3.texture.FrameBuffer) }
  852. * method is called.</li>
  853. * <li>The translucent queue of the ViewPort is sorted and then flushed
  854. * (see {@link #renderTranslucentQueue(com.jme3.renderer.ViewPort) })</li>
  855. * <li>If any objects remained in the render queue, they are removed
  856. * from the queue. This is generally objects added to the
  857. * {@link RenderQueue#renderShadowQueue(com.jme3.renderer.queue.RenderQueue.ShadowMode, com.jme3.renderer.RenderManager, com.jme3.renderer.Camera, boolean)
  858. * shadow queue}
  859. * which were not rendered because of a missing shadow renderer.</li>
  860. * </ul>
  861. *
  862. * @param vp
  863. * @param tpf
  864. */
  865. public void renderViewPort(ViewPort vp, float tpf) {
  866. if (!vp.isEnabled()) {
  867. return;
  868. }
  869. List<SceneProcessor> processors = vp.getProcessors();
  870. if (processors.isEmpty()) {
  871. processors = null;
  872. }
  873. if (processors != null) {
  874. for (SceneProcessor proc : processors) {
  875. if (!proc.isInitialized()) {
  876. proc.initialize(this, vp);
  877. }
  878. proc.preFrame(tpf);
  879. }
  880. }
  881. renderer.setFrameBuffer(vp.getOutputFrameBuffer());
  882. setCamera(vp.getCamera(), false);
  883. if (vp.isClearDepth() || vp.isClearColor() || vp.isClearStencil()) {
  884. if (vp.isClearColor()) {
  885. renderer.setBackgroundColor(vp.getBackgroundColor());
  886. }
  887. renderer.clearBuffers(vp.isClearColor(),
  888. vp.isClearDepth(),
  889. vp.isClearStencil());
  890. }
  891. List<Spatial> scenes = vp.getScenes();
  892. for (int i = scenes.size() - 1; i >= 0; i--) {
  893. renderScene(scenes.get(i), vp);
  894. }
  895. if (processors != null) {
  896. for (SceneProcessor proc : processors) {
  897. proc.postQueue(vp.getQueue());
  898. }
  899. }
  900. flushQueue(vp);
  901. if (processors != null) {
  902. for (SceneProcessor proc : processors) {
  903. proc.postFrame(vp.getOutputFrameBuffer());
  904. }
  905. }
  906. //renders the translucent objects queue after processors have been rendered
  907. renderTranslucentQueue(vp);
  908. // clear any remaining spatials that were not rendered.
  909. clearQueue(vp);
  910. }
  911. /**
  912. * Called by the application to render any ViewPorts
  913. * added to this RenderManager.
  914. * <p>
  915. * Renders any viewports that were added using the following methods:
  916. * <ul>
  917. * <li>{@link #createPreView(java.lang.String, com.jme3.renderer.Camera) }</li>
  918. * <li>{@link #createMainView(java.lang.String, com.jme3.renderer.Camera) }</li>
  919. * <li>{@link #createPostView(java.lang.String, com.jme3.renderer.Camera) }</li>
  920. * </ul>
  921. *
  922. * @param tpf Time per frame value
  923. */
  924. public void render(float tpf, boolean mainFrameBufferActive) {
  925. if (renderer instanceof NullRenderer) {
  926. return;
  927. }
  928. this.shader = renderer.getCaps().contains(Caps.GLSL100);
  929. for (int i = 0; i < preViewPorts.size(); i++) {
  930. ViewPort vp = preViewPorts.get(i);
  931. if (vp.getOutputFrameBuffer() != null || mainFrameBufferActive){
  932. renderViewPort(vp, tpf);
  933. }
  934. }
  935. for (int i = 0; i < viewPorts.size(); i++) {
  936. ViewPort vp = viewPorts.get(i);
  937. if (vp.getOutputFrameBuffer() != null || mainFrameBufferActive){
  938. renderViewPort(vp, tpf);
  939. }
  940. }
  941. for (int i = 0; i < postViewPorts.size(); i++) {
  942. ViewPort vp = postViewPorts.get(i);
  943. if (vp.getOutputFrameBuffer() != null || mainFrameBufferActive){
  944. renderViewPort(vp, tpf);
  945. }
  946. }
  947. }
  948. }