"""Contains the BufferViewer class, which is used as a debugging aid when debugging render-to-texture effects. It shows different views at the bottom of the screen showing the various render targets.""" __all__ = ['BufferViewer'] from panda3d.core import * from direct.task import Task from direct.task.TaskManagerGlobal import taskMgr from direct.directnotify.DirectNotifyGlobal import * from direct.showbase.DirectObject import DirectObject import math class BufferViewer(DirectObject): notify = directNotify.newCategory('BufferViewer') def __init__(self, win, parent): """Access: private. Constructor.""" self.enabled = 0 size = ConfigVariableDouble('buffer-viewer-size', '0 0') self.sizex = size[0] self.sizey = size[1] self.position = ConfigVariableString('buffer-viewer-position', "lrcorner").getValue() self.layout = ConfigVariableString('buffer-viewer-layout', "hline").getValue() self.include = "all" self.exclude = "none" self.cullbin = "fixed" self.cullsort = 10000 self.win = win self.engine = GraphicsEngine.getGlobalPtr() self.renderParent = parent self.cards = [] self.cardindex = 0 self.cardmaker = CardMaker("cubemaker") self.cardmaker.setFrame(-1,1,-1,1) self.task = 0 self.dirty = 1 self.accept("render-texture-targets-changed", self.refreshReadout) if (ConfigVariableBool("show-buffers", 0).getValue()): self.enable(1) def refreshReadout(self): """Force the readout to be refreshed. This is usually invoked by GraphicsOutput::add_render_texture (via an event handler). However, it is also possible to invoke it manually. Currently, the only time I know of that this is necessary is after a window resize (and I ought to fix that).""" self.dirty = 1 # Call enabled again, mainly to ensure that the task has been # started. self.enable(self.enabled) def isValidTextureSet(self, x): """Access: private. Returns true if the parameter is a list of GraphicsOutput and Texture, or the keyword 'all'.""" if (isinstance(x, list)): for elt in x: if (self.isValidTextureSet(elt)==0): return 0 else: return (x=="all") or (isinstance(x, Texture)) or (isinstance(x, GraphicsOutput)) def isEnabled(self): """Returns true if the buffer viewer is currently enabled.""" return self.enabled def enable(self, x): """Turn the buffer viewer on or off. The initial state of the buffer viewer depends on the Config variable 'show-buffers'.""" if (x != 0) and (x != 1): BufferViewer.notify.error('invalid parameter to BufferViewer.enable') return self.enabled = x self.dirty = 1 if (x and self.task == 0): self.task = taskMgr.add(self.maintainReadout, "buffer-viewer-maintain-readout", priority=1) def toggleEnable(self): """Toggle the buffer viewer on or off. The initial state of the enable flag depends on the Config variable 'show-buffers'.""" self.enable(1-self.enabled) def setCardSize(self, x, y): """Set the size of each card. The units are relative to render2d (ie, 1x1 card is not square). If one of the dimensions is zero, then the viewer will choose a value for that dimension so as to ensure that the aspect ratio of the card matches the aspect ratio of the source-window. If both dimensions are zero, the viewer uses a heuristic to choose a reasonable size for the card. The initial value is (0, 0).""" if (x < 0) or (y < 0): BufferViewer.notify.error('invalid parameter to BufferViewer.setCardSize') return self.sizex = x self.sizey = y self.dirty = 1 def setPosition(self, pos): """Set the position of the cards. The valid values are: * llcorner - put them in the lower-left corner of the window * lrcorner - put them in the lower-right corner of the window * ulcorner - put them in the upper-left corner of the window * urcorner - put them in the upper-right corner of the window * window - put them in a separate window The initial value is 'lrcorner'.""" valid=["llcorner","lrcorner","ulcorner","urcorner","window"] if (valid.count(pos)==0): BufferViewer.notify.error('invalid parameter to BufferViewer.setPosition') BufferViewer.notify.error('valid parameters are: llcorner, lrcorner, ulcorner, urcorner, window') return if (pos == "window"): BufferViewer.notify.error('BufferViewer.setPosition - "window" mode not implemented yet.') return self.position = pos self.dirty = 1 def setLayout(self, lay): """Set the layout of the cards. The valid values are: * vline - display them in a vertical line * hline - display them in a horizontal line * vgrid - display them in a vertical grid * hgrid - display them in a horizontal grid * cycle - display one card at a time, using selectCard/advanceCard The default value is 'hline'.""" valid=["vline","hline","vgrid","hgrid","cycle"] if (valid.count(lay)==0): BufferViewer.notify.error('invalid parameter to BufferViewer.setLayout') BufferViewer.notify.error('valid parameters are: vline, hline, vgrid, hgrid, cycle') return self.layout = lay self.dirty = 1 def selectCard(self, i): """Only useful when using setLayout('cycle'). Sets the index that selects which card to display. The index is taken modulo the actual number of cards.""" self.cardindex = i self.dirty = 1 def advanceCard(self): """Only useful when using setLayout('cycle'). Increments the index that selects which card to display. The index is taken modulo the actual number of cards.""" self.cardindex += 1 self.dirty = 1 def setInclude(self, x): """Set the include-set for the buffer viewer. The include-set specifies which of the render-to-texture targets to display. Valid inputs are the string 'all' (display every render-to-texture target), or a list of GraphicsOutputs or Textures. The initial value is 'all'.""" if (self.isValidTextureSet(x)==0): BufferViewer.notify.error('setInclude: must be list of textures and buffers, or "all"') return self.include = x self.dirty = 1 def setExclude(self, x): """Set the exclude-set for the buffer viewer. The exclude-set should be a list of GraphicsOutputs and Textures to ignore. The exclude-set is subtracted from the include-set (so the excludes effectively override the includes.) The initial value is the empty list.""" if (self.isValidTextureSet(x)==0): BufferViewer.notify.error('setExclude: must be list of textures and buffers') return self.exclude = x self.dirty = 1 def setSort(self, bin, sort): """Set the cull-bin and sort-order for the output cards. The default value is 'fixed', 10000.""" self.cullbin = bin self.cullsort = sort self.dirty = 1 def setRenderParent(self, renderParent): """Set the scene graph root to which the output cards should be parented. The default is render2d. """ self.renderParent = renderParent self.dirty = 1 def analyzeTextureSet(self, x, set): """Access: private. Converts a list of GraphicsObject, GraphicsEngine, and Texture into a table of Textures.""" if (isinstance(x, list)): for elt in x: self.analyzeTextureSet(elt, set) elif (isinstance(x, Texture)): set[x] = 1 elif (isinstance(x, GraphicsOutput)): for itex in range(x.countTextures()): tex = x.getTexture(itex) set[tex] = 1 elif (isinstance(x, GraphicsEngine)): for iwin in range(x.getNumWindows()): win = x.getWindow(iwin) self.analyzeTextureSet(win, set) elif (x=="all"): self.analyzeTextureSet(self.engine, set) else: return def makeFrame(self, sizex, sizey): """Access: private. Each texture card is displayed with a two-pixel wide frame (a ring of black and a ring of white). This routine builds the frame geometry. It is necessary to be precise so that the frame exactly aligns to pixel boundaries, and so that it doesn't overlap the card at all.""" format=GeomVertexFormat.getV3cp() vdata=GeomVertexData('card-frame', format, Geom.UHDynamic) vwriter=GeomVertexWriter(vdata, 'vertex') cwriter=GeomVertexWriter(vdata, 'color') ringoffset = [0, 1, 1, 2] ringbright = [0, 0, 1, 1] for ring in range(4): offsetx = (ringoffset[ring]*2.0) / float(sizex) offsety = (ringoffset[ring]*2.0) / float(sizey) bright = ringbright[ring] vwriter.addData3f(-1-offsetx, 0, -1-offsety) vwriter.addData3f(1+offsetx, 0, -1-offsety) vwriter.addData3f(1+offsetx, 0, 1+offsety) vwriter.addData3f(-1-offsetx, 0, 1+offsety) cwriter.addData3f(bright, bright, bright) cwriter.addData3f(bright, bright, bright) cwriter.addData3f(bright, bright, bright) cwriter.addData3f(bright, bright, bright) triangles=GeomTriangles(Geom.UHStatic) for i in range(2): delta = i*8 triangles.addVertices(0+delta, 4+delta, 1+delta) triangles.addVertices(1+delta, 4+delta, 5+delta) triangles.addVertices(1+delta, 5+delta, 2+delta) triangles.addVertices(2+delta, 5+delta, 6+delta) triangles.addVertices(2+delta, 6+delta, 3+delta) triangles.addVertices(3+delta, 6+delta, 7+delta) triangles.addVertices(3+delta, 7+delta, 0+delta) triangles.addVertices(0+delta, 7+delta, 4+delta) triangles.closePrimitive() geom=Geom(vdata) geom.addPrimitive(triangles) geomnode=GeomNode("card-frame") geomnode.addGeom(geom) return NodePath(geomnode) def maintainReadout(self, task): """Access: private. Whenever necessary, rebuilds the entire display from scratch. This is only done when the configuration parameters have changed.""" # If nothing has changed, don't update. if (self.dirty==0): return Task.cont self.dirty = 0 # Delete the old set of cards. for card in self.cards: card.removeNode() self.cards = [] # If not enabled, return. if (self.enabled == 0): self.task = 0 return Task.done # Generate the include and exclude sets. exclude = {} include = {} self.analyzeTextureSet(self.exclude, exclude) self.analyzeTextureSet(self.include, include) # Use a custom sampler when applying the textures. This fixes # wrap issues and prevents depth compare on shadow maps. sampler = SamplerState() sampler.setWrapU(SamplerState.WM_clamp) sampler.setWrapV(SamplerState.WM_clamp) sampler.setWrapW(SamplerState.WM_clamp) sampler.setMinfilter(SamplerState.FT_linear) sampler.setMagfilter(SamplerState.FT_nearest) # Generate a list of cards and the corresponding windows. cards = [] wins = [] for iwin in range(self.engine.getNumWindows()): win = self.engine.getWindow(iwin) for itex in range(win.countTextures()): tex = win.getTexture(itex) if (tex in include) and (tex not in exclude): if (tex.getTextureType() == Texture.TTCubeMap): for face in range(6): self.cardmaker.setUvRangeCube(face) card = NodePath(self.cardmaker.generate()) card.setTexture(tex, sampler) cards.append(card) elif (tex.getTextureType() == Texture.TT2dTextureArray): for layer in range(tex.getZSize()): self.cardmaker.setUvRange((0, 1, 1, 0), (0, 0, 1, 1),\ (layer, layer, layer, layer)) card = NodePath(self.cardmaker.generate()) # 2D texture arrays are not supported by # the fixed-function pipeline, so we need to # enable the shader generator to view them. card.setShaderAuto() card.setTexture(tex, sampler) cards.append(card) else: card = win.getTextureCard() card.setTexture(tex, sampler) cards.append(card) wins.append(win) exclude[tex] = 1 self.cards = cards if (len(cards)==0): self.task = 0 return Task.done ncards = len(cards) # Decide how many rows and columns to use for the layout. if (self.layout == "hline"): rows = 1 cols = ncards elif (self.layout == "vline"): rows = ncards cols = 1 elif (self.layout == "hgrid"): rows = int(math.sqrt(ncards)) cols = rows if (rows * cols < ncards): cols += 1 if (rows * cols < ncards): rows += 1 elif (self.layout == "vgrid"): rows = int(math.sqrt(ncards)) cols = rows if (rows * cols < ncards): rows += 1 if (rows * cols < ncards): cols += 1 elif (self.layout == "cycle"): rows = 1 cols = 1 else: BufferViewer.notify.error('shouldnt ever get here in BufferViewer.maintainReadout') # Choose an aspect ratio for the cards. All card size # calculations are done in pixel-units, using integers, # in order to ensure that everything ends up neatly on # a pixel boundary. aspectx = wins[0].getXSize() aspecty = wins[0].getYSize() for win in wins: if (win.getXSize()*aspecty) != (win.getYSize()*aspectx): aspectx = 1 aspecty = 1 # Choose a card size. If the user didn't specify a size, # use a heuristic, otherwise, just follow orders. The # heuristic uses an initial card size of 42.66666667% of # the screen vertically, which comes to 256 pixels on # an 800x600 display. Then, it double checks that the # readout will fit on the screen, and if not, it shrinks it. bordersize = 4.0 if (float(self.sizex)==0.0) and (float(self.sizey)==0.0): sizey = int(0.4266666667 * self.win.getYSize()) sizex = (sizey * aspectx) // aspecty v_sizey = (self.win.getYSize() - (rows-1) - (rows*2)) // rows v_sizex = (v_sizey * aspectx) // aspecty if (v_sizey < sizey) or (v_sizex < sizex): sizey = v_sizey sizex = v_sizex adjustment = 2 h_sizex = float (self.win.getXSize() - adjustment) / float (cols) h_sizex -= bordersize if (h_sizex < 1.0): h_sizex = 1.0 h_sizey = (h_sizex * aspecty) // aspectx if (h_sizey < sizey) or (h_sizex < sizex): sizey = h_sizey sizex = h_sizex else: sizex = int(self.sizex * 0.5 * self.win.getXSize()) sizey = int(self.sizey * 0.5 * self.win.getYSize()) if (sizex == 0): sizex = (sizey*aspectx) // aspecty if (sizey == 0): sizey = (sizex*aspecty) // aspectx # Convert from pixels to render2d-units. fsizex = (2.0 * sizex) / float(self.win.getXSize()) fsizey = (2.0 * sizey) / float(self.win.getYSize()) fpixelx = 2.0 / float(self.win.getXSize()) fpixely = 2.0 / float(self.win.getYSize()) # Choose directional offsets if (self.position == "llcorner"): dirx = -1.0 diry = -1.0 elif (self.position == "lrcorner"): dirx = 1.0 diry = -1.0 elif (self.position == "ulcorner"): dirx = -1.0 diry = 1.0 elif (self.position == "urcorner"): dirx = 1.0 diry = 1.0 else: BufferViewer.notify.error('window mode not implemented yet') # Create the frame frame = self.makeFrame(sizex, sizey) # Now, position the cards on the screen. # For each card, create a frame consisting of eight quads. for r in range(rows): for c in range(cols): index = c + r*cols if (index < ncards): index = (index + self.cardindex) % len(cards) posx = dirx * (1.0 - ((c + 0.5) * (fsizex + fpixelx * bordersize))) - (fpixelx * dirx) posy = diry * (1.0 - ((r + 0.5) * (fsizey + fpixely * bordersize))) - (fpixely * diry) placer = NodePath("card-structure") placer.setPos(Point3.rfu(posx, 0, posy)) placer.setScale(Vec3.rfu(fsizex*0.5, 1.0, fsizey*0.5)) placer.setBin(self.cullbin, self.cullsort) placer.reparentTo(self.renderParent) frame.instanceTo(placer) cards[index].reparentTo(placer) cards[index] = placer return Task.cont # Snake-case aliases, for people who prefer these. advance_card = advanceCard analyze_texture_set = analyzeTextureSet is_enabled = isEnabled is_valid_texture_set = isValidTextureSet maintain_readout = maintainReadout make_frame = makeFrame refresh_readout = refreshReadout select_card = selectCard set_card_size = setCardSize set_exclude = setExclude set_include = setInclude set_layout = setLayout set_position = setPosition set_render_parent = setRenderParent set_sort = setSort toggle_enable = toggleEnable