Crop Compositing Node:

This commit adds a new node, crop, to the compositor. This node can be used to
crop input image. It has two modes of operation. It can either crop image
size (Crop Image Size option) or crop while retaining original size of the
image. This latter mode can be used to preview the crop.

Use X1, Y1, X2, Y2 controls to manage the area to be cropped.

Note that I added a check for image preview min and max values to node_update.
This is because it could give inappropriate values in certain cases when
Crop Image Size option was toggled (values such as x1=0, y1=0, x2=60, y2=0 would
result in eternal loop due to bad min and max (min bigger than max!)). The check 
makes sure that min and max values are always valid.
This commit is contained in:
Juho Vepsalainen
2007-10-29 14:37:19 +00:00
parent 8a3f926938
commit d6db819a83
6 changed files with 179 additions and 2 deletions

View File

@@ -302,6 +302,7 @@ void set_node_shader_lamp_loop(void (*lamp_loop_func)(struct ShadeInput *, str
#define CMP_NODE_GAMMA 250 #define CMP_NODE_GAMMA 250
#define CMP_NODE_INVERT 251 #define CMP_NODE_INVERT 251
#define CMP_NODE_NORMALIZE 252 #define CMP_NODE_NORMALIZE 252
#define CMP_NODE_CROP 253
#define CMP_NODE_GLARE 301 #define CMP_NODE_GLARE 301
#define CMP_NODE_TONEMAP 302 #define CMP_NODE_TONEMAP 302

View File

@@ -2363,6 +2363,7 @@ static void registerCompositNodes(ListBase *ntypelist)
nodeRegisterType(ntypelist, &cmp_node_rotate); nodeRegisterType(ntypelist, &cmp_node_rotate);
nodeRegisterType(ntypelist, &cmp_node_scale); nodeRegisterType(ntypelist, &cmp_node_scale);
nodeRegisterType(ntypelist, &cmp_node_flip); nodeRegisterType(ntypelist, &cmp_node_flip);
nodeRegisterType(ntypelist, &cmp_node_crop);
nodeRegisterType(ntypelist, &cmp_node_displace); nodeRegisterType(ntypelist, &cmp_node_displace);
nodeRegisterType(ntypelist, &cmp_node_mapuv); nodeRegisterType(ntypelist, &cmp_node_mapuv);

View File

@@ -214,6 +214,10 @@ typedef struct NodeChroma {
float key[4]; float key[4];
} NodeChroma; } NodeChroma;
typedef struct NodeTwoXYs {
short x1, x2, y1, y2;
} NodeTwoXYs;
typedef struct NodeGeometry { typedef struct NodeGeometry {
char uvname[32]; char uvname[32];
char colname[32]; char colname[32];

View File

@@ -94,6 +94,7 @@ extern bNodeType cmp_node_translate;
extern bNodeType cmp_node_rotate; extern bNodeType cmp_node_rotate;
extern bNodeType cmp_node_scale; extern bNodeType cmp_node_scale;
extern bNodeType cmp_node_flip; extern bNodeType cmp_node_flip;
extern bNodeType cmp_node_crop;
extern bNodeType cmp_node_displace; extern bNodeType cmp_node_displace;
extern bNodeType cmp_node_mapuv; extern bNodeType cmp_node_mapuv;

View File

@@ -0,0 +1,119 @@
/**
*
*
* ***** BEGIN GPL LICENSE BLOCK *****
*
* This program is free software; you can redistribute it and/or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation; either version 2
* of the License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software Foundation,
* Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
*
* The Original Code is Copyright (C) 2006 Blender Foundation.
* All rights reserved.
*
* The Original Code is: all of this file.
*
* Contributor(s): Juho Vepsäläinen
*
* ***** END GPL LICENSE BLOCK *****
*/
#include "../CMP_util.h"
/* **************** Crop ******************** */
static bNodeSocketType cmp_node_crop_in[]= {
{ SOCK_RGBA, 1, "Image", 0.8f, 0.8f, 0.8f, 1.0f, 0.0f, 1.0f},
{ -1, 0, "" }
};
static bNodeSocketType cmp_node_crop_out[]= {
{ SOCK_RGBA, 0, "Image", 0.0f, 0.0f, 0.0f, 1.0f, 0.0f, 1.0f},
{ -1, 0, "" }
};
static void node_composit_exec_crop(void *data, bNode *node, bNodeStack **in, bNodeStack **out)
{
if(in[0]->data) {
NodeTwoXYs *ntxy= node->storage;
CompBuf *cbuf= in[0]->data;
CompBuf *stackbuf;
int x, y;
float *srcfp, *outfp;
rcti outputrect;
/* check input image size */
if(cbuf->x <= ntxy->x1 + 1)
ntxy->x1= cbuf->x - 1;
if(cbuf->y <= ntxy->y1 + 1)
ntxy->y1= cbuf->y - 1;
if(cbuf->x <= ntxy->x2 + 1)
ntxy->x2= cbuf->x - 1;
if(cbuf->y <= ntxy->y2 + 1)
ntxy->y2= cbuf->y - 1;
/* figure out the minimums and maximums */
outputrect.xmax=MAX2(ntxy->x1, ntxy->x2) + 1;
outputrect.xmin=MIN2(ntxy->x1, ntxy->x2);
outputrect.ymax=MAX2(ntxy->y1, ntxy->y2) + 1;
outputrect.ymin=MIN2(ntxy->y1, ntxy->y2);
if(node->custom1) {
/* this option crops the image size too */
stackbuf= get_cropped_compbuf(&outputrect, cbuf->rect, cbuf->x, cbuf->y, cbuf->type);
}
else {
/* this option won't crop the size of the image as well */
/* allocate memory for the output image */
stackbuf = alloc_compbuf(cbuf->x, cbuf->y, cbuf->type, 1);
/* select the cropped part of the image and set it to the output */
for(y=outputrect.ymin; y<outputrect.ymax; y++){
srcfp= cbuf->rect + (y * cbuf->x + outputrect.xmin) * cbuf->type;
outfp= stackbuf->rect + (y * stackbuf->x + outputrect.xmin) * stackbuf->type;
for(x=outputrect.xmin; x<outputrect.xmax; x++, outfp+= stackbuf->type, srcfp+= cbuf->type)
memcpy(outfp, srcfp, sizeof(float)*stackbuf->type);
}
}
out[0]->data= stackbuf;
}
}
static void node_composit_init_crop(bNode* node)
{
NodeTwoXYs *nxy= MEM_callocN(sizeof(NodeTwoXYs), "node xy data");
node->storage= nxy;
nxy->x1= 0;
nxy->x2= 0;
nxy->y1= 0;
nxy->y2= 0;
}
bNodeType cmp_node_crop= {
/* *next,*prev */ NULL, NULL,
/* type code */ CMP_NODE_CROP,
/* name */ "Crop",
/* width+range */ 140, 100, 320,
/* class+opts */ NODE_CLASS_DISTORT, NODE_OPTIONS,
/* input sock */ cmp_node_crop_in,
/* output sock */ cmp_node_crop_out,
/* storage */ "NodeTwoXYs",
/* execfunc */ node_composit_exec_crop,
/* butfunc */ NULL,
/* initfunc */ node_composit_init_crop,
/* freestoragefunc */ node_free_standard_storage,
/* copystoragefunc */ node_copy_standard_storage,
/* id */ NULL
};

View File

@@ -1330,6 +1330,50 @@ static int node_composit_buts_flip(uiBlock *block, bNodeTree *ntree, bNode *node
return 20; return 20;
} }
static int node_composit_buts_crop(uiBlock *block, bNodeTree *ntree, bNode *node, rctf *butr)
{
if(block) {
NodeTwoXYs *ntxy= node->storage;
uiBut *bt;
char elementheight = 19;
short dx= (butr->xmax-butr->xmin)/2;
short dy= butr->ymax - elementheight;
short xymin= 0, xymax= 10000;
uiBlockBeginAlign(block);
/* crop image size toggle */
uiDefButS(block, TOG, B_NODE_EXEC+node->nr, "Crop Image Size",
butr->xmin, dy, dx*2, elementheight,
&node->custom1, 0, 0, 0, 0, "Crop the size of the input image.");
dy-=elementheight;
/* x1 */
bt=uiDefButS(block, NUM, B_NODE_EXEC+node->nr, "X1:",
butr->xmin, dy, dx, elementheight,
&ntxy->x1, xymin, xymax, 0, 0, "");
/* y1 */
bt=uiDefButS(block, NUM, B_NODE_EXEC+node->nr, "Y1:",
butr->xmin+dx, dy, dx, elementheight,
&ntxy->y1, xymin, xymax, 0, 0, "");
dy-=elementheight;
/* x2 */
bt=uiDefButS(block, NUM, B_NODE_EXEC+node->nr, "X2:",
butr->xmin, dy, dx, elementheight,
&ntxy->x2, xymin, xymax, 0, 0, "");
/* y2 */
bt=uiDefButS(block, NUM, B_NODE_EXEC+node->nr, "Y2:",
butr->xmin+dx, dy, dx, elementheight,
&ntxy->y2, xymin, xymax, 0, 0, "");
uiBlockEndAlign(block);
}
return 60;
}
static int node_composit_buts_splitviewer(uiBlock *block, bNodeTree *ntree, bNode *node, rctf *butr) static int node_composit_buts_splitviewer(uiBlock *block, bNodeTree *ntree, bNode *node, rctf *butr)
{ {
if(block) { if(block) {
@@ -1758,6 +1802,9 @@ static void node_composit_set_butfunc(bNodeType *ntype)
case CMP_NODE_VALTORGB: case CMP_NODE_VALTORGB:
ntype->butfunc= node_buts_valtorgb; ntype->butfunc= node_buts_valtorgb;
break; break;
case CMP_NODE_CROP:
ntype->butfunc= node_composit_buts_crop;
break;
case CMP_NODE_BLUR: case CMP_NODE_BLUR:
ntype->butfunc= node_composit_buts_blur; ntype->butfunc= node_composit_buts_blur;
break; break;
@@ -2134,6 +2181,10 @@ static void node_update(bNode *node)
} }
dy= node->prvr.ymin - NODE_DYS/2; dy= node->prvr.ymin - NODE_DYS/2;
/* make sure that maximums are bigger or equal to minimums */
if(node->prvr.xmax < node->prvr.xmin) SWAP(float, node->prvr.xmax, node->prvr.xmin);
if(node->prvr.ymax < node->prvr.ymin) SWAP(float, node->prvr.ymax, node->prvr.ymin);
} }
else { else {
float oldh= node->prvr.ymax - node->prvr.ymin; float oldh= node->prvr.ymax - node->prvr.ymin;