2d map, physics, and vptree implementation for sv
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 

840 lines
28 KiB

#include "ofApp.h"
//--------------------------------------------------------------
void ofApp::setup(){
/* allocated fbo's */
map_fbo.allocate(ofGetWindowWidth() / 2, map_h, GL_RGB);
map_fbo_alpha.allocate(map_fbo.getWidth(), map_h, GL_RGBA);
map_fbo_post.allocate(map_fbo.getWidth(), map_h, GL_RGBA);
portrait_fbo.allocate((ofGetWindowWidth() / 2), map_h, GL_RGBA); // this shoould be full with
portrait_pre_fbo.allocate((ofGetWindowWidth() / 3) * 1, map_h, GL_RGB); // this should be full, and same below
portrait_pre_fbo_alpha.allocate((ofGetWindowWidth() / 3) * 1, map_h, GL_RGBA);
portrait_cropped.allocate((ofGetWindowWidth() / 3) * 1, map_h, GL_RGB);
portrait_cropped_alpha.allocate((ofGetWindowWidth() / 3) * 1, map_h, GL_RGBA);
comp_fbo.allocate(ofGetWindowWidth(), map_h, GL_RGBA);
model_outptut_fbo.allocate(map_fbo.getWidth(), map_h, GL_RGB);
model_esp_out_fbo.allocate(128 * 2, 168 * 2, GL_RGB);
model_portrait_out_fbo.allocate(portrait_pre_fbo_alpha.getWidth(), portrait_pre_fbo_alpha.getHeight(), GL_RGB);
/* k-d image comp (4-images) */
kd_out.allocate(128 * 2, 168 * 2, GL_RGB);
esp_comp_fbo.allocate(128 * 2, 168 * 2, GL_RGB);
/* input images for model */
model_image_esp.allocate(128 * 2, 168 * 2, OF_IMAGE_COLOR);
model_image.allocate(map_fbo.getWidth(), map_h, OF_IMAGE_COLOR);
model_image_portrait.allocate(portrait_pre_fbo_alpha.getWidth(), portrait_pre_fbo_alpha.getHeight(), OF_IMAGE_COLOR);
model_image_portrait_cropped.allocate(portrait_pre_fbo_alpha.getWidth(), portrait_pre_fbo_alpha.getHeight(), OF_IMAGE_COLOR);
alpha_demo.allocate(portrait_pre_fbo_alpha.getWidth(), portrait_pre_fbo_alpha.getHeight(), OF_IMAGE_COLOR_ALPHA);
/* allocated ofImages in esp_images */
for(int i = 0; i < 4; i++){
ofFbo temp;
temp.allocate(128, 168, GL_RGB);
esp_images.push_back(temp);
}
/* of settings (not sure why putting this at the start of setup breaks the map - but it works!) */
ofSetVerticalSync(true);
ofDisableArbTex();
ofEnableDepthTest();
ofDisableAntiAliasing();
ofEnableAlphaBlending();
/* load */
ofLoadImage(bayer, "images/bayer.png");
bayer.setTextureWrap(GL_REPEAT, GL_REPEAT);
shaders.load("shaders/dither");
esp_shader.load("shaders/espDepth");
vert_snap.load("shaders/vertex_snap");
p_depth.load("shaders/p_depth");
map_depth.load("shaders/map_depth");
ORTCHAR_T* modelPath = "/home/cailean/Desktop/openframeworks/of_v0.12.0_linux64gcc6_release/apps/myApps/image-to-mesh/bin/data/models/depth_anything_v2_vits.onnx";
ORTCHAR_T* modelPath_Small = "/home/cailean/Desktop/openframeworks/of_v0.12.0_linux64gcc6_release/apps/myApps/image-to-mesh/bin/data/models/depth_anything_v2_vitb.onnx";
ORTCHAR_T* modelPath_Yolo = "/home/cailean/Desktop/openframeworks/of_v0.12.0_linux64gcc6_release/apps/myApps/image-to-mesh/bin/data/models/yolov5s-face.onnx";
ORTCHAR_T* modelPath_Indoor_Dynamic = "/home/cailean/Desktop/openframeworks/of_v0.12.0_linux64gcc6_release/apps/myApps/image-to-mesh/bin/data/models/depth_anything_v2_vitl.onnx";
bullet.setup();
/* pi mapper setup */
// mapper.registerFboSource(projection_src);
// mapper.setup();
// projection_src.setFbo(&comp_fbo);
/* onnx setup */
depth_onnx.Setup(modelPath, false, true);
depth_onnx_esp.Setup(modelPath, false, true);
depth_onnx_portrait.Setup(modelPath, false, true);
yolo_onnx.Setup(modelPath_Yolo, false, true);
/* multi-thread setup */
depth_thread.setup(&model_image, &model_outptut_fbo, &depth_onnx);
depth_esp.setup(&model_image_esp, &model_esp_out_fbo, &depth_onnx_esp);
depth_portrait.setup(&model_image_portrait_cropped, &model_portrait_out_fbo, &depth_onnx_portrait);
yolo.setupYolo(&model_image_portrait, &detected_faces, &yolo_onnx, &face_detector);
/* camera settings for portrait */
CameraPosition cp;
cp.position = glm::vec3(-131.228, 502.624, 706.1);
cp.rotation = glm::quat(0.898726, -0.436474, -0.0351978, -0.0170607);
cp.scale = 0.313542;
cam_positions.push_back(cp);
cp.position = glm::vec3(7.987, -195.329, 813.56);
cp.rotation = glm::quat(0.993, 0.117, 0.005, -0.00057);
cam_positions.push_back(cp);
cp.position = glm::vec3(160.81, -87.86, 731.575);
cp.rotation = glm::quat(0.980, 0.027, 0.1934, -0.0053);
cam_positions.push_back(cp);
/* settings */
portrait_camera.enableOrtho();
portrait_camera.setNearClip(-10000);
portrait_camera.setFarClip(10000);
portrait_camera.setPosition(cam_positions[0].position);
portrait_camera.setOrientation(cam_positions[0].rotation);
portrait_camera.setScale(cam_positions[0].scale);
portrait_camera.removeAllInteractions();
portrait_camera.disableMouseInput();
createNodes("data/json/sv_embeddings.json");
buildMeshes();
buildKDTree();
bullet.setNodes(nodes);
server = std::make_unique<Server>(6762, embed, vp_tree, nodes, false, "192.168.0.253", 2000, "search");
server->start();
last_updated_time = ofGetElapsedTimef();
last_updated_time_esp = ofGetElapsedTimef();
}
//--------------------------------------------------------------
void ofApp::update(){
server->update(esp_comp_fbo);
float current_time = ofGetElapsedTimef();
Node n = server->getChosenNode();
// Check if node has changed
if(n.tsne_position != last_chosen_node.tsne_position) {
portrait_needs_update = true;
last_chosen_node = n;
} else {
portrait_needs_update = false;
}
if(current_time - last_updated_time >= 0.1){
buildKDTree();
getNearestImages();
last_updated_time = current_time;
}
if(current_time - last_updated_time_esp >= 4.0){
depth_esp.update();
depth_onnx_esp.SetPixels(model_esp_out_fbo);
last_updated_time_esp = current_time;
}
if(ofGetFrameNum() < 1){
depth_thread.start();
depth_esp.start();
depth_portrait.start();
yolo.start();
}
/* write pixels to model input image */
try{
map_pixels.allocate(map_fbo.getWidth(), map_h, OF_PIXELS_RGB);
map_fbo.readToPixels(map_pixels);
model_image.setFromPixels(map_pixels);
} catch (exception e) {
ofLogError() << "couldn't set pixels!";
}
try{
depth_thread.update();
/* set output to fbo's */
depth_onnx.SetPixels(model_outptut_fbo);
} catch (exception e){
std::cout << "Model did not run" << std::endl;
}
//mapper.update();
bullet.update(server->is_active, server->getChosenNode(), nn_nodes);
if(tsne_update_complete) {
tsne_update_complete = false; // Reset flag
onTSNEUpdateComplete(); // Call your main thread function
}
if(ofGetElapsedTimef() > tsne_start_time + TNSE_DURATION){
tsne_iter_idx = (tsne_iter_idx + 1) % 9;
updateTSNEPositions(nodes);
}
}
//--------------------------------------------------------------
void ofApp::draw(){
ofPushStyle();
map_fbo_alpha.begin();
ofClear(0);
bullet.draw();
map_fbo_alpha.end();
ofPopStyle();
map_fbo.begin();
ofClear(0, 0, 0, 0);
map_fbo_alpha.draw(0, 0);
map_fbo.end();
ofTexture t = map_fbo.getTexture();
ofTexture d = model_outptut_fbo.getTexture();
esp_comp_fbo.begin();
esp_shader.begin();
esp_shader.setUniformTexture("tex0", model_esp_out_fbo.getTexture(), 0);
model_esp_out_fbo.draw(0, 0);
esp_comp_fbo.end();
esp_shader.end();
portrait_fbo.begin();
ofBackground(0, 0, 0, 0);
drawPortraitZ();
portrait_fbo.end();
/* white depth map pass */
map_fbo_post.begin();
ofClear(0);
map_depth.begin();
map_depth.setUniformTexture("tex0", d, 0);
model_outptut_fbo.draw(0, 0);
map_depth.end();
map_fbo_post.end();
drawPortrait();
comp_fbo.begin();
shaders.begin();
portrait_fbo.draw(ofGetWindowWidth() / 2, 0);
shaders.setUniformTexture("tex0", t, 0);
shaders.setUniformTexture("tex1", bayer, 1);
shaders.setUniform2f("resolution", map_w, map_h);
shaders.setUniform1f("time", ofGetElapsedTimef());
shaders.setUniform1i("frame", ofGetFrameNum());
map_fbo_post.draw(0, 0);
map_fbo_alpha.draw(0,0);
shaders.end();
comp_fbo.end();
comp_fbo.getTexture().setTextureMinMagFilter(GL_NEAREST, GL_NEAREST);
//mapper.draw();
ofTranslate(0, 60);
comp_fbo.draw(0, 0);
if(bullet.print_debug)
server->print();
}
void ofApp::drawPortrait(){
portrait_pre_fbo.begin();
ofClear(0, 0, 0, 0);
float scale = min(
(float)portrait_pre_fbo.getWidth() / last_chosen_node.img.getWidth(),
(float)portrait_pre_fbo.getHeight() / last_chosen_node.img.getHeight()
);
float scaledWidth = last_chosen_node.img.getWidth() * scale;
float scaledHeight = last_chosen_node.img.getHeight() * scale;
float xPos = (portrait_pre_fbo.getWidth() - scaledWidth) / 2;
float yPos = (portrait_pre_fbo.getHeight() - scaledHeight) / 2;
ofPushMatrix();
// Move to center of FBO
ofTranslate(portrait_pre_fbo.getWidth()/2, portrait_pre_fbo.getHeight()); // 0.8
// Move back by half the scaled image dimensions
ofTranslate(-scaledWidth/2, -scaledHeight);
// Draw at 0,0 since we've already translated
last_chosen_node.img.draw(0, 0, scaledWidth, scaledHeight);
ofPopMatrix();
portrait_pre_fbo.end();
portrait_pre_fbo_alpha.begin();
ofPushMatrix();
ofClear(0, 0, 0, 0);
ofTranslate(portrait_pre_fbo.getWidth()/2, portrait_pre_fbo.getHeight());
// Move back by half the scaled image dimensions
ofTranslate(-scaledWidth/2, -scaledHeight);
last_chosen_node.img.draw(0, 0, scaledWidth, scaledHeight);
ofPopMatrix();
portrait_pre_fbo_alpha.end();
portrait_pre_fbo_alpha.readToPixels(alpha_pixels);
alpha_demo.setFromPixels(alpha_pixels);
float current_time = ofGetElapsedTimef();
// we have full portrait & alpha portrait.
if(portrait_needs_update) {
yolo.resetInferenceFlag();
ofPixels pix;
portrait_pre_fbo.readToPixels(pix);
model_image_portrait.setFromPixels(pix);
detected_faces.clear();
yolo.update(); // This triggers the thread to run
portrait_needs_update = false;
}
if(yolo.checkInferenceComplete()) {
/* check if face is detected, and crop potrait */
if(detected_faces.size() > 0){
for(auto& f : detected_faces){
float crop_width = f.box.x2 - f.box.x1;
float crop_height = f.box.y2 - f.box.y1;
// Calculate scaling to fit in FBO while maintaining aspect ratio
float scale = min(
(float)portrait_cropped.getWidth() / crop_width,
(float)portrait_cropped.getHeight() / crop_height
);
float scaledWidth = crop_width * scale;
float scaledHeight = crop_height * scale;
// Calculate position to center in FBO
float x = (portrait_cropped.getWidth() - scaledWidth) / 2;
float y = (portrait_cropped.getHeight() - scaledHeight) / 2;
portrait_cropped.begin();
ofClear(ofColor::black);
ofPushMatrix();
// Move to center position
ofTranslate(x, y);
// Draw the cropped section at the calculated size
model_image_portrait.drawSubsection(
0, 0, // Draw at translated position
scaledWidth, scaledHeight, // Draw at scaled size
f.box.x1, f.box.y1, // Start crop from face detection
crop_width, crop_height // Amount to crop
);
ofPopMatrix();
portrait_cropped.end();
portrait_cropped_alpha.begin();
ofClear(0);
ofPushMatrix();
// Move to center position
ofTranslate(x, y);
// Draw the cropped section at the calculated size
alpha_demo.drawSubsection(
0, 0, // Draw at translated position
scaledWidth, scaledHeight, // Draw at scaled size
f.box.x1, f.box.y1, // Start crop from face detection
crop_width, crop_height // Amount to crop
);
ofPopMatrix();
portrait_cropped_alpha.end();
ofPixels c_pix;
portrait_cropped.readToPixels(c_pix);
model_image_portrait_cropped.setFromPixels(c_pix);
}
} else {
/* don't use cropped image for depth model */
ofPixels c_pix;
portrait_pre_fbo.readToPixels(c_pix);
model_image_portrait_cropped.setFromPixels(c_pix);
}
/* run depth model on cropped portrait */
depth_portrait.update();
yolo.resetInferenceFlag();
}
if(depth_portrait.checkInferenceComplete()){
depth_onnx_portrait.SetPixels(model_portrait_out_fbo);
float planeScale = 1.0;
int planeWidth = portrait_pre_fbo_alpha.getWidth() * planeScale;
int planeHeight = portrait_pre_fbo_alpha.getHeight() * planeScale;
int planeGridSize = plane_size;
int planeColumns = planeWidth / planeGridSize;
int planeRows = planeHeight / planeGridSize;
plane.set(planeWidth, planeHeight, planeColumns, planeRows, OF_PRIMITIVE_TRIANGLES);
custom_mesh = createCustomPlane(planeWidth, planeHeight, planeColumns, planeRows);
/* send osc message when new node selected*/
server->sendOSCMessage();
depth_portrait.resetInferenceFlag();
}
}
void ofApp::drawPortraitZ(){
ofBackgroundGradient(ofColor::white, ofColor::orange, OF_GRADIENT_LINEAR);
/* if no faces found use full portrait texture */
ofTexture tex_color;
if(detected_faces.size() > 0){
tex_color = portrait_cropped_alpha.getTexture();
} else {
tex_color = portrait_pre_fbo_alpha.getTexture();
}
ofTexture tex_depth = model_portrait_out_fbo.getTexture();
float minDepth = std::numeric_limits<float>::max();
float maxDepth = std::numeric_limits<float>::lowest();
ofPixels depth_pix;
tex_depth.readToPixels(depth_pix);
float max_r = 0;
// Loop through only the R channel
for(int i = 0; i < depth_pix.getWidth() * depth_pix.getHeight(); i++) {
float depth = depth_pix[i * depth_pix.getNumChannels()] / 255.0f; // Access just R channel
if(max_r < depth){
max_r = depth;
}
}
portrait_camera.begin();
ofEnableDepthTest();
float time = ofGetElapsedTimef() / 20;
float p_noise_x = ofSignedNoise(time, time) * 20;
float p_noise_y = ofSignedNoise(time + 100, time + 100) * 20;
// Second plane with offset
ofPushMatrix();
p_depth.begin();
p_depth.setUniform1f("gridSize", g_size);
p_depth.setUniform2f("resolution", glm::vec2(portrait_fbo.getWidth(), portrait_fbo.getHeight()));
p_depth.setUniform2f("threshold", glm::vec2(minDepth, max_r));
p_depth.setUniformTexture("tex0", tex_color, 0);
p_depth.setUniformTexture("tex1", tex_depth, 1);
ofFill();
// Offset the second plane - adjust these values as needed
ofTranslate(p_noise_x - 50, p_noise_y - 0, -150); // moved right (+200), up (+100) and back (-300)
ofRotateDeg(60, 0, 1, 0);
plane.draw();
p_depth.end();
ofPopMatrix();
ofDisableDepthTest();
// Second plane with offset
ofPushMatrix();
p_depth.begin();
p_depth.setUniform1f("gridSize", g_size);
p_depth.setUniform2f("resolution", glm::vec2(portrait_fbo.getWidth(), portrait_fbo.getHeight()));
p_depth.setUniform2f("threshold", glm::vec2(minDepth, max_r));
p_depth.setUniformTexture("tex0", tex_color, 0);
p_depth.setUniformTexture("tex1", tex_depth, 1);
ofFill();
// Offset the second plane - adjust these values as needed
ofTranslate(p_noise_x - 140, p_noise_y - 0, -150); // moved right (+200), up (+100) and back (-300)
ofRotateDeg(-60, 0, 1, 0);
plane.draw();
p_depth.end();
ofPopMatrix();
ofDisableDepthTest();
ofPushMatrix();
p_depth.begin();
p_depth.setUniform1f("gridSize", g_size);
p_depth.setUniform2f("resolution", glm::vec2(portrait_fbo.getWidth(), portrait_fbo.getHeight()));
p_depth.setUniform2f("threshold", glm::vec2(minDepth, max_r));
p_depth.setUniformTexture("tex0", tex_color, 0);
p_depth.setUniformTexture("tex1", tex_depth, 1);
ofFill();
ofTranslate(p_noise_x, p_noise_y, 0);
plane.draw();
p_depth.end();
ofPopMatrix();
portrait_camera.end();
}
/* creates an fbo with four cropped images, to preprare for model input */
void ofApp::getNearestImages(){
nn_nodes.clear();
glm::vec3 cam_position = bullet.getCameraPosition();
queryKD(cam_position, 4);
kd_out.begin();
ofClear(255, 255, 255, 0);
ofImage selected_image;
int imageIndex = 0;
for(auto& img : esp_images){
/* add to nn_nodes array */
nn_nodes.push_back(&nodes[kd_result[imageIndex].second]);
selected_image = nodes[kd_result[imageIndex].second].img;
// Calculate the scaling factor
float widthRatio = 128.0f / selected_image.getWidth();
float heightRatio = 168.0f / selected_image.getHeight();
float scale = std::max(widthRatio, heightRatio);
// Calculate new dimensions
int newWidth = std::ceil(selected_image.getWidth() * scale);
int newHeight = std::ceil(selected_image.getHeight() * scale);
// Resize the image
selected_image.resize(newWidth, newHeight);
// Calculate the crop position to center the image
int cropX = (newWidth - 128) / 2;
int cropY = (newHeight - 168) / 2;
// Ensure crop dimensions don't exceed the image size
int cropWidth = std::min(128, newWidth);
int cropHeight = std::min(168, newHeight);
// Calculate the drawing position based on the image index
int drawX = (imageIndex % 2) * 128;
int drawY = (imageIndex / 2) * 168;
// Draw the resized and cropped image
selected_image.drawSubsection(drawX, drawY, cropWidth, cropHeight, cropX, cropY);
imageIndex++;
if (imageIndex >= 4) break; // Stop after drawing 4 images
}
kd_out.end();
kd_out.readToPixels(esp_comp_pixels);
model_image_esp.setFromPixels(esp_comp_pixels);
}
void ofApp::createNodes(std::string json_path){
/* read in json */
json = ofLoadJson(json_path);
if(!json.is_array()){
ofLogError() << "json is not array";
return;
} else {
ofLogNotice() << "json loaded";
}
/* setup nodes */
bool addNode = true; // Toggle flag to track every second node
for(const auto& j : json) {
if(j.contains("vector") && j["vector"].is_array() && j["error"] == 0) {
if(addNode) { // Only process when flag is true
Node n;
n.img.load(j["image"]);
n.tex = n.img.getTexture();
n.error = j["error"];
std::vector<float> t_embedding;
int count = 0;
for (const auto& value: j["vector"]) {
if(value.is_number()) {
t_embedding.push_back(value.get<float>());
count++;
} else {
ofLogError() << "Vector value is not a number";
}
}
n.raw_embedding = t_embedding;
nodes.push_back(n);
}
addNode = !addNode; // Toggle the flag after each valid node
}
}
ofLogNotice() << "node count: " << nodes.size();
/* build vp_tree */
auto vectors = createDoubleVectorFromNodes(nodes);
vp_tree.buildTree(vectors);
ofLogNotice() << "vpt built";
/* run tsne */
vector<vector<float>> tsne_input;
std::vector<std::vector<double>> tsne_points;
for(const auto& n : nodes){
tsne_input.push_back(n.raw_embedding);
}
tsne_points = tsne.run(tsne_input, dims, perplexity, theta, normalise, runManually);
// Option 1: Using nested loops
for(size_t i = 0; i < 9; i++) {
for(size_t t = 0; t <= 100; t++) {
tsne_points = tsne.iterate();
}
point_iterations.push_back(tsne_points);
}
for(size_t i = 0; i < tsne_points.size(); i++){
const auto& vec = point_iterations[0][i];
auto& n = nodes[i];
n.tsne_position = (glm::vec3(((vec[0] * 2) - 1) * tsne_scale, ((vec[1] * 2) - 1) * tsne_scale, -5.0f));
}
}
void ofApp::updateTSNEPositions(vector<Node>& _nodes){
if(tsne_updating) {
ofLog() << "TSNE update already in progress";
return;
}
tsne_updating = true;
tsne_update_complete = false; // Reset flag
tsne_thread = std::thread(&ofApp::updateTSNEPositionsThreaded, this, std::ref(_nodes));
tsne_thread.detach();
}
void ofApp::updateTSNEPositionsThreaded(vector<Node>& _nodes){
ofLog() << "Starting TSNE update in thread";
{
std::lock_guard<std::mutex> lock(nodes_mutex);
for(size_t i = 0; i < point_iterations[tsne_iter_idx].size(); i++) {
const auto& vec = point_iterations[tsne_iter_idx][i];
nodes[i].tsne_position = glm::vec3(
((vec[0] * 2) - 1) * tsne_scale,
((vec[1] * 2) - 1) * tsne_scale,
-5.0f
);
}
}
tsne_updating = false;
tsne_update_complete = true; // Set completion flag
ofLog() << "TSNE update complete";
}
std::vector<std::vector<double>> ofApp::createDoubleVectorFromNodes(const std::vector<Node>& nodes) {
std::vector<std::vector<double>> result;
result.reserve(nodes.size());
for (const auto& node : nodes) {
std::vector<double> doubleVector;
doubleVector.reserve(node.raw_embedding.size());
for (const float& value : node.raw_embedding) {
doubleVector.push_back(static_cast<double>(value));
}
result.push_back(std::move(doubleVector));
}
return result;
}
void ofApp::exit(){
std::cout << "closing server" << std::endl;
server->close();
}
void ofApp::buildMeshes(){
vector<ofMesh> mesh_list;
ofLogNotice() << "building meshes: " << nodes.size();
for(auto& n : nodes){
mesh_list = mesh_generator.generateSimplifiedMesh(n.img);
bullet.addMesh(mesh_list[0], mesh_list[1], n);
mesh_list.clear();
}
ofLogNotice() << "finished building meshes";
}
void ofApp::buildKDTree(){
pointVec kd_points;
for(const auto& n : nodes){
glm::vec3 n_pos = n.collider->getPosition();
std::vector< double > p = {
static_cast<double>(n_pos.x),
static_cast<double>(n_pos.y)
};
kd_points.push_back(p);
}
kd_tree = std::make_unique<KDTree>(kd_points);
kd_result.resize(4);
}
void ofApp::queryKD(glm::vec3& _position, int k){
kd_result.clear();
vector<double> kd_input = {
static_cast<double>(_position.x),
static_cast<double>(_position.y)
};
auto res = kd_tree->nearest_pointIndices(kd_input, k);
for (const auto& r : res) {
kd_result.push_back(r);
}
size_t index = kd_result[0].second;
//std::cout << "camera positions: " << _position << ", nearest node: "<< nodes[index].tsne_position << " " << index << std::endl;
}
void ofApp::keyPressed(int key) {
switch(key) {
case OF_KEY_UP:
g_size += 5;
ofLog() << "Grid Size: " << g_size;
break;
case OF_KEY_DOWN:
g_size -= 5;
ofLog() << "Grid Size: " << g_size;
break;
case 'q':
plane_size += 5;
ofLog() << "Plane Size: " << plane_size;
break;
case 'w':
plane_size -=5;
ofLog() << "Plane Size: " << plane_size;
break;
case 'c':
// cam_pos_idx++;
// if(cam_pos_idx > cam_positions.size() - 1)
// cam_pos_idx = 0;
portrait_camera.setPosition(cam_positions[cam_pos_idx].position);
portrait_camera.setOrientation(cam_positions[cam_pos_idx].rotation);
portrait_camera.setScale(cam_positions[cam_pos_idx].scale);
case 'p':
if(bullet.print_debug){
bullet.print_debug = false;
} else {
bullet.print_debug = true;
}
std::cout << portrait_camera.getPosition() << " : " <<portrait_camera.getOrientationQuat()<< " : " << portrait_camera.getScale() << std::endl;
break;
case 't':
tsne_iter_idx++;
updateTSNEPositions(nodes);
}
//mapper.keyPressed(key);
}
// Your main thread function:
void ofApp::onTSNEUpdateComplete() {
// Do whatever you need to do after TSNE update
// This will run on the main thread
bullet.updateTSNEPosition(nodes);
tsne_start_time = ofGetElapsedTimef();
ofLog() << "Handling TSNE update completion on main thread";
}
void ofApp::keyReleased(int key){
//mapper.keyReleased(key);
}
//--------------------------------------------------------------
void ofApp::mouseDragged(int x, int y, int button){
//mapper.mouseDragged(x, y, button);
}
//--------------------------------------------------------------
void ofApp::mousePressed(int x, int y, int button){
//mapper.mousePressed(x, y, button);
}
//--------------------------------------------------------------
void ofApp::mouseReleased(int x, int y, int button){
//mapper.mouseReleased(x, y, button);
}
ofMesh ofApp::createCustomPlane(float width, float height, int numX, int numY) {
ofMesh mesh;
mesh.setMode(OF_PRIMITIVE_TRIANGLES);
// Create vertices with non-uniform distribution
for(int y = 0; y <= numY; y++) {
for(int x = 0; x <= numX; x++) {
// Get base uniform position
float xPos = (float)x/numX * width;
float yPos = (float)y/numY * height;
float texX = (float)x/numX;
float texY = 1.0 - (float)y/numY;
// Add random offset but keep edges fixed
if(x != 0 && x != numX && y != 0 && y != numY) {
// Random offset proportional to cell size
float cellWidth = width/numX;
float cellHeight = height/numY;
xPos += ofRandom(-cellWidth * 0.2, cellWidth * 0.2);
yPos += ofRandom(-cellHeight * 0.2, cellHeight * 0.2);
}
mesh.addVertex(glm::vec3(xPos, yPos, 0));
// Add texture coordinates (normalized)
mesh.addTexCoord(glm::vec2(texX, texY));
mesh.addColor(ofColor(255, 255, 255, 255));
}
}
// Create triangles
for(int y = 0; y < numY; y++) {
for(int x = 0; x < numX; x++) {
// Get indices for corners of quad
int nw = y * (numX + 1) + x;
int ne = nw + 1;
int sw = nw + (numX + 1);
int se = sw + 1;
// Add triangles
mesh.addIndex(nw);
mesh.addIndex(ne);
mesh.addIndex(se);
mesh.addIndex(nw);
mesh.addIndex(se);
mesh.addIndex(sw);
}
}
return mesh;
}