diff --git a/.gitignore b/.gitignore
index c18cae5a1f720af0f84a79bfe0061c4d30a15105..d7d4599d47adc998a0d92afd718804f56edbfb3b 100644
--- a/.gitignore
+++ b/.gitignore
@@ -1,6 +1,7 @@
 .project
 .cproject
 .vs/
+.vscode/
 .idea/
 .editorconfig
 
diff --git a/config/Sources.cmake b/config/Sources.cmake
index f25104e9df248d07dd0cd864d56645db8ed422d7..9a9f55747194ab9972254d5bf57ac11735607b06 100644
--- a/config/Sources.cmake
+++ b/config/Sources.cmake
@@ -2,4 +2,8 @@
 set(vkcv_sources
 		${vkcv_source}/vkcv/Context.hpp
 		${vkcv_source}/vkcv/Context.cpp
+		${vkcv_source}/vkcv/Window.hpp
+		${vkcv_source}/vkcv/Window.cpp
+		${vkcv_source}/vkcv/CoreManager.hpp
+		${vkcv_source}/vkcv/CoreManager.cpp
 )
diff --git a/projects/first_triangle/src/main.cpp b/projects/first_triangle/src/main.cpp
index 46f617bc6f7691dd5227c8a472634a08679ed385..cc592f468e0c3e95d64a1558404873c4ca19f8b9 100644
--- a/projects/first_triangle/src/main.cpp
+++ b/projects/first_triangle/src/main.cpp
@@ -1,9 +1,17 @@
 #include <iostream>
 #include <vkcv/Context.hpp>
+#include <vkcv/Window.hpp>
 
 int main(int argc, const char** argv) {
+    const char* applicationName = "First Triangle";
+	vkcv::Window window = vkcv::Window::create(
+            applicationName,
+        800,
+        600,
+		false
+	);
 	vkcv::Context context = vkcv::Context::create(
-		"First Triangle",
+            applicationName,
 		VK_MAKE_VERSION(0, 0, 1),
 		20,
 		{vk::QueueFlagBits::eGraphics, vk::QueueFlagBits::eTransfer}
@@ -24,5 +32,8 @@ int main(int argc, const char** argv) {
 		default: std::cout << "Unknown GPU vendor?! Either you're on an exotic system or your driver is broken..." << std::endl;
 	}
 
+	while (window.isWindowOpen()) {
+		window.pollEvents();
+	}
 	return 0;
 }
diff --git a/src/vkcv/CoreManager.cpp b/src/vkcv/CoreManager.cpp
new file mode 100644
index 0000000000000000000000000000000000000000..7205e81ec05c5ee4d50ed6e3cf16a049fce9d921
--- /dev/null
+++ b/src/vkcv/CoreManager.cpp
@@ -0,0 +1,27 @@
+#define GLFW_INCLUDE_VULKAN
+
+#include "CoreManager.hpp"
+
+namespace vkcv {
+
+    int glfwCounter = 0;
+
+    void initGLFW() {
+
+        if (glfwCounter == 0) {
+            int glfwSuccess = glfwInit();
+
+            if (glfwSuccess == GLFW_FALSE) {
+                throw std::runtime_error("Could not initialize GLFW");
+            }
+        }
+        glfwCounter++;
+    }
+
+    void terminateGLFW() {
+        if (glfwCounter == 1) {
+            glfwTerminate();
+        }
+        glfwCounter--;
+    }
+}
\ No newline at end of file
diff --git a/src/vkcv/CoreManager.hpp b/src/vkcv/CoreManager.hpp
new file mode 100644
index 0000000000000000000000000000000000000000..9e874310792ec6ba27510f1a9d57fcdc8cc57a9a
--- /dev/null
+++ b/src/vkcv/CoreManager.hpp
@@ -0,0 +1,19 @@
+#ifndef VKCV_COREMANAGER_HPP
+#define VKCV_COREMANAGER_HPP
+
+#include <GLFW/glfw3.h>
+#include <stdexcept>
+
+namespace vkcv {
+
+    /**
+     * initialize GLFW if not initialized
+     */
+    void initGLFW();
+    /**
+     * terminate GLFW
+     */
+    void terminateGLFW();
+}
+
+#endif //VKCV_COREMANAGER_HPP
diff --git a/src/vkcv/Window.cpp b/src/vkcv/Window.cpp
new file mode 100644
index 0000000000000000000000000000000000000000..da88e8615d7867922511d9ca3009a3adec97d80a
--- /dev/null
+++ b/src/vkcv/Window.cpp
@@ -0,0 +1,50 @@
+#include "Window.hpp"
+#include "CoreManager.hpp"
+
+namespace vkcv {
+
+    Window::Window(GLFWwindow *window)
+            : m_window(window) {
+    }
+
+    Window::~Window() {
+        glfwDestroyWindow(m_window);
+        vkcv::terminateGLFW();
+    }
+
+    Window Window::create(const char *windowTitle, int width, int height, bool resizable) {
+        vkcv::initGLFW();
+        width = std::max(width, 1);
+        height = std::max(height, 1);
+
+        glfwWindowHint(GLFW_CLIENT_API, GLFW_NO_API);
+        glfwWindowHint(GLFW_RESIZABLE, resizable ? GLFW_TRUE : GLFW_FALSE);
+        GLFWwindow *window;
+        window = glfwCreateWindow(width, height, windowTitle, nullptr, nullptr);
+        return Window(window);
+    }
+
+    bool Window::isWindowOpen() const {
+        return !glfwWindowShouldClose(m_window);
+    }
+
+    void Window::pollEvents() {
+        glfwPollEvents();
+    }
+
+    GLFWwindow *Window::getWindow() const {
+        return m_window;
+    }
+
+    int Window::getWidth() const {
+        int width;
+        glfwGetWindowSize(m_window, &width, nullptr);
+        return width;
+    }
+
+    int Window::getHeight() const {
+        int height;
+        glfwGetWindowSize(m_window, nullptr, &height);
+        return height;
+    }
+}
\ No newline at end of file
diff --git a/src/vkcv/Window.hpp b/src/vkcv/Window.hpp
new file mode 100644
index 0000000000000000000000000000000000000000..2c79287b703065e29e2d32338d9c440fa3123c88
--- /dev/null
+++ b/src/vkcv/Window.hpp
@@ -0,0 +1,41 @@
+#pragma once
+
+#include <vulkan/vulkan.hpp>
+
+#define GLFW_INCLUDE_VULKAN
+
+#include <GLFW/glfw3.h>
+
+namespace vkcv {
+
+    class Window final {
+    private:
+        explicit Window(GLFWwindow *window);
+
+        GLFWwindow *m_window;
+
+    public:
+        static Window create(const char *windowTitle, int width = -1, int height = -1, bool resizable = false);
+
+        [[nodiscard]]
+        bool isWindowOpen() const;
+
+        static void pollEvents();
+
+        [[nodiscard]]
+        GLFWwindow *getWindow() const;
+
+        [[nodiscard]]
+        int getWidth() const;
+
+        [[nodiscard]]
+        int getHeight() const;
+
+        Window &operator=(const Window &other) = delete;
+
+        Window &operator=(Window &&other) = default;
+
+        virtual ~Window();
+
+    };
+}
\ No newline at end of file