support VRMC_materials_mtoon and KHR_materials_unlit

This commit is contained in:
HifiExperiments 2024-04-14 16:23:45 -07:00
parent f1475e49ee
commit 8f27a4bf2b
11 changed files with 379 additions and 53 deletions

View file

@ -333,6 +333,14 @@ void AvatarDoctor::diagnoseTextures() {
addTextureToList(material.occlusionTexture);
addTextureToList(material.scatteringTexture);
addTextureToList(material.lightmapTexture);
if (material.isMToonMaterial) {
addTextureToList(material.shadeTexture);
addTextureToList(material.shadingShiftTexture);
addTextureToList(material.matcapTexture);
addTextureToList(material.rimTexture);
addTextureToList(material.uvAnimationTexture);
}
}
for (const auto& materialMapping : model->getMaterialMapping()) {

View file

@ -136,6 +136,14 @@ AvatarProject* AvatarProject::createAvatarProject(const QString& projectsFolder,
addTextureToList(material.occlusionTexture);
addTextureToList(material.scatteringTexture);
addTextureToList(material.lightmapTexture);
if (material.isMToonMaterial) {
addTextureToList(material.shadeTexture);
addTextureToList(material.shadingShiftTexture);
addTextureToList(material.matcapTexture);
addTextureToList(material.rimTexture);
addTextureToList(material.uvAnimationTexture);
}
}
QDir textureDir(textureFolder.isEmpty() ? fbxInfo.absoluteDir() : textureFolder);

View file

@ -269,19 +269,32 @@ void MaterialBaker::setMaterials(const QHash<QString, hfm::Material>& materials,
_materialResource = NetworkMaterialResourcePointer(new NetworkMaterialResource(), [](NetworkMaterialResource* ptr) { ptr->deleteLater(); });
for (auto& material : materials) {
_materialResource->parsedMaterials.names.push_back(material.name.toStdString());
_materialResource->parsedMaterials.networkMaterials[material.name.toStdString()] = std::make_shared<NetworkMaterial>(material, baseURL);
if (!material.isMToonMaterial) {
_materialResource->parsedMaterials.networkMaterials[material.name.toStdString()] = std::make_shared<NetworkMaterial>(material, baseURL);
} else {
_materialResource->parsedMaterials.networkMaterials[material.name.toStdString()] = std::make_shared<NetworkMToonMaterial>(material, baseURL);
}
// Store any embedded texture content
addTexture(material.name, image::TextureUsage::NORMAL_TEXTURE, material.normalTexture);
addTexture(material.name, image::TextureUsage::ALBEDO_TEXTURE, material.albedoTexture);
addTexture(material.name, image::TextureUsage::GLOSS_TEXTURE, material.glossTexture);
addTexture(material.name, image::TextureUsage::ROUGHNESS_TEXTURE, material.roughnessTexture);
addTexture(material.name, image::TextureUsage::SPECULAR_TEXTURE, material.specularTexture);
addTexture(material.name, image::TextureUsage::METALLIC_TEXTURE, material.metallicTexture);
addTexture(material.name, image::TextureUsage::EMISSIVE_TEXTURE, material.emissiveTexture);
addTexture(material.name, image::TextureUsage::OCCLUSION_TEXTURE, material.occlusionTexture);
addTexture(material.name, image::TextureUsage::SCATTERING_TEXTURE, material.scatteringTexture);
addTexture(material.name, image::TextureUsage::LIGHTMAP_TEXTURE, material.lightmapTexture);
if (!material.isMToonMaterial) {
addTexture(material.name, image::TextureUsage::GLOSS_TEXTURE, material.glossTexture);
addTexture(material.name, image::TextureUsage::ROUGHNESS_TEXTURE, material.roughnessTexture);
addTexture(material.name, image::TextureUsage::SPECULAR_TEXTURE, material.specularTexture);
addTexture(material.name, image::TextureUsage::METALLIC_TEXTURE, material.metallicTexture);
addTexture(material.name, image::TextureUsage::OCCLUSION_TEXTURE, material.occlusionTexture);
addTexture(material.name, image::TextureUsage::SCATTERING_TEXTURE, material.scatteringTexture);
addTexture(material.name, image::TextureUsage::LIGHTMAP_TEXTURE, material.lightmapTexture);
} else {
addTexture(material.name, image::TextureUsage::ALBEDO_TEXTURE, material.shadeTexture);
addTexture(material.name, image::TextureUsage::ROUGHNESS_TEXTURE, material.shadingShiftTexture);
addTexture(material.name, image::TextureUsage::EMISSIVE_TEXTURE, material.matcapTexture);
addTexture(material.name, image::TextureUsage::ALBEDO_TEXTURE, material.rimTexture);
addTexture(material.name, image::TextureUsage::ROUGHNESS_TEXTURE, material.uvAnimationTexture);
}
}
}

View file

@ -478,10 +478,9 @@ public:
protected:
std::string _name { "" };
mutable MaterialKey _key { 0 };
private:
std::string _model { HIFI_PBR };
private:
// Material properties
glm::vec3 _emissive { DEFAULT_EMISSIVE };
float _opacity { DEFAULT_OPACITY };

View file

@ -47,6 +47,24 @@ void HFMMaterial::getTextureNames(QSet<QString>& textureList) const {
if (!lightmapTexture.isNull()) {
textureList.insert(lightmapTexture.name);
}
if (isMToonMaterial) {
if (!shadeTexture.isNull()) {
textureList.insert(shadeTexture.name);
}
if (!shadingShiftTexture.isNull()) {
textureList.insert(shadingShiftTexture.name);
}
if (!matcapTexture.isNull()) {
textureList.insert(matcapTexture.name);
}
if (!rimTexture.isNull()) {
textureList.insert(rimTexture.name);
}
if (!uvAnimationTexture.isNull()) {
textureList.insert(uvAnimationTexture.name);
}
}
}
void HFMMaterial::setMaxNumPixelsPerTexture(int maxNumPixels) {
@ -61,6 +79,12 @@ void HFMMaterial::setMaxNumPixelsPerTexture(int maxNumPixels) {
occlusionTexture.maxNumPixels = maxNumPixels;
scatteringTexture.maxNumPixels = maxNumPixels;
lightmapTexture.maxNumPixels = maxNumPixels;
shadeTexture.maxNumPixels = maxNumPixels;
shadingShiftTexture.maxNumPixels = maxNumPixels;
matcapTexture.maxNumPixels = maxNumPixels;
rimTexture.maxNumPixels = maxNumPixels;
uvAnimationTexture.maxNumPixels = maxNumPixels;
}
bool HFMMaterial::needTangentSpace() const {
@ -312,6 +336,13 @@ void HFMModel::debugDump() {
qCDebug(modelformat) << " useMetallicMap =" << mat.useMetallicMap;
qCDebug(modelformat) << " useEmissiveMap =" << mat.useEmissiveMap;
qCDebug(modelformat) << " useOcclusionMap =" << mat.useOcclusionMap;
qCDebug(modelformat) << " isMToonMaterial =" << mat.isMToonMaterial;
qCDebug(modelformat) << " shadeTexture =" << mat.shadeTexture.filename;
qCDebug(modelformat) << " shadingShiftTexture =" << mat.shadingShiftTexture.filename;
qCDebug(modelformat) << " matcapTexture =" << mat.matcapTexture.filename;
qCDebug(modelformat) << " rimTexture =" << mat.rimTexture.filename;
qCDebug(modelformat) << " uvAnimationTexture =" << mat.uvAnimationTexture.filename;
}
qCDebug(modelformat) << "---------------- Joints ----------------";

View file

@ -224,6 +224,14 @@ public:
bool useEmissiveMap { false };
bool useOcclusionMap { false };
bool isMToonMaterial { false };
Texture shadeTexture;
Texture shadingShiftTexture;
Texture matcapTexture;
Texture rimTexture;
Texture uvAnimationTexture;
bool needTangentSpace() const;
};

View file

@ -325,7 +325,11 @@ void GeometryResource::setGeometryDefinition(HFMModel::Pointer hfmModel, const M
QHash<QString, size_t> materialIDAtlas;
for (const HFMMaterial& material : _hfmModel->materials) {
materialIDAtlas[material.materialID] = _materials.size();
_materials.push_back(std::make_shared<NetworkMaterial>(material, _textureBaseURL));
if (!material.isMToonMaterial) {
_materials.push_back(std::make_shared<NetworkMaterial>(material, _textureBaseURL));
} else {
_materials.push_back(std::make_shared<NetworkMToonMaterial>(material, _textureBaseURL));
}
}
std::shared_ptr<GeometryMeshes> meshes = std::make_shared<GeometryMeshes>();
@ -357,7 +361,11 @@ void GeometryResource::setTextures() {
if (_hfmModel) {
if (DependencyManager::get<TextureCache>()) {
for (const HFMMaterial& material : _hfmModel->materials) {
_materials.push_back(std::make_shared<NetworkMaterial>(material, _textureBaseURL));
if (!material.isMToonMaterial) {
_materials.push_back(std::make_shared<NetworkMaterial>(material, _textureBaseURL));
} else {
_materials.push_back(std::make_shared<NetworkMToonMaterial>(material, _textureBaseURL));
}
}
} else {
qDebug() << "GeometryResource::setTextures: TextureCache dependency not available, skipping textures";
@ -436,7 +444,11 @@ Geometry::Geometry(const Geometry& geometry) {
_materials.reserve(geometry._materials.size());
for (const auto& material : geometry._materials) {
_materials.push_back(std::make_shared<NetworkMaterial>(*material));
if (!material->isMToon()) {
_materials.push_back(std::make_shared<NetworkMaterial>(*material));
} else if (auto mToonMaterial = std::static_pointer_cast<NetworkMToonMaterial>(material)) {
_materials.push_back(std::make_shared<NetworkMToonMaterial>(*mToonMaterial));
}
}
_animGraphOverrideUrl = geometry._animGraphOverrideUrl;
@ -452,9 +464,13 @@ void Geometry::setTextures(const QVariantMap& textureMap) {
// FIXME: The Model currently caches the materials (waste of space!)
// so they must be copied in the Geometry copy-ctor
// if (material->isOriginal()) {
//if (material->isOriginal()) {
// // Copy the material to avoid mutating the cached version
// material = std::make_shared<NetworkMaterial>(*material);
// if (!material->isMToon()) {
// material = std::make_shared<NetworkMaterial>(*material);
// } else {
// material = std::make_shared<NetworkMToonMaterial>(*material);
// }
//}
material->setTextures(textureMap);

View file

@ -1,7 +1,7 @@
set(TARGET_NAME model-serializers)
setup_hifi_library()
link_hifi_libraries(shared graphics networking image hfm)
link_hifi_libraries(shared graphics networking image hfm procedural material-networking ktx shaders)
include_hifi_library_headers(gpu image)
target_draco()

View file

@ -38,6 +38,7 @@
#include <PathUtils.h>
#include <image/ColorChannel.h>
#include <BlendshapeConstants.h>
#include <procedural/ProceduralMaterialCache.h>
#include "FBXSerializer.h"
@ -109,7 +110,7 @@ bool GLTFSerializer::getSkinInverseBindMatrices(std::vector<std::vector<float>>&
if (matricesAccessor.type != cgltf_type_mat4) {
return false;
}
matrices.resize(matricesAccessor.count * 16);
matrices.resize((int)matricesAccessor.count * 16);
size_t numFloats = cgltf_accessor_unpack_floats(&matricesAccessor, matrices.data(), matricesAccessor.count * 16);
Q_ASSERT(numFloats == matricesAccessor.count * 16);
inverseBindMatrixValues.push_back(std::vector<float>(matrices.begin(), matrices.end()));
@ -125,7 +126,7 @@ bool GLTFSerializer::generateTargetData(cgltf_accessor *accessor, float weight,
if (accessor->type != cgltf_type_vec3) {
return false;
}
storedValues.resize(accessor->count * 3);
storedValues.resize((int)accessor->count * 3);
size_t numFloats = cgltf_accessor_unpack_floats(accessor, storedValues.data(), accessor->count * 3);
if (numFloats != accessor->count * 3) {
return false;
@ -147,8 +148,8 @@ bool findNodeInPointerArray(const cgltf_node *nodePointer, cgltf_node **nodes, s
return false;
}
template<typename T> bool findPointerInArray(const T *pointer, const T *array, size_t arraySize, size_t &index) {
for (size_t i = 0; i < arraySize; i++) {
template<typename T> bool findPointerInArray(const T *pointer, const T *array, size_t arraySize, int &index) {
for (int i = 0; i < arraySize; i++) {
if (&array[i] == pointer) {
index = i;
return true;
@ -175,18 +176,18 @@ bool findAttribute(const QString &name, const cgltf_attribute *attributes, size_
bool GLTFSerializer::buildGeometry(HFMModel& hfmModel, const hifi::VariantHash& mapping, const hifi::URL& url) {
hfmModel.originalURL = url.toString();
size_t numNodes = _data->nodes_count;
int numNodes = (int)_data->nodes_count;
//Build dependencies
QVector<int> parents;
QVector<int> sortedNodes;
parents.fill(-1, numNodes);
sortedNodes.reserve(numNodes);
for(size_t index = 0; index < numNodes; index++) {
for(int index = 0; index < numNodes; index++) {
auto &node = _data->nodes[index];
for(size_t childIndexInParent = 0; childIndexInParent < node.children_count; childIndexInParent++) {
cgltf_node *child = node.children[childIndexInParent];
size_t childIndex = 0;
int childIndex = 0;
if (!findPointerInArray(child, _data->nodes, _data->nodes_count, childIndex)) {
qDebug(modelformat) << "findPointerInArray failed for model: " << _url;
hfmModel.loadErrorCount++;
@ -201,7 +202,7 @@ bool GLTFSerializer::buildGeometry(HFMModel& hfmModel, const hifi::VariantHash&
typedef QVector<glm::mat4> NodeTransforms;
QVector<NodeTransforms> transforms;
transforms.resize(numNodes);
for (size_t index = 0; index < numNodes; index++) {
for (int index = 0; index < numNodes; index++) {
// collect node transform
auto &node = _data->nodes[index];
transforms[index].push_back(getModelTransform(node));
@ -219,7 +220,7 @@ bool GLTFSerializer::buildGeometry(HFMModel& hfmModel, const hifi::VariantHash&
QVector<bool> hasBeenSorted;
hasBeenSorted.fill(false, numNodes);
{
size_t i = 0; // initial index
int i = 0; // initial index
while (i < numNodes) {
int currentNode = sortedNodes[i];
int parentIndex = parents[currentNode];
@ -227,7 +228,7 @@ bool GLTFSerializer::buildGeometry(HFMModel& hfmModel, const hifi::VariantHash&
hasBeenSorted[currentNode] = true;
++i;
} else {
size_t j = i + 1; // index of node to be sorted
int j = i + 1; // index of node to be sorted
while (j < numNodes) {
int nextNode = sortedNodes[j];
parentIndex = parents[nextNode];
@ -249,7 +250,7 @@ bool GLTFSerializer::buildGeometry(HFMModel& hfmModel, const hifi::VariantHash&
// Build map from original to new indices
QVector<int> originalToNewNodeIndexMap;
originalToNewNodeIndexMap.fill(-1, numNodes);
for (size_t i = 0; i < numNodes; ++i) {
for (int i = 0; i < numNodes; ++i) {
originalToNewNodeIndexMap[sortedNodes[i]] = i;
}
@ -311,7 +312,7 @@ bool GLTFSerializer::buildGeometry(HFMModel& hfmModel, const hifi::VariantHash&
return false;
}
for (size_t jointIndex = 0; jointIndex < numNodes; ++jointIndex) {
for (int jointIndex = 0; jointIndex < numNodes; ++jointIndex) {
int nodeIndex = sortedNodes[jointIndex];
auto joint = hfmModel.joints[jointIndex];
@ -365,18 +366,17 @@ bool GLTFSerializer::buildGeometry(HFMModel& hfmModel, const hifi::VariantHash&
materialIDs.push_back(mid);
}
for (size_t i = 0; i < (size_t)materialIDs.size(); ++i) {
for (int i = 0; i < materialIDs.size(); ++i) {
QString& matid = materialIDs[i];
hfmModel.materials[matid] = HFMMaterial();
HFMMaterial& hfmMaterial = hfmModel.materials[matid];
hfmMaterial._material = std::make_shared<graphics::Material>();
hfmMaterial.name = hfmMaterial.materialID = matid;
setHFMMaterial(hfmMaterial, _data->materials[i]);
}
// Build meshes
size_t nodeCount = 0;
int nodeCount = 0;
hfmModel.meshExtents.reset();
for (int nodeIndex : sortedNodes) {
auto& node = _data->nodes[nodeIndex];
@ -394,7 +394,7 @@ bool GLTFSerializer::buildGeometry(HFMModel& hfmModel, const hifi::VariantHash&
cluster.inverseBindTransform = Transform(cluster.inverseBindMatrix);
mesh.clusters.append(cluster);
} else { // skinned model
for (size_t j = 0; j < numNodes; ++j) {
for (int j = 0; j < numNodes; ++j) {
HFMCluster cluster;
cluster.jointIndex = j;
cluster.inverseBindMatrix = jointInverseBindTransforms[j];
@ -450,7 +450,7 @@ bool GLTFSerializer::buildGeometry(HFMModel& hfmModel, const hifi::VariantHash&
QVector<float> weights;
int weightStride = 4;
indices.resize(indicesAccessor->count);
indices.resize((int)indicesAccessor->count);
size_t readIndicesCount = cgltf_accessor_unpack_indices(indicesAccessor, indices.data(), sizeof(unsigned int), indicesAccessor->count);
if (readIndicesCount != indicesAccessor->count) {
@ -480,6 +480,7 @@ bool GLTFSerializer::buildGeometry(HFMModel& hfmModel, const hifi::VariantHash&
return false;
}
auto accessor = primitive.attributes[attributeIndex].data;
int accessorCount = (int)accessor->count;
if (key == "POSITION") {
if (accessor->type != cgltf_type_vec3) {
@ -488,7 +489,7 @@ bool GLTFSerializer::buildGeometry(HFMModel& hfmModel, const hifi::VariantHash&
continue;
}
vertices.resize(accessor->count * 3);
vertices.resize(accessorCount * 3);
size_t floatCount = cgltf_accessor_unpack_floats(accessor, vertices.data(), accessor->count * 3);
if (floatCount != accessor->count * 3) {
qWarning(modelformat) << "There was a problem reading glTF POSITION data for model " << _url;
@ -502,7 +503,7 @@ bool GLTFSerializer::buildGeometry(HFMModel& hfmModel, const hifi::VariantHash&
continue;
}
normals.resize(accessor->count * 3);
normals.resize(accessorCount * 3);
size_t floatCount = cgltf_accessor_unpack_floats(accessor, normals.data(), accessor->count * 3);
if (floatCount != accessor->count * 3) {
qWarning(modelformat) << "There was a problem reading glTF NORMAL data for model " << _url;
@ -520,7 +521,7 @@ bool GLTFSerializer::buildGeometry(HFMModel& hfmModel, const hifi::VariantHash&
continue;
}
tangents.resize(accessor->count * tangentStride);
tangents.resize(accessorCount * tangentStride);
size_t floatCount = cgltf_accessor_unpack_floats(accessor, tangents.data(), accessor->count * tangentStride);
if (floatCount != accessor->count * tangentStride) {
qWarning(modelformat) << "There was a problem reading glTF TANGENT data for model " << _url;
@ -535,7 +536,7 @@ bool GLTFSerializer::buildGeometry(HFMModel& hfmModel, const hifi::VariantHash&
continue;
}
texcoords.resize(accessor->count * 2);
texcoords.resize(accessorCount * 2);
size_t floatCount = cgltf_accessor_unpack_floats(accessor, texcoords.data(), accessor->count * 2);
if (floatCount != accessor->count * 2) {
qWarning(modelformat) << "There was a problem reading glTF TEXCOORD_0 data for model " << _url;
@ -549,7 +550,7 @@ bool GLTFSerializer::buildGeometry(HFMModel& hfmModel, const hifi::VariantHash&
continue;
}
texcoords2.resize(accessor->count * 2);
texcoords2.resize(accessorCount * 2);
size_t floatCount = cgltf_accessor_unpack_floats(accessor, texcoords2.data(), accessor->count * 2);
if (floatCount != accessor->count * 2) {
qWarning(modelformat) << "There was a problem reading glTF TEXCOORD_1 data for model " << _url;
@ -567,7 +568,7 @@ bool GLTFSerializer::buildGeometry(HFMModel& hfmModel, const hifi::VariantHash&
continue;
}
colors.resize(accessor->count * colorStride);
colors.resize(accessorCount * colorStride);
size_t floatCount = cgltf_accessor_unpack_floats(accessor, colors.data(), accessor->count * colorStride);
if (floatCount != accessor->count * colorStride) {
qWarning(modelformat) << "There was a problem reading glTF COLOR_0 data for model " << _url;
@ -589,12 +590,12 @@ bool GLTFSerializer::buildGeometry(HFMModel& hfmModel, const hifi::VariantHash&
continue;
}
joints.resize(accessor->count * jointStride);
joints.resize(accessorCount * jointStride);
cgltf_uint jointIndices[4];
for (size_t i = 0; i < accessor->count; i++) {
cgltf_accessor_read_uint(accessor, i, jointIndices, jointStride);
for (int component = 0; component < jointStride; component++) {
joints[i * jointStride + component] = (uint16_t)jointIndices[component];
joints[(int)i * jointStride + component] = (uint16_t)jointIndices[component];
}
}
@ -613,7 +614,7 @@ bool GLTFSerializer::buildGeometry(HFMModel& hfmModel, const hifi::VariantHash&
continue;
}
weights.resize(accessor->count * weightStride);
weights.resize(accessorCount * weightStride);
size_t floatCount = cgltf_accessor_unpack_floats(accessor, weights.data(), accessor->count * weightStride);
if (floatCount != accessor->count * weightStride) {
qWarning(modelformat) << "There was a problem reading glTF WEIGHTS_0 data for model " << _url;
@ -917,7 +918,7 @@ bool GLTFSerializer::buildGeometry(HFMModel& hfmModel, const hifi::VariantHash&
continue;
}
size_t jointIndex = 0;
int jointIndex = 0;
if (!findPointerInArray(node.skin->joints[clusterJoints[c]], _data->nodes, _data->nodes_count, jointIndex)) {
qCWarning(modelformat) << "Cannot find the joint " << node.skin->joints[clusterJoints[c]]->name <<" in joint array";
hfmModel.loadErrorCount++;
@ -960,7 +961,7 @@ bool GLTFSerializer::buildGeometry(HFMModel& hfmModel, const hifi::VariantHash&
}
}
size_t materialIndex = 0;
int materialIndex = 0;
if (primitive.material != nullptr && !findPointerInArray(primitive.material, _data->materials, _data->materials_count, materialIndex)) {
qCWarning(modelformat) << "GLTFSerializer::buildGeometry: Invalid material pointer";
hfmModel.loadErrorCount++;
@ -1012,7 +1013,7 @@ bool GLTFSerializer::buildGeometry(HFMModel& hfmModel, const hifi::VariantHash&
// If an FST isn't being used and the model is likely from ReadyPlayerMe, add blendshape synonyms.
QVector<QString> fileTargetNames;
fileTargetNames.reserve(node.mesh->target_names_count);
fileTargetNames.reserve((int)node.mesh->target_names_count);
for (size_t i = 0; i < node.mesh->target_names_count; i++) {
fileTargetNames.push_back(QString(node.mesh->target_names[i]));
}
@ -1048,7 +1049,7 @@ bool GLTFSerializer::buildGeometry(HFMModel& hfmModel, const hifi::VariantHash&
auto keys = blendshapeIndices.keys();
auto values = blendshapeIndices.values();
QVector<QString> names;
names.reserve(node.mesh->target_names_count);
names.reserve((int)node.mesh->target_names_count);
for (size_t i = 0; i < node.mesh->target_names_count; i++) {
names.push_back(QString(node.mesh->target_names[i]));
}
@ -1303,9 +1304,9 @@ HFMTexture GLTFSerializer::getHFMTexture(const cgltf_texture *texture) {
cgltf_buffer_view *bufferView = image->buffer_view;
size_t offset = bufferView->offset;
size_t length = bufferView->size;
int length = (int)bufferView->size;
size_t imageIndex = 0;
int imageIndex = 0;
if (!findPointerInArray(image, _data->images, _data->images_count, imageIndex)) {
// This should never happen. It would mean a bug in cgltf library.
qDebug(modelformat) << "GLTFSerializer::getHFMTexture: can't find texture in the array";
@ -1328,6 +1329,113 @@ HFMTexture GLTFSerializer::getHFMTexture(const cgltf_texture *texture) {
}
void GLTFSerializer::setHFMMaterial(HFMMaterial& hfmMat, const cgltf_material& material) {
hfmMat._material = std::make_shared<graphics::Material>();
for (size_t i = 0; i < material.extensions_count; i++) {
auto& extension = material.extensions[i];
if (extension.name != nullptr) {
if (strcmp(extension.name, "VRMC_materials_mtoon") == 0 && extension.data != nullptr) {
hfmMat.isMToonMaterial = true;
auto mToonMaterial = std::make_shared<NetworkMToonMaterial>();
QJsonDocument mToonExtension = QJsonDocument::fromJson(extension.data);
if (!mToonExtension.isNull()) {
if (mToonExtension["shadeColorFactor"].isArray()) {
auto array = mToonExtension["shadeColorFactor"].toArray();
glm::vec3 shadeLinear = glm::vec3(array[0].toDouble(), array[1].toDouble(), array[2].toDouble());
glm::vec3 shade = ColorUtils::tosRGBVec3(shadeLinear);
mToonMaterial->setShade(shade);
}
if (mToonExtension["shadeMultiplyTexture"].isObject()) {
QJsonObject object = mToonExtension["shadeMultiplyTexture"].toObject();
if (object["index"].isDouble() && object["index"].toInt() < _data->textures_count) {
hfmMat.shadeTexture = getHFMTexture(&_data->textures[object["index"].toInt()]);
}
}
if (mToonExtension["shadingShiftFactor"].isDouble()) {
mToonMaterial->setShadingShift(mToonExtension["shadingShiftFactor"].toDouble());
}
if (mToonExtension["shadingShiftTexture"].isObject()) {
QJsonObject object = mToonExtension["shadingShiftTexture"].toObject();
if (object["index"].isDouble() && object["index"].toInt() < _data->textures_count) {
hfmMat.shadingShiftTexture = getHFMTexture(&_data->textures[object["index"].toInt()]);
}
}
if (mToonExtension["shadingToonyFactor"].isDouble()) {
mToonMaterial->setShadingToony(mToonExtension["shadingToonyFactor"].toDouble());
}
if (mToonExtension["matcapFactor"].isArray()) {
auto array = mToonExtension["matcapFactor"].toArray();
glm::vec3 matcapLinear = glm::vec3(array[0].toDouble(), array[1].toDouble(), array[2].toDouble());
glm::vec3 matcap = ColorUtils::tosRGBVec3(matcapLinear);
mToonMaterial->setMatcap(matcap);
}
if (mToonExtension["matcapTexture"].isObject()) {
QJsonObject object = mToonExtension["matcapTexture"].toObject();
if (object["index"].isDouble() && object["index"].toInt() < _data->textures_count) {
hfmMat.matcapTexture = getHFMTexture(&_data->textures[object["index"].toInt()]);
}
}
if (mToonExtension["parametricRimColorFactor"].isArray()) {
auto array = mToonExtension["parametricRimColorFactor"].toArray();
glm::vec3 parametricRimLinear = glm::vec3(array[0].toDouble(), array[1].toDouble(), array[2].toDouble());
glm::vec3 parametricRim = ColorUtils::tosRGBVec3(parametricRimLinear);
mToonMaterial->setParametricRim(parametricRim);
}
if (mToonExtension["parametricRimFresnelPowerFactor"].isDouble()) {
mToonMaterial->setParametricRimFresnelPower(mToonExtension["parametricRimFresnelPowerFactor"].toDouble());
}
if (mToonExtension["parametricRimLiftFactor"].isDouble()) {
mToonMaterial->setParametricRimLift(mToonExtension["parametricRimLiftFactor"].toDouble());
}
if (mToonExtension["rimMultiplyTexture"].isObject()) {
QJsonObject object = mToonExtension["rimMultiplyTexture"].toObject();
if (object["index"].isDouble() && object["index"].toInt() < _data->textures_count) {
hfmMat.rimTexture = getHFMTexture(&_data->textures[object["index"].toInt()]);
}
}
if (mToonExtension["rimLightingMixFactor"].isDouble()) {
mToonMaterial->setRimLightingMix(mToonExtension["rimLightingMixFactor"].toDouble());
}
// FIXME: Outlines are currently disabled because they're buggy
//if (mToonExtension["outlineWidthMode"].isString()) {
// QString outlineWidthMode = mToonExtension["outlineWidthMode"].toString();
// if (outlineWidthMode == "none") {
// mToonMaterial->setOutlineWidthMode(NetworkMToonMaterial::OutlineWidthMode::OUTLINE_NONE);
// } else if (outlineWidthMode == "worldCoordinates") {
// mToonMaterial->setOutlineWidthMode(NetworkMToonMaterial::OutlineWidthMode::OUTLINE_WORLD);
// } else if (outlineWidthMode == "screenCoordinates") {
// mToonMaterial->setOutlineWidthMode(NetworkMToonMaterial::OutlineWidthMode::OUTLINE_SCREEN);
// }
//}
if (mToonExtension["outlineWidthFactor"].isDouble()) {
mToonMaterial->setOutlineWidth(mToonExtension["outlineWidthFactor"].toDouble());
}
if (mToonExtension["outlineColorFactor"].isArray()) {
auto array = mToonExtension["outlineColorFactor"].toArray();
glm::vec3 outlineLinear = glm::vec3(array[0].toDouble(), array[1].toDouble(), array[2].toDouble());
glm::vec3 outline = ColorUtils::tosRGBVec3(outlineLinear);
mToonMaterial->setOutline(outline);
}
if (mToonExtension["uvAnimationMaskTexture"].isObject()) {
QJsonObject object = mToonExtension["uvAnimationMaskTexture"].toObject();
if (object["index"].isDouble() && object["index"].toInt() < _data->textures_count) {
hfmMat.uvAnimationTexture = getHFMTexture(&_data->textures[object["index"].toInt()]);
}
}
if (mToonExtension["uvAnimationScrollXSpeedFactor"].isDouble()) {
mToonMaterial->setUVAnimationScrollXSpeed(mToonExtension["uvAnimationScrollXSpeedFactor"].toDouble());
}
if (mToonExtension["uvAnimationScrollYSpeedFactor"].isDouble()) {
mToonMaterial->setUVAnimationScrollYSpeed(mToonExtension["uvAnimationScrollYSpeedFactor"].toDouble());
}
if (mToonExtension["uvAnimationRotationSpeedFactor"].isDouble()) {
mToonMaterial->setUVAnimationRotationSpeed(mToonExtension["uvAnimationRotationSpeedFactor"].toDouble());
}
}
hfmMat._material = mToonMaterial;
}
}
}
if (material.alpha_mode == cgltf_alpha_mode_opaque) {
hfmMat._material->setOpacityMapMode(graphics::MaterialKey::OPACITY_MAP_OPAQUE);
} else if (material.alpha_mode == cgltf_alpha_mode_mask) {
@ -1340,6 +1448,11 @@ void GLTFSerializer::setHFMMaterial(HFMMaterial& hfmMat, const cgltf_material& m
hfmMat._material->setOpacityCutoff(material.alpha_cutoff);
// VRMC_materials_mtoon takes precedence over KHR_materials_unlit
if (!hfmMat.isMToonMaterial) {
hfmMat._material->setUnlit(material.unlit);
}
if (material.double_sided) {
hfmMat._material->setCullFaceMode(graphics::MaterialKey::CullFaceMode::CULL_NONE);
}

View file

@ -1134,6 +1134,65 @@ bool NetworkMaterial::checkResetOpacityMap() {
return false;
}
NetworkMToonMaterial::NetworkMToonMaterial(const HFMMaterial& material, const QUrl& textureBaseUrl) :
NetworkMaterial(*material._material)
{
_name = material.name.toStdString();
if (!material.albedoTexture.filename.isEmpty()) {
auto map = fetchTextureMap(textureBaseUrl, material.albedoTexture, image::TextureUsage::ALBEDO_TEXTURE, MapChannel::ALBEDO_MAP);
if (map) {
_albedoTransform = material.albedoTexture.transform;
map->setTextureTransform(_albedoTransform);
if (!material.opacityTexture.filename.isEmpty()) {
if (material.albedoTexture.filename == material.opacityTexture.filename) {
// Best case scenario, just indicating that the albedo map contains transparency
// TODO: Different albedo/opacity maps are not currently supported
map->setUseAlphaChannel(true);
}
}
}
setTextureMap(MapChannel::ALBEDO_MAP, map);
}
if (!material.normalTexture.filename.isEmpty()) {
auto type = (material.normalTexture.isBumpmap ? image::TextureUsage::BUMP_TEXTURE : image::TextureUsage::NORMAL_TEXTURE);
auto map = fetchTextureMap(textureBaseUrl, material.normalTexture, type, MapChannel::NORMAL_MAP);
setTextureMap(MapChannel::NORMAL_MAP, map);
}
if (!material.emissiveTexture.filename.isEmpty()) {
auto map = fetchTextureMap(textureBaseUrl, material.emissiveTexture, image::TextureUsage::EMISSIVE_TEXTURE, MapChannel::EMISSIVE_MAP);
setTextureMap(MapChannel::EMISSIVE_MAP, map);
}
if (!material.shadeTexture.filename.isEmpty()) {
auto map = fetchTextureMap(textureBaseUrl, material.shadeTexture, image::TextureUsage::ALBEDO_TEXTURE, (MapChannel)MToonMapChannel::SHADE_MAP);
setTextureMap((MapChannel)MToonMapChannel::SHADE_MAP, map);
}
if (!material.shadingShiftTexture.filename.isEmpty()) {
auto map = fetchTextureMap(textureBaseUrl, material.shadingShiftTexture, image::TextureUsage::ROUGHNESS_TEXTURE, (MapChannel)MToonMapChannel::SHADING_SHIFT_MAP);
setTextureMap((MapChannel)MToonMapChannel::SHADING_SHIFT_MAP, map);
}
if (!material.matcapTexture.filename.isEmpty()) {
auto map = fetchTextureMap(textureBaseUrl, material.matcapTexture, image::TextureUsage::EMISSIVE_TEXTURE, (MapChannel)MToonMapChannel::MATCAP_MAP);
setTextureMap((MapChannel)MToonMapChannel::MATCAP_MAP, map);
}
if (!material.rimTexture.filename.isEmpty()) {
auto map = fetchTextureMap(textureBaseUrl, material.rimTexture, image::TextureUsage::ALBEDO_TEXTURE, (MapChannel)MToonMapChannel::RIM_MAP);
setTextureMap((MapChannel)MToonMapChannel::RIM_MAP, map);
}
if (!material.uvAnimationTexture.filename.isEmpty()) {
auto map = fetchTextureMap(textureBaseUrl, material.uvAnimationTexture, image::TextureUsage::ROUGHNESS_TEXTURE, (MapChannel)MToonMapChannel::UV_ANIMATION_MASK_MAP);
setTextureMap((MapChannel)MToonMapChannel::UV_ANIMATION_MASK_MAP, map);
}
}
NetworkMToonMaterial::NetworkMToonMaterial(const NetworkMToonMaterial& material) :
NetworkMaterial(material),
_shade(material._shade),
@ -1152,6 +1211,72 @@ NetworkMToonMaterial::NetworkMToonMaterial(const NetworkMToonMaterial& material)
_outline(material._outline)
{}
void NetworkMToonMaterial::setTextures(const QVariantMap& textureMap) {
_isOriginal = false;
const auto& albedoName = getTextureName(MapChannel::ALBEDO_MAP);
const auto& normalName = getTextureName(MapChannel::NORMAL_MAP);
const auto& emissiveName = getTextureName(MapChannel::EMISSIVE_MAP);
const auto& shadeName = getTextureName((MapChannel)MToonMapChannel::SHADE_MAP);
const auto& shadingShiftName = getTextureName((MapChannel)MToonMapChannel::SHADING_SHIFT_MAP);
const auto& matcapName = getTextureName((MapChannel)MToonMapChannel::MATCAP_MAP);
const auto& rimName = getTextureName((MapChannel)MToonMapChannel::RIM_MAP);
const auto& uvAnimationMaskName = getTextureName((MapChannel)MToonMapChannel::UV_ANIMATION_MASK_MAP);
if (!albedoName.isEmpty()) {
auto url = textureMap.contains(albedoName) ? textureMap[albedoName].toUrl() : QUrl();
auto map = fetchTextureMap(url, image::TextureUsage::ALBEDO_TEXTURE, MapChannel::ALBEDO_MAP);
if (map) {
map->setTextureTransform(_albedoTransform);
// when reassigning the albedo texture we also check for the alpha channel used as opacity
map->setUseAlphaChannel(true);
}
setTextureMap(MapChannel::ALBEDO_MAP, map);
}
if (!normalName.isEmpty()) {
auto url = textureMap.contains(normalName) ? textureMap[normalName].toUrl() : QUrl();
auto map = fetchTextureMap(url, image::TextureUsage::NORMAL_TEXTURE, MapChannel::NORMAL_MAP);
setTextureMap(MapChannel::NORMAL_MAP, map);
}
if (!emissiveName.isEmpty()) {
auto url = textureMap.contains(emissiveName) ? textureMap[emissiveName].toUrl() : QUrl();
auto map = fetchTextureMap(url, image::TextureUsage::EMISSIVE_TEXTURE, MapChannel::EMISSIVE_MAP);
setTextureMap(MapChannel::EMISSIVE_MAP, map);
}
if (!shadeName.isEmpty()) {
auto url = textureMap.contains(shadeName) ? textureMap[shadeName].toUrl() : QUrl();
auto map = fetchTextureMap(url, image::TextureUsage::ALBEDO_TEXTURE, (MapChannel)MToonMapChannel::SHADE_MAP);
setTextureMap((MapChannel)MToonMapChannel::SHADE_MAP, map);
}
if (!shadingShiftName.isEmpty()) {
auto url = textureMap.contains(shadingShiftName) ? textureMap[shadingShiftName].toUrl() : QUrl();
auto map = fetchTextureMap(url, image::TextureUsage::ROUGHNESS_TEXTURE, (MapChannel)MToonMapChannel::SHADING_SHIFT_MAP);
setTextureMap((MapChannel)MToonMapChannel::SHADING_SHIFT_MAP, map);
}
if (!matcapName.isEmpty()) {
auto url = textureMap.contains(matcapName) ? textureMap[matcapName].toUrl() : QUrl();
auto map = fetchTextureMap(url, image::TextureUsage::EMISSIVE_TEXTURE, (MapChannel)MToonMapChannel::MATCAP_MAP);
setTextureMap((MapChannel)MToonMapChannel::MATCAP_MAP, map);
}
if (!rimName.isEmpty()) {
auto url = textureMap.contains(rimName) ? textureMap[rimName].toUrl() : QUrl();
auto map = fetchTextureMap(url, image::TextureUsage::ALBEDO_TEXTURE, (MapChannel)MToonMapChannel::RIM_MAP);
setTextureMap((MapChannel)MToonMapChannel::RIM_MAP, map);
}
if (!uvAnimationMaskName.isEmpty()) {
auto url = textureMap.contains(uvAnimationMaskName) ? textureMap[uvAnimationMaskName].toUrl() : QUrl();
auto map = fetchTextureMap(url, image::TextureUsage::ROUGHNESS_TEXTURE, (MapChannel)MToonMapChannel::UV_ANIMATION_MASK_MAP);
setTextureMap((MapChannel)MToonMapChannel::UV_ANIMATION_MASK_MAP, map);
}
}
std::string NetworkMToonMaterial::getOutlineWidthModeName(OutlineWidthMode mode) {
const std::string names[3] = { "none", "worldCoordinates", "screenCoordinates" };
return names[mode];

View file

@ -26,6 +26,7 @@ public:
NetworkMaterial() : _textures(MapChannel::NUM_MAP_CHANNELS) {}
NetworkMaterial(const HFMMaterial& material, const QUrl& textureBaseUrl);
NetworkMaterial(const NetworkMaterial& material);
NetworkMaterial(const graphics::Material material) : graphics::Material(material) {}
void setAlbedoMap(const QUrl& url, bool useAlphaChannel);
void setNormalMap(const QUrl& url, bool isBumpmap);
@ -60,7 +61,7 @@ protected:
static const QString NO_TEXTURE;
const QString& getTextureName(MapChannel channel);
void setTextures(const QVariantMap& textureMap);
virtual void setTextures(const QVariantMap& textureMap);
const bool& isOriginal() const { return _isOriginal; }
@ -68,22 +69,26 @@ protected:
image::TextureUsage::Type type, MapChannel channel);
graphics::TextureMapPointer fetchTextureMap(const QUrl& url, image::TextureUsage::Type type, MapChannel channel);
Transform _albedoTransform;
bool _isOriginal{ true };
private:
// Helpers for the ctors
QUrl getTextureUrl(const QUrl& baseUrl, const HFMTexture& hfmTexture);
Transform _albedoTransform;
Transform _lightmapTransform;
vec2 _lightmapParams;
bool _isOriginal { true };
};
class NetworkMToonMaterial : public NetworkMaterial {
public:
NetworkMToonMaterial() : NetworkMaterial() {}
NetworkMToonMaterial() : NetworkMaterial() { _model = VRM_MTOON; }
NetworkMToonMaterial(const HFMMaterial& material, const QUrl& textureBaseUrl);
NetworkMToonMaterial(const NetworkMToonMaterial& material);
void setTextures(const QVariantMap& textureMap) override;
enum MToonMapChannel {
// Keep aligned with graphics/ShaderConstants.h and graphics-scripting/ScriptableModel.cpp
SHADE_MAP = MapChannel::ROUGHNESS_MAP,