mirror of
https://github.com/game-stop/veejay.git
synced 2025-12-15 04:10:00 +01:00
293 lines
7.5 KiB
C
293 lines
7.5 KiB
C
/*
|
|
* LIBVJE - veejay fx library
|
|
*
|
|
* Copyright(C)2015 Niels Elburg <nwelburg@gmail.com>
|
|
* See COPYING for software license and distribution details
|
|
*/
|
|
/*
|
|
* EffecTV - Realtime Digital Video Effector
|
|
* Copyright (C) 2001-2006 FUKUCHI Kentaro
|
|
*
|
|
* DisplayWall
|
|
* Copyright (C) 2005-2006 FUKUCHI Kentaro
|
|
*
|
|
*/
|
|
|
|
#ifndef IS_LIVIDO_PLUGIN
|
|
#define IS_LIVIDO_PLUGIN
|
|
#endif
|
|
|
|
#include "../libplugger/specs/livido.h"
|
|
LIVIDO_PLUGIN
|
|
#include "utils.h"
|
|
#include "livido-utils.c"
|
|
|
|
typedef struct
|
|
{
|
|
int *vecx;
|
|
int *vecy;
|
|
int scale;
|
|
int dx, dy;
|
|
int bx, by;
|
|
int speed;
|
|
int speedi;
|
|
int cx, cy;
|
|
} displaywall_t;
|
|
|
|
static void displaywall_initVec(displaywall_t *wall, int w, int h)
|
|
{
|
|
int x, y, i;
|
|
double vx, vy;
|
|
|
|
i = 0;
|
|
for(y=0; y < h; y++) {
|
|
for(x=0; x < w; x++) {
|
|
vx = (double)(x - wall->cx) / w;
|
|
vy = (double)(y - wall->cy) / w;
|
|
|
|
vx *= 1.0 - vx * vx * 0.4;
|
|
vy *= 1.0 - vx * vx * 0.8;
|
|
vx *= 1.0 - (double)y / h * 0.15;
|
|
wall->vecx[i] = vx * w;
|
|
wall->vecy[i] = vy * w;
|
|
|
|
i++;
|
|
}
|
|
}
|
|
}
|
|
|
|
static void displaywall_tick( displaywall_t *wall, int video_width, int video_height )
|
|
{
|
|
wall->speed += wall->speedi;
|
|
if(wall->speed < 0) wall->speed = 0;
|
|
|
|
wall->bx += wall->dx * wall->speed;
|
|
wall->by += wall->dy * wall->speed;
|
|
while(wall->bx < 0) wall->bx += video_width;
|
|
while(wall->bx >= video_width) wall->bx -= video_width;
|
|
while(wall->by < 0) wall->by += video_height;
|
|
while(wall->by >= video_height) wall->by -= video_height;
|
|
|
|
if(wall->scale == 1) {
|
|
wall->bx = wall->cx;
|
|
wall->by = wall->cy;
|
|
}
|
|
}
|
|
|
|
static int displaywall_draw(displaywall_t *wall, uint8_t *src, uint8_t *dst, int video_width, int video_height)
|
|
{
|
|
int x, y, i;
|
|
int px, py;
|
|
|
|
i = 0;
|
|
for(y=0; y<video_height; y++) {
|
|
for(x=0; x<video_width; x++) {
|
|
px = wall->bx + wall->vecx[i] * wall->scale;
|
|
py = wall->by + wall->vecy[i] * wall->scale;
|
|
while(px < 0) px += video_width;
|
|
while(px >= video_width) px -= video_width;
|
|
while(py < 0) py += video_height;
|
|
while(py >= video_height) py -= video_height;
|
|
|
|
dst[i++] = src[py * video_width + px];
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
int init_instance( livido_port_t *my_instance )
|
|
{
|
|
int w = 0, h = 0;
|
|
lvd_extract_dimensions( my_instance, "out_channels", &w, &h );
|
|
|
|
int video_area = w * h * 3;
|
|
|
|
displaywall_t *entry = (displaywall_t*) livido_malloc(sizeof(displaywall_t));
|
|
if(entry == NULL) {
|
|
return LIVIDO_ERROR_MEMORY_ALLOCATION;
|
|
}
|
|
|
|
livido_memset( entry, 0, sizeof(displaywall_t));
|
|
|
|
entry->vecx = (int *)livido_malloc(sizeof(int) * video_area);
|
|
entry->vecy = (int *)livido_malloc(sizeof(int) * video_area);
|
|
|
|
if(entry->vecx == NULL || entry->vecy == NULL) {
|
|
if(entry->vecx) free(entry->vecx);
|
|
if(entry->vecy) free(entry->vecy);
|
|
free(entry);
|
|
return LIVIDO_ERROR_MEMORY_ALLOCATION;
|
|
}
|
|
|
|
entry->scale = 3;
|
|
entry->dx = 0;
|
|
entry->dy = 0;
|
|
entry->speed = 10;
|
|
entry->cx = w / 2;
|
|
entry->cy = h / 2;
|
|
|
|
displaywall_initVec(entry, w, h);
|
|
|
|
livido_property_set( my_instance, "PLUGIN_private", LIVIDO_ATOM_TYPE_VOIDPTR,1, &entry);
|
|
|
|
return LIVIDO_NO_ERROR;
|
|
}
|
|
|
|
|
|
int deinit_instance( livido_port_t *my_instance )
|
|
{
|
|
displaywall_t *wall = NULL;
|
|
livido_property_get( my_instance, "PLUGIN_private", 0, &wall );
|
|
|
|
if( wall ) {
|
|
if( wall->vecx ) free(wall->vecx);
|
|
if( wall->vecy ) free(wall->vecy);
|
|
free(wall);
|
|
}
|
|
|
|
return LIVIDO_NO_ERROR;
|
|
}
|
|
|
|
|
|
int process_instance( livido_port_t *my_instance, double timecode )
|
|
{
|
|
uint8_t *A[4] = {NULL,NULL,NULL,NULL};
|
|
uint8_t *O[4]= {NULL,NULL,NULL,NULL};
|
|
|
|
int palette;
|
|
int w;
|
|
int h;
|
|
|
|
int error = lvd_extract_channel_values( my_instance, "out_channels", 0, &w,&h, O,&palette );
|
|
if( error != LIVIDO_NO_ERROR )
|
|
return LIVIDO_ERROR_NO_OUTPUT_CHANNELS;
|
|
|
|
error = lvd_extract_channel_values( my_instance, "in_channels" , 0, &w, &h, A, &palette );
|
|
if( error != LIVIDO_NO_ERROR )
|
|
return LIVIDO_ERROR_NO_INPUT_CHANNELS;
|
|
|
|
int scale = lvd_extract_param_index( my_instance,"in_parameters", 0 );
|
|
int mode = lvd_extract_param_index( my_instance,"in_parameters", 1 );
|
|
int shift = 1;
|
|
|
|
if( palette == LIVIDO_PALETTE_YUV444P )
|
|
shift = 0;
|
|
|
|
displaywall_t *wall = NULL;
|
|
livido_property_get( my_instance, "PLUGIN_private", 0, &wall );
|
|
|
|
switch( mode ) {
|
|
case 0:
|
|
wall->dx = 0; wall->dy = 1;
|
|
break;
|
|
case 1:
|
|
wall->dy = 1; wall->dx = 0;
|
|
break;
|
|
case 2:
|
|
wall->dy = -1; wall->dx = 0;
|
|
break;
|
|
case 3:
|
|
wall->dy = 0; wall->dx = -1;
|
|
break;
|
|
}
|
|
|
|
wall->scale = scale;
|
|
|
|
displaywall_tick( wall, w, h );
|
|
|
|
displaywall_draw( wall, A[0], O[0], w, h);
|
|
displaywall_draw( wall, A[1], O[1], w >> shift, h );
|
|
displaywall_draw( wall, A[2], O[2], w >> shift, h );
|
|
|
|
return LIVIDO_NO_ERROR;
|
|
}
|
|
|
|
livido_port_t *livido_setup(livido_setup_t list[], int version)
|
|
|
|
{
|
|
LIVIDO_IMPORT(list);
|
|
|
|
livido_port_t *port = NULL;
|
|
livido_port_t *in_params[2];
|
|
livido_port_t *in_chans[3];
|
|
livido_port_t *out_chans[1];
|
|
livido_port_t *info = NULL;
|
|
livido_port_t *filter = NULL;
|
|
|
|
//@ setup root node, plugin info
|
|
info = livido_port_new( LIVIDO_PORT_TYPE_PLUGIN_INFO );
|
|
port = info;
|
|
|
|
livido_set_string_value( port, "maintainer", "Niels");
|
|
livido_set_string_value( port, "version","1");
|
|
|
|
filter = livido_port_new( LIVIDO_PORT_TYPE_FILTER_CLASS );
|
|
livido_set_int_value( filter, "api_version", LIVIDO_API_VERSION );
|
|
|
|
//@ setup function pointers
|
|
livido_set_voidptr_value( filter, "deinit_func", &deinit_instance );
|
|
livido_set_voidptr_value( filter, "init_func", &init_instance );
|
|
livido_set_voidptr_value( filter, "process_func", &process_instance );
|
|
port = filter;
|
|
|
|
//@ meta information
|
|
livido_set_string_value( port, "name", "Displaywall (EffecTV)");
|
|
livido_set_string_value( port, "description", "Displaywall");
|
|
livido_set_string_value( port, "author", "Kentaro Fukuchi"); //ported to livido
|
|
|
|
livido_set_int_value( port, "flags", 0);
|
|
livido_set_string_value( port, "license", "GPL2");
|
|
livido_set_int_value( port, "version", 1);
|
|
|
|
//@ some palettes veejay-classic uses
|
|
int palettes0[] = {
|
|
LIVIDO_PALETTE_YUV444P,
|
|
0,
|
|
};
|
|
|
|
//@ setup output channel
|
|
out_chans[0] = livido_port_new( LIVIDO_PORT_TYPE_CHANNEL_TEMPLATE );
|
|
port = out_chans[0];
|
|
|
|
livido_set_string_value( port, "name", "Output Channel");
|
|
livido_set_int_array( port, "palette_list", 2, palettes0);
|
|
livido_set_int_value( port, "flags", 0);
|
|
|
|
in_chans[0] = livido_port_new( LIVIDO_PORT_TYPE_CHANNEL_TEMPLATE );
|
|
port = in_chans[0];
|
|
livido_set_string_value( port, "name", "Input Channel");
|
|
livido_set_int_array( port, "palette_list", 2, palettes0);
|
|
livido_set_int_value( port, "flags", 0);
|
|
|
|
//@ setup parameters (INDEX type, 0-255)
|
|
in_params[0] = livido_port_new( LIVIDO_PORT_TYPE_PARAMETER_TEMPLATE );
|
|
port = in_params[0];
|
|
|
|
livido_set_string_value(port, "name", "Scale" );
|
|
livido_set_string_value(port, "kind", "INDEX" );
|
|
livido_set_int_value( port, "min", 2 );
|
|
livido_set_int_value( port, "max", 10 );
|
|
livido_set_int_value( port, "default", 2 );
|
|
livido_set_string_value( port, "description" ,"Scale");
|
|
|
|
|
|
in_params[1] = livido_port_new( LIVIDO_PORT_TYPE_PARAMETER_TEMPLATE );
|
|
port = in_params[1];
|
|
|
|
livido_set_string_value(port, "name", "Mode" );
|
|
livido_set_string_value(port, "kind", "INDEX" );
|
|
livido_set_int_value( port, "min", 0 );
|
|
livido_set_int_value( port, "max", 3 );
|
|
livido_set_int_value( port, "default", 0 );
|
|
livido_set_string_value( port, "description" ,"Mode");
|
|
|
|
//@ setup the nodes
|
|
livido_set_portptr_array( filter, "in_parameter_templates",2, in_params );
|
|
livido_set_portptr_array( filter, "out_channel_templates", 1, out_chans );
|
|
livido_set_portptr_array( filter, "in_channel_templates",1, in_chans );
|
|
|
|
livido_set_portptr_value(info, "filters", filter);
|
|
return info;
|
|
}
|