layer1/SceneRender.cpp (1,389 lines of code) (raw):

/* * (c) Schrodinger, Inc. */ #include <algorithm> #include"Scene.h" #include"SceneRay.h" #include"ScenePicking.h" #include"ShaderMgr.h" #include"CGO.h" #include"Matrix.h" #include"PyMOLOptions.h" #include"Util.h" #include"main.h" #include"Control.h" #include"Editor.h" #include"Executive.h" #include"P.h" #include"Err.h" /* EXPERIMENTAL VOLUME RAYTRACING DATA */ extern float *rayDepthPixels; extern int rayVolume, rayWidth, rayHeight; void SetDrawBufferForStereo(PyMOLGlobals * G, CScene *I, int stereo_mode, int times, int fog_active, int offscreen); void SceneDrawStencilInBuffer(PyMOLGlobals * G, CScene *I, int stereo_mode); void SceneRenderStereoLoop(PyMOLGlobals * G, int timesArg, int must_render_stereo, int stereo_mode, short render_to_texture, int x, int y, int oversize_width, int oversize_height, int stereo_double_pump_mono, int curState, float *normal, SceneUnitContext *context, float width_scale, int fog_active, int onlySelections, int noAA); void SceneRenderAA(PyMOLGlobals * G); void PrepareViewPortForStereoImpl(PyMOLGlobals * G, CScene *I, int stereo_mode, short offscreen, int times, int x, int y, int oversize_width, int oversize_height, GLenum draw_mode, int position /* left=0, right=1 */); void PrepareViewPortForMonoInitializeViewPort(PyMOLGlobals * G, CScene *I, int stereo_mode, short offscreen, int times, int x, int y, int oversize_width, int oversize_height); void PrepareViewPortForStereo(PyMOLGlobals * G, CScene *I, int stereo_mode, short offscreen, int times, int x, int y, int oversize_width, int oversize_height); void PrepareViewPortForStereo2nd(PyMOLGlobals * G, CScene *I, int stereo_mode, short offscreen, int times, int x, int y, int oversize_width, int oversize_height); void InitializeViewPortToScreenBlock(PyMOLGlobals * G, CScene *I, int x, int y, int oversize_width, int oversize_height, int *stereo_mode, float *width_scale); void SceneSetPrepareViewPortForStereo(PyMOLGlobals *G, void (*prepareViewPortForStereo)(PyMOLGlobals *, CScene *, int, short, int, int, int, int, int), int times, int x, int y, int oversize_width, int oversize_height, int stereo_mode, float width_scale); CGO *GenerateUnitScreenCGO(PyMOLGlobals * G); static int stereo_via_stencil(int stereo_mode) { switch (stereo_mode) { case cStereo_stencil_by_row: case cStereo_stencil_by_column: case cStereo_stencil_checkerboard: case cStereo_stencil_custom: return true; } return false; } static int render_stereo_blend_into_full_screen(int stereo_mode) { switch (stereo_mode) { case cStereo_stencil_by_row: case cStereo_stencil_by_column: case cStereo_stencil_checkerboard: case cStereo_stencil_custom: case cStereo_anaglyph: case cStereo_dynamic: case cStereo_clone_dynamic: return true; } return false; } void GridGetGLViewport(PyMOLGlobals * G, GridInfo * I) { #ifdef _PYMOL_IOS { int width, height; SceneGetWidthHeight(G, &width, &height); I->cur_view[0] = I->cur_view[1] = 0; I->cur_view[2] = width; I->cur_view[3] = height; } #else glGetIntegerv(GL_VIEWPORT, I->cur_view); #endif } void GridSetGLViewport(GridInfo * I, int slot) { if(slot) I->slot = slot + I->first_slot - 1; else I->slot = slot; /* if we are in grid mode, then prepare the grid slot viewport */ if(slot < 0) { glViewport(I->cur_view[0], I->cur_view[1], I->cur_view[2], I->cur_view[3]); } else if(!slot) { /* slot 0 is the full screen */ int vx = 0; int vw = I->cur_view[2] / I->n_col; int vy = 0; int vh = I->cur_view[3] / I->n_row; if(I->n_col < I->n_row) { vw *= I->n_col; vh *= I->n_col; } else { vw *= I->n_row; vh *= I->n_row; } vx += I->cur_view[0] + (I->cur_view[2] - vw) / 2; vy += I->cur_view[1]; glViewport(vx, vy, vw, vh); ScenePrepareUnitContext(&I->context, vw, vh); } else { int abs_grid_slot = slot - I->first_slot; int grid_col = abs_grid_slot % I->n_col; int grid_row = (abs_grid_slot / I->n_col); int vx = (grid_col * I->cur_view[2]) / I->n_col; int vw = ((grid_col + 1) * I->cur_view[2]) / I->n_col - vx; int vy = I->cur_view[3] - ((grid_row + 1) * I->cur_view[3]) / I->n_row; int vh = (I->cur_view[3] - ((grid_row) * I->cur_view[3]) / I->n_row) - vy; vx += I->cur_view[0]; vy += I->cur_view[1]; I->cur_viewport_size[0] = vw; I->cur_viewport_size[1] = vh; glViewport(vx, vy, vw, vh); ScenePrepareUnitContext(&I->context, vw, vh); } } static void glBlendFunc_default() { if (glBlendFuncSeparate) { glBlendFuncSeparate( GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA, GL_ONE, GL_ONE_MINUS_SRC_ALPHA); } else { // OpenGL 1.x (e.g. remote desktop on Windows) glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); } } /*========================================================================*/ /* SceneRender: Responsible for rendering the scene, whether its picking (SceneRenderPicking) or rendering (SceneRenderStereoLoop). It also takes calls anti-aliasing (SceneRenderAA) if necessary after rendering and before selection markers. */ void SceneRender(PyMOLGlobals * G, Picking * pick, int x, int y, Multipick * smp, int oversize_width, int oversize_height, int click_side, int force_copy) { /* think in terms of the camera's world */ CScene *I = G->Scene; float normal[4] = { 0.0, 0.0, 1.0, 0.0 }; float aspRat = ((float) I->Width) / ((float) I->Height); float height, width; double start_time = 0.0; int view_save[4]; int curState; int must_render_stereo = false; int stereo_double_pump_mono = false; GLenum render_buffer; SceneUnitContext context; float width_scale = 0.0F; int stereo_mode = I->StereoMode; int stereo = SettingGetGlobal_i(G, cSetting_stereo); int grid_mode = SettingGetGlobal_i(G, cSetting_grid_mode); bool use_shaders = SettingGetGlobal_b(G, cSetting_use_shaders); int fog_active = false; int last_grid_active = I->grid.active; int grid_size = 0; short oneAA = 0; I->n_texture_refreshes = 0; #if defined(_WEBGL) && defined(PYMOL_EVAL) if (!OrthoEvalCheck(G)) return; #endif PRINTFD(G, FB_Scene) " SceneRender: entered. pick %p x %d y %d smp %p\n", (void *) pick, x, y, (void *) smp ENDFD; G->ShaderMgr->Check_Reload(); if(grid_mode) { grid_size = SceneGetGridSize(G, grid_mode); GridUpdate(&I->grid, aspRat, grid_mode, grid_size); if(I->grid.active) aspRat *= I->grid.asp_adjust; } else { I->grid.active = false; } if (last_grid_active != I->grid.active || grid_size != I->last_grid_size){ G->ShaderMgr->ResetUniformSet(); } I->last_grid_size = grid_size; G->ShaderMgr->FreeAllVBOs(); #ifndef _PYMOL_IOS SceneUpdateAnimation(G); #endif if(SceneMustDrawBoth(G)) { render_buffer = GL_BACK_LEFT; } else { render_buffer = G->DRAW_BUFFER0; // GL_BACK } switch (stereo_mode) { case cStereo_walleye: case cStereo_crosseye: aspRat = aspRat / 2; case cStereo_sidebyside: case cStereo_anaglyph: oneAA = stereo ? 1 : 0; break; default: oneAA = stereo ? 0 : 1; } if(G->HaveGUI && G->ValidContext) { if(Feedback(G, FB_OpenGL, FB_Debugging)) PyMOLCheckOpenGLErr("SceneRender checkpoint 0"); must_render_stereo = (stereo && stereo_mode != 0); // are we doing stereo? if(!must_render_stereo) { if(G->StereoCapable && SettingGet_i(G, NULL, NULL, cSetting_stereo_double_pump_mono)) { /* force stereo rendering */ must_render_stereo = true; stereo_double_pump_mono = true; } } /* if we seem to be configured for hardware stereo, but can't actually do it, then fallback on mono -- this would happen for instance if fullscreen is stereo-component and windowed is not */ if(must_render_stereo && (stereo_mode < cStereo_crosseye) && !(G->StereoCapable)) { must_render_stereo = false; } /* If we are rendering a stereo_mode that stencils, define the stencil buffer */ if(must_render_stereo && stereo_via_stencil(stereo_mode)) { if(!I->StencilValid) { SceneDrawStencilInBuffer(G, I, stereo_mode); I->StencilValid = true; } } render_buffer = G->DRAW_BUFFER0; // GL_BACK if(must_render_stereo) { switch (stereo_mode) { case cStereo_quadbuffer: /* hardware stereo */ case cStereo_clone_dynamic: render_buffer = GL_BACK_LEFT; break; } } OrthoDrawBuffer(G, render_buffer); if(Feedback(G, FB_OpenGL, FB_Debugging)) PyMOLCheckOpenGLErr("SceneRender checkpoint 1"); #ifdef _PYMOL_IOS { int width, height; SceneGetWidthHeight(G, &width, &height); view_save[0] = view_save[1] = 0; view_save[2] = width; view_save[3] = height; } #else glGetIntegerv(GL_VIEWPORT, (GLint *) (void *) view_save); #endif InitializeViewPortToScreenBlock(G, I, x, y, oversize_width, oversize_height, &stereo_mode, &width_scale); if(!(pick || smp)) bg_grad(G); #ifndef _WEBGL glLineWidth(SettingGetGlobal_f(G, cSetting_line_width)); #endif glEnable(GL_DEPTH_TEST); /* get matrixes for unit objects */ #ifndef PURE_OPENGL_ES_2 if(SettingGetGlobal_b(G, cSetting_line_smooth)) { if(!(pick || smp)) { glEnable(GL_LINE_SMOOTH); glHint(GL_LINE_SMOOTH_HINT, GL_NICEST); } } else { glDisable(GL_LINE_SMOOTH); } glPointSize(SettingGetGlobal_f(G, cSetting_dot_width)); if (ALWAYS_IMMEDIATE_OR(!use_shaders)) { glEnable(GL_NORMALIZE); /* get rid of this to boost performance */ glMatrixMode(GL_MODELVIEW); glLoadIdentity(); /* must be done with identity MODELVIEW */ SceneProgramLighting(G); } #endif ScenePrepareUnitContext(&context, I->Width, I->Height); /* do standard 3D objects */ /* Set up the clipping planes */ if(SettingGetGlobal_b(G, cSetting_all_states)) { curState = -1; } else { curState = std::max(-1, SettingGetGlobal_i(G, cSetting_state) - 1); } if(!SettingGetGlobal_b(G, cSetting_ortho)) { double xmin, xmax, ymin, ymax; ymax = I->FrontSafe * GetFovWidth(G) / 2.0; ymin = -ymax; xmin = ymin * aspRat; xmax = ymax * aspRat; glFrustum44f(I->ProjectionMatrix, xmin, xmax, ymin, ymax, I->FrontSafe, I->BackSafe); } else { height = std::max(R_SMALL4, -I->Pos[2]) * GetFovWidth(G) / 2.f; width = height * aspRat; glOrtho44f(I->ProjectionMatrix, -width, width, -height, height, I->FrontSafe, I->BackSafe); } #ifndef PURE_OPENGL_ES_2 if (ALWAYS_IMMEDIATE_OR(!use_shaders)) { glMatrixMode(GL_PROJECTION); glLoadMatrixf(I->ProjectionMatrix); glMatrixMode(GL_MODELVIEW); } #endif ScenePrepareMatrix(G, 0); /* get the Z axis vector for sorting transparent objects */ if(SettingGetGlobal_b(G, cSetting_transparency_global_sort) && SettingGetGlobal_b(G, cSetting_transparency_mode)) { if(!I->AlphaCGO) I->AlphaCGO = CGONew(G); } else { CGOFree(I->AlphaCGO); } /* make note of how large pixels are at the origin */ I->VertexScale = SceneGetScreenVertexScale(G, NULL); /* determine the direction in which we are looking relative */ /* 2. set the normals to reflect light back at the camera */ float zAxis[4] = { 0.0, 0.0, 1.0, 0.0 }; MatrixInvTransformC44fAs33f3f(I->RotMatrix, zAxis, normal); copy3f(normal, I->ViewNormal); if(SettingGetGlobal_b(G, cSetting_normal_workaround)) { I->LinesNormal[0] = 0.0; I->LinesNormal[1] = 0.0; I->LinesNormal[2] = 1.0; /* for versions of GL that don't transform GL_LINES normals */ } else { I->LinesNormal[0] = I->ViewNormal[0]; I->LinesNormal[1] = I->ViewNormal[1]; I->LinesNormal[2] = I->ViewNormal[2]; } PRINTFD(G, FB_Scene) " SceneRender: matrices loaded. rendering objects...\n" ENDFD; /* 1. render all objects */ if(pick || smp) { SceneRenderPicking(G, stereo_mode, &click_side, stereo_double_pump_mono, pick, x, y, smp, &context, render_buffer); } else { int times = 1; short render_to_texture_for_pp = 0; /* STANDARD RENDERING */ glEnable(GL_BLEND); glBlendFunc_default(); glEnable(GL_DITHER); #ifndef PURE_OPENGL_ES_2 if (ALWAYS_IMMEDIATE_OR(!use_shaders)) { glColorMaterial(GL_FRONT_AND_BACK, GL_AMBIENT_AND_DIFFUSE); glEnable(GL_COLOR_MATERIAL); glShadeModel(SettingGetGlobal_b(G, cSetting_pick_shading) ? GL_FLAT : GL_SMOOTH); if (use_shaders) { glDisable(GL_ALPHA_TEST); } else { // for immediate mode labels (with shaders, this would cause the OS X R9 bugs!) glAlphaFunc(GL_GREATER, 0.05F); glEnable(GL_ALPHA_TEST); } if(G->Option->multisample) glEnable(0x809D); /* GL_MULTISAMPLE_ARB */ glColor4ub(255, 255, 255, 255); glNormal3fv(normal); } #endif fog_active = SceneSetFog(G); #ifndef _PYMOL_NO_AA_SHADERS if (!oversize_width && !oversize_height){ render_to_texture_for_pp = SettingGetGlobal_i(G, cSetting_antialias_shader); } if(render_to_texture_for_pp) { if (!must_render_stereo || oneAA){ G->ShaderMgr->bindOffscreen(I->Width, I->Height, &I->grid); bg_grad(G); } } #endif /* rendering for visualization */ /*** THIS IS AN UGLY EXPERIMENTAL *** VOLUME + RAYTRACING COMPOSITION CODE ***/ if (rayVolume && rayDepthPixels) { SceneRenderRayVolume(G, I); rayVolume--; } /*** END OF EXPERIMENTAL CODE ***/ switch (stereo_mode) { case cStereo_clone_dynamic: case cStereo_dynamic: times = 2; break; } PRINTFD(G, FB_Scene) " SceneRender: I->StereoMode %d must_render_stereo %d\n StereoCapable %d\n", stereo_mode, must_render_stereo, G->StereoCapable ENDFD; SceneRenderStereoLoop(G, times, must_render_stereo, stereo_mode, render_to_texture_for_pp, x, y, oversize_width, oversize_height, stereo_double_pump_mono, curState, normal, &context, width_scale, fog_active, 0 /*onlySelections*/, oneAA); if(render_to_texture_for_pp) { /* BEGIN rendering the selection markers, should we put all of this into a function, so it can be called above as well? */ #ifndef PURE_OPENGL_ES_2 if (!must_render_stereo || oneAA){ SceneSetPrepareViewPortForStereo(G, PrepareViewPortForMonoInitializeViewPort, times, x, y, oversize_width, oversize_height, stereo_mode, width_scale); SceneRenderAA(G); } #endif SceneRenderStereoLoop(G, times, must_render_stereo, stereo_mode, 0, x, y, oversize_width, oversize_height, stereo_double_pump_mono, curState, normal, &context, width_scale, fog_active, 1 /*onlySelections*/, oneAA); } #ifndef PURE_OPENGL_ES_2 if (ALWAYS_IMMEDIATE_OR(!use_shaders)) { glDisable(GL_FOG); glDisable(GL_LIGHTING); glDisable(GL_LIGHT0); glDisable(GL_LIGHT1); glDisable(GL_COLOR_MATERIAL); glDisable(GL_DITHER); } #endif } #ifndef PURE_OPENGL_ES_2 if (ALWAYS_IMMEDIATE_OR(!use_shaders)) { glLineWidth(1.0); glDisable(GL_LINE_SMOOTH); glDisable(GL_BLEND); glDisable(GL_NORMALIZE); glDisable(GL_DEPTH_TEST); glDisable(GL_ALPHA_TEST); if(G->Option->multisample) glDisable(0x809D); /* GL_MULTISAMPLE_ARB */ } #endif glViewport(view_save[0], view_save[1], view_save[2], view_save[3]); if(Feedback(G, FB_OpenGL, FB_Debugging)) PyMOLCheckOpenGLErr("SceneRender final checkpoint"); } PRINTFD(G, FB_Scene) " SceneRender: rendering complete.\n" ENDFD; if(!(pick || smp)) { /* update frames per second field */ I->LastRender = UtilGetSeconds(G); I->ApproxRenderTime = I->LastRender - start_time; if(I->CopyNextFlag) { start_time = I->LastRender - start_time; if((start_time > 0.10) || (MainSavingUnderWhileIdle())) if(!(ControlIdling(G))) if(SettingGetGlobal_b(G, cSetting_cache_display)) { if(!I->CopyType) { SceneCopy(G, render_buffer, false, false); } } } else { I->CopyNextFlag = true; } if(force_copy && !(I->CopyType)) { SceneCopy(G, render_buffer, true, false); I->CopyType = 2; /* do not display force copies */ } } PRINTFD(G, FB_Scene) " SceneRender: leaving...\n" ENDFD; } void AppendCopyWithChangedShader(PyMOLGlobals * G, CGO *destCGO, CGO *srcCGO, int frommode, int tomode){ CGO *cgo = CGONew(G); CGOAppendNoStop(cgo, srcCGO); CGOChangeShadersTo(cgo, frommode, tomode); CGOAppendNoStop(destCGO, cgo); CGOFreeWithoutVBOs(cgo); } /* SceneRenderAA: renders Anti-aliasing from the I->offscreen_texture texture, depending on the antialias_shader setting, FXAA (1 stage) or SMAA (3 stages) are rendered using framebuffers I->offscreen2_fb, I->offscreen3_fb, and into the screen block */ void SceneRenderAA(PyMOLGlobals * G){ #ifndef _PYMOL_NO_AA_SHADERS CScene *I = G->Scene; int ok = true; glBindFramebufferEXT(GL_FRAMEBUFFER_EXT, G->ShaderMgr->default_framebuffer_id); if (!I->offscreenCGO) { CGO *unitCGO = GenerateUnitScreenCGO(G); ok &= unitCGO!=NULL; if (ok){ int offscreen = SettingGetGlobal_b(G, cSetting_antialias_shader); I->offscreenCGO = CGONew(G); switch (offscreen){ case 0: break; case 1: // fxaa AppendCopyWithChangedShader(G, I->offscreenCGO, unitCGO, GL_DEFAULT_SHADER_WITH_SETTINGS, GL_FXAA_SHADER); break; default: AppendCopyWithChangedShader(G, I->offscreenCGO, unitCGO, GL_DEFAULT_SHADER_WITH_SETTINGS, GL_SMAA1_SHADER); if (offscreen!=3){ // not 1nd Pass as output CGODisable(I->offscreenCGO, GL_SMAA1_SHADER); AppendCopyWithChangedShader(G, I->offscreenCGO, unitCGO, GL_DEFAULT_SHADER_WITH_SETTINGS, GL_SMAA2_SHADER); CGODisable(I->offscreenCGO, GL_SMAA2_SHADER); if (offscreen!=4){ // not 2nd Pass as output AppendCopyWithChangedShader(G, I->offscreenCGO, unitCGO, GL_DEFAULT_SHADER_WITH_SETTINGS, GL_SMAA3_SHADER); CGODisable(I->offscreenCGO, GL_SMAA3_SHADER); } } break; } CGOStop(I->offscreenCGO); CGOFreeWithoutVBOs(unitCGO); I->offscreenCGO->use_shader = true; } else { I->offscreenCGO = NULL; } } if (ok && I->offscreenCGO) { { CGORenderGL(I->offscreenCGO, NULL, NULL, NULL, NULL, NULL); { CShaderPrg * shaderPrg = G->ShaderMgr->Get_Current_Shader(); if (shaderPrg) shaderPrg->Disable(); } glBindTexture(GL_TEXTURE_2D, 0); glEnable(GL_DEPTH_TEST); } glBindFramebufferEXT(GL_FRAMEBUFFER_EXT, G->ShaderMgr->default_framebuffer_id); } #endif } static void SceneRenderAllObject(PyMOLGlobals * G, CScene *I, SceneUnitContext * context, RenderInfo *info, float *normal, int state, CObject *obj, GridInfo * grid, int *slot_vla, int fat) { if (!SceneGetDrawFlag(grid, slot_vla, obj->grid_slot)) return; if (obj->Context == 2) return; auto use_shader = info->use_shaders; #ifndef _WEBGL glLineWidth(fat ? 3.0 : 1.0); #endif switch (obj->Context) { case 1: /* unit context */ // e.g. Gadgets/Ramps { float projSave[16]; copy44f(I->ProjectionMatrix, projSave); if (grid->active) { context = &grid->context; } glOrtho44f(I->ProjectionMatrix, context->unit_left, context->unit_right, context->unit_top, context->unit_bottom, context->unit_front, context->unit_back); #ifndef PURE_OPENGL_ES_2 if (ALWAYS_IMMEDIATE_OR(!use_shader)) { glPushAttrib(GL_LIGHTING_BIT); glMatrixMode(GL_PROJECTION); glLoadMatrixf(I->ProjectionMatrix); glMatrixMode(GL_MODELVIEW); glPushMatrix(); glLoadIdentity(); float vv[4] = { 0.f, 0.f, -1.f, 0.f }, dif[4] = { 1.f, 1.f, 1.f, 1.f }; glLightfv(GL_LIGHT0, GL_POSITION, vv); glLightfv(GL_LIGHT0, GL_DIFFUSE, dif); glNormal3f(0.0F, 0.0F, 1.0F); } #endif info->state = ObjectGetCurrentState(obj, false); obj->fRender(obj, info); copy44f(projSave, I->ProjectionMatrix); #ifndef PURE_OPENGL_ES_2 if (ALWAYS_IMMEDIATE_OR(!use_shader)) { glMatrixMode(GL_PROJECTION); glLoadMatrixf(I->ProjectionMatrix); glMatrixMode(GL_MODELVIEW); glPopMatrix(); glPopAttrib(); } #endif } break; case 0: /* context/grid 0 is all slots */ default: ScenePushModelViewMatrix(G); #ifndef PURE_OPENGL_ES_2 if (normal && Feedback(G, FB_OpenGL, FB_Debugging)) glNormal3fv(normal); #endif if((!grid->active) || (grid->mode < 2)) { info->state = ObjectGetCurrentState(obj, false); obj->fRender(obj, info); } else if(grid->slot) { if (grid->mode == 2) { if((info->state = state + grid->slot - 1) >= 0) obj->fRender(obj, info); } else if (grid->mode == 3) { info->state = grid->slot - obj->grid_slot - 1; if (info->state >= 0 && info->state < obj->getNFrame()) obj->fRender(obj, info); } } ScenePopModelViewMatrix(G, !use_shader); break; } } /*======================================================================== * SceneRenderAll: Renders all CObjects in the scene * * context: context info * normal: initial normal (for immediate mode) * pickVLA: for picking * pass: which pass (opaque, antialias, transparent) * fat: wide lines (i.e., for picking) * width_scale: specifies width_scale and sampling * grid: grid information * dynamic_pass: for specific stereo modes dynamic and clone_dynamic * which: 0 - all objects * 1 - only gadgets * 2 - only non-gadgets * 3 - gadgets last */ void SceneRenderAll(PyMOLGlobals * G, SceneUnitContext * context, float *normal, std::vector<Picking>* pickVLA, int pass, int fat, float width_scale, GridInfo * grid, int dynamic_pass, short which_objects, bool picking32bit) { CScene *I = G->Scene; int state = SceneGetState(G); RenderInfo info; #if defined(_WEBGL) && defined(PYMOL_EVAL) if (!OrthoEvalCheck(G)) return; #endif UtilZeroMem(&info, sizeof(RenderInfo)); info.pick = pickVLA; info.pass = pass; info.vertex_scale = I->VertexScale; info.fog_start = I->FogStart; info.fog_end = I->FogEnd; info.front = I->FrontSafe; info.use_shaders = SettingGetGlobal_b(G, cSetting_use_shaders); #if defined(_PYMOL_IOS) && !defined(_WEBGL) /* For now, on IOS, just crank up the sampling for text, TODO : need to figure out where this change should really be */ info.sampling = 2; #else info.sampling = 1; #endif info.alpha_cgo = I->AlphaCGO; info.ortho = SettingGetGlobal_b(G, cSetting_ortho); if(I->StereoMode && dynamic_pass && (!info.pick)) { int stereo_mode = SettingGetGlobal_i(G, cSetting_stereo_mode); switch (stereo_mode) { case cStereo_dynamic: case cStereo_clone_dynamic: info.line_lighting = true; break; } } if(I->StereoMode) { float buffer; float stAng, stShift; stAng = SettingGetGlobal_f(G, cSetting_stereo_angle); stShift = SettingGetGlobal_f(G, cSetting_stereo_shift); stShift = (float) (stShift * fabs(I->Pos[2]) / 100.0); stAng = (float) (stAng * atan(stShift / fabs(I->Pos[2])) * 90.0 / cPI); buffer = fabs(I->Width * I->VertexScale * tan(cPI * stAng / 180.0)); info.stereo_front = I->FrontSafe + buffer; } else { info.stereo_front = I->FrontSafe; } info.back = I->BackSafe; SceneGetViewNormal(G, info.view_normal); if(info.alpha_cgo && (pass == 1)) { CGOReset(info.alpha_cgo); CGOSetZVector(info.alpha_cgo, I->ModMatrix[2], I->ModMatrix[6], I->ModMatrix[10]); } if(SettingGetGlobal_b(G, cSetting_dynamic_width)) { info.dynamic_width = true; info.dynamic_width_factor = SettingGetGlobal_f(G, cSetting_dynamic_width_factor); info.dynamic_width_min = SettingGetGlobal_f(G, cSetting_dynamic_width_min); info.dynamic_width_max = SettingGetGlobal_f(G, cSetting_dynamic_width_max); } if(width_scale != 0.0F) { info.width_scale_flag = true; info.width_scale = width_scale; info.sampling = (int) info.width_scale; if(info.sampling < 1) info.sampling = 1; } if (picking32bit){ info.setUCColorFromIndex = SetUCColorFromIndex_32bit; info.setUCColorToZero = SetUCColorToZero_32bit; info.picking_32bit = true; } else { info.setUCColorFromIndex = SetUCColorFromIndex_16bit; info.setUCColorToZero = SetUCColorToZero_16bit; info.picking_32bit = false; } { int *slot_vla = I->SlotVLA; switch (which_objects){ case 0: for ( auto it = I->Obj.begin(); it != I->Obj.end(); ++it) { /* EXPERIMENTAL RAY-VOLUME COMPOSITION CODE */ /* if(rec->obj->fRender) { */ if((*it)->fRender && (!rayVolume || (*it)->type==cObjectVolume)) { SceneRenderAllObject(G, I, context, &info, normal, state, *it, grid, slot_vla, fat); } } break; case 1: for ( auto it = I->GadgetObjs.begin(); it != I->GadgetObjs.end(); ++it) { if((*it)->fRender) SceneRenderAllObject(G, I, context, &info, normal, state, *it, grid, slot_vla, fat); } break; case 2: for ( auto it = I->NonGadgetObjs.begin(); it != I->NonGadgetObjs.end(); ++it) { if((*it)->fRender) SceneRenderAllObject(G, I, context, &info, normal, state, *it, grid, slot_vla, fat); } break; case 3: // Gadgets Last for ( auto it = I->NonGadgetObjs.begin(); it != I->NonGadgetObjs.end(); ++it) { /* EXPERIMENTAL RAY-VOLUME COMPOSITION CODE */ /* if(rec->obj->fRender) { */ if((*it)->fRender && (!rayVolume || (*it)->type==cObjectVolume)) { SceneRenderAllObject(G, I, context, &info, normal, state, *it, grid, slot_vla, fat); } } for ( auto it = I->GadgetObjs.begin(); it != I->GadgetObjs.end(); ++it) { if((*it)->fRender) SceneRenderAllObject(G, I, context, &info, normal, state, *it, grid, slot_vla, fat); } break; } } if(info.alpha_cgo) { CGOStop(info.alpha_cgo); /* this only works when all objects are rendered in the same frame of reference */ if(pass == -1) { CGORenderGLAlpha(info.alpha_cgo, &info, 0); } } } /*==================================================================================*/ /* DoRendering: This is the function that is responsible for looping through each rendering pass (opaque, then antialiased, then transparent) for each grid slot (only one grid slot if full screen). It also implements transparency_mode 3, (weighted, blended order-independent transparency) which renders the opaque/antialiased passes to the offscreen texture with a depth texture, renders the transparent pass to the OIT offscreen texture, calls OIT_copy to copy the opaque to the screen (if necessary, i.e., not already rendering to AA texture), and then calls the OIT rendering pass that computes the resulting image. This function also renders only the selections (onlySelections) for all grids or the full screen. */ void DoRendering(PyMOLGlobals * G, CScene *I, short offscreen, GridInfo *grid, int times, int curState, float *normal, SceneUnitContext *context, float width_scale, short onlySelections, short excludeSelections){ int pass; bool use_shaders = (bool)SettingGetGlobal_b(G, cSetting_use_shaders); bool t_mode_3_os = use_shaders && SettingGetGlobal_i(G, cSetting_transparency_mode) == 3; bool t_mode_3 = !onlySelections && t_mode_3_os; GLint currentFrameBuffer; #if !defined(PURE_OPENGL_ES_2) || defined(_WEBGL) if (t_mode_3){ glGetIntegerv(GL_FRAMEBUFFER_BINDING, &currentFrameBuffer); // currentFrameBuffer: 0 - rendering to screen, need to render opaque to offscreen buffer // non-0 - already rendering to AA texture, need to use I->offscreen_depth_rb // transparent (OIT) pass // In the case of jymol the currentFramebuffer is not 0 so we are checking against the default // framebuffer if (currentFrameBuffer == G->ShaderMgr->default_framebuffer_id){ G->ShaderMgr->bindOffscreen(I->Width, I->Height, &I->grid); bg_grad(G); } glEnable(GL_DEPTH_TEST); } #endif if(grid->active)// && !offscreen) GridGetGLViewport(G, grid); { int slot; bool cont = true; bool t_first_pass = true; G->ShaderMgr->stereo_draw_buffer_pass = 0; for(pass = 1; cont && pass > -2; pass--) { /* render opaque, then antialiased, then transparent... */ #if !defined(PURE_OPENGL_ES_2) || defined(_WEBGL) if (t_mode_3 && pass == -1){ { CShaderPrg * shaderPrg = G->ShaderMgr->Get_Current_Shader(); if (shaderPrg) shaderPrg->Disable(); } int drawbuf = 1; if (TM3_IS_ONEBUF){ if (!t_first_pass){ G->ShaderMgr->stereo_draw_buffer_pass = 1; } drawbuf = t_first_pass ? 1 : 2; } G->ShaderMgr->bindOffscreenOIT(I->Width, I->Height, drawbuf); G->ShaderMgr->bindOffscreenOITFBO(drawbuf); // for transparency pass, render to OIT texture if (currentFrameBuffer == G->ShaderMgr->default_framebuffer_id){ SceneInitializeViewport(G, 2); } } #endif for(slot = 0; slot <= grid->last_slot; slot++) { if(grid->active) { GridSetGLViewport(grid, slot); } else if (slot){ break; // if grid is off, then just get out of loop after 1st pass (full screen) } /* render picked atoms */ /* render the debugging CGO */ #ifdef PURE_OPENGL_ES_2 if (!onlySelections){ EditorRender(G, curState); CGORenderGL(G->DebugCGO, NULL, NULL, NULL, NULL, NULL); } #else if (!use_shaders) glPushMatrix(); /* 2 */ if (!onlySelections && !t_mode_3) EditorRender(G, curState); if (!use_shaders){ glPopMatrix(); /* 1 */ glPushMatrix(); /* 2 */ } if (!onlySelections){ if (!use_shaders) glNormal3fv(normal); CGORenderGL(G->DebugCGO, NULL, NULL, NULL, NULL, NULL); } if (!use_shaders){ glPopMatrix(); /* 1 */ glPushMatrix(); /* 2 */ } #endif /* render all objects */ if (!onlySelections){ #if !defined(PURE_OPENGL_ES_2) || defined(_WEBGL) if (t_mode_3){ if (pass == 1){ EditorRender(G, curState); } // transparency-mode == 3 render all objects for this pass SceneRenderAll(G, context, normal, NULL, pass, false, width_scale, grid, times, 2 /* non-gadgets */, false); // opaque } else { #else { #endif // transparency-mode != 3 render all objects for each pass for(pass = 1; pass > -2; pass--) { /* render opaque, then antialiased, then transparent... */ SceneRenderAll(G, context, normal, NULL, pass, false, width_scale, grid, times, 3 /* gadgets last */, false); } cont = false; } } else if (t_mode_3_os && pass > 0){ /* opaque pass*/ // onlySelections and t_mode_3, render only gadgets glEnable(GL_BLEND); // need to blend for the text onto the gadget background glBlendFunc_default(); SceneRenderAll(G, context, normal, NULL, -1 /* gadgets render in transp pass */, false, width_scale, grid, times, 1 /* only gadgets */, false); glDisable(GL_BLEND); } #ifdef PURE_OPENGL_ES_2 if (!excludeSelections){ if (!grid->active || slot > 0){ /* slot 0 is the full screen in grid mode, so don't render selections */ int s = grid->active && grid->mode==1 ? slot : 0; ExecutiveRenderSelections(G, curState, s, grid); } } #else if (!use_shaders){ glPopMatrix(); /* 1 */ /* render selections */ glPushMatrix(); /* 2 */ glNormal3fv(normal); } if (!t_mode_3 && !excludeSelections){ if (!grid->active || slot > 0){ /* slot 0 is the full screen in grid mode, so don't render selections */ int s = grid->active && grid->mode==1 ? slot : 0; ExecutiveRenderSelections(G, curState, s, grid); } } if (!use_shaders){ glPopMatrix(); /* 1 */ } #endif } // end slot loop if (TM3_IS_ONEBUF){ if (t_mode_3 && pass == -1 && t_first_pass){ pass++; t_first_pass = false; continue; } } #if !defined(PURE_OPENGL_ES_2) || defined(_WEBGL) if (t_mode_3 && pass == -1){ glBindFramebufferEXT(GL_FRAMEBUFFER_EXT, currentFrameBuffer); glBindTexture(GL_TEXTURE_2D, 0); if(grid->active) GridSetGLViewport(grid, -1); if (currentFrameBuffer == G->ShaderMgr->default_framebuffer_id){ // if rendering to screen, need to render offscreen opaque to screen SceneInitializeViewport(G, 0); if (!I->offscreenOIT_CGO_copy){ // TODO G->ShaderMgr->Reload_Copy_Shaders(); I->offscreenOIT_CGO_copy = GenerateUnitScreenCGO(G); CGOChangeShadersTo(I->offscreenOIT_CGO_copy, GL_DEFAULT_SHADER_WITH_SETTINGS, GL_OIT_COPY_SHADER); I->offscreenOIT_CGO_copy->use_shader = true; } CGORenderGL(I->offscreenOIT_CGO_copy, NULL, NULL, NULL, NULL, NULL); } if (!I->offscreenOIT_CGO){ I->offscreenOIT_CGO = GenerateUnitScreenCGO(G); CGOChangeShadersTo(I->offscreenOIT_CGO, GL_DEFAULT_SHADER_WITH_SETTINGS, GL_OIT_SHADER); I->offscreenOIT_CGO->use_shader = true; } CGORenderGL(I->offscreenOIT_CGO, NULL, NULL, NULL, NULL, NULL); glBlendFunc_default(); if ((currentFrameBuffer == G->ShaderMgr->default_framebuffer_id) && t_mode_3){ // onlySelections and t_mode_3, render only gadgets SceneRenderAll(G, context, normal, NULL, -1 /* gadgets render in transp pass */, false, width_scale, grid, times, 1 /* only gadgets */, false); } glDisable(GL_BLEND); glDepthMask(GL_TRUE); if (!excludeSelections){ GridGetGLViewport(G, grid); for(slot = 0; slot <= grid->last_slot; slot++) { if(grid->active) { GridSetGLViewport(grid, slot); } if (!grid->active || slot > 0){ /* slot 0 is the full screen in grid mode, so don't render selections */ int s = grid->active && grid->mode==1 ? slot : 0; ExecutiveRenderSelections(G, curState, s, grid); } } } } #endif } } if(grid->active) GridSetGLViewport(grid, -1); } /*==================================================================================*/ /* SceneRenderStereoLoop: This is the function that is responsible for rendering all objects either in a monoscopic or stereo display. It prepares the viewport, offscreen textures (if necessary), draws the background (if necessary) and calls DoRendering either once (for monoscopic) or twice (for stereo) */ void SceneRenderStereoLoop(PyMOLGlobals * G, int timesArg, int must_render_stereo, int stereo_mode, short render_to_texture, int x, int y, int oversize_width, int oversize_height, int stereo_double_pump_mono, int curState, float *normal, SceneUnitContext *context, float width_scale, int fog_active, int onlySelections, int noAA){ CScene *I = G->Scene; int times = timesArg; short offscreen_aa = !onlySelections && render_to_texture && !noAA; bool use_shaders = (bool)SettingGetGlobal_b(G, cSetting_use_shaders); // only cStereo_clone_dynamic and cStereo_dynamic has times=2, otherwise times=1 while(times--) { if(must_render_stereo) { bool anaglyph = G->ShaderMgr && stereo_mode==cStereo_anaglyph; /* STEREO RENDERING (real or double-pumped) */ PRINTFD(G, FB_Scene) " SceneRender: left hand stereo...\n" ENDFD; /* LEFT HAND STEREO */ if (anaglyph){ G->ShaderMgr->stereo_flag = -1; // left eye G->ShaderMgr->stereo_blend = 0; } SceneSetPrepareViewPortForStereo(G, PrepareViewPortForStereo, times, x, y, oversize_width, oversize_height, stereo_mode, width_scale); if (!offscreen_aa){ PrepareViewPortForStereo(G, I, stereo_mode, render_to_texture, times, x, y, oversize_width, oversize_height); } #ifndef PURE_OPENGL_ES_2 if (use_shaders) glPushMatrix(); // 1 if (offscreen_aa){ G->ShaderMgr->bindOffscreen(I->Width, I->Height, &I->grid); bg_grad(G); } #endif ScenePrepareMatrix(G, stereo_double_pump_mono ? 0 : 1); DoRendering(G, I, render_to_texture, &I->grid, times, curState, normal, context, width_scale, onlySelections, render_to_texture); #ifndef PURE_OPENGL_ES_2 if (use_shaders) glPopMatrix(); // 0 #endif PRINTFD(G, FB_Scene) " SceneRender: right hand stereo...\n" ENDFD; if (offscreen_aa){ SceneRenderAA(G); } /* RIGHT HAND STEREO */ if (anaglyph){ G->ShaderMgr->stereo_flag = 1; // right eye G->ShaderMgr->stereo_blend = render_stereo_blend_into_full_screen(stereo_mode); } SceneSetPrepareViewPortForStereo(G, PrepareViewPortForStereo2nd, times, x, y, oversize_width, oversize_height, stereo_mode, width_scale); if (!offscreen_aa){ PrepareViewPortForStereo2nd(G, I, stereo_mode, render_to_texture, times, x, y, oversize_width, oversize_height); } #ifndef PURE_OPENGL_ES_2 if (!use_shaders) glPushMatrix(); // 1 if (offscreen_aa){ G->ShaderMgr->bindOffscreen(I->Width, I->Height, &I->grid); } if (offscreen_aa || (stereo_mode == cStereo_quadbuffer && !onlySelections) // PYMOL-2342 ) { bg_grad(G); } #endif ScenePrepareMatrix(G, stereo_double_pump_mono ? 0 : 2); glClear(GL_DEPTH_BUFFER_BIT); DoRendering(G, I, render_to_texture, &I->grid, times, curState, normal, context, width_scale, onlySelections, render_to_texture); if (anaglyph){ G->ShaderMgr->stereo_flag = 0; G->ShaderMgr->stereo_blend = 0; } #ifndef PURE_OPENGL_ES_2 if (!use_shaders) glPopMatrix(); // 0 #endif /* restore draw buffer */ if (offscreen_aa){ SceneRenderAA(G); } SetDrawBufferForStereo(G, I, stereo_mode, times, fog_active, render_to_texture); } else { if (G->ShaderMgr){ G->ShaderMgr->stereo_flag = 0; G->ShaderMgr->stereo_blend = 0; } /* MONOSCOPING RENDERING (not double-pumped) */ if(!I->grid.active && render_to_texture) { glViewport(0, 0, I->Width, I->Height); if (!onlySelections) bg_grad(G); } if(Feedback(G, FB_OpenGL, FB_Debugging)) PyMOLCheckOpenGLErr("Before mono rendering"); SceneSetPrepareViewPortForStereo(G, PrepareViewPortForMonoInitializeViewPort, times, x, y, oversize_width, oversize_height, stereo_mode, width_scale); DoRendering(G, I, render_to_texture, &I->grid, times, curState, normal, context, width_scale, onlySelections, render_to_texture); if(Feedback(G, FB_OpenGL, FB_Debugging)) PyMOLCheckOpenGLErr("during mono rendering"); } } } void PrepareViewPortForMonoInitializeViewPort(PyMOLGlobals * G, CScene *I, int stereo_mode, short offscreen, int times, int x, int y, int oversize_width, int oversize_height){ float width_scale; InitializeViewPortToScreenBlock(G, I, x, y, oversize_width, oversize_height, &stereo_mode, &width_scale); } void PrepareViewPortForStereo(PyMOLGlobals * G, CScene *I, int stereo_mode, short offscreen, int times, int x, int y, int oversize_width, int oversize_height){ PrepareViewPortForStereoImpl(G, I, stereo_mode, offscreen, times, x, y, oversize_width, oversize_height, GL_BACK_LEFT, 0); } void PrepareViewPortForStereo2nd(PyMOLGlobals * G, CScene *I, int stereo_mode, short offscreen, int times, int x, int y, int oversize_width, int oversize_height){ PrepareViewPortForStereoImpl(G, I, stereo_mode, offscreen, times, x, y, oversize_width, oversize_height, GL_BACK_RIGHT, 1); } void InitializeViewPortToScreenBlock(PyMOLGlobals * G, CScene *I, int x, int y, int oversize_width, int oversize_height, int *stereo_mode, float *width_scale){ if(oversize_width && oversize_height) { int want_view[4]; int got_view[4]; want_view[0] = I->rect.left + x; want_view[1] = I->rect.bottom + y; want_view[2] = oversize_width; want_view[3] = oversize_height; glViewport(want_view[0], want_view[1], want_view[2], want_view[3]); #ifdef _PYMOL_IOS { int width, height; SceneGetWidthHeight(G, &width, &height); got_view[0] = got_view[1] = 0; got_view[2] = width; got_view[3] = height; } #else glGetIntegerv(GL_VIEWPORT, (GLint *) (void *) got_view); #endif #ifndef _WEBGL if((got_view[0] != want_view[0]) || (got_view[1] != want_view[1]) || (got_view[2] != want_view[2]) || (got_view[3] != want_view[3])) { PRINTFB(G, FB_Scene, FB_Warnings) "Scene-Warning: glViewport failure.\n" ENDFB(G); } #endif switch (*stereo_mode) { case cStereo_geowall: *stereo_mode = 0; break; } *width_scale = ((float) (oversize_width)) / I->Width; } else { glViewport(I->rect.left, I->rect.bottom, I->Width, I->Height); } } void SceneSetPrepareViewPortForStereo(PyMOLGlobals *G, void (*prepareViewPortForStereo)(PyMOLGlobals *, CScene *, int, short, int, int, int, int, int), int times, int x, int y, int oversize_width, int oversize_height, int stereo_mode, float width_scale){ CScene *I = G->Scene; I->vp_prepareViewPortForStereo = prepareViewPortForStereo; I->vp_times = times; I->vp_x = x; I->vp_y = y; I->vp_owidth = oversize_width; I->vp_oheight = oversize_height; I->vp_stereo_mode = stereo_mode; I->vp_width_scale = width_scale; } /* PrepareViewPortForStereoImpl : sets up viewport and GL state for stereo_modes */ void PrepareViewPortForStereoImpl(PyMOLGlobals * G, CScene *I, int stereo_mode, short offscreen, int times, int x, int y, int oversize_width, int oversize_height, GLenum draw_mode, int position /* left=0, right=1 */){ int position_inv = 1 - position; switch (stereo_mode) { case cStereo_quadbuffer: /* hardware */ OrthoDrawBuffer(G, draw_mode); glViewport(I->rect.left, I->rect.bottom, I->Width, I->Height); break; case cStereo_crosseye: /* side by side, crosseye */ if (offscreen){ glViewport(position_inv * I->Width / 2, 0, I->Width / 2, I->Height); } else if(oversize_width && oversize_height) { glViewport(I->rect.left + (position_inv * oversize_width / 2) + x, I->rect.bottom + y, oversize_width / 2, oversize_height); } else { glViewport(I->rect.left + (position_inv * I->Width / 2), I->rect.bottom, I->Width / 2, I->Height); } break; case cStereo_walleye: case cStereo_sidebyside: if (offscreen){ glViewport(position * I->Width / 2, 0, I->Width / 2, I->Height); } else if(oversize_width && oversize_height) { glViewport(I->rect.left + (position * oversize_width / 2) + x, I->rect.bottom + y, oversize_width / 2, oversize_height); } else { glViewport(I->rect.left + (position * I->Width / 2), I->rect.bottom, I->Width / 2, I->Height); } break; case cStereo_geowall: if (offscreen){ glViewport(position * I->Width / 2, 0, I->Width / 2, I->Height); } else { glViewport(I->rect.left + (position * G->Option->winX / 2), I->rect.bottom, I->Width, I->Height); } break; case cStereo_stencil_by_row: case cStereo_stencil_by_column: case cStereo_stencil_checkerboard: if(I->StencilValid) { glStencilFunc(GL_EQUAL, position_inv, 1); glStencilOp(GL_KEEP, GL_KEEP, GL_KEEP); glEnable(GL_STENCIL_TEST); } break; case cStereo_stencil_custom: #ifdef _PYMOL_SHARP3D sharp3d_begin_left_stereo(); #endif break; case cStereo_anaglyph: /* glClear(GL_ACCUM_BUFFER_BIT); */ #ifdef _PYMOL_IOS if (position) glClear(GL_DEPTH_BUFFER_BIT); glEnable(GL_BLEND); glBlendEquation(GL_FUNC_ADD); glColorMask(position_inv, position, position, true); #else if (TM3_IS_ONEBUF){ glColorMask(position_inv, position, position, true); } else { if (GLEW_VERSION_3_0 && SettingGetGlobal_i(G, cSetting_transparency_mode) == 3){ // if GL 3.0 is available, use glColorMaski to mask only first draw buffer // for anaglyph in transparency_mode 3 glColorMaski(0, position_inv, position, position, true); } else { glColorMask(position_inv, position, position, true); } } if (position) glClear(GL_DEPTH_BUFFER_BIT); #endif break; #ifndef PURE_OPENGL_ES_2 case cStereo_clone_dynamic: if (position_inv){ glClear(GL_ACCUM_BUFFER_BIT); OrthoDrawBuffer(G, GL_BACK_LEFT); if(times) { float dynamic_strength = SettingGetGlobal_f(G, cSetting_stereo_dynamic_strength); float vv[4] = { 0.75F, 0.75F, 0.75F, 1.0F }; vv[0] = dynamic_strength; vv[1] = dynamic_strength; vv[2] = dynamic_strength; glMaterialfv(GL_FRONT_AND_BACK, GL_EMISSION, vv); glAccum(GL_ADD, 0.5); glDisable(GL_FOG); } } else { GLenum err; if(times) { glAccum(GL_ACCUM, -0.5); } else { glAccum(GL_ACCUM, 0.5); } if((err = glGetError())) { PRINTFB(G, FB_Scene, FB_Errors) "Stereo Error 0x%x: stereo_mode=12 clone_dynamic requires access to the accumulation buffer,\n" "you need to start PyMOL with the -t argument, setting back to default\n", err ENDFB(G); SettingSetGlobal_i(G, cSetting_stereo_mode, cStereo_crosseye); SceneSetStereo(G, 0); return; } glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); } break; case cStereo_dynamic: if (position_inv){ if(times) { float dynamic_strength = SettingGetGlobal_f(G, cSetting_stereo_dynamic_strength); float vv[4] = { 0.75F, 0.75F, 0.75F, 1.0F }; vv[0] = dynamic_strength; vv[1] = dynamic_strength; vv[2] = dynamic_strength; glClearAccum(0.5, 0.5, 0.5, 0.5); glClear(GL_ACCUM_BUFFER_BIT); glMaterialfv(GL_FRONT_AND_BACK, GL_EMISSION, vv); glDisable(GL_FOG); glViewport(I->rect.left + G->Option->winX / 2, I->rect.bottom, I->Width, I->Height); } else { glClearAccum(0.0, 0.0, 0.0, 0.0); glClear(GL_ACCUM_BUFFER_BIT); glViewport(I->rect.left, I->rect.bottom, I->Width, I->Height); } } else { GLenum err; if(times) { glAccum(GL_ACCUM, -0.5); } else { glAccum(GL_ACCUM, 0.5); glEnable(GL_SCISSOR_TEST); } if((err = glGetError())) { if (SettingGetGlobal_i(G, cSetting_stereo_mode) != cStereo_crosseye){ PRINTFB(G, FB_Scene, FB_Errors) "Stereo Error 0x%x: stereo_mode=11 dynamic requires access to the accumulation buffer,\n" "you need to start PyMOL with the -t argument, setting back to default\n", err ENDFB(G); SettingSetGlobal_i(G, cSetting_stereo_mode, cStereo_crosseye); SceneSetStereo(G, 0); } return; } glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); if(!times) { glDisable(GL_SCISSOR_TEST); } } break; #endif } } /* SetDrawBufferForStereo : called after 2nd/right eye rendered in stereo to reset GL state properly based on what was changed in PrepareViewPortForStereoImpl per stereo_mode */ void SetDrawBufferForStereo(PyMOLGlobals * G, CScene *I, int stereo_mode, int times, int fog_active, int offscreen){ switch (stereo_mode) { case cStereo_quadbuffer: OrthoDrawBuffer(G, GL_BACK_LEFT); /* leave us in a stereo context (avoids problems with cards than can't handle use of mono contexts) */ break; case cStereo_crosseye: case cStereo_walleye: case cStereo_sidebyside: OrthoDrawBuffer(G, GL_BACK); break; case cStereo_geowall: break; case cStereo_stencil_by_row: case cStereo_stencil_by_column: case cStereo_stencil_checkerboard: glDisable(GL_STENCIL_TEST); break; case cStereo_stencil_custom: #ifdef _PYMOL_SHARP3D sharp3d_end_stereo(); #endif break; case cStereo_anaglyph: glColorMask(true, true, true, true); break; case cStereo_clone_dynamic: #ifndef PURE_OPENGL_ES_2 glAccum(GL_ACCUM, 0.5); #endif if(times) { float vv[4] = { 0.0F, 0.0F, 0.0F, 0.0F }; #ifndef PURE_OPENGL_ES_2 glMaterialfv(GL_FRONT_AND_BACK, GL_EMISSION, vv); if(fog_active) glEnable(GL_FOG); #endif glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); OrthoDrawBuffer(G, GL_BACK_RIGHT); } #ifndef PURE_OPENGL_ES_2 glAccum(GL_RETURN, 1.0); #endif OrthoDrawBuffer(G, GL_BACK_LEFT); break; case cStereo_dynamic: #ifndef PURE_OPENGL_ES_2 glAccum(GL_ACCUM, 0.5); #endif if(times) { float vv[4] = { 0.0F, 0.0F, 0.0F, 0.0F }; #ifndef PURE_OPENGL_ES_2 glMaterialfv(GL_FRONT_AND_BACK, GL_EMISSION, vv); if(fog_active) glEnable(GL_FOG); #endif glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); } #ifndef PURE_OPENGL_ES_2 glAccum(GL_RETURN, 1.0); #endif if(times) { glViewport(I->rect.left, I->rect.bottom, I->Width + 2, I->Height + 2); glScissor(I->rect.left - 1, I->rect.bottom - 1, I->Width + 2, I->Height + 2); glEnable(GL_SCISSOR_TEST); glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); glDisable(GL_SCISSOR_TEST); } else { glDisable(GL_SCISSOR_TEST); } break; } } void SceneInitializeViewport(PyMOLGlobals * G, int offscreen){ CScene *I = G->Scene; if (offscreen == 1 || offscreen == 2) glViewport(0, 0, I->Width, I->Height); else { if (I->vp_prepareViewPortForStereo){ GLint currentFrameBuffer; glGetIntegerv(GL_FRAMEBUFFER_BINDING, &currentFrameBuffer); if (currentFrameBuffer == G->ShaderMgr->default_framebuffer_id){ // if writing to screen, then set viewport to screen float width_scale; // this is called before preparing view port, since the prepare function // doesn't setup/change the viewport in all modes InitializeViewPortToScreenBlock(G, I, I->vp_x, I->vp_y, I->vp_owidth, I->vp_oheight, &I->vp_stereo_mode, &width_scale); } I->vp_prepareViewPortForStereo(G, I, I->vp_stereo_mode, 0, I->vp_times, I->vp_x, I->vp_y, I->vp_owidth, I->vp_oheight); } else { PRINTFB(G, FB_Scene, FB_Errors) " SceneInitializeViewport: I->vp_prepareViewPortForStereo=NULL\n" ENDFB(G); } } } void SceneDrawStencilInBuffer(PyMOLGlobals * G, CScene *I, int stereo_mode){ GLint viewport[4]; #ifdef _PYMOL_IOS { int width, height; SceneGetWidthHeight(G, &width, &height); viewport[0] = viewport[1] = 0; viewport[2] = width; viewport[3] = height; } #else glGetIntegerv(GL_VIEWPORT, viewport); #endif #ifndef PURE_OPENGL_ES_2 glMatrixMode(GL_PROJECTION); glPushMatrix(); glLoadIdentity(); glOrtho(0, viewport[2], 0, viewport[3], -10.0, 10.0); glMatrixMode(GL_MODELVIEW); glPushMatrix(); glLoadIdentity(); glTranslatef(0.33F, 0.33F, 0.0F); glDisable(GL_ALPHA_TEST); glDisable(GL_LIGHTING); glDisable(GL_FOG); glDisable(GL_NORMALIZE); glDisable(GL_COLOR_MATERIAL); glDisable(GL_LINE_SMOOTH); glShadeModel(SettingGetGlobal_b(G, cSetting_pick_shading) ? GL_FLAT : GL_SMOOTH); glDisable(0x809D); /* GL_MULTISAMPLE_ARB */ #endif glDisable(GL_DEPTH_TEST); glDisable(GL_DITHER); glDisable(GL_BLEND); glDisable(GL_STENCIL_TEST); glClearStencil(0); glColorMask(false, false, false, false); glDepthMask(false); glClear(GL_STENCIL_BUFFER_BIT); glEnable(GL_STENCIL_TEST); glStencilFunc(GL_ALWAYS, 1, 1); glStencilOp(GL_KEEP, GL_KEEP, GL_REPLACE); #ifndef PURE_OPENGL_ES_2 { int h = viewport[3], w = viewport[2]; glLineWidth(1.0); switch (stereo_mode) { case cStereo_stencil_by_row: { int parity = I->StencilParity; int y; glBegin(GL_LINES); for(y = 0; y < h; y += 2) { glVertex2i(0, y + parity); glVertex2i(w, y + parity); } glEnd(); } break; case cStereo_stencil_by_column: { int x; glBegin(GL_LINES); for(x = 0; x < w; x += 2) { glVertex2i(x, 0); glVertex2i(x, h); } glEnd(); } break; case cStereo_stencil_checkerboard: { int i, m = 2 * ((h > w) ? h : w); glBegin(GL_LINES); for(i = 0; i < m; i += 2) { glVertex2i(i, 0); glVertex2i(0, i); } glEnd(); } break; } } #endif glColorMask(true, true, true, true); glDepthMask(true); #ifndef PURE_OPENGL_ES_2 glMatrixMode(GL_MODELVIEW); glPopMatrix(); glMatrixMode(GL_PROJECTION); glPopMatrix(); #endif } CGO *GenerateUnitScreenCGO(PyMOLGlobals * G){ int ok = true; CGO *cgo2 = NULL, *ret = NULL; CGO *cgo = CGONew(G); ok &= CGOBegin(cgo, GL_TRIANGLE_STRIP); if (ok) ok &= CGOVertex(cgo, -1.f, -1.f, 0.98f); if (ok) ok &= CGOVertex(cgo, 1.f, -1.f, 0.98f); if (ok) ok &= CGOVertex(cgo, -1.f, 1.f, 0.98f); if (ok) ok &= CGOVertex(cgo, 1.f, 1.f, 0.98f); if (ok) ok &= CGOEnd(cgo); if (ok) ok &= CGOStop(cgo); if (ok) cgo2 = CGOCombineBeginEnd(cgo, 0); CHECKOK(ok, cgo2); CGOFree(cgo); if (ok) ret = CGOOptimizeToVBONotIndexed(cgo2, 0); CGOFree(cgo2); return ret; }