commit
6c2490090d
@ -0,0 +1,19 @@
|
||||
; Top-most EditorConfig file
|
||||
root = true
|
||||
|
||||
; 4-column tab indentation
|
||||
[*.cpp]
|
||||
indent_style = tab
|
||||
indent_size = 4
|
||||
|
||||
[*.c]
|
||||
indent_style = tab
|
||||
indent_size = 4
|
||||
|
||||
[*.h]
|
||||
indent_style = tab
|
||||
indent_size = 4
|
||||
|
||||
[*.py]
|
||||
indent_style = tab
|
||||
indent_size = 4
|
@ -0,0 +1,25 @@
|
||||
FIND_PATH(
|
||||
assimp_INCLUDE_DIRS
|
||||
NAMES assimp/postprocess.h assimp/scene.h assimp/version.h assimp/config.h assimp/cimport.h
|
||||
PATHS /usr/local/include/
|
||||
)
|
||||
|
||||
FIND_LIBRARY(
|
||||
assimp_LIBRARIES
|
||||
NAMES assimp
|
||||
PATHS /usr/local/lib/
|
||||
)
|
||||
|
||||
IF (assimp_INCLUDE_DIRS AND assimp_LIBRARIES)
|
||||
SET(assimp_FOUND TRUE)
|
||||
ENDIF (assimp_INCLUDE_DIRS AND assimp_LIBRARIES)
|
||||
|
||||
IF (assimp_FOUND)
|
||||
IF (NOT assimp_FIND_QUIETLY)
|
||||
MESSAGE(STATUS "Found asset importer library: ${assimp_LIBRARIES}")
|
||||
ENDIF (NOT assimp_FIND_QUIETLY)
|
||||
ELSE (assimp_FOUND)
|
||||
IF (assimp_FIND_REQUIRED)
|
||||
MESSAGE(FATAL_ERROR "Could not find asset importer library")
|
||||
ENDIF (assimp_FIND_REQUIRED)
|
||||
ENDIF (assimp_FOUND)
|
@ -0,0 +1,210 @@
|
||||
/**
|
||||
* \file tool_logger.h
|
||||
* \brief CToolLogger
|
||||
* \date 2012-02-19 10:33GMT
|
||||
* \author Jan Boon (Kaetemi)
|
||||
* Tool logger is fully implemented in header so small tools do not
|
||||
* need to link to this library unnecessarily.
|
||||
* NOTE: Needs to be changed not to use time_nl and string_common.
|
||||
*/
|
||||
|
||||
/*
|
||||
* Copyright (C) 2012 by authors
|
||||
*
|
||||
* This file is part of RYZOM CORE PIPELINE.
|
||||
* RYZOM CORE PIPELINE 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.
|
||||
*
|
||||
* RYZOM CORE PIPELINE 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 RYZOM CORE PIPELINE; see the file COPYING. If not, see
|
||||
* <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
#ifndef NLMISC_TOOL_LOGGER_H
|
||||
#define NLMISC_TOOL_LOGGER_H
|
||||
#include <nel/misc/types_nl.h>
|
||||
|
||||
// STL includes
|
||||
#include <string>
|
||||
#include <stdio.h>
|
||||
#include <stdlib.h>
|
||||
|
||||
// NeL includes
|
||||
#include <nel/misc/time_nl.h>
|
||||
#include <nel/misc/string_common.h>
|
||||
|
||||
// Project includes
|
||||
|
||||
#ifdef ERROR
|
||||
#undef ERROR
|
||||
#endif
|
||||
|
||||
#ifdef NL_DEBUG_H
|
||||
#define tlerror(toolLogger, path, error, ...) nlwarning(error, __VA_ARGS__), toolLogger.writeError(NLMISC::ERROR, path, error, __VA_ARGS__)
|
||||
#define tlwarning(toolLogger, path, error, ...) nlwarning(error, __VA_ARGS__), toolLogger.writeError(NLMISC::WARNING, path, error, __VA_ARGS__)
|
||||
#define tlmessage(toolLogger, path, error, ...) nlinfo(error, __VA_ARGS__), toolLogger.writeError(NLMISC::MESSAGE, path, error, __VA_ARGS__)
|
||||
#else
|
||||
#define tlerror(toolLogger, path, error, ...) toolLogger.writeError(NLMISC::ERROR, path, error, __VA_ARGS__)
|
||||
#define tlwarning(toolLogger, path, error, ...) toolLogger.writeError(NLMISC::WARNING, path, error, __VA_ARGS__)
|
||||
#define tlmessage(toolLogger, path, error, ...) toolLogger.writeError(NLMISC::MESSAGE, path, error, __VA_ARGS__)
|
||||
#endif
|
||||
|
||||
namespace NLMISC {
|
||||
|
||||
enum TError
|
||||
{
|
||||
ERROR,
|
||||
WARNING,
|
||||
MESSAGE,
|
||||
};
|
||||
|
||||
enum TDepend
|
||||
{
|
||||
BUILD,
|
||||
DIRECTORY,
|
||||
RUNTIME,
|
||||
};
|
||||
|
||||
const std::string s_ErrorHeader = "type\tpath\ttime\terror";
|
||||
const std::string s_DependHeader = "type\toutput_file\tinput_file";
|
||||
|
||||
/**
|
||||
* \brief CToolLogger
|
||||
* \date 2012-02-19 10:33GMT
|
||||
* \author Jan Boon (Kaetemi)
|
||||
* CToolLogger
|
||||
*/
|
||||
class CToolLogger
|
||||
{
|
||||
private:
|
||||
FILE *m_ErrorLog;
|
||||
FILE *m_DependLog;
|
||||
|
||||
public:
|
||||
inline CToolLogger() : m_ErrorLog(NULL), m_DependLog(NULL)
|
||||
{
|
||||
|
||||
}
|
||||
|
||||
inline ~CToolLogger()
|
||||
{
|
||||
release();
|
||||
}
|
||||
|
||||
inline void initError(const std::string &errorLog)
|
||||
{
|
||||
releaseError();
|
||||
|
||||
m_ErrorLog = fopen(errorLog.c_str(), "wt");
|
||||
fwrite(s_ErrorHeader.c_str(), 1, s_ErrorHeader.length(), m_ErrorLog);
|
||||
fwrite("\n", 1, 1, m_ErrorLog);
|
||||
fflush(m_ErrorLog);
|
||||
|
||||
}
|
||||
|
||||
inline void initDepend(const std::string &dependLog)
|
||||
{
|
||||
releaseDepend();
|
||||
|
||||
m_DependLog = fopen(dependLog.c_str(), "wt");
|
||||
fwrite(s_DependHeader.c_str(), 1, s_DependHeader.length(), m_DependLog);
|
||||
fwrite("\n", 1, 1, m_DependLog);
|
||||
// fflush(m_DependLog);
|
||||
}
|
||||
|
||||
inline void writeError(TError type, const char *path, const char *error, ...)
|
||||
{
|
||||
if (m_ErrorLog)
|
||||
{
|
||||
switch (type)
|
||||
{
|
||||
case ERROR:
|
||||
fwrite("ERROR", 1, 5, m_ErrorLog);
|
||||
break;
|
||||
case WARNING:
|
||||
fwrite("WARNING", 1, 7, m_ErrorLog);
|
||||
break;
|
||||
case MESSAGE:
|
||||
fwrite("MESSAGE", 1, 7, m_ErrorLog);
|
||||
break;
|
||||
}
|
||||
fwrite("\t", 1, 1, m_ErrorLog);
|
||||
fprintf(m_ErrorLog, "%s", path);
|
||||
fwrite("\t", 1, 1, m_ErrorLog);
|
||||
std::string time = NLMISC::toString(NLMISC::CTime::getSecondsSince1970());
|
||||
fwrite(time.c_str(), 1, time.length(), m_ErrorLog);
|
||||
fwrite("\t", 1, 1, m_ErrorLog);
|
||||
va_list args;
|
||||
va_start(args, error);
|
||||
vfprintf(m_ErrorLog, error, args);
|
||||
va_end(args);
|
||||
fwrite("\n", 1, 1, m_ErrorLog);
|
||||
fflush(m_ErrorLog);
|
||||
}
|
||||
}
|
||||
|
||||
/// inputFile can only be file. [? May be not-yet-existing file for expected input for future build runs. ?] Directories are handled on process level. [? You should call this before calling writeError on inputFile, so the error is also linked from the outputFile. ?]
|
||||
inline void writeDepend(TDepend type, const char *outputFile, const char *inputFile)
|
||||
{
|
||||
if (m_DependLog)
|
||||
{
|
||||
switch (type)
|
||||
{
|
||||
case BUILD:
|
||||
fwrite("BUILD", 1, 5, m_DependLog);
|
||||
break;
|
||||
case DIRECTORY:
|
||||
fwrite("DIRECTORY", 1, 9, m_DependLog);
|
||||
break;
|
||||
case RUNTIME:
|
||||
fwrite("RUNTIME", 1, 7, m_DependLog);
|
||||
break;
|
||||
}
|
||||
fwrite("\t", 1, 1, m_DependLog);
|
||||
fprintf(m_DependLog, "%s", outputFile);
|
||||
fwrite("\t", 1, 1, m_DependLog);
|
||||
fprintf(m_DependLog, "%s", inputFile);
|
||||
fwrite("\n", 1, 1, m_DependLog);
|
||||
// fflush(m_DependLog);
|
||||
}
|
||||
}
|
||||
|
||||
inline void releaseError()
|
||||
{
|
||||
if (m_ErrorLog)
|
||||
{
|
||||
fflush(m_ErrorLog);
|
||||
fclose(m_ErrorLog);
|
||||
m_ErrorLog = NULL;
|
||||
}
|
||||
}
|
||||
|
||||
inline void releaseDepend()
|
||||
{
|
||||
if (m_DependLog)
|
||||
{
|
||||
fflush(m_DependLog);
|
||||
fclose(m_DependLog);
|
||||
m_DependLog = NULL;
|
||||
}
|
||||
}
|
||||
|
||||
inline void release()
|
||||
{
|
||||
releaseError();
|
||||
releaseDepend();
|
||||
}
|
||||
}; /* class CToolLogger */
|
||||
|
||||
} /* namespace NLMISC */
|
||||
|
||||
#endif /* #ifndef NLMISC_TOOL_LOGGER_H */
|
||||
|
||||
/* end of file */
|
@ -0,0 +1,45 @@
|
||||
/**
|
||||
* \file tool_logger.cpp
|
||||
* \brief CToolLogger
|
||||
* \date 2012-02-19 10:33GMT
|
||||
* \author Jan Boon (Kaetemi)
|
||||
* CToolLogger
|
||||
*/
|
||||
|
||||
/*
|
||||
* Copyright (C) 2012 by authors
|
||||
*
|
||||
* This file is part of RYZOM CORE PIPELINE.
|
||||
* RYZOM CORE PIPELINE 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.
|
||||
*
|
||||
* RYZOM CORE PIPELINE 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 RYZOM CORE PIPELINE; see the file COPYING. If not, see
|
||||
* <http://www.gnu.org/licenses/>.
|
||||
*/
|
||||
|
||||
#include "stdmisc.h"
|
||||
#include "nel/misc/tool_logger.h"
|
||||
|
||||
// STL includes
|
||||
|
||||
// NeL includes
|
||||
// #include <nel/misc/debug.h>
|
||||
|
||||
// Project includes
|
||||
|
||||
namespace NLMISC {
|
||||
|
||||
// Tool logger is fully implemented in header so small tools do not need to link to this library unnecessarily.
|
||||
void dummy_tool_logger_cpp() { }
|
||||
|
||||
} /* namespace NLMISC */
|
||||
|
||||
/* end of file */
|
@ -0,0 +1,12 @@
|
||||
FILE(GLOB SRCS *.cpp)
|
||||
FILE(GLOB HDRS *.h)
|
||||
|
||||
SOURCE_GROUP("" FILES ${SRCS} ${HDRS})
|
||||
|
||||
ADD_EXECUTABLE(mesh_export ${SRCS} ${HDRS})
|
||||
|
||||
TARGET_LINK_LIBRARIES(mesh_export mesh_utils nel3d nelmisc)
|
||||
NL_DEFAULT_PROPS(mesh_export "NeL, Tools, 3D: Mesh Export")
|
||||
NL_ADD_RUNTIME_FLAGS(mesh_export)
|
||||
|
||||
INSTALL(TARGETS mesh_export RUNTIME DESTINATION ${NL_BIN_PREFIX} COMPONENT tools3d)
|
@ -0,0 +1,77 @@
|
||||
// NeL - MMORPG Framework <http://dev.ryzom.com/projects/nel/>
|
||||
// Copyright (C) 2015 Winch Gate Property Limited
|
||||
// Author: Jan Boon <jan.boon@kaetemi.be>
|
||||
//
|
||||
// This program is free software: you can redistribute it and/or modify
|
||||
// it under the terms of the GNU Affero General Public License as
|
||||
// published by the Free Software Foundation, either version 3 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 Affero General Public License for more details.
|
||||
//
|
||||
// You should have received a copy of the GNU Affero General Public License
|
||||
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
#include <nel/misc/types_nl.h>
|
||||
#include "../mesh_utils/mesh_utils.h"
|
||||
|
||||
#include <nel/misc/cmd_args.h>
|
||||
#include <nel/misc/path.h>
|
||||
|
||||
#include <nel/3d/register_3d.h>
|
||||
#include <nel/3d/scene.h>
|
||||
|
||||
int printHelp(const NLMISC::CCmdArgs &args)
|
||||
{
|
||||
printf("NeL Mesh Export\n");
|
||||
return EXIT_SUCCESS;
|
||||
}
|
||||
|
||||
int main(int argc, char *argv[])
|
||||
{
|
||||
NLMISC::CApplicationContext app;
|
||||
|
||||
NLMISC::CCmdArgs args;
|
||||
args.setArgs(argc, (const char **)argv);
|
||||
|
||||
if (args.getArgs().size() == 1
|
||||
|| args.haveArg('h')
|
||||
|| args.haveLongArg("help"))
|
||||
return printHelp(args);
|
||||
|
||||
const NLMISC::CSString &filePath = args.getArgs().back();
|
||||
if (!NLMISC::CFile::fileExists(filePath))
|
||||
{
|
||||
printHelp(args);
|
||||
nlerror("File '%s' does not exist", filePath);
|
||||
return EXIT_FAILURE;
|
||||
}
|
||||
|
||||
CMeshUtilsSettings settings;
|
||||
settings.SourceFilePath = filePath;
|
||||
|
||||
if (args.haveArg('d'))
|
||||
settings.DestinationDirectoryPath = args.getArg('d');
|
||||
if (settings.DestinationDirectoryPath.empty())
|
||||
settings.DestinationDirectoryPath = args.getLongArg("dst");
|
||||
if (settings.DestinationDirectoryPath.empty())
|
||||
settings.DestinationDirectoryPath = filePath + "_export";
|
||||
settings.DestinationDirectoryPath += "/";
|
||||
|
||||
settings.ToolDependLog = args.getLongArg("dependlog");
|
||||
if (settings.ToolDependLog.empty())
|
||||
settings.ToolDependLog = settings.DestinationDirectoryPath + "depend.log";
|
||||
settings.ToolErrorLog = args.getLongArg("errorlog");
|
||||
if (settings.ToolErrorLog.empty())
|
||||
settings.ToolErrorLog = settings.DestinationDirectoryPath + "error.log";
|
||||
|
||||
NL3D::CScene::registerBasics();
|
||||
NL3D::registerSerial3d();
|
||||
|
||||
return exportScene(settings);
|
||||
}
|
||||
|
||||
/* end of file */
|
@ -0,0 +1,16 @@
|
||||
FILE(GLOB SRCS *.cpp)
|
||||
FILE(GLOB HDRS *.h)
|
||||
|
||||
SOURCE_GROUP("" FILES ${SRCS} ${HDRS})
|
||||
|
||||
INCLUDE_DIRECTORIES(${assimp_INCLUDE_DIRS})
|
||||
|
||||
NL_TARGET_LIB(mesh_utils ${SRCS} ${HDRS})
|
||||
|
||||
TARGET_LINK_LIBRARIES(mesh_utils ${assimp_LIBRARIES} nelmisc nel3d)
|
||||
NL_DEFAULT_PROPS(mesh_utils "NeL, Tools, 3D: Mesh Utils")
|
||||
NL_ADD_RUNTIME_FLAGS(mesh_utils)
|
||||
|
||||
IF((WITH_INSTALL_LIBRARIES AND WITH_STATIC) OR NOT WITH_STATIC)
|
||||
INSTALL(TARGETS mesh_utils LIBRARY DESTINATION ${NL_LIB_PREFIX} ARCHIVE DESTINATION ${NL_LIB_PREFIX} COMPONENT tools3d)
|
||||
ENDIF((WITH_INSTALL_LIBRARIES AND WITH_STATIC) OR NOT WITH_STATIC)
|
@ -0,0 +1,228 @@
|
||||
// NeL - MMORPG Framework <http://dev.ryzom.com/projects/nel/>
|
||||
// Copyright (C) 2015 Winch Gate Property Limited
|
||||
// Author: Jan Boon <jan.boon@kaetemi.be>
|
||||
//
|
||||
// This program is free software: you can redistribute it and/or modify
|
||||
// it under the terms of the GNU Affero General Public License as
|
||||
// published by the Free Software Foundation, either version 3 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 Affero General Public License for more details.
|
||||
//
|
||||
// You should have received a copy of the GNU Affero General Public License
|
||||
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
#include <nel/misc/types_nl.h>
|
||||
#include "assimp_shape.h"
|
||||
|
||||
#include <assimp/postprocess.h>
|
||||
#include <assimp/scene.h>
|
||||
#include <assimp/Importer.hpp>
|
||||
|
||||
#define NL_NODE_INTERNAL_TYPE aiNode
|
||||
#define NL_SCENE_INTERNAL_TYPE aiScene
|
||||
#include "scene_context.h"
|
||||
|
||||
#include <nel/misc/debug.h>
|
||||
#include <nel/misc/path.h>
|
||||
#include <nel/misc/tool_logger.h>
|
||||
|
||||
#include <nel/3d/mesh.h>
|
||||
#include <nel/3d/texture_file.h>
|
||||
|
||||
using namespace std;
|
||||
using namespace NLMISC;
|
||||
using namespace NL3D;
|
||||
|
||||
// http://assimp.sourceforge.net/lib_html/materials.html
|
||||
|
||||
inline CRGBA convColor(const aiColor3D &ac, uint8 a = 255)
|
||||
{
|
||||
return CRGBA(ac.r * 255.99f, ac.g * 255.99f, ac.b * 255.99f);
|
||||
}
|
||||
|
||||
inline CRGBA convColor(const aiColor4D &ac)
|
||||
{
|
||||
return CRGBA(ac.r * 255.99f, ac.g * 255.99f, ac.b * 255.99f, ac.a * 255.99f);
|
||||
}
|
||||
|
||||
void assimpMaterial(NL3D::CMaterial &mat, CMeshUtilsContext &context, const aiMaterial *am)
|
||||
{
|
||||
aiString amname;
|
||||
if (am->Get(AI_MATKEY_NAME, amname) != aiReturn_SUCCESS)
|
||||
amname = "";
|
||||
|
||||
mat.initLighted();
|
||||
mat.setShader(CMaterial::Normal);
|
||||
|
||||
int i;
|
||||
float f;
|
||||
aiColor3D c3;
|
||||
aiColor4D c4;
|
||||
|
||||
if (am->Get(AI_MATKEY_TWOSIDED, i) == aiReturn_SUCCESS)
|
||||
mat.setDoubleSided(i != 0);
|
||||
|
||||
if (am->Get(AI_MATKEY_BLEND_FUNC, i) == aiReturn_SUCCESS) switch ((aiBlendMode)i)
|
||||
{
|
||||
case aiBlendMode_Default:
|
||||
mat.setSrcBlend(CMaterial::srcalpha);
|
||||
mat.setDstBlend(CMaterial::invsrcalpha);
|
||||
break;
|
||||
case aiBlendMode_Additive:
|
||||
mat.setSrcBlend(CMaterial::one);
|
||||
mat.setDstBlend(CMaterial::one);
|
||||
break;
|
||||
}
|
||||
|
||||
// Colors follow GL convention
|
||||
// "While the ambient, diffuse, specular and emission
|
||||
// "material parameters all have alpha components, only the diffuse"
|
||||
// "alpha component is used in the lighting computation."
|
||||
if (am->Get(AI_MATKEY_COLOR_DIFFUSE, c3) == aiReturn_SUCCESS)
|
||||
mat.setDiffuse(convColor(c3));
|
||||
|
||||
if (am->Get(AI_MATKEY_OPACITY, f) == aiReturn_SUCCESS)
|
||||
mat.setOpacity(f * 255.99f);
|
||||
|
||||
if (am->Get(AI_MATKEY_COLOR_AMBIENT, c3) == aiReturn_SUCCESS)
|
||||
mat.setAmbient(convColor(c3));
|
||||
|
||||
if (am->Get(AI_MATKEY_SHININESS, f) == aiReturn_SUCCESS)
|
||||
mat.setShininess(f); // (float)pow(2.0, f * 10.0) * 4.f;
|
||||
|
||||
if (am->Get(AI_MATKEY_COLOR_SPECULAR, c3) == aiReturn_SUCCESS)
|
||||
mat.setSpecular(convColor(c3));
|
||||
|
||||
if (am->Get(AI_MATKEY_SHININESS_STRENGTH, f) == aiReturn_SUCCESS)
|
||||
mat.setSpecular(CRGBAF(mat.getSpecular()) * f);
|
||||
else
|
||||
mat.setSpecular(NLMISC::CRGBA::Black);
|
||||
|
||||
if (am->Get(AI_MATKEY_COLOR_EMISSIVE, c3) == aiReturn_SUCCESS)
|
||||
mat.setEmissive(convColor(c3));
|
||||
|
||||
// Textures
|
||||
unsigned int texCount = am->GetTextureCount(aiTextureType_DIFFUSE);
|
||||
if (texCount > IDRV_MAT_MAXTEXTURES)
|
||||
{
|
||||
tlwarning(context.ToolLogger, context.Settings.SourceFilePath.c_str(),
|
||||
"Material '%s' has more than %i textures (%i textures found)", amname.C_Str(), IDRV_MAT_MAXTEXTURES, texCount);
|
||||
texCount = IDRV_MAT_MAXTEXTURES;
|
||||
}
|
||||
|
||||
for (unsigned int ti = 0; ti < texCount; ++ti)
|
||||
{
|
||||
aiString path;
|
||||
aiTextureMapping mapping;
|
||||
unsigned int uvindex;
|
||||
float blend; // Partially supported
|
||||
aiTextureOp op;
|
||||
aiTextureMapMode mapmode;
|
||||
if (am->GetTexture(aiTextureType_DIFFUSE, ti, &path, &mapping, &uvindex, &blend, &op, &mapmode) != aiReturn_SUCCESS)
|
||||
{
|
||||
tlerror(context.ToolLogger, context.Settings.SourceFilePath.c_str(),
|
||||
"Failed to get texture %i in material '%s'", ti, amname.C_Str());
|
||||
break;
|
||||
}
|
||||
|
||||
std::string fileName = CFile::getFilename(CPath::standardizePath(path.C_Str(), false));
|
||||
std::string knownPath = CPath::lookup(fileName, false, false, false);
|
||||
if (knownPath.empty())
|
||||
{
|
||||
tlwarning(context.ToolLogger, context.Settings.SourceFilePath.c_str(),
|
||||
"Texture '%s' referenced in material '%s' but not found in the database search paths", fileName.c_str(), amname.C_Str());
|
||||
}
|
||||
|
||||
// NeL supports bitmap and cubemap, but we import only basic bitmap here. Cubemap can be inserted from the mesh editor tool
|
||||
// NeL also has fancy multi-bitmap thing to switch between summer and winter and so on. Same story
|
||||
CSmartPtr<CTextureFile> tex = new CTextureFile();
|
||||
tex->setFileName(fileName);
|
||||
tex->setWrapS(mapmode == aiTextureMapMode_Clamp ? ITexture::Clamp : ITexture::Repeat);
|
||||
tex->setWrapT(mapmode == aiTextureMapMode_Clamp ? ITexture::Clamp : ITexture::Repeat);
|
||||
mat.setTexture(ti, tex);
|
||||
|
||||
// TODO uvindex for uv routing (probably necessary during shape import - if so also need to also ask the uv channel in the editor and store in meta)
|
||||
|
||||
// TODO aiTextureMapping texcoordgen if useful to import
|
||||
|
||||
mat.texEnvArg0Alpha(ti, CMaterial::Texture, CMaterial::SrcAlpha);
|
||||
mat.texEnvArg0RGB(ti, CMaterial::Texture, CMaterial::SrcColor);
|
||||
mat.texEnvArg1Alpha(ti, ti == 0 ? CMaterial::Diffuse : CMaterial::Previous, CMaterial::SrcAlpha);
|
||||
mat.texEnvArg1RGB(ti, ti == 0 ? CMaterial::Diffuse : CMaterial::Previous, CMaterial::SrcColor);
|
||||
switch (op)
|
||||
{
|
||||
case aiTextureOp_Multiply:
|
||||
default:
|
||||
mat.texEnvOpAlpha(ti, CMaterial::Modulate);
|
||||
mat.texEnvOpRGB(ti, CMaterial::Modulate);
|
||||
break;
|
||||
case aiTextureOp_Add:
|
||||
mat.texEnvOpAlpha(ti, CMaterial::Add);
|
||||
mat.texEnvOpRGB(ti, CMaterial::Add);
|
||||
break;
|
||||
case aiTextureOp_Subtract:
|
||||
mat.texEnvArg0Alpha(ti, CMaterial::Texture, CMaterial::InvSrcAlpha);
|
||||
mat.texEnvArg0RGB(ti, CMaterial::Texture, CMaterial::InvSrcColor);
|
||||
mat.texEnvOpAlpha(ti, CMaterial::Add);
|
||||
mat.texEnvOpRGB(ti, CMaterial::Add);
|
||||
break;
|
||||
case aiTextureOp_SignedAdd:
|
||||
mat.texEnvOpAlpha(ti, CMaterial::AddSigned);
|
||||
mat.texEnvOpRGB(ti, CMaterial::AddSigned);
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
CSmartPtr<CMaterial> assimpMaterial(CMeshUtilsContext &context, const aiMaterial *am)
|
||||
{
|
||||
CSmartPtr<CMaterial> matp = new CMaterial();
|
||||
CMaterial &mat = *matp;
|
||||
assimpMaterial(mat, context, am);
|
||||
return matp;
|
||||
}
|
||||
|
||||
void assimpMaterials(CMeshUtilsContext &context)
|
||||
{
|
||||
set<CSString> materialNames;
|
||||
|
||||
const aiScene *scene = context.InternalScene;
|
||||
for (unsigned int mi = 0; mi < scene->mNumMaterials; ++mi)
|
||||
{
|
||||
const aiMaterial *am = scene->mMaterials[mi];
|
||||
|
||||
for (unsigned int pi = 0; pi < am->mNumProperties; ++pi) // DEBUG
|
||||
{ // DEBUG
|
||||
const aiMaterialProperty *amp = am->mProperties[pi];
|
||||
printf("%s\n", amp->mKey.C_Str());
|
||||
} // DEBUG
|
||||
|
||||
aiString amname;
|
||||
if (am->Get(AI_MATKEY_NAME, amname) != aiReturn_SUCCESS)
|
||||
{
|
||||
tlerror(context.ToolLogger, context.Settings.SourceFilePath.c_str(),
|
||||
"Material has no name");
|
||||
continue;
|
||||
}
|
||||
|
||||
if (materialNames.find(amname.C_Str()) != materialNames.end())
|
||||
{
|
||||
tlerror(context.ToolLogger, context.Settings.SourceFilePath.c_str(),
|
||||
"Material name '%s' used more than once", amname.C_Str());
|
||||
continue;
|
||||
}
|
||||
|
||||
if (context.SceneMeta.Materials.find(amname.C_Str())
|
||||
== context.SceneMeta.Materials.end())
|
||||
{
|
||||
materialNames.insert(amname.C_Str());
|
||||
context.SceneMeta.Materials[amname.C_Str()] = assimpMaterial(context, am);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/* end of file */
|
@ -0,0 +1,28 @@
|
||||
// NeL - MMORPG Framework <http://dev.ryzom.com/projects/nel/>
|
||||
// Copyright (C) 2015 Winch Gate Property Limited
|
||||
// Author: Jan Boon <jan.boon@kaetemi.be>
|
||||
//
|
||||
// This program is free software: you can redistribute it and/or modify
|
||||
// it under the terms of the GNU Affero General Public License as
|
||||
// published by the Free Software Foundation, either version 3 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 Affero General Public License for more details.
|
||||
//
|
||||
// You should have received a copy of the GNU Affero General Public License
|
||||
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
#include <nel/misc/types_nl.h>
|
||||
|
||||
namespace NL3D {
|
||||
class CMaterial;
|
||||
}
|
||||
|
||||
struct CMeshUtilsContext;
|
||||
void assimpMaterial(NL3D::CMaterial &mat, CMeshUtilsContext &context, const aiMaterial *am);
|
||||
void assimpMaterials(CMeshUtilsContext &context);
|
||||
|
||||
/* end of file */
|
@ -0,0 +1,379 @@
|
||||
// NeL - MMORPG Framework <http://dev.ryzom.com/projects/nel/>
|
||||
// Copyright (C) 2015 Winch Gate Property Limited
|
||||
// Author: Jan Boon <jan.boon@kaetemi.be>
|
||||
//
|
||||
// This program is free software: you can redistribute it and/or modify
|
||||
// it under the terms of the GNU Affero General Public License as
|
||||
// published by the Free Software Foundation, either version 3 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 Affero General Public License for more details.
|
||||
//
|
||||
// You should have received a copy of the GNU Affero General Public License
|
||||
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
#include <nel/misc/types_nl.h>
|
||||
#include "assimp_shape.h"
|
||||
|
||||
#include <assimp/postprocess.h>
|
||||
#include <assimp/scene.h>
|
||||
#include <assimp/Importer.hpp>
|
||||
|
||||
#define NL_NODE_INTERNAL_TYPE aiNode
|
||||
#define NL_SCENE_INTERNAL_TYPE aiScene
|
||||
#include "scene_context.h"
|
||||
|
||||
#include <nel/misc/debug.h>
|
||||
#include <nel/misc/path.h>
|
||||
#include <nel/misc/tool_logger.h>
|
||||
|
||||
#include <nel/3d/mesh.h>
|
||||
|
||||
#include "assimp_material.h"
|
||||
|
||||
using namespace std;
|
||||
using namespace NLMISC;
|
||||
using namespace NL3D;
|
||||
|
||||
// TODO: buildParticleSystem ??
|
||||
// TODO: buildWaveMakerShape ??
|
||||
// TODO: buildRemanence ??
|
||||
// TODO: buildFlare ??
|
||||
// Probably specific settings we can only do in meta editor on a dummy node..
|
||||
// TODO: pacs prim
|
||||
|
||||
// TODO: buildWaterShape specifics when node has water material
|
||||
|
||||
// TODO: CMeshMultiLod::CMeshMultiLodBuild multiLodBuild; export_mesh.cpp ln 228
|
||||
// TODO: LOD MRM
|
||||
|
||||
// TODO: Skinned - reverse transform by skeleton root bone to align?
|
||||
|
||||
/*inline CMatrix convMatrix(const aiMatrix4x4 &tf)
|
||||
{
|
||||
CMatrix m;
|
||||
for (int i = 0; i < 16; ++i)
|
||||
m.set(&tf.a1);
|
||||
return m;
|
||||
}*/
|
||||
|
||||
inline CVector convVector(const aiVector3D &av)
|
||||
{
|
||||
return CVector(av.x, av.y, av.z); // COORDINATE CONVERSION
|
||||
}
|
||||
|
||||
inline CRGBA convColor(const aiColor4D &ac)
|
||||
{
|
||||
return CRGBA(ac.r * 255.99f, ac.g * 255.99f, ac.b * 255.99f, ac.a * 255.99f);
|
||||
}
|
||||
|
||||
inline CUVW convUvw(const aiVector3D &av)
|
||||
{
|
||||
return CUVW(av.x, -av.y, av.z); // UH OH COORDINATE CONVERSION ?! ONLY FOR TEXTURES !!
|
||||
}
|
||||
|
||||
inline CQuat convQuat(const aiQuaternion &aq)
|
||||
{
|
||||
return CQuat(aq.x, aq.y, aq.z, aq.w);
|
||||
}
|
||||
|
||||
void assimpBuildBaseMesh(CMeshBase::CMeshBaseBuild &buildBaseMesh, CMeshUtilsContext &context, CNodeContext &nodeContext)
|
||||
{
|
||||
const aiNode *node = nodeContext.InternalNode;
|
||||
// Reference CExportNel::buildBaseMeshInterface
|
||||
|
||||
// Load materials
|
||||
buildBaseMesh.Materials.resize(node->mNumMeshes);
|
||||
|
||||
for (unsigned int mi = 0; mi < node->mNumMeshes; ++mi)
|
||||
{
|
||||
const aiMesh *mesh = context.InternalScene->mMeshes[node->mMeshes[mi]];
|
||||
const aiMaterial *am = context.InternalScene->mMaterials[mesh->mMaterialIndex];
|
||||
|
||||
aiString amname;
|
||||
if (am->Get(AI_MATKEY_NAME, amname) != aiReturn_SUCCESS)
|
||||
{
|
||||
tlerror(context.ToolLogger, context.Settings.SourceFilePath.c_str(),
|
||||
"Material used by node '%s' has no name", node->mName.C_Str()); // TODO: Maybe autogen names by index in mesh or node if this is actually a thing
|
||||
assimpMaterial(buildBaseMesh.Materials[mi], context, am);
|
||||
}
|
||||
else
|
||||
{
|
||||
buildBaseMesh.Materials[mi] = *context.SceneMeta.Materials[amname.C_Str()];
|
||||
}
|
||||
}
|
||||
|
||||
// Positioning
|
||||
const aiMatrix4x4 &root = context.InternalScene->mRootNode->mTransformation;
|
||||
const aiMatrix4x4 &tf = nodeContext.InternalNode->mTransformation; // COORDINATE CONVERSION HERE INSTEAD OF PER VERTEX ??
|
||||
aiVector3D scaling;
|
||||
aiQuaternion rotation;
|
||||
aiVector3D position;
|
||||
tf.Decompose(scaling, rotation, position);
|
||||
buildBaseMesh.DefaultScale = convVector(scaling);
|
||||
buildBaseMesh.DefaultRotQuat = convQuat(rotation);
|
||||
buildBaseMesh.DefaultRotEuler = CVector(0, 0, 0);
|
||||
buildBaseMesh.DefaultPivot = CVector(0, 0, 0);
|
||||
buildBaseMesh.DefaultPos = convVector(position);
|
||||
if (buildBaseMesh.DefaultScale.x != 1.0f || buildBaseMesh.DefaultScale.y != 1.0f || buildBaseMesh.DefaultScale.z != 1.0f)
|
||||
{
|
||||
tlmessage(context.ToolLogger, context.Settings.SourceFilePath.c_str(),
|
||||
"Node '%s' has a scaled transformation. This may be a mistake", node->mName.C_Str());
|
||||
}
|
||||
|
||||
// Meta
|
||||
// dst.CollisionMeshGeneration = src.CollisionMeshGeneration;
|
||||
|
||||
// TODO: Morph
|
||||
}
|
||||
|
||||
bool assimpBuildMesh(CMesh::CMeshBuild &buildMesh, CMeshBase::CMeshBaseBuild &buildBaseMesh, CMeshUtilsContext &context, CNodeContext &nodeContext)
|
||||
{
|
||||
// TODO
|
||||
// *** If the mesh is skined, vertices will be exported in world space.
|
||||
// *** If the mesh is not skined, vertices will be exported in offset space.
|
||||
|
||||
// TODO Support skinning
|
||||
|
||||
const aiNode *node = nodeContext.InternalNode;
|
||||
nlassert(node->mNumMeshes);
|
||||
|
||||
// Basic validations before processing starts
|
||||
for (unsigned int mi = 0; mi < node->mNumMeshes; ++mi)
|
||||
{
|
||||
// TODO: Maybe needs to be the same count too for all meshes, so compare with mesh 0
|
||||
const aiMesh *mesh = context.InternalScene->mMeshes[node->mMeshes[mi]];
|
||||
if (mesh->GetNumColorChannels() > 2)
|
||||
{
|
||||
tlerror(context.ToolLogger, context.Settings.SourceFilePath.c_str(),
|
||||
"(%s) mesh->GetNumColorChannels() > 2", node->mName.C_Str());
|
||||
return false;
|
||||
}
|
||||
if (mesh->GetNumUVChannels() > CVertexBuffer::MaxStage)
|
||||
{
|
||||
tlerror(context.ToolLogger, context.Settings.SourceFilePath.c_str(),
|
||||
"(%s) mesh->GetNumUVChannels() > CVertexBuffer::MaxStage", node->mName.C_Str());
|
||||
return false;
|
||||
}
|
||||
if (!mesh->HasNormals())
|
||||
{
|
||||
tlerror(context.ToolLogger, context.Settings.SourceFilePath.c_str(),
|
||||
"(%s) !mesh->HasNormals()", node->mName.C_Str());
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
// Default vertex flags
|
||||
buildMesh.VertexFlags = CVertexBuffer::PositionFlag | CVertexBuffer::NormalFlag;
|
||||
|
||||
// TODO: UV Channels routing to correct texture stage
|
||||
for (uint i = 0; i < CVertexBuffer::MaxStage; ++i)
|
||||
buildMesh.UVRouting[i] = i;
|
||||
|
||||
// Meshes in assimp are separated per material, so we need to re-merge them for the mesh build process
|
||||
// This process also deduplicates vertices
|
||||
bool cleanupMesh = true;
|
||||
sint32 numVertices = 0;
|
||||
for (unsigned int mi = 0; mi < node->mNumMeshes; ++mi)
|
||||
numVertices += context.InternalScene->mMeshes[node->mMeshes[mi]]->mNumVertices;
|
||||
buildMesh.Vertices.resize(numVertices);
|
||||
numVertices = 0;
|
||||
map<CVector, sint32> vertexIdentifiers;
|
||||
vector<vector<sint32> > vertexRemapping;
|
||||
vertexRemapping.resize(node->mNumMeshes);
|
||||
for (unsigned int mi = 0; mi < node->mNumMeshes; ++mi)
|
||||
{
|
||||
const aiMesh *mesh = context.InternalScene->mMeshes[node->mMeshes[mi]];
|
||||
vertexRemapping[mi].resize(mesh->mNumVertices);
|
||||
for (unsigned int vi = 0; vi < mesh->mNumVertices; ++vi)
|
||||
{
|
||||
CVector vec = convVector(mesh->mVertices[vi]);
|
||||
map<CVector, sint32>::iterator vecit = vertexIdentifiers.find(vec);
|
||||
if (vecit == vertexIdentifiers.end())
|
||||
{
|
||||
buildMesh.Vertices[numVertices] = vec;
|
||||
if (cleanupMesh) vertexIdentifiers[vec] = numVertices; // Don't remap if we don't wan't to lose vertex indices
|
||||
vertexRemapping[mi][vi] = numVertices;
|
||||
++numVertices;
|
||||
}
|
||||
else
|
||||
{
|
||||
vertexRemapping[mi][vi] = vecit->second;
|
||||
}
|
||||
}
|
||||
}
|
||||
buildMesh.Vertices.resize(numVertices);
|
||||
|
||||
// Process all faces
|
||||
// WONT IMPLEMENT: Radial faces generation... is linked to smoothing group...
|
||||
// leave radial normals generation to modeling tool for now...
|
||||
sint32 numFaces = 0;
|
||||
for (unsigned int mi = 0; mi < node->mNumMeshes; ++mi)
|
||||
numFaces += context.InternalScene->mMeshes[node->mMeshes[mi]]->mNumFaces;
|
||||
buildMesh.Faces.resize(numFaces);
|
||||
numFaces = 0;
|
||||
unsigned int refNumColorChannels = context.InternalScene->mMeshes[node->mMeshes[0]]->GetNumColorChannels();
|
||||
unsigned int refNumUVChannels = context.InternalScene->mMeshes[node->mMeshes[0]]->GetNumUVChannels();
|
||||
for (unsigned int mi = 0; mi < node->mNumMeshes; ++mi)
|
||||
{
|
||||
const aiMesh *mesh = context.InternalScene->mMeshes[node->mMeshes[mi]];
|
||||
|
||||
// Get channel numbers
|
||||
unsigned int numColorChannels = mesh->GetNumColorChannels();
|
||||
if (numColorChannels > 2)
|
||||
{
|
||||
tlerror(context.ToolLogger, context.Settings.SourceFilePath.c_str(),
|
||||
"Shape '%s' has too many color channels in mesh %i (%i channels found)", node->mName.C_Str(), mi, numColorChannels);
|
||||
}
|
||||
if (numColorChannels > 0)
|
||||
{
|
||||
buildMesh.VertexFlags |= CVertexBuffer::PrimaryColorFlag;
|
||||
if (numColorChannels > 1)
|
||||
{
|
||||
buildMesh.VertexFlags |= CVertexBuffer::SecondaryColorFlag;
|
||||
}
|
||||
}
|
||||
unsigned int numUVChannels = mesh->GetNumUVChannels();
|
||||
if (numUVChannels > CVertexBuffer::MaxStage)
|
||||
{
|
||||
tlerror(context.ToolLogger, context.Settings.SourceFilePath.c_str(),
|
||||
"Shape '%s' has too many uv channels in mesh %i (%i channels found)", node->mName.C_Str(), mi, numUVChannels);
|
||||
numUVChannels = CVertexBuffer::MaxStage;
|
||||
}
|
||||
for (unsigned int ui = 0; ui < numUVChannels; ++ui)
|
||||
buildMesh.VertexFlags |= (CVertexBuffer::TexCoord0Flag << ui); // TODO: Coord UV tex stage rerouting
|
||||
|
||||
// TODO: Channels do in fact differ between submeshes, so we need to correctly recount and reroute the materials properly
|
||||
if (numColorChannels != refNumColorChannels)
|
||||
tlerror(context.ToolLogger, context.Settings.SourceFilePath.c_str(),
|
||||
"Shape '%s' mismatch of nb color channel in mesh '%i', please contact developer", node->mName.C_Str(), mi);
|
||||
if (numUVChannels != refNumUVChannels)
|
||||
tlerror(context.ToolLogger, context.Settings.SourceFilePath.c_str(),
|
||||
"Shape '%s' mismatch of nb uv channel in mesh '%i', please contact developer", node->mName.C_Str(), mi);
|
||||
|
||||
for (unsigned int fi = 0; fi < mesh->mNumFaces; ++fi)
|
||||
{
|
||||
const aiFace &af = mesh->mFaces[fi];
|
||||
if (af.mNumIndices != 3)
|
||||
{
|
||||
tlerror(context.ToolLogger, context.Settings.SourceFilePath.c_str(),
|
||||
"(%s) Face %i on mesh %i has %i faces", node->mName.C_Str(), fi, mi, af.mNumIndices);
|
||||
continue; // return false; Keep going, just drop the face for better user experience
|
||||
}
|
||||
if (cleanupMesh)
|
||||
{
|
||||
if (vertexRemapping[mi][af.mIndices[0]] == vertexRemapping[mi][af.mIndices[1]]
|
||||
|| vertexRemapping[mi][af.mIndices[1]] == vertexRemapping[mi][af.mIndices[2]]
|
||||
|| vertexRemapping[mi][af.mIndices[2]] == vertexRemapping[mi][af.mIndices[0]])
|
||||
continue; // Not a triangle
|
||||
}
|
||||
CMesh::CFace &face = buildMesh.Faces[numFaces];
|
||||
face.MaterialId = mi;
|
||||
face.SmoothGroup = 0; // No smoothing groups (bitfield)
|
||||
face.Corner[0].Vertex = vertexRemapping[mi][af.mIndices[0]];
|
||||
face.Corner[1].Vertex = vertexRemapping[mi][af.mIndices[1]];
|
||||
face.Corner[2].Vertex = vertexRemapping[mi][af.mIndices[2]];
|
||||
face.Corner[0].Normal = convVector(mesh->mNormals[af.mIndices[0]]);
|
||||
face.Corner[1].Normal = convVector(mesh->mNormals[af.mIndices[1]]);
|
||||
face.Corner[2].Normal = convVector(mesh->mNormals[af.mIndices[2]]);
|
||||
// TODO: If we want normal maps, we need to add tangent vectors to CFace and build process
|
||||
// UV channels
|
||||
for (unsigned int ui = 0; ui < numUVChannels; ++ui) // TODO: UV Rerouting
|
||||
{
|
||||
face.Corner[0].Uvws[ui] = convUvw(mesh->mTextureCoords[ui][af.mIndices[0]]);
|
||||
face.Corner[1].Uvws[ui] = convUvw(mesh->mTextureCoords[ui][af.mIndices[1]]);
|
||||
face.Corner[2].Uvws[ui] = convUvw(mesh->mTextureCoords[ui][af.mIndices[2]]);
|
||||
}
|
||||
for (unsigned int ui = numUVChannels; ui < CVertexBuffer::MaxStage; ++ui)
|
||||
{
|
||||
face.Corner[0].Uvws[ui] = CUVW(0, 0, 0);
|
||||
face.Corner[1].Uvws[ui] = CUVW(0, 0, 0);
|
||||
face.Corner[2].Uvws[ui] = CUVW(0, 0, 0);
|
||||
}
|
||||
// Primary and secondary color channels
|
||||
if (numColorChannels > 0) // TODO: Verify
|
||||
{
|
||||
face.Corner[0].Color = convColor(mesh->mColors[0][af.mIndices[0]]);
|
||||
face.Corner[1].Color = convColor(mesh->mColors[0][af.mIndices[1]]);
|
||||
face.Corner[2].Color = convColor(mesh->mColors[0][af.mIndices[2]]);
|
||||
}
|
||||
else
|
||||
{
|
||||
face.Corner[0].Color = CRGBA(255, 255, 255, 255);
|
||||
face.Corner[1].Color = CRGBA(255, 255, 255, 255);
|
||||
face.Corner[2].Color = CRGBA(255, 255, 255, 255);
|
||||
}
|
||||
if (numColorChannels > 1) // TODO: Verify
|
||||
{
|
||||
face.Corner[0].Specular = convColor(mesh->mColors[1][af.mIndices[0]]);
|
||||
face.Corner[1].Specular = convColor(mesh->mColors[1][af.mIndices[1]]);
|
||||
face.Corner[2].Specular = convColor(mesh->mColors[1][af.mIndices[2]]);
|
||||
}
|
||||
else
|
||||
{
|
||||
face.Corner[0].Specular = CRGBA(255, 255, 255, 255);
|
||||
face.Corner[1].Specular = CRGBA(255, 255, 255, 255);
|
||||
face.Corner[2].Specular = CRGBA(255, 255, 255, 255);
|
||||
}
|
||||
// TODO: Color modulate, alpha, use color alpha for vp tree, etc
|
||||
++numFaces;
|
||||
}
|
||||
}
|
||||
if (numFaces != buildMesh.Faces.size())
|
||||
{
|
||||
tlmessage(context.ToolLogger, context.Settings.SourceFilePath.c_str(),
|
||||
"Removed %u degenerate faces in shape '%s'", (uint32)(buildMesh.Faces.size() - numFaces), node->mName.C_Str());
|
||||
buildMesh.Faces.resize(numFaces);
|
||||
}
|
||||
|
||||
// clear for MRM info
|
||||
buildMesh.Interfaces.clear();
|
||||
buildMesh.InterfaceLinks.clear();
|
||||
|
||||
// TODO: Export VP
|
||||
buildMesh.MeshVertexProgram = NULL;
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
bool assimpShape(CMeshUtilsContext &context, CNodeContext &nodeContext)
|
||||
{
|
||||
// Reference: export_mesh.cpp, buildShape
|
||||
nodeContext.Shape = NULL;
|
||||
|
||||
const aiNode *node = nodeContext.InternalNode;
|
||||
nlassert(node->mNumMeshes);
|
||||
|
||||
// Fill the build interface of CMesh
|
||||
CMeshBase::CMeshBaseBuild buildBaseMesh;
|
||||
assimpBuildBaseMesh(buildBaseMesh, context, nodeContext);
|
||||
|
||||
CMesh::CMeshBuild buildMesh;
|
||||
if (!assimpBuildMesh(buildMesh, buildBaseMesh, context, nodeContext))
|
||||
return false;
|
||||
|
||||
// Make a CMesh object
|
||||
CMesh *mesh = new CMesh();
|
||||
|
||||
// Build the mesh with the build interface
|
||||
mesh->build(buildBaseMesh, buildMesh);
|
||||
|
||||
// TODO
|
||||
// Reference: export_mesh.cpp, buildShape
|
||||
// Must be done after the build to update vertex links
|
||||
// Pass to buildMeshMorph if the original mesh is skinned or not
|
||||
// buildMeshMorph(buildMesh, node, time, nodeMap != NULL);
|
||||
// mesh->setBlendShapes(buildMesh.BlendShapes);
|
||||
|
||||
// optimize number of material
|
||||
// mesh->optimizeMaterialUsage(materialRemap);
|
||||
|
||||
// Store mesh in context
|
||||
nodeContext.Shape = mesh;
|
||||
return true;
|
||||
}
|
||||
|
||||
/* end of file */
|
@ -0,0 +1,25 @@
|
||||
// NeL - MMORPG Framework <http://dev.ryzom.com/projects/nel/>
|
||||
// Copyright (C) 2015 Winch Gate Property Limited
|
||||
// Author: Jan Boon <jan.boon@kaetemi.be>
|
||||
//
|
||||
// This program is free software: you can redistribute it and/or modify
|
||||
// it under the terms of the GNU Affero General Public License as
|
||||
// published by the Free Software Foundation, either version 3 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 Affero General Public License for more details.
|
||||
//
|
||||
// You should have received a copy of the GNU Affero General Public License
|
||||
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
#include <nel/misc/types_nl.h>
|
||||
|
||||
struct CMeshUtilsContext;
|
||||
struct CNodeContext;
|
||||
|
||||
bool assimpShape(CMeshUtilsContext &context, CNodeContext &nodeContext);
|
||||
|
||||
/* end of file */
|
@ -0,0 +1,107 @@
|
||||
// NeL - MMORPG Framework <http://dev.ryzom.com/projects/nel/>
|
||||
// Copyright (C) 2015 Winch Gate Property Limited
|
||||
// Author: Jan Boon <jan.boon@kaetemi.be>
|
||||
//
|
||||
// This program is free software: you can redistribute it and/or modify
|
||||
// it under the terms of the GNU Affero General Public License as
|
||||
// published by the Free Software Foundation, either version 3 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 Affero General Public License for more details.
|
||||
//
|
||||
// You should have received a copy of the GNU Affero General Public License
|
||||
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
#include <nel/misc/types_nl.h>
|
||||
#include "database_config.h"
|
||||
|
||||
#include <nel/misc/debug.h>
|
||||
#include <nel/misc/path.h>
|
||||
#include <nel/misc/config_file.h>
|
||||
|
||||
using namespace std;
|
||||
using namespace NLMISC;
|
||||
|
||||
TPathString CDatabaseConfig::s_RootPath;
|
||||
NLMISC::CConfigFile *CDatabaseConfig::s_ConfigFile = NULL;
|
||||
CDatabaseConfig CDatabaseConfig::s_Instance;
|
||||
uint32 CDatabaseConfig::s_ConfigFileModification;
|
||||
|
||||
static std::set<TPathString> s_SearchPaths;
|
||||
|
||||
void CDatabaseConfig::cleanup()
|
||||
{
|
||||
delete CDatabaseConfig::s_ConfigFile;
|
||||
CDatabaseConfig::s_ConfigFile = NULL;
|
||||
}
|
||||
|
||||
CDatabaseConfig::~CDatabaseConfig()
|
||||
{
|
||||
cleanup();
|
||||
}
|
||||
|
||||
bool CDatabaseConfig::init(const std::string &asset)
|
||||
{
|
||||
// release();
|
||||
|
||||
TPathString rootPath = NLMISC::CPath::standardizePath(asset, false);
|
||||
TPathString configPath = rootPath + "/database.cfg";
|
||||
while (!CFile::fileExists(configPath))
|
||||
{
|
||||
int sep = CFile::getLastSeparator(rootPath);
|
||||
if (sep == string::npos)
|
||||
return false;
|
||||
|
||||
rootPath = rootPath.substr(0, sep);
|
||||
if (rootPath.empty())
|
||||
return false;
|
||||
|
||||
configPath = rootPath + "/database.cfg";
|
||||
}
|
||||
|
||||
rootPath += "/";
|
||||
uint32 configFileModification = CFile::getFileModificationDate(configPath);
|
||||
if (rootPath == s_RootPath && s_ConfigFileModification == configFileModification)
|
||||
return true; // Do not reload
|
||||
|
||||
nldebug("Initializing database config '%s'", configPath.c_str());
|
||||
release();
|
||||
|
||||
s_RootPath = rootPath;
|
||||
s_ConfigFileModification = configFileModification;
|
||||
|
||||
s_ConfigFile = new CConfigFile();
|
||||
s_ConfigFile->load(configPath);
|
||||
return true;
|
||||
}
|
||||
|
||||
void CDatabaseConfig::initTextureSearchDirectories()
|
||||
{
|
||||
searchDirectories("TextureSearchDirectories");
|
||||
}
|
||||
|
||||
void CDatabaseConfig::searchDirectories(const char *var)
|
||||
{
|
||||
CConfigFile::CVar &paths = s_ConfigFile->getVar(var);
|
||||
for (uint i = 0; i < paths.size(); i++)
|
||||
{
|
||||
TPathString path = paths.asString(i);
|
||||
if (s_SearchPaths.find(path) == s_SearchPaths.end())
|
||||
{
|
||||
CPath::addSearchPath(s_RootPath + path);
|
||||
s_SearchPaths.insert(path);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void CDatabaseConfig::release()
|
||||
{
|
||||
s_SearchPaths.clear();
|
||||
CPath::clearMap();
|
||||
cleanup();
|
||||
}
|
||||
|
||||
/* end of file */
|
@ -0,0 +1,58 @@
|
||||
// NeL - MMORPG Framework <http://dev.ryzom.com/projects/nel/>
|
||||
// Copyright (C) 2015 Winch Gate Property Limited
|
||||
// Author: Jan Boon <jan.boon@kaetemi.be>
|
||||
//
|
||||
// This program is free software: you can redistribute it and/or modify
|
||||
// it under the terms of the GNU Affero General Public License as
|
||||
// published by the Free Software Foundation, either version 3 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 Affero General Public License for more details.
|
||||
//
|
||||
// You should have received a copy of the GNU Affero General Public License
|
||||
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
#include <nel/misc/types_nl.h>
|
||||
|
||||
namespace NLMISC {
|
||||
class CConfigFile;
|
||||
}
|
||||
|
||||
#ifdef NL_OS_WINDOWS
|
||||
#include <nel/misc/sstring.h>
|
||||
typedef NLMISC::CSString TPathString;
|
||||
#else
|
||||
typedef std::string TPathString;
|
||||
#endif
|
||||
|
||||
/// Asset database configuration
|
||||
class CDatabaseConfig
|
||||
{
|
||||
public:
|
||||
~CDatabaseConfig();
|
||||
|
||||
/// Searches for the configuration for the specified asset path by recursively going through all parent directories looking for 'database.cfg', initializes and applies the configuration.
|
||||
static bool init(const std::string &asset);
|
||||
static void release();
|
||||
|
||||
static void initTextureSearchDirectories();
|
||||
|
||||
static inline const TPathString &rootPath() { return s_RootPath; }
|
||||
static inline TPathString configPath() { return s_RootPath + "/database.cfg"; }
|
||||
|
||||
private:
|
||||
static void cleanup();
|
||||
static void searchDirectories(const char *var);
|
||||
|
||||
static CDatabaseConfig s_Instance;
|
||||
static uint32 s_ConfigFileModification;
|
||||
|
||||
static TPathString s_RootPath;
|
||||
static NLMISC::CConfigFile *s_ConfigFile;
|
||||
|
||||
};
|
||||
|
||||
/* end of file */
|
@ -0,0 +1,348 @@
|
||||
// NeL - MMORPG Framework <http://dev.ryzom.com/projects/nel/>
|
||||
// Copyright (C) 2015 Winch Gate Property Limited
|
||||
// Author: Jan Boon <jan.boon@kaetemi.be>
|
||||
//
|
||||
// This program is free software: you can redistribute it and/or modify
|
||||
// it under the terms of the GNU Affero General Public License as
|
||||
// published by the Free Software Foundation, either version 3 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 Affero General Public License for more details.
|
||||
//
|
||||
// You should have received a copy of the GNU Affero General Public License
|
||||
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
#include <nel/misc/types_nl.h>
|
||||
#include "mesh_utils.h"
|
||||
|
||||
#include <nel/misc/debug.h>
|
||||
#include <nel/misc/tool_logger.h>
|
||||
#include <nel/misc/sstring.h>
|
||||
#include <nel/misc/file.h>
|
||||
#include <nel/misc/path.h>
|
||||
|
||||
#include <nel/3d/shape.h>
|
||||
#include <nel/3d/mesh.h>
|
||||
#include <nel/3d/texture_file.h>
|
||||
|
||||
#include "database_config.h"
|
||||
#include "scene_meta.h"
|
||||
|
||||
#include <assimp/postprocess.h>
|
||||
#include <assimp/scene.h>
|
||||
#include <assimp/Importer.hpp>
|
||||
|
||||
#define NL_NODE_INTERNAL_TYPE aiNode
|
||||
#define NL_SCENE_INTERNAL_TYPE aiScene
|
||||
#include "scene_context.h"
|
||||
|
||||
#include "assimp_material.h"
|
||||
#include "assimp_shape.h"
|
||||
|
||||
CMeshUtilsSettings::CMeshUtilsSettings()
|
||||
{
|
||||
/*ShapeDirectory = "shape";
|
||||
IGDirectory = "ig";
|
||||
SkelDirectory = "skel";*/
|
||||
}
|
||||
|
||||
void importShapes(CMeshUtilsContext &context, const aiNode *node)
|
||||
{
|
||||
if (node != context.InternalScene->mRootNode)
|
||||
{
|
||||
CNodeContext &nodeContext = context.Nodes[node->mName.C_Str()];
|
||||
CNodeMeta &nodeMeta = context.SceneMeta.Nodes[node->mName.C_Str()];
|
||||
if (nodeMeta.ExportMesh == TMeshShape && nodeMeta.InstanceName.empty())
|
||||
{
|
||||
if (node->mNumMeshes)
|
||||
{
|
||||
nldebug("Shape '%s' found containing '%u' meshes", node->mName.C_Str(), node->mNumMeshes);
|
||||
assimpShape(context, nodeContext);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
for (unsigned int i = 0; i < node->mNumChildren; ++i)
|
||||
importShapes(context, node->mChildren[i]);
|
||||
}
|
||||
|
||||
void validateInternalNodeNames(CMeshUtilsContext &context, const aiNode *node)
|
||||
{
|
||||
if (!node->mParent || node == context.InternalScene->mRootNode)
|
||||
{
|
||||
// do nothing
|
||||
}
|
||||
else if (node->mName.length == 0)
|
||||
{
|
||||
tlwarning(context.ToolLogger, context.Settings.SourceFilePath.c_str(),
|
||||
"Node has no name");
|
||||
}
|
||||
else
|
||||
{
|
||||
CNodeContext &nodeContext = context.Nodes[node->mName.C_Str()];
|
||||
|
||||
if (nodeContext.InternalNode && nodeContext.InternalNode != node)
|
||||
{
|
||||
tlerror(context.ToolLogger, context.Settings.SourceFilePath.c_str(),
|
||||
"Node name '%s' appears multiple times", node->mName.C_Str());
|
||||
}
|
||||
else
|
||||
{
|
||||
nodeContext.InternalNode = node;
|
||||
}
|
||||
}
|
||||
|
||||
for (unsigned int i = 0; i < node->mNumChildren; ++i)
|
||||
validateInternalNodeNames(context, node->mChildren[i]);
|
||||
}
|
||||
|
||||
void flagAssimpBones(CMeshUtilsContext &context)
|
||||
{
|
||||
// Find out which nodes are bones by checking the mesh meta info
|
||||
const aiScene *scene = context.InternalScene;
|
||||
for (unsigned int i = 0; i < scene->mNumMeshes; ++i)
|
||||
{
|
||||
// nldebug("FOUND MESH '%s'\n", scene->mMeshes[i]->mName.C_Str());
|
||||
const aiMesh *mesh = scene->mMeshes[i];
|
||||
for (unsigned int j = 0; j < mesh->mNumBones; ++j)
|
||||
{
|
||||
CNodeContext &nodeContext = context.Nodes[mesh->mBones[j]->mName.C_Str()];
|
||||
if (!nodeContext.InternalNode)
|
||||
{
|
||||
tlerror(context.ToolLogger, context.Settings.SourceFilePath.c_str(),
|
||||
"Bone '%s' has no associated node", mesh->mBones[j]->mName.C_Str());
|
||||
}
|
||||
else
|
||||
{
|
||||
// Flag as bone
|
||||
nodeContext.IsBone = true;
|
||||
|
||||
// Flag all parents as bones
|
||||
/*const aiNode *parent = nodeContext.InternalNode;
|
||||
while (parent = parent->mParent) if (parent->mName.length)
|
||||
{
|
||||
context.Nodes[parent->mName.C_Str()].IsBone = true;
|
||||
}*/
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Find out which nodes are bones by checking the animation info
|
||||
// TODO
|
||||
}
|
||||
|
||||
void flagRecursiveBones(CMeshUtilsContext &context, CNodeContext &nodeContext, bool autoStop = false)
|
||||
{
|
||||
nodeContext.IsBone = true;
|
||||
const aiNode *node = nodeContext.InternalNode;
|
||||
nlassert(node);
|
||||
for (unsigned int i = 0; i < node->mNumChildren; ++i)
|
||||
{
|
||||
CNodeContext &ctx = context.Nodes[node->mName.C_Str()];
|
||||
if (autoStop && ctx.IsBone)
|
||||
continue;
|
||||
flagRecursiveBones(context, ctx);
|
||||
}
|
||||
}
|
||||
|
||||
void flagMetaBones(CMeshUtilsContext &context)
|
||||
{
|
||||
for (TNodeContextMap::iterator it(context.Nodes.begin()), end(context.Nodes.end()); it != end; ++it)
|
||||
{
|
||||
CNodeContext &ctx = it->second;
|
||||
CNodeMeta &meta = context.SceneMeta.Nodes[it->first];
|
||||
if (meta.ExportBone == TBoneForce)
|
||||
ctx.IsBone = true;
|
||||
else if (meta.ExportBone == TBoneRoot)
|
||||
flagRecursiveBones(context, ctx);
|
||||
}
|
||||
}
|
||||
|
||||
void flagLocalParentBones(CMeshUtilsContext &context, CNodeContext &nodeContext)
|
||||
{
|
||||
const aiNode *node = nodeContext.InternalNode;
|
||||
}
|
||||
|
||||
void flagAllParentBones(CMeshUtilsContext &context, CNodeContext &nodeContext, bool autoStop = false)
|
||||
{
|
||||
const aiNode *parent = nodeContext.InternalNode;
|
||||
while (parent = parent->mParent) if (parent->mName.length && parent != context.InternalScene->mRootNode)
|
||||
{
|
||||
CNodeContext &ctx = context.Nodes[parent->mName.C_Str()];
|
||||
if (autoStop && ctx.IsBone)
|
||||
break;
|
||||
ctx.IsBone = true;
|
||||
}
|
||||
}
|
||||
|
||||
bool hasIndirectParentBone(CMeshUtilsContext &context, CNodeContext &nodeContext)
|
||||
{
|
||||
const aiNode *parent = nodeContext.InternalNode;
|
||||
while (parent = parent->mParent) if (parent->mName.length && parent != context.InternalScene->mRootNode)
|
||||
if (context.Nodes[parent->mName.C_Str()].IsBone) return true;
|
||||
return false;
|
||||
}
|
||||
|
||||
void flagExpandedBones(CMeshUtilsContext &context)
|
||||
{
|
||||
switch (context.SceneMeta.SkeletonMode)
|
||||
{
|
||||
case TSkelLocal:
|
||||
for (TNodeContextMap::iterator it(context.Nodes.begin()), end(context.Nodes.end()); it != end; ++it)
|
||||
{
|
||||
CNodeContext &nodeContext = it->second;
|
||||
if (nodeContext.IsBone && hasIndirectParentBone(context, nodeContext))
|
||||
flagAllParentBones(context, nodeContext, true);
|
||||
}
|
||||
break;
|
||||
case TSkelRoot:
|
||||
for (TNodeContextMap::iterator it(context.Nodes.begin()), end(context.Nodes.end()); it != end; ++it)
|
||||
{
|
||||
CNodeContext &nodeContext = it->second;
|
||||
if (nodeContext.IsBone)
|
||||
flagAllParentBones(context, nodeContext, true);
|
||||
}
|
||||
break;
|
||||
case TSkelFull:
|
||||
for (TNodeContextMap::iterator it(context.Nodes.begin()), end(context.Nodes.end()); it != end; ++it)
|
||||
{
|
||||
CNodeContext &nodeContext = it->second;
|
||||
if (nodeContext.IsBone)
|
||||
flagAllParentBones(context, nodeContext, true);
|
||||
}
|
||||
for (TNodeContextMap::iterator it(context.Nodes.begin()), end(context.Nodes.end()); it != end; ++it)
|
||||
{
|
||||
CNodeContext &nodeContext = it->second;
|
||||
if (nodeContext.IsBone)
|
||||
flagRecursiveBones(context, nodeContext, true);
|
||||
}
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
void exportShapes(CMeshUtilsContext &context)
|
||||
{
|
||||
for (TNodeContextMap::iterator it(context.Nodes.begin()), end(context.Nodes.end()); it != end; ++it)
|
||||
{
|
||||
CNodeContext &nodeContext = it->second;
|
||||
if (nodeContext.Shape)
|
||||
{
|
||||
std::string shapePath = NLMISC::CPath::standardizePath(context.Settings.DestinationDirectoryPath, true) + it->first + ".shape";
|
||||
context.ToolLogger.writeDepend(NLMISC::BUILD, shapePath.c_str(), "*");
|
||||
NLMISC::COFile f;
|
||||
if (f.open(shapePath, false, false, true))
|
||||
{
|
||||
try
|
||||
{
|
||||
NL3D::CShapeStream shapeStream(nodeContext.Shape);
|
||||
shapeStream.serial(f);
|
||||
f.close();
|
||||
}
|
||||
catch (...)
|
||||
{
|
||||
tlerror(context.ToolLogger, context.Settings.SourceFilePath.c_str(),
|
||||
"Shape '%s' serialization failed!", it->first.c_str());
|
||||
}
|
||||
}
|
||||
if (NL3D::CMeshBase *mesh = dynamic_cast<NL3D::CMeshBase *>(nodeContext.Shape.getPtr()))
|
||||
{
|
||||
for (uint mi = 0; mi < mesh->getNbMaterial(); ++mi)
|
||||
{
|
||||
NL3D::CMaterial &mat = mesh->getMaterial(mi);
|
||||
for (uint ti = 0; ti < NL3D::IDRV_MAT_MAXTEXTURES; ++ti)
|
||||
{
|
||||
if (NL3D::ITexture *itex = mat.getTexture(ti))
|
||||
{
|
||||
if (NL3D::CTextureFile *tex = dynamic_cast<NL3D::CTextureFile *>(itex))
|
||||
{
|
||||
std::string fileName = tex->getFileName();
|
||||
std::string knownPath = NLMISC::CPath::lookup(fileName, false, false, false);
|
||||
if (!knownPath.empty())
|
||||
{
|
||||
context.ToolLogger.writeDepend(NLMISC::RUNTIME, shapePath.c_str(), knownPath.c_str());
|
||||
}
|
||||
else
|
||||
{
|
||||
// TODO: Move this warning into nelmeta serialization so it's shown before export
|
||||
tlwarning(context.ToolLogger, context.Settings.SourceFilePath.c_str(),
|
||||
"Texture '%s' referenced in material but not found in the database search paths", fileName.c_str());
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// TODO: Separate load scene and save scene functions
|
||||
int exportScene(const CMeshUtilsSettings &settings)
|
||||
{
|
||||
CMeshUtilsContext context(settings);
|
||||
NLMISC::CFile::createDirectoryTree(settings.DestinationDirectoryPath);
|
||||
|
||||
if (!settings.ToolDependLog.empty())
|
||||
context.ToolLogger.initDepend(settings.ToolDependLog);
|
||||
if (!settings.ToolErrorLog.empty())
|
||||
context.ToolLogger.initError(settings.ToolErrorLog);
|
||||
context.ToolLogger.writeDepend(NLMISC::BUILD, "*", NLMISC::CPath::standardizePath(context.Settings.SourceFilePath, false).c_str()); // Base input file
|
||||
|
||||
// Apply database configuration
|
||||
CDatabaseConfig::init(settings.SourceFilePath);
|
||||
CDatabaseConfig::initTextureSearchDirectories();
|
||||
|
||||
Assimp::Importer importer;
|
||||
const aiScene *scene = importer.ReadFile(settings.SourceFilePath, 0
|
||||
| aiProcess_Triangulate
|
||||
| aiProcess_ValidateDataStructure
|
||||
| aiProcess_GenNormals // Or GenSmoothNormals? TODO: Validate smoothness between material boundaries!
|
||||
); // aiProcess_SplitLargeMeshes | aiProcess_LimitBoneWeights
|
||||
if (!scene)
|
||||
{
|
||||
const char *errs = importer.GetErrorString();
|
||||
if (errs) tlerror(context.ToolLogger, context.Settings.SourceFilePath.c_str(), "Assimp failed to load the scene: '%s'", errs);
|
||||
else tlerror(context.ToolLogger, context.Settings.SourceFilePath.c_str(), "Unable to load scene");
|
||||
return EXIT_FAILURE;
|
||||
}
|
||||
// aiProcess_Triangulate
|
||||
// aiProcess_ValidateDataStructure: TODO: Catch Assimp error output stream
|
||||
// aiProcess_RemoveRedundantMaterials: Not used because we may override materials with NeL Material from meta
|
||||
// aiProcess_ImproveCacheLocality: TODO: Verify this does not modify vertex indices
|
||||
//scene->mRootNode->mMetaData
|
||||
|
||||
context.InternalScene = scene;
|
||||
if (context.SceneMeta.load(context.Settings.SourceFilePath))
|
||||
context.ToolLogger.writeDepend(NLMISC::BUILD, "*", context.SceneMeta.metaFilePath().c_str()); // Meta input file
|
||||
|
||||
validateInternalNodeNames(context, context.InternalScene->mRootNode);
|
||||
|
||||
// -- SKEL FLAG --
|
||||
flagAssimpBones(context);
|
||||
flagMetaBones(context);
|
||||
flagExpandedBones(context);
|
||||
// TODO
|
||||
// [
|
||||
// Only necessary in TSkelLocal
|
||||
// For each shape test if all the bones have the same root bones for their skeleton
|
||||
// 1) Iterate each until a different is found
|
||||
// 2) When a different root is found, connect the two to the nearest common bone
|
||||
// ]
|
||||
// -- SKEL FLAG --
|
||||
|
||||
// First import materials
|
||||
assimpMaterials(context);
|
||||
|
||||
// Import shapes
|
||||
importShapes(context, context.InternalScene->mRootNode);
|
||||
|
||||
// Export shapes
|
||||
exportShapes(context);
|
||||
|
||||
return EXIT_SUCCESS;
|
||||
}
|
||||
|
||||
/* end of file */
|
@ -0,0 +1,44 @@
|
||||
// NeL - MMORPG Framework <http://dev.ryzom.com/projects/nel/>
|
||||
// Copyright (C) 2015 Winch Gate Property Limited
|
||||
// Author: Jan Boon <jan.boon@kaetemi.be>
|
||||
//
|
||||
// This program is free software: you can redistribute it and/or modify
|
||||
// it under the terms of the GNU Affero General Public License as
|
||||
// published by the Free Software Foundation, either version 3 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 Affero General Public License for more details.
|
||||
//
|
||||
// You should have received a copy of the GNU Affero General Public License
|
||||
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
#ifndef NL_MESH_UTILS_H
|
||||
#define NL_MESH_UTILS_H
|
||||
#include <nel/misc/types_nl.h>
|
||||
|
||||
#include <string>
|
||||
|
||||
struct CMeshUtilsSettings
|
||||
{
|
||||
CMeshUtilsSettings();
|
||||
|
||||
// Absolute Paths
|
||||
std::string SourceFilePath;
|
||||
std::string DestinationDirectoryPath;
|
||||
std::string ToolDependLog;
|
||||
std::string ToolErrorLog;
|
||||
|
||||
// Relative Directories
|
||||
/*std::string ShapeDirectory;
|
||||
std::string IGDirectory;
|
||||
std::string SkelDirectory;*/
|
||||
};
|
||||
|
||||
int exportScene(const CMeshUtilsSettings &settings);
|
||||
|
||||
#endif /* NL_MESH_UTILS_H */
|
||||
|
||||
/* end of file */
|
@ -0,0 +1,30 @@
|
||||
// NeL - MMORPG Framework <http://dev.ryzom.com/projects/nel/>
|
||||
// Copyright (C) 2015 Winch Gate Property Limited
|
||||
// Author: Jan Boon <jan.boon@kaetemi.be>
|
||||
//
|
||||
// This program is free software: you can redistribute it and/or modify
|
||||
// it under the terms of the GNU Affero General Public License as
|
||||
// published by the Free Software Foundation, either version 3 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 Affero General Public License for more details.
|
||||
//
|
||||
// You should have received a copy of the GNU Affero General Public License
|
||||
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
#include <nel/misc/types_nl.h>
|
||||
#include "scene_context.h"
|
||||
|
||||
#include <nel/misc/debug.h>
|
||||
#include <nel/misc/path.h>
|
||||
#include <nel/misc/tool_logger.h>
|
||||
|
||||
using namespace std;
|
||||
using namespace NLMISC;
|
||||
|
||||
void dummy_scene_context_cpp();
|
||||
|
||||
/* end of file */
|
@ -0,0 +1,81 @@
|
||||
// NeL - MMORPG Framework <http://dev.ryzom.com/projects/nel/>
|
||||
// Copyright (C) 2015 Winch Gate Property Limited
|
||||
// Author: Jan Boon <jan.boon@kaetemi.be>
|
||||
//
|
||||
// This program is free software: you can redistribute it and/or modify
|
||||
// it under the terms of the GNU Affero General Public License as
|
||||
// published by the Free Software Foundation, either version 3 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 Affero General Public License for more details.
|
||||
//
|
||||
// You should have received a copy of the GNU Affero General Public License
|
||||
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
#ifndef NL_SCENE_CONTEXT_H
|
||||
#define NL_SCENE_CONTEXT_H
|
||||
#include <nel/misc/types_nl.h>
|
||||
|
||||
#include "mesh_utils.h"
|
||||
#include "scene_meta.h"
|
||||
|
||||
#include <nel/misc/sstring.h>
|
||||
#include <nel/misc/tool_logger.h>
|
||||
#include <nel/misc/smart_ptr.h>
|
||||
#include <nel/misc/matrix.h>
|
||||
|
||||
#include <nel/3d/shape.h>
|
||||
|
||||
#ifndef NL_NODE_INTERNAL_TYPE
|
||||
#define NL_NODE_INTERNAL_TYPE void
|
||||
#endif
|
||||
#ifndef NL_SCENE_INTERNAL_TYPE
|
||||
#define NL_SCENE_INTERNAL_TYPE void
|
||||
#endif
|
||||
|
||||
namespace NL3D {
|
||||
class IShape;
|
||||
class CMaterial;
|
||||
}
|
||||
|
||||
struct CNodeContext
|
||||
{
|
||||
CNodeContext() :
|
||||
InternalNode(NULL),
|
||||
IsBone(false)
|
||||
{
|
||||
|
||||
}
|
||||
|
||||
const NL_NODE_INTERNAL_TYPE *InternalNode;
|
||||
bool IsBone;
|
||||
|
||||
// NLMISC::CMatrix Transform; // TODO
|
||||
NLMISC::CSmartPtr<NL3D::IShape> Shape;
|
||||
};
|
||||
|
||||
typedef std::map<NLMISC::CSString, CNodeContext> TNodeContextMap;
|
||||
struct CMeshUtilsContext
|
||||
{
|
||||
CMeshUtilsContext(const CMeshUtilsSettings &settings) : Settings(settings), InternalScene(NULL)
|
||||
{
|
||||
|
||||
}
|
||||
|
||||
const CMeshUtilsSettings &Settings;
|
||||
|
||||
NLMISC::CToolLogger ToolLogger;
|
||||
|
||||
const NL_SCENE_INTERNAL_TYPE *InternalScene;
|
||||
CSceneMeta SceneMeta;
|
||||
|
||||
TNodeContextMap Nodes; // Impl note: Should never end up containing the scene root node.
|
||||
// std::map<const aiMesh *, NLMISC::CSString> MeshNames; // Maps meshes to a node name ********************* todo ***************
|
||||
};
|
||||
|
||||
#endif /* NL_SCENE_CONTEXT_H */
|
||||
|
||||
/* end of file */
|
@ -0,0 +1,100 @@
|
||||
// NeL - MMORPG Framework <http://dev.ryzom.com/projects/nel/>
|
||||
// Copyright (C) 2015 Winch Gate Property Limited
|
||||
// Author: Jan Boon <jan.boon@kaetemi.be>
|
||||
//
|
||||
// This program is free software: you can redistribute it and/or modify
|
||||
// it under the terms of the GNU Affero General Public License as
|
||||
// published by the Free Software Foundation, either version 3 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 Affero General Public License for more details.
|
||||
//
|
||||
// You should have received a copy of the GNU Affero General Public License
|
||||
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
#include <nel/misc/types_nl.h>
|
||||
#include "scene_meta.h"
|
||||
|
||||
#include <nel/misc/debug.h>
|
||||
#include <nel/misc/stream.h>
|
||||
#include <nel/misc/file.h>
|
||||
|
||||
#include <nel/3d/material.h>
|
||||
|
||||
using namespace std;
|
||||
using namespace NLMISC;
|
||||
|
||||
CNodeMeta::CNodeMeta() :
|
||||
AddToIG(true),
|
||||
ExportMesh(TMeshShape),
|
||||
ExportBone(TBoneAuto),
|
||||
AutoAnim(false)
|
||||
{
|
||||
|
||||
}
|
||||
|
||||
void CNodeMeta::serial(NLMISC::IStream &s)
|
||||
{
|
||||
uint version = s.serialVersion(1);
|
||||
s.serial(AddToIG);
|
||||
s.serial((uint32 &)ExportMesh);
|
||||
s.serial((uint32 &)ExportBone);
|
||||
s.serial(InstanceShape);
|
||||
s.serial(InstanceName);
|
||||
s.serial(InstanceGroupName);
|
||||
s.serial(AutoAnim);
|
||||
}
|
||||
|
||||
CSceneMeta::CSceneMeta() :
|
||||
ImportShape(true),
|
||||
ImportSkel(true),
|
||||
ImportAnim(true),
|
||||
ImportCmb(true),
|
||||
ImportIG(true),
|
||||
ExportDefaultIG(false),
|
||||
SkeletonMode(TSkelRoot)
|
||||
{
|
||||
|
||||
}
|
||||
|
||||
bool CSceneMeta::load(const std::string &filePath)
|
||||
{
|
||||
m_MetaFilePath = NLMISC::CPath::standardizePath(filePath + ".nelmeta", false);
|
||||
if (CFile::fileExists(m_MetaFilePath))
|
||||
{
|
||||
CIFile f(m_MetaFilePath);
|
||||
serial(f);
|
||||
f.close();
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
void CSceneMeta::save()
|
||||
{
|
||||
COFile f(m_MetaFilePath, false, false, true);
|
||||
serial(f);
|
||||
f.close();
|
||||
}
|
||||
|
||||
void CSceneMeta::serial(NLMISC::IStream &s)
|
||||
{
|
||||
uint version = s.serialVersion(1);
|
||||
|
||||
s.serial(ImportShape);
|
||||
s.serial(ImportSkel);
|
||||
s.serial(ImportAnim);
|
||||
s.serial(ImportCmb);
|
||||
s.serial(ImportIG);
|
||||
|
||||
s.serial(ExportDefaultIG);
|
||||
s.serial((uint32 &)SkeletonMode);
|
||||
|
||||
s.serialCont(Nodes);
|
||||
s.serialPtrCont(Materials);
|
||||
}
|
||||
|
||||
/* end of file */
|
@ -0,0 +1,108 @@
|
||||
// NeL - MMORPG Framework <http://dev.ryzom.com/projects/nel/>
|
||||
// Copyright (C) 2015 Winch Gate Property Limited
|
||||
// Author: Jan Boon <jan.boon@kaetemi.be>
|
||||
//
|
||||
// This program is free software: you can redistribute it and/or modify
|
||||
// it under the terms of the GNU Affero General Public License as
|
||||
// published by the Free Software Foundation, either version 3 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 Affero General Public License for more details.
|
||||
//
|
||||
// You should have received a copy of the GNU Affero General Public License
|
||||
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||
|
||||
#ifndef NL_SCENE_META_H
|
||||
#define NL_SCENE_META_H
|
||||
#include <nel/misc/types_nl.h>
|
||||
|
||||
#include <nel/misc/sstring.h>
|
||||
#include <nel/misc/smart_ptr.h>
|
||||
|
||||
#include <nel/3d/material.h>
|
||||
|
||||
namespace NLMISC {
|
||||
class IStream;
|
||||
}
|
||||
|
||||
namespace NL3D {
|
||||
class CMaterial;
|
||||
}
|
||||
|
||||
enum TMesh
|
||||
{
|
||||
TMeshDisabled = 0,
|
||||
TMeshShape = 1,
|
||||
TMeshCollisionInt = 2,
|
||||
TMeshCollisionExt = 3,
|
||||
TMeshZone = 4,
|
||||
TMeshPortal = 5,
|
||||
TMeshCluster = 6,
|
||||
};
|
||||
|
||||
enum TBone
|
||||
{
|
||||
TBoneAuto = 0,
|
||||
TBoneForce = 1, // Force this node to be part of a skeleton
|
||||
TBoneRoot = 2, // Make this node the skeleton root, it will be exported using the scene name. There can only be one (editor should keep track and disable)
|
||||
};
|
||||
|
||||
struct CNodeMeta
|
||||
{
|
||||
CNodeMeta();
|
||||
|
||||
bool AddToIG; // Add this node to an instance group
|
||||
TMesh ExportMesh;
|
||||
TBone ExportBone;
|
||||
|
||||
std::string InstanceShape;
|
||||
std::string InstanceName;
|
||||
std::string InstanceGroupName;
|
||||
|
||||
bool AutoAnim;
|
||||
// std::vector<NLMISC::CSString> Materials; // In case there's an issue with nameless materials in some format... Map to material entirely in the meta editor.
|
||||
|
||||
void serial(NLMISC::IStream &s);
|
||||
};
|
||||
|
||||
enum TSkel
|
||||
{
|
||||
TSkelLocal = 0, // Export smallest skeleton possible from connected bones
|
||||
TSkelRoot = 1, // Export skeleton from a direct child node in the scene root node
|
||||
TSkelFull = 2, // Include all connected child nodes in the skeleton
|
||||
};
|
||||
|
||||
typedef std::map<NLMISC::CSString, NLMISC::CSmartPtr<NL3D::CMaterial> > TMaterialMap;
|
||||
struct CSceneMeta
|
||||
{
|
||||
CSceneMeta();
|
||||
|
||||
bool ImportShape;
|
||||
bool ImportSkel;
|
||||
bool ImportAnim;
|
||||
bool ImportCmb;
|
||||
bool ImportIG;
|
||||
|
||||
bool ExportDefaultIG; // Export a default instance group from nodes the scene that do not have an instance group set
|
||||
TSkel SkeletonMode;
|
||||
|
||||
std::map<NLMISC::CSString, CNodeMeta> Nodes;
|
||||
TMaterialMap Materials;
|
||||
|
||||
const std::string &metaFilePath() const { return m_MetaFilePath; }
|
||||
|
||||
bool load(const std::string &filePath);
|
||||
void save();
|
||||
void serial(NLMISC::IStream &s);
|
||||
|
||||
private:
|
||||
std::string m_MetaFilePath;
|
||||
|
||||
};
|
||||
|
||||
#endif NL_SCENE_META_H
|
||||
|
||||
/* end of file */
|
Loading…
Reference in New Issue