/* * Copyright (c) 2015 scrawl <scrawl@baseoftrash.de> * * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation files (the "Software"), to deal * in the Software without restriction, including without limitation the rights * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in * all copies or substantial portions of the Software. * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN * THE SOFTWARE. */ #include "material.hpp" #include <iostream> #include <osg/Depth> #include <osg/TexEnvCombine> #include <osg/Texture2D> #include <osg/TexMat> #include <osg/Material> #include <osg/io_utils> namespace Terrain { FixedFunctionTechnique::FixedFunctionTechnique(const std::vector<osg::ref_ptr<osg::Texture2D> >& layers, const std::vector<osg::ref_ptr<osg::Texture2D> >& blendmaps) { bool firstLayer = true; int i=0; for (std::vector<osg::ref_ptr<osg::Texture2D> >::const_iterator it = layers.begin(); it != layers.end(); ++it) { osg::ref_ptr<osg::StateSet> stateset (new osg::StateSet); if (!firstLayer) { stateset->setMode(GL_BLEND, osg::StateAttribute::ON); osg::ref_ptr<osg::Depth> depth (new osg::Depth); depth->setFunction(osg::Depth::EQUAL); stateset->setAttributeAndModes(depth, osg::StateAttribute::ON); } int texunit = 0; if(!firstLayer) { osg::ref_ptr<osg::Texture2D> blendmap = blendmaps.at(i++); stateset->setTextureAttributeAndModes(texunit, blendmap.get()); // This is to map corner vertices directly to the center of a blendmap texel. osg::Matrixf texMat; float scale = (16/(16.f+1.f)); texMat.preMultTranslate(osg::Vec3f(0.5f, 0.5f, 0.f)); texMat.preMultScale(osg::Vec3f(scale, scale, 1.f)); texMat.preMultTranslate(osg::Vec3f(-0.5f, -0.5f, 0.f)); stateset->setTextureAttributeAndModes(texunit, new osg::TexMat(texMat)); ++texunit; } // Add the actual layer texture multiplied by the alpha map. osg::ref_ptr<osg::Texture2D> tex = *it; stateset->setTextureAttributeAndModes(texunit, tex.get()); osg::ref_ptr<osg::TexMat> texMat (new osg::TexMat); float scale = 16.f; texMat->setMatrix(osg::Matrix::scale(osg::Vec3f(scale,scale,1.f))); stateset->setTextureAttributeAndModes(texunit, texMat, osg::StateAttribute::ON); firstLayer = false; addPass(stateset); } } Effect::Effect(const std::vector<osg::ref_ptr<osg::Texture2D> > &layers, const std::vector<osg::ref_ptr<osg::Texture2D> > &blendmaps) : mLayers(layers) , mBlendmaps(blendmaps) { osg::ref_ptr<osg::Material> material (new osg::Material); material->setColorMode(osg::Material::AMBIENT_AND_DIFFUSE); getOrCreateStateSet()->setAttributeAndModes(material, osg::StateAttribute::ON); selectTechnique(0); } bool Effect::define_techniques() { addTechnique(new FixedFunctionTechnique(mLayers, mBlendmaps)); return true; } }