mirror of
https://github.com/MaSzyna-EU07/maszyna.git
synced 2026-03-22 15:05:03 +01:00
200 lines
4.5 KiB
C++
200 lines
4.5 KiB
C++
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
#include "stdafx.h"
|
|
#include "widgets/cameraview_extcam.h"
|
|
#include "stb/stb_image.h"
|
|
#include "Globals.h"
|
|
#include "Logs.h"
|
|
#include "extras/piped_proc.h"
|
|
|
|
ui::cameraview_panel::cameraview_panel()
|
|
: ui_panel(STR_C("Camera preview"), false)
|
|
{
|
|
size_min = { -2, -2 };
|
|
}
|
|
|
|
void cameraview_window_callback(ImGuiSizeCallbackData *data) {
|
|
data->DesiredSize.y = data->DesiredSize.x * (float)Global.extcam_res.y / (float)Global.extcam_res.x;
|
|
}
|
|
|
|
ui::cameraview_panel::~cameraview_panel() {
|
|
record_state = STOPPING;
|
|
if (record_workthread.joinable())
|
|
record_workthread.join();
|
|
|
|
capture_state = STOPPING;
|
|
if (capture_workthread.joinable())
|
|
capture_workthread.join();
|
|
}
|
|
|
|
void ui::cameraview_panel::render()
|
|
{
|
|
if (Global.extcam_cmd.empty())
|
|
return;
|
|
|
|
if (record_state == STOPPING_DONE) {
|
|
record_workthread.join();
|
|
record_state = STOPPED;
|
|
}
|
|
|
|
if (capture_state == STOPPING_DONE) {
|
|
capture_workthread.join();
|
|
capture_state = STOPPED;
|
|
}
|
|
|
|
if (capture_state == STOPPED) {
|
|
capture_state = STARTING;
|
|
capture_workthread = std::thread(&cameraview_panel::capture_func, this);
|
|
}
|
|
|
|
if (recording && !(record_state == RUNNING || record_state == STARTING)) {
|
|
record_state = STARTING;
|
|
record_workthread = std::thread(&cameraview_panel::record_func, this);
|
|
}
|
|
if (!recording && !(record_state == STOPPED || record_state == STOPPING_DONE)) {
|
|
record_state = STOPPING;
|
|
}
|
|
|
|
if (!is_open)
|
|
return;
|
|
|
|
ImGui::SetNextWindowSizeConstraints(ImVec2(200, 200), ImVec2(2500, 2500), cameraview_window_callback);
|
|
|
|
auto const panelname{(title.empty() ? m_name : title) + "###" + m_name};
|
|
if (ImGui::Begin(panelname.c_str(), &is_open)) {
|
|
render_contents();
|
|
}
|
|
|
|
ImGui::End();
|
|
}
|
|
|
|
bool ui::cameraview_panel::set_state(bool rec)
|
|
{
|
|
if (recording == rec)
|
|
return false;
|
|
|
|
recording = rec;
|
|
|
|
return true;
|
|
}
|
|
|
|
void ui::cameraview_panel::render_contents()
|
|
{
|
|
if (!texture) {
|
|
texture.emplace();
|
|
texture->make_stub();
|
|
}
|
|
|
|
texture->bind(0);
|
|
|
|
{
|
|
std::lock_guard<std::mutex> lock(mutex);
|
|
if (image_ptr) {
|
|
|
|
glActiveTexture(GL_TEXTURE0);
|
|
|
|
glTexImage2D(
|
|
GL_TEXTURE_2D, 0,
|
|
GL_SRGB8_ALPHA8,
|
|
Global.extcam_res.x, Global.extcam_res.y, 0,
|
|
GL_RGB, GL_UNSIGNED_BYTE, image_ptr);
|
|
image_ptr = nullptr;
|
|
|
|
if (Global.python_mipmaps) {
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR_MIPMAP_LINEAR);
|
|
glGenerateMipmap(GL_TEXTURE_2D);
|
|
}
|
|
else {
|
|
glTexParameteri(GL_TEXTURE_2D, GL_TEXTURE_MIN_FILTER, GL_LINEAR);
|
|
}
|
|
}
|
|
}
|
|
|
|
ImVec2 surface_size = ImGui::GetContentRegionAvail();
|
|
ImGui::Image(reinterpret_cast<void*>(texture->id), surface_size);
|
|
}
|
|
|
|
void ui::cameraview_panel::capture_func()
|
|
{
|
|
std::string cmdline = Global.extcam_cmd;
|
|
|
|
piped_proc proc(cmdline);
|
|
|
|
size_t frame_size = Global.extcam_res.x * Global.extcam_res.y * 3;
|
|
uint8_t *read_buffer = new uint8_t[frame_size];
|
|
uint8_t *active_buffer = new uint8_t[frame_size];
|
|
|
|
size_t bufpos = 0;
|
|
|
|
capture_state = RUNNING;
|
|
while (capture_state == RUNNING) {
|
|
size_t read = 0;
|
|
read = proc.read(read_buffer + bufpos, frame_size - bufpos);
|
|
|
|
if (!read)
|
|
break;
|
|
bufpos += read;
|
|
|
|
if (bufpos != frame_size)
|
|
continue;
|
|
|
|
bufpos = 0;
|
|
|
|
std::lock_guard<std::mutex> lock(mutex);
|
|
image_ptr = read_buffer;
|
|
read_buffer = active_buffer;
|
|
active_buffer = image_ptr;
|
|
|
|
frame_cnt++;
|
|
notify_var.notify_one();
|
|
}
|
|
|
|
std::lock_guard<std::mutex> lock(mutex);
|
|
image_ptr = nullptr;
|
|
delete[] read_buffer;
|
|
delete[] active_buffer;
|
|
|
|
capture_state = STOPPING_DONE;
|
|
}
|
|
|
|
void ui::cameraview_panel::record_func()
|
|
{
|
|
std::string cmdline = Global.extcam_rec;
|
|
|
|
if (!rec_name.empty()) {
|
|
const std::string magic{"{RECORD}"};
|
|
size_t pos = cmdline.find(magic);
|
|
if (pos != -1)
|
|
cmdline.replace(pos, magic.size(), rec_name);
|
|
}
|
|
|
|
piped_proc proc(cmdline, true);
|
|
|
|
size_t frame_size = Global.extcam_res.x * Global.extcam_res.y * 3;
|
|
uint8_t *read_buffer = new uint8_t[frame_size];
|
|
uint32_t last_cnt = 0;
|
|
size_t bufpos = frame_size;
|
|
|
|
record_state = RUNNING;
|
|
while (record_state == RUNNING) {
|
|
if (bufpos == frame_size)
|
|
{
|
|
std::unique_lock<std::mutex> lock(mutex);
|
|
auto r = notify_var.wait_for(lock, std::chrono::milliseconds(50), [this, last_cnt]{return last_cnt != frame_cnt;});
|
|
last_cnt = frame_cnt;
|
|
if (!image_ptr || !r)
|
|
continue;
|
|
|
|
bufpos = 0;
|
|
memcpy(read_buffer, image_ptr, frame_size);
|
|
}
|
|
bufpos += proc.write(read_buffer + bufpos, frame_size - bufpos);
|
|
}
|
|
|
|
delete[] read_buffer;
|
|
|
|
record_state = STOPPING_DONE;
|
|
}
|