Skip to content

Kuesa viewer Example

viewer-example.png

Setup Window

Kuesa::Serenity::Window can be conveniently used to create a Window suitable for Serenity.

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
int main(int ac, char **av)
{
    if (ac < 2) {
        spdlog::warn("Usage: viewer path_to_gltf");
        return -1;
    }

    ::Serenity::GuiApplication app;

    // Setup Window
    Kuesa::Serenity::Window w;
    auto engine = std::make_unique<::Serenity::AspectEngine>();
    w.visible.valueChanged().connect([&app](const bool &visible) {
        if (visible == false)
            app.quit();
    });
    w.title = makeBinding(makeTitle(w.width, w.height, engine->fps));
    w.width = 1920;
    w.height = 1080;
    w.visible = true;

Filename: viewer/main.cpp

Importing a glTF2 File

We add the default Kuesa Layers to the [LayerManager ] so that meshes can be tagged with the appropriate layer mask based on their material properties upon importing.

This will be required to filter which Entities to render at the appropriate time.

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
std::unique_ptr<::Serenity::Entity> createKuesaScene(const String &gltfPath, ::Serenity::LayerManager *layerManager)
{
    Kuesa::GLTF2Import::GLTF2Context ctx;
    Kuesa::GLTF2Import::GLTF2Parser parser;
    Kuesa::Serenity::SceneGenerator generator(nullptr, layerManager);

    parser.setContext(&ctx);
    generator.setContext(&ctx);

    // Parse GLTF2 File -> fills the GLTF2Context
    const bool parsingSuccessful = parser.parse(gltfPath);
    assert(parsingSuccessful);

    // Generate Serenity Scene (from trhe GLTF2Context)
    generator.generateContent();

    // Insert the generated Kuesa Entities into rootEntity
    auto rootEntity = std::make_unique<::Serenity::Entity>();
    const std::vector<::Serenity::Entity *> &sceneRoots = generator.sceneRoots();

    {
        // Add a default light
        ::Serenity::Entity *e = rootEntity->createChildEntity<::Serenity::Entity>();
        ::Serenity::Light *l = e->createComponent<::Serenity::Light>();
        l->type = ::Serenity::Light::Type::Point;
        ::Serenity::SrtTransform *t = e->createComponent<::Serenity::SrtTransform>();
        t->translation = glm::vec3(0.0f, 40.0f, 0.0f);
    }

    for (::Serenity::Entity *sceneRoot : sceneRoots) {
        // Take ownership
        std::unique_ptr<::Serenity::Entity> ownedPtr(sceneRoot);
        rootEntity->addChildEntity(std::move(ownedPtr));
    }

    return std::move(rootEntity);
}

Filename: viewer/main.cpp

We make use of the GLTF2Importer to load a glTF2 file and feed our AssectCollections .

1
2
3
4
5
6
7
    // Setup Kuesa Scene
    ::Serenity::LayerManager layerManager;
    layerManager.addLayer(Kuesa::Serenity::Layers::ZFillLayerName);
    layerManager.addLayer(Kuesa::Serenity::Layers::OpaqueLayerName);
    layerManager.addLayer(Kuesa::Serenity::Layers::AlphaLayerName);
    layerManager.addLayer(Kuesa::Serenity::Layers::SkyboxLayerName);
    layerManager.addLayer(Kuesa::Serenity::Layers::BackgroundLayerName);

Filename: viewer/main.cpp

Setting up the Camera

1
2
3
4
5
6
7
    ::Serenity::Camera *defaultCamera = createDefaultCamera(root.get(), w);

    // Camera Controller
    Kuesa::Serenity::CameraController *controller = w.createChild<Kuesa::Serenity::CameraController>();
    controller->window = &w;
    controller->camera = defaultCamera;
    w.cameraController = controller;

Filename: viewer/main.cpp

Using a CameraController

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
::Serenity::Camera *createDefaultCamera(::Serenity::Entity *rootEntity, ::Serenity::Window &w)
{
    // Add Camera into the Scene
    ::Serenity::Camera *camera = rootEntity->createChildEntity<::Serenity::Camera>();
    camera->lookAt(glm::vec3(0.0f, 10.0f, 15.0f),
                   glm::vec3(0.0f, 0.0f, 0.0f),
                   glm::vec3(0.0f, 1.0f, 0.0f));
    camera->lens()->setPerspectiveProjection(45.0f,
                                             float(w.width()) / w.height(),
                                             1.0f, 1000.0f);
    camera->lens()->aspectRatio = makeBinding(updateAspectRatio(w.width, w.height));

    return camera;
}

Filename: viewer/main.cpp

Setting up Serenity

We start by creating the various aspects we will make use of.

1
2
3
4
    // Setup Serenity Engine
    auto renderAspect = engine->createAspect<::Serenity::RenderAspect>(std::make_unique<::Serenity::VulkanDevice>());
    auto spatialAspect = engine->createAspect<::Serenity::SpatialAspect>();
    auto logicAspect = engine->createAspect<::Serenity::LogicAspect>();

Filename: viewer/main.cpp

And then we configure the Serenity Rendering algorithm. Rendering will be split in 3 phases, one for Opaque, one for Skybox and one for Alpha objects.

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
    auto algo = std::make_unique<::Serenity::ForwardAlgorithm>();

    auto createOpaquePhase = [&layerManager]() {
        ::Serenity::ForwardAlgorithm::RenderPhase opaquePhase{
            layerManager.layerMask({ Kuesa::Serenity::Layers::OpaqueLayerName }),
            ::Serenity::ForwardAlgorithm::RenderPhase::Type::Opaque,
            ::Serenity::LayerFilterType::AcceptAny
        };

        auto depthState = std::make_shared<::Serenity::DepthStencilState>();
        depthState->depthTestEnabled = true;
        depthState->depthWriteEnabled = true;
        depthState->depthCompareOp = ::Serenity::CompareOp::Less;
        opaquePhase.renderStates.setDepthStencilState(std::move(depthState));

        auto rasterizerState = std::make_shared<::Serenity::RasterizerState>();
        rasterizerState->cullMode = ::Serenity::CullModeFlag::Back;
        opaquePhase.renderStates.setRasterizerState(std::move(rasterizerState));

        return opaquePhase;
    };

    auto createSkyboxPhase = [&layerManager]() {
        ::Serenity::ForwardAlgorithm::RenderPhase opaquePhase{
            layerManager.layerMask({ Kuesa::Serenity::Layers::SkyboxLayerName, Kuesa::Serenity::Layers::BackgroundLayerName }),
            ::Serenity::ForwardAlgorithm::RenderPhase::Type::Opaque,
            ::Serenity::LayerFilterType::AcceptAny
        };

        auto depthState = std::make_shared<::Serenity::DepthStencilState>();
        depthState->depthTestEnabled = true;
        depthState->depthWriteEnabled = false;
        depthState->depthCompareOp = ::Serenity::CompareOp::LessOrEqual;
        opaquePhase.renderStates.setDepthStencilState(std::move(depthState));

        auto rasterizerState = std::make_shared<::Serenity::RasterizerState>();
        rasterizerState->cullMode = ::Serenity::CullModeFlag::None;
        opaquePhase.renderStates.setRasterizerState(std::move(rasterizerState));

        return opaquePhase;
    };

    auto createAlphaPhase = [&layerManager]() {
        ::Serenity::ForwardAlgorithm::RenderPhase alphaPhase{
            layerManager.layerMask({ Kuesa::Serenity::Layers::AlphaLayerName }),
            ::Serenity::ForwardAlgorithm::RenderPhase::Type::Alpha,
            ::Serenity::LayerFilterType::AcceptAll
        };

        auto depthState = std::make_shared<::Serenity::DepthStencilState>();
        depthState->depthTestEnabled = true;
        depthState->depthWriteEnabled = false;
        depthState->depthCompareOp = ::Serenity::CompareOp::Less;
        alphaPhase.renderStates.setDepthStencilState(std::move(depthState));

        auto blendState = std::make_shared<::Serenity::ColorBlendState>();
        ::Serenity::ColorBlendState::AttachmentBlendState attachmentBlendState;

        attachmentBlendState.blendingEnabled = true;
        attachmentBlendState.alphaBlendOp = ::Serenity::BlendOp::Add;
        attachmentBlendState.colorBlendOp = ::Serenity::BlendOp::Add;
        attachmentBlendState.sourceAlphaBlendFactor = ::Serenity::BlendFactor::Zero;
        attachmentBlendState.destinationAlphaBlendFactor = ::Serenity::BlendFactor::OneMinusSourceAlpha;
        attachmentBlendState.sourceColorBlendFactor = ::Serenity::BlendFactor::SourceAlpha;
        attachmentBlendState.destinationColorBlendFactor = ::Serenity::BlendFactor::OneMinusSourceAlpha;
        blendState->attachmentBlendStates = { attachmentBlendState };

        alphaPhase.renderStates.setColorBlendState(std::move(blendState));

        return alphaPhase;
    };

    // ImGUI Overlay
    Serenity::AbstractOverlay *overlay = createOverlay(w, engine.get(), algo.get());

    // Specify RT
    algo->renderTargetRefs = { Serenity::RenderTargetRef::fromWindow(&w, renderAspect->device()->instance()) };
    // Specify which RenderTarget to render to, which render phases and which camera to use
    algo->renderViews = {
        Serenity::ForwardAlgorithm::RenderView(0,
                                               { createOpaquePhase(),
                                                 createSkyboxPhase(),
                                                 createAlphaPhase() },
                                               overlay,
                                               defaultCamera)
    };

Filename: viewer/main.cpp

Finally we can set the Render Algorithm and the scene root before starting the Serenity engine.

1
2
3
4
    renderAspect->setRenderAlgorithm(std::move(algo));
    engine->setRootEntity(std::move(root));

    engine->running = true;

Filename: viewer/main.cpp


Updated on 2022-10-18 at 11:12:52 +0200