Files
maszyna/widgets/cameraview_extcam.cpp
2026-03-14 19:16:48 +00:00

200 lines
4.6 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 "utilities/Globals.h"
#include "utilities/translation.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;
}