| 
									
										
										
										
											2011-02-23 10:52:22 +00:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2010-10-07 00:24:42 +00:00
										 |  |  |  * ***** BEGIN GPL LICENSE BLOCK ***** | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is free software; you can redistribute it and/or | 
					
						
							|  |  |  |  * modify it under the terms of the GNU General Public License | 
					
						
							|  |  |  |  * as published by the Free Software Foundation; either version 2 | 
					
						
							|  |  |  |  * of the License, or (at your option) any later version. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is distributed in the hope that it will be useful, | 
					
						
							|  |  |  |  * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  |  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the | 
					
						
							|  |  |  |  * GNU General Public License for more details. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You should have received a copy of the GNU General Public License | 
					
						
							|  |  |  |  * along with this program; if not, write to the Free Software Foundation, | 
					
						
							|  |  |  |  * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Contributor(s): Chingiz Dyussenov, Arystanbek Dyussenov, Jan Diederich, Tod Liverseed, | 
					
						
							|  |  |  |  *                 Nathan Letwory | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * ***** END GPL LICENSE BLOCK ***** | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-02-27 20:30:35 +00:00
										 |  |  | /** \file blender/collada/EffectExporter.cpp
 | 
					
						
							|  |  |  |  *  \ingroup collada | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-07 00:24:42 +00:00
										 |  |  | #include <map>
 | 
					
						
							| 
									
										
										
										
											2012-06-22 16:16:58 +00:00
										 |  |  | #include <set>
 | 
					
						
							| 
									
										
										
										
											2010-10-07 00:24:42 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "COLLADASWEffectProfile.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "EffectExporter.h"
 | 
					
						
							| 
									
										
										
										
											2012-06-12 21:25:29 +00:00
										 |  |  | #include "DocumentExporter.h"
 | 
					
						
							| 
									
										
										
										
											2010-10-07 00:24:42 +00:00
										 |  |  | #include "MaterialExporter.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "DNA_mesh_types.h"
 | 
					
						
							|  |  |  | #include "DNA_texture_types.h"
 | 
					
						
							| 
									
										
										
										
											2011-09-12 13:20:24 +00:00
										 |  |  | #include "DNA_world_types.h"
 | 
					
						
							| 
									
										
										
										
											2010-10-07 00:24:42 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "BKE_customdata.h"
 | 
					
						
							| 
									
										
										
										
											2012-06-22 16:16:58 +00:00
										 |  |  | #include "BKE_mesh.h"
 | 
					
						
							|  |  |  | #include "BKE_material.h"
 | 
					
						
							| 
									
										
										
										
											2010-10-07 00:24:42 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "collada_internal.h"
 | 
					
						
							|  |  |  | #include "collada_utils.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // OB_MESH is assumed
 | 
					
						
							|  |  |  | static std::string getActiveUVLayerName(Object *ob) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-06-12 22:05:33 +00:00
										 |  |  | 	Mesh *me = (Mesh *)ob->data; | 
					
						
							| 
									
										
										
										
											2010-10-07 00:24:42 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	int num_layers = CustomData_number_of_layers(&me->fdata, CD_MTFACE); | 
					
						
							|  |  |  | 	if (num_layers) | 
					
						
							|  |  |  | 		return std::string(bc_CustomData_get_active_layer_name(&me->fdata, CD_MTFACE)); | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 	return ""; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-12 22:05:33 +00:00
										 |  |  | EffectsExporter::EffectsExporter(COLLADASW::StreamWriter *sw, const ExportSettings *export_settings) : COLLADASW::LibraryEffects(sw), export_settings(export_settings) { | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2011-03-18 14:06:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | bool EffectsExporter::hasEffects(Scene *sce) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	Base *base = (Base *)sce->base.first; | 
					
						
							|  |  |  | 	 | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 	while (base) { | 
					
						
							| 
									
										
										
										
											2012-06-12 22:05:33 +00:00
										 |  |  | 		Object *ob = base->object; | 
					
						
							| 
									
										
										
										
											2011-03-18 14:06:13 +00:00
										 |  |  | 		int a; | 
					
						
							| 
									
										
										
										
											2012-04-28 06:31:57 +00:00
										 |  |  | 		for (a = 0; a < ob->totcol; a++) { | 
					
						
							| 
									
										
										
										
											2012-06-12 22:05:33 +00:00
										 |  |  | 			Material *ma = give_current_material(ob, a + 1); | 
					
						
							| 
									
										
										
										
											2011-03-18 14:06:13 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			// no material, but check all of the slots
 | 
					
						
							|  |  |  | 			if (!ma) continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			return true; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2012-06-12 22:05:33 +00:00
										 |  |  | 		base = base->next; | 
					
						
							| 
									
										
										
										
											2011-03-18 14:06:13 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-09-07 18:23:30 +00:00
										 |  |  | void EffectsExporter::exportEffects(Scene *sce) | 
					
						
							| 
									
										
										
										
											2010-10-07 00:24:42 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2012-03-24 07:52:14 +00:00
										 |  |  | 	if (hasEffects(sce)) { | 
					
						
							| 
									
										
										
										
											2011-09-12 13:20:24 +00:00
										 |  |  | 		this->scene = sce; | 
					
						
							| 
									
										
										
										
											2011-03-18 14:06:13 +00:00
										 |  |  | 		openLibrary(); | 
					
						
							|  |  |  | 		MaterialFunctor mf; | 
					
						
							| 
									
										
										
										
											2012-06-12 21:25:29 +00:00
										 |  |  | 		mf.forEachMaterialInExportSet<EffectsExporter>(sce, *this, this->export_settings->export_set); | 
					
						
							| 
									
										
										
										
											2010-10-07 00:24:42 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-03-18 14:06:13 +00:00
										 |  |  | 		closeLibrary(); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2010-10-07 00:24:42 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2011-03-22 15:28:56 +00:00
										 |  |  | void EffectsExporter::writeBlinn(COLLADASW::EffectProfile &ep, Material *ma) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	COLLADASW::ColorOrTexture cot; | 
					
						
							|  |  |  | 	ep.setShaderType(COLLADASW::EffectProfile::BLINN); | 
					
						
							|  |  |  | 	// shininess
 | 
					
						
							| 
									
										
										
										
											2012-04-29 15:47:02 +00:00
										 |  |  | 	ep.setShininess(ma->har, false, "shininess"); | 
					
						
							| 
									
										
										
										
											2011-03-22 15:28:56 +00:00
										 |  |  | 	// specular
 | 
					
						
							|  |  |  | 	cot = getcol(ma->specr, ma->specg, ma->specb, 1.0f); | 
					
						
							| 
									
										
										
										
											2012-06-12 22:05:33 +00:00
										 |  |  | 	ep.setSpecular(cot, false, "specular"); | 
					
						
							| 
									
										
										
										
											2011-03-22 15:28:56 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EffectsExporter::writeLambert(COLLADASW::EffectProfile &ep, Material *ma) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	COLLADASW::ColorOrTexture cot; | 
					
						
							|  |  |  | 	ep.setShaderType(COLLADASW::EffectProfile::LAMBERT); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void EffectsExporter::writePhong(COLLADASW::EffectProfile &ep, Material *ma) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	COLLADASW::ColorOrTexture cot; | 
					
						
							|  |  |  | 	ep.setShaderType(COLLADASW::EffectProfile::PHONG); | 
					
						
							|  |  |  | 	// shininess
 | 
					
						
							| 
									
										
										
										
											2012-06-12 22:05:33 +00:00
										 |  |  | 	ep.setShininess(ma->har, false, "shininess"); | 
					
						
							| 
									
										
										
										
											2011-03-22 15:28:56 +00:00
										 |  |  | 	// specular
 | 
					
						
							|  |  |  | 	cot = getcol(ma->specr, ma->specg, ma->specb, 1.0f); | 
					
						
							| 
									
										
										
										
											2012-06-12 22:05:33 +00:00
										 |  |  | 	ep.setSpecular(cot, false, "specular"); | 
					
						
							| 
									
										
										
										
											2011-03-22 15:28:56 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-22 16:16:58 +00:00
										 |  |  | void EffectsExporter::writeTextures(COLLADASW::EffectProfile &ep, | 
					
						
							|  |  |  | 									std::string &key, | 
					
						
							|  |  |  | 									COLLADASW::Sampler *sampler,  | 
					
						
							|  |  |  | 									MTex *t, Image *ima, | 
					
						
							|  |  |  | 									std::string &uvname ) { | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 	// Image not set for texture
 | 
					
						
							|  |  |  | 	if (!ima) return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// color
 | 
					
						
							|  |  |  | 	if (t->mapto & (MAP_COL | MAP_COLSPEC)) { | 
					
						
							|  |  |  | 		ep.setDiffuse(createTexture(ima, uvname, sampler), false, "diffuse"); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// ambient
 | 
					
						
							|  |  |  | 	if (t->mapto & MAP_AMB) { | 
					
						
							|  |  |  | 		ep.setAmbient(createTexture(ima, uvname, sampler), false, "ambient"); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// specular
 | 
					
						
							|  |  |  | 	if (t->mapto & MAP_SPEC) { | 
					
						
							|  |  |  | 		ep.setSpecular(createTexture(ima, uvname, sampler), false, "specular"); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// emission
 | 
					
						
							|  |  |  | 	if (t->mapto & MAP_EMIT) { | 
					
						
							|  |  |  | 		ep.setEmission(createTexture(ima, uvname, sampler), false, "emission"); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// reflective
 | 
					
						
							|  |  |  | 	if (t->mapto & MAP_REF) { | 
					
						
							|  |  |  | 		ep.setReflective(createTexture(ima, uvname, sampler)); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// alpha
 | 
					
						
							|  |  |  | 	if (t->mapto & MAP_ALPHA) { | 
					
						
							|  |  |  | 		ep.setTransparent(createTexture(ima, uvname, sampler)); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// extension:
 | 
					
						
							|  |  |  | 	// Normal map --> Must be stored with <extra> tag as different technique, 
 | 
					
						
							|  |  |  | 	// since COLLADA doesn't support normal maps, even in current COLLADA 1.5.
 | 
					
						
							|  |  |  | 	if (t->mapto & MAP_NORM) { | 
					
						
							|  |  |  | 		COLLADASW::Texture texture(key); | 
					
						
							|  |  |  | 		texture.setTexcoord(uvname); | 
					
						
							|  |  |  | 		texture.setSampler(*sampler); | 
					
						
							|  |  |  | 		// technique FCOLLADA, with the <bump> tag, is most likely the best understood,
 | 
					
						
							|  |  |  | 		// most widespread de-facto standard.
 | 
					
						
							|  |  |  | 		texture.setProfileName("FCOLLADA"); | 
					
						
							|  |  |  | 		texture.setChildElementName("bump"); | 
					
						
							|  |  |  | 		ep.addExtraTechniqueColorOrTexture(COLLADASW::ColorOrTexture(texture)); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-07 00:24:42 +00:00
										 |  |  | void EffectsExporter::operator()(Material *ma, Object *ob) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	// create a list of indices to textures of type TEX_IMAGE
 | 
					
						
							|  |  |  | 	std::vector<int> tex_indices; | 
					
						
							| 
									
										
										
										
											2012-07-08 21:37:59 +00:00
										 |  |  | 	if (this->export_settings->include_material_textures) | 
					
						
							| 
									
										
										
										
											2012-06-23 22:03:31 +00:00
										 |  |  | 		createTextureIndices(ma, tex_indices); | 
					
						
							| 
									
										
										
										
											2010-10-07 00:24:42 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	openEffect(translate_id(id_name(ma)) + "-effect"); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	COLLADASW::EffectProfile ep(mSW); | 
					
						
							|  |  |  | 	ep.setProfileType(COLLADASW::EffectProfile::COMMON); | 
					
						
							|  |  |  | 	ep.openProfile(); | 
					
						
							|  |  |  | 	// set shader type - one of three blinn, phong or lambert
 | 
					
						
							| 
									
										
										
										
											2012-06-12 22:05:33 +00:00
										 |  |  | 	if (ma->spec > 0.0f) { | 
					
						
							| 
									
										
										
										
											2011-03-22 22:00:26 +00:00
										 |  |  | 		if (ma->spec_shader == MA_SPEC_BLINN) { | 
					
						
							|  |  |  | 			writeBlinn(ep, ma); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else { | 
					
						
							|  |  |  | 			// \todo figure out handling of all spec+diff shader combos blender has, for now write phong
 | 
					
						
							|  |  |  | 			// for now set phong in case spec shader is not blinn
 | 
					
						
							|  |  |  | 			writePhong(ep, ma); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2012-03-24 06:38:07 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2012-03-24 07:52:14 +00:00
										 |  |  | 		if (ma->diff_shader == MA_DIFF_LAMBERT) { | 
					
						
							| 
									
										
										
										
											2011-03-22 22:00:26 +00:00
										 |  |  | 			writeLambert(ep, ma); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		else { | 
					
						
							| 
									
										
										
										
											2012-06-12 22:05:33 +00:00
										 |  |  | 			// \todo figure out handling of all spec+diff shader combos blender has, for now write phong
 | 
					
						
							|  |  |  | 			writePhong(ep, ma); | 
					
						
							| 
									
										
										
										
											2011-03-22 22:00:26 +00:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2010-10-07 00:24:42 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2011-03-22 22:00:26 +00:00
										 |  |  | 	 | 
					
						
							| 
									
										
										
										
											2010-10-07 00:24:42 +00:00
										 |  |  | 	// index of refraction
 | 
					
						
							|  |  |  | 	if (ma->mode & MA_RAYTRANSP) { | 
					
						
							| 
									
										
										
										
											2012-04-29 15:47:02 +00:00
										 |  |  | 		ep.setIndexOfRefraction(ma->ang, false, "index_of_refraction"); | 
					
						
							| 
									
										
										
										
											2010-10-07 00:24:42 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 	else { | 
					
						
							| 
									
										
										
										
											2012-04-29 15:47:02 +00:00
										 |  |  | 		ep.setIndexOfRefraction(1.0f, false, "index_of_refraction"); | 
					
						
							| 
									
										
										
										
											2010-10-07 00:24:42 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	COLLADASW::ColorOrTexture cot; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// transparency
 | 
					
						
							|  |  |  | 	if (ma->mode & MA_TRANSP) { | 
					
						
							|  |  |  | 		// Tod: because we are in A_ONE mode transparency is calculated like this:
 | 
					
						
							| 
									
										
										
										
											2012-04-29 15:47:02 +00:00
										 |  |  | 		ep.setTransparency(ma->alpha, false, "transparency"); | 
					
						
							| 
									
										
										
										
											2010-10-07 00:24:42 +00:00
										 |  |  | 		// cot = getcol(1.0f, 1.0f, 1.0f, 1.0f);
 | 
					
						
							|  |  |  | 		// ep.setTransparent(cot);
 | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// emission
 | 
					
						
							| 
									
										
										
										
											2012-06-12 22:05:33 +00:00
										 |  |  | 	cot = getcol(ma->emit, ma->emit, ma->emit, 1.0f); | 
					
						
							| 
									
										
										
										
											2012-04-29 15:47:02 +00:00
										 |  |  | 	ep.setEmission(cot, false, "emission"); | 
					
						
							| 
									
										
										
										
											2010-10-07 00:24:42 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// diffuse multiplied by diffuse intensity
 | 
					
						
							|  |  |  | 	cot = getcol(ma->r * ma->ref, ma->g * ma->ref, ma->b * ma->ref, 1.0f); | 
					
						
							| 
									
										
										
										
											2012-04-29 15:47:02 +00:00
										 |  |  | 	ep.setDiffuse(cot, false, "diffuse"); | 
					
						
							| 
									
										
										
										
											2010-10-07 00:24:42 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// ambient
 | 
					
						
							| 
									
										
										
										
											2011-09-12 13:20:24 +00:00
										 |  |  | 	/* ma->ambX is calculated only on render, so lets do it here manually and not rely on ma->ambX. */ | 
					
						
							| 
									
										
										
										
											2012-03-24 07:52:14 +00:00
										 |  |  | 	if (this->scene->world) | 
					
						
							| 
									
										
										
										
											2012-06-12 22:05:33 +00:00
										 |  |  | 		cot = getcol(this->scene->world->ambr * ma->amb, this->scene->world->ambg * ma->amb, this->scene->world->ambb * ma->amb, 1.0f); | 
					
						
							| 
									
										
										
										
											2011-09-12 13:59:23 +00:00
										 |  |  | 	else | 
					
						
							|  |  |  | 		cot = getcol(ma->amb, ma->amb, ma->amb, 1.0f); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-04-29 15:47:02 +00:00
										 |  |  | 	ep.setAmbient(cot, false, "ambient"); | 
					
						
							| 
									
										
										
										
											2010-10-07 00:24:42 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// reflective, reflectivity
 | 
					
						
							|  |  |  | 	if (ma->mode & MA_RAYMIRROR) { | 
					
						
							|  |  |  | 		cot = getcol(ma->mirr, ma->mirg, ma->mirb, 1.0f); | 
					
						
							|  |  |  | 		ep.setReflective(cot); | 
					
						
							|  |  |  | 		ep.setReflectivity(ma->ray_mirror); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// else {
 | 
					
						
							| 
									
										
										
										
											2012-06-12 22:05:33 +00:00
										 |  |  | 	//  cot = getcol(ma->specr, ma->specg, ma->specb, 1.0f);
 | 
					
						
							|  |  |  | 	//  ep.setReflective(cot);
 | 
					
						
							|  |  |  | 	//  ep.setReflectivity(ma->spec);
 | 
					
						
							| 
									
										
										
										
											2010-10-07 00:24:42 +00:00
										 |  |  | 	// }
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// specular
 | 
					
						
							|  |  |  | 	if (ep.getShaderType() != COLLADASW::EffectProfile::LAMBERT) { | 
					
						
							|  |  |  | 		cot = getcol(ma->specr * ma->spec, ma->specg * ma->spec, ma->specb * ma->spec, 1.0f); | 
					
						
							| 
									
										
										
										
											2012-04-29 15:47:02 +00:00
										 |  |  | 		ep.setSpecular(cot, false, "specular"); | 
					
						
							| 
									
										
										
										
											2010-10-07 00:24:42 +00:00
										 |  |  | 	}	 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// XXX make this more readable if possible
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// create <sampler> and <surface> for each image
 | 
					
						
							|  |  |  | 	COLLADASW::Sampler samplers[MAX_MTEX]; | 
					
						
							|  |  |  | 	//COLLADASW::Surface surfaces[MAX_MTEX];
 | 
					
						
							|  |  |  | 	//void *samp_surf[MAX_MTEX][2];
 | 
					
						
							|  |  |  | 	void *samp_surf[MAX_MTEX][1]; | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	// image to index to samp_surf map
 | 
					
						
							|  |  |  | 	// samp_surf[index] stores 2 pointers, sampler and surface
 | 
					
						
							|  |  |  | 	std::map<std::string, int> im_samp_map; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	unsigned int a, b; | 
					
						
							|  |  |  | 	for (a = 0, b = 0; a < tex_indices.size(); a++) { | 
					
						
							|  |  |  | 		MTex *t = ma->mtex[tex_indices[a]]; | 
					
						
							|  |  |  | 		Image *ima = t->tex->ima; | 
					
						
							|  |  |  | 		 | 
					
						
							|  |  |  | 		// Image not set for texture
 | 
					
						
							| 
									
										
										
										
											2012-03-24 07:52:14 +00:00
										 |  |  | 		if (!ima) continue; | 
					
						
							| 
									
										
										
										
											2010-10-07 00:24:42 +00:00
										 |  |  | 		 | 
					
						
							|  |  |  | 		std::string key(id_name(ima)); | 
					
						
							|  |  |  | 		key = translate_id(key); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// create only one <sampler>/<surface> pair for each unique image
 | 
					
						
							|  |  |  | 		if (im_samp_map.find(key) == im_samp_map.end()) { | 
					
						
							|  |  |  | 			// //<newparam> <surface> <init_from>
 | 
					
						
							|  |  |  | 			// COLLADASW::Surface surface(COLLADASW::Surface::SURFACE_TYPE_2D,
 | 
					
						
							| 
									
										
										
										
											2012-06-12 22:05:33 +00:00
										 |  |  | 			//                         key + COLLADASW::Surface::SURFACE_SID_SUFFIX);
 | 
					
						
							| 
									
										
										
										
											2010-10-07 00:24:42 +00:00
										 |  |  | 			// COLLADASW::SurfaceInitOption sio(COLLADASW::SurfaceInitOption::INIT_FROM);
 | 
					
						
							|  |  |  | 			// sio.setImageReference(key);
 | 
					
						
							|  |  |  | 			// surface.setInitOption(sio);
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			// COLLADASW::NewParamSurface surface(mSW);
 | 
					
						
							|  |  |  | 			// surface->setParamType(COLLADASW::CSW_SURFACE_TYPE_2D);
 | 
					
						
							|  |  |  | 			 | 
					
						
							|  |  |  | 			//<newparam> <sampler> <source>
 | 
					
						
							|  |  |  | 			COLLADASW::Sampler sampler(COLLADASW::Sampler::SAMPLER_TYPE_2D, | 
					
						
							| 
									
										
										
										
											2012-06-12 22:05:33 +00:00
										 |  |  | 			                           key + COLLADASW::Sampler::SAMPLER_SID_SUFFIX, | 
					
						
							|  |  |  | 			                           key + COLLADASW::Sampler::SURFACE_SID_SUFFIX); | 
					
						
							| 
									
										
										
										
											2010-10-07 00:24:42 +00:00
										 |  |  | 			sampler.setImageId(key); | 
					
						
							|  |  |  | 			// copy values to arrays since they will live longer
 | 
					
						
							|  |  |  | 			samplers[a] = sampler; | 
					
						
							|  |  |  | 			//surfaces[a] = surface;
 | 
					
						
							|  |  |  | 			 | 
					
						
							|  |  |  | 			// store pointers so they can be used later when we create <texture>s
 | 
					
						
							|  |  |  | 			samp_surf[b][0] = &samplers[a]; | 
					
						
							|  |  |  | 			//samp_surf[b][1] = &surfaces[a];
 | 
					
						
							|  |  |  | 			 | 
					
						
							|  |  |  | 			im_samp_map[key] = b; | 
					
						
							|  |  |  | 			b++; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-22 16:16:58 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	std::set<Image *> uv_textures; | 
					
						
							| 
									
										
										
										
											2012-06-23 22:03:31 +00:00
										 |  |  | 	if (ob->type == OB_MESH && ob->totcol && this->export_settings->include_uv_textures) { | 
					
						
							| 
									
										
										
										
											2012-06-22 16:16:58 +00:00
										 |  |  | 		Mesh *me     = (Mesh *) ob->data; | 
					
						
							|  |  |  | 		BKE_mesh_tessface_ensure(me); | 
					
						
							|  |  |  | 		for (int i = 0; i < me->pdata.totlayer; i++) { | 
					
						
							|  |  |  | 			if (me->pdata.layers[i].type == CD_MTEXPOLY) { | 
					
						
							|  |  |  | 				MTexPoly *txface = (MTexPoly *)me->pdata.layers[i].data; | 
					
						
							|  |  |  | 				MFace *mface = me->mface; | 
					
						
							|  |  |  | 				for (int j = 0; j < me->totpoly; j++, mface++, txface++) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					Material *mat = give_current_material(ob, mface->mat_nr + 1); | 
					
						
							|  |  |  | 					if (mat != ma)  | 
					
						
							|  |  |  | 						continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					Image *ima = txface->tpage; | 
					
						
							|  |  |  | 					if (ima == NULL) | 
					
						
							|  |  |  | 						continue; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 					bool not_in_list = uv_textures.find(ima)==uv_textures.end(); | 
					
						
							|  |  |  | 					if (not_in_list) { | 
					
						
							|  |  |  | 						std::string name = id_name(ima); | 
					
						
							|  |  |  | 						std::string key(name); | 
					
						
							|  |  |  | 						key = translate_id(key); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 						// create only one <sampler>/<surface> pair for each unique image
 | 
					
						
							|  |  |  | 						if (im_samp_map.find(key) == im_samp_map.end()) { | 
					
						
							|  |  |  | 							//<newparam> <sampler> <source>
 | 
					
						
							|  |  |  | 							COLLADASW::Sampler sampler(COLLADASW::Sampler::SAMPLER_TYPE_2D, | 
					
						
							|  |  |  | 													   key + COLLADASW::Sampler::SAMPLER_SID_SUFFIX, | 
					
						
							|  |  |  | 													   key + COLLADASW::Sampler::SURFACE_SID_SUFFIX); | 
					
						
							|  |  |  | 							sampler.setImageId(key); | 
					
						
							|  |  |  | 							samplers[a] = sampler; | 
					
						
							|  |  |  | 							samp_surf[b][0] = &samplers[a]; | 
					
						
							|  |  |  | 							im_samp_map[key] = b; | 
					
						
							|  |  |  | 							b++; | 
					
						
							|  |  |  | 							a++; | 
					
						
							|  |  |  | 							uv_textures.insert(ima); | 
					
						
							|  |  |  | 						} | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-07 00:24:42 +00:00
										 |  |  | 	// used as fallback when MTex->uvname is "" (this is pretty common)
 | 
					
						
							|  |  |  | 	// it is indeed the correct value to use in that case
 | 
					
						
							|  |  |  | 	std::string active_uv(getActiveUVLayerName(ob)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// write textures
 | 
					
						
							|  |  |  | 	// XXX very slow
 | 
					
						
							|  |  |  | 	for (a = 0; a < tex_indices.size(); a++) { | 
					
						
							|  |  |  | 		MTex *t = ma->mtex[tex_indices[a]]; | 
					
						
							|  |  |  | 		Image *ima = t->tex->ima; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		std::string key(id_name(ima)); | 
					
						
							|  |  |  | 		key = translate_id(key); | 
					
						
							|  |  |  | 		int i = im_samp_map[key]; | 
					
						
							|  |  |  | 		std::string uvname = strlen(t->uvname) ? t->uvname : active_uv; | 
					
						
							| 
									
										
										
										
											2012-06-22 16:16:58 +00:00
										 |  |  | 		COLLADASW::Sampler *sampler = (COLLADASW::Sampler *)samp_surf[i][0]; | 
					
						
							|  |  |  | 		writeTextures(ep, key, sampler, t, ima, uvname); | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2010-10-07 00:24:42 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-06-22 16:16:58 +00:00
										 |  |  | 	std::set<Image *>::iterator uv_t_iter; | 
					
						
							|  |  |  | 	for(uv_t_iter = uv_textures.begin(); uv_t_iter != uv_textures.end(); uv_t_iter++ ) { | 
					
						
							|  |  |  | 		Image *ima = *uv_t_iter; | 
					
						
							|  |  |  | 		std::string key(id_name(ima)); | 
					
						
							|  |  |  | 		key = translate_id(key); | 
					
						
							|  |  |  | 		int i = im_samp_map[key]; | 
					
						
							|  |  |  | 		COLLADASW::Sampler *sampler = (COLLADASW::Sampler *)samp_surf[i][0]; | 
					
						
							|  |  |  | 		ep.setDiffuse(createTexture(ima, active_uv, sampler), false, "diffuse"); | 
					
						
							| 
									
										
										
										
											2010-10-07 00:24:42 +00:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2012-06-22 16:16:58 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-10-07 00:24:42 +00:00
										 |  |  | 	// performs the actual writing
 | 
					
						
							|  |  |  | 	ep.addProfileElements(); | 
					
						
							|  |  |  | 	bool twoSided = false; | 
					
						
							|  |  |  | 	if (ob->type == OB_MESH && ob->data) { | 
					
						
							| 
									
										
										
										
											2012-06-12 22:05:33 +00:00
										 |  |  | 		Mesh *me = (Mesh *)ob->data; | 
					
						
							| 
									
										
										
										
											2010-10-07 00:24:42 +00:00
										 |  |  | 		if (me->flag & ME_TWOSIDED) | 
					
						
							|  |  |  | 			twoSided = true; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if (twoSided) | 
					
						
							| 
									
										
										
										
											2011-03-23 00:03:52 +00:00
										 |  |  | 		ep.addExtraTechniqueParameter("GOOGLEEARTH", "double_sided", 1); | 
					
						
							| 
									
										
										
										
											2010-10-07 00:24:42 +00:00
										 |  |  | 	ep.addExtraTechniques(mSW); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	ep.closeProfile(); | 
					
						
							|  |  |  | 	if (twoSided) | 
					
						
							|  |  |  | 		mSW->appendTextBlock("<extra><technique profile=\"MAX3D\"><double_sided>1</double_sided></technique></extra>"); | 
					
						
							|  |  |  | 	closeEffect();	 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | COLLADASW::ColorOrTexture EffectsExporter::createTexture(Image *ima, | 
					
						
							| 
									
										
										
										
											2012-06-12 22:05:33 +00:00
										 |  |  |                                                          std::string& uv_layer_name, | 
					
						
							|  |  |  |                                                          COLLADASW::Sampler *sampler | 
					
						
							|  |  |  |                                                          /*COLLADASW::Surface *surface*/) | 
					
						
							| 
									
										
										
										
											2010-10-07 00:24:42 +00:00
										 |  |  | { | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	COLLADASW::Texture texture(translate_id(id_name(ima))); | 
					
						
							|  |  |  | 	texture.setTexcoord(uv_layer_name); | 
					
						
							|  |  |  | 	//texture.setSurface(*surface);
 | 
					
						
							|  |  |  | 	texture.setSampler(*sampler); | 
					
						
							|  |  |  | 	 | 
					
						
							|  |  |  | 	COLLADASW::ColorOrTexture cot(texture); | 
					
						
							|  |  |  | 	return cot; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | COLLADASW::ColorOrTexture EffectsExporter::getcol(float r, float g, float b, float a) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2012-04-29 15:47:02 +00:00
										 |  |  | 	COLLADASW::Color color(r, g, b, a); | 
					
						
							| 
									
										
										
										
											2010-10-07 00:24:42 +00:00
										 |  |  | 	COLLADASW::ColorOrTexture cot(color); | 
					
						
							|  |  |  | 	return cot; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | //returns the array of mtex indices which have image 
 | 
					
						
							|  |  |  | //need this for exporting textures
 | 
					
						
							|  |  |  | void EffectsExporter::createTextureIndices(Material *ma, std::vector<int> &indices) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	indices.clear(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for (int a = 0; a < MAX_MTEX; a++) { | 
					
						
							|  |  |  | 		if (ma->mtex[a] && | 
					
						
							| 
									
										
										
										
											2012-03-28 05:03:24 +00:00
										 |  |  | 		    ma->mtex[a]->tex && | 
					
						
							|  |  |  | 		    ma->mtex[a]->tex->type == TEX_IMAGE && | 
					
						
							|  |  |  | 		    ma->mtex[a]->texco == TEXCO_UV) | 
					
						
							|  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2010-10-07 00:24:42 +00:00
										 |  |  | 			indices.push_back(a); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } |