/***************************************************************************** Major portions of this software are copyrighted by the Medical College of Wisconsin, 1994-2000, and are released under the Gnu General Public License, Version 2. See the file README.Copyright for details. ******************************************************************************/ /* rcr todo: * * - reload bigmode widgets */ /*---------------------------------------------------------------------- $Id: plug_crender.c,v 1.37 2007/03/20 16:43:04 rwcox Exp $ ---------------------------------------------------------------------- */ #define PLUG_CRENDER_VERSION "Version 1.9 " /***********************************************************************/ /* VERSION HISTORY */ static char g_cren_hist[] = "plug_crender.c version history:\n" "\n" " 0.0 Original version is plug_render.c, by RW Cox.\n" "\n" " 1.0 08 March 2002 [rickr]\n" " - initial release, plug_crender.c\n" " - replaced MREN_ library calls with CREN_ calls\n" " (from the new Cox RENder library)\n" "\n" " 1.3 01 July 2002 [rickr]\n" " - re-orient overlay and underlay to rai\n" " o see dset_or and fset_or\n" "\n" " 1.4 29 July 2002 [rickr]\n" " - underlay data set grid need not match that of overlay\n" " o see gcr.mset and new_fset\n" "\n" " 1.5 05 August 2002 [rickr]\n" " - aligned crosshairs to resampled grid\n" " o see RCREND_xhair_underlay/RCREND_xhair_overlay\n" " o passing mset to both RCREND_xhair_XXX functions\n" " - added ENTRY(), EXRETURN and RETURN() statements\n" "\n" " 1.6 26 September 2004 [rickr]\n" " - draw crosshairs directly onto the rendered image\n" " o see gcr.hairs\n" " - added sneaky (shhhh...) debugging interface\n" " o access via 'dh' in opacity box\n" "\n" " 1.7 23 October 2002 [rickr] - incremental rotation is now the default\n" "\n" " 1.8 22 July 2003 [rickr] - handle bigmode color bar (see v1.8)\n" "\n" " 1.9 27 July 2004 [rickr]\n" " - updated calls to r_new_resam_dset, passing sublist\n" " (now resampling is done only for displayed sub-bricks)\n" " - created printable history through debug interface\n" " (via 'dh', where 'd?' is now debug help\n" "\n" " 1.9a 22 March 2005 [rickr] - removed all tabs\n" " 1.10 4 April 2006 [rickr] - handle NPANE_BIG > 128 for bigmode\n" " 1.11 16 Jan 2007 [rickr] - Pos for bigmode had not been properly applied\n" "\n"; /***********************************************************************/ #include "afni.h" #include "cox_render.h" #include "mcw_graf.h" #include "parser.h" #include #include "rickr/r_new_resam_dset.h" #include "rickr/r_idisp.h" #ifndef ALLOW_PLUGINS # error "Plugins not properly set up -- see machdep.h" #endif #undef RCREND_DEBUG /* #define ONLY_AXIAL rickr - 1/10/02 */ /*********************************************************************** Plugin to render a volume dataset. Makes a custom interface. Modified from plug_drawdset.c by RWCox - February 1999. ************************************************************************/ /*---------- prototypes for internal routines ----------*/ char * RCREND_main( PLUGIN_interface * ) ; /* called from AFNI */ void RCREND_make_widgets(void) ; /* initialization */ void RCREND_done_CB ( Widget , XtPointer , XtPointer ) ; /* Done button */ void RCREND_draw_CB ( Widget , XtPointer , XtPointer ) ; /* Draw button */ void RCREND_help_CB ( Widget , XtPointer , XtPointer ) ; /* Help button */ void RCREND_reload_CB ( Widget , XtPointer , XtPointer ) ; /* Reload button */ void RCREND_choose_CB ( Widget , XtPointer , XtPointer ) ; /* Choose button */ void RCREND_xhair_CB ( Widget , XtPointer , XtPointer ) ; /* See Xhairs toggle */ void RCREND_dynamic_CB( Widget , XtPointer , XtPointer ) ; /* DynaDraw toggle */ void RCREND_accum_CB ( Widget , XtPointer , XtPointer ) ; /* Accumulate toggle */ void RCREND_angle_CB ( MCW_arrowval * , XtPointer ) ; /* Angle arrowvals */ void RCREND_param_CB ( MCW_arrowval * , XtPointer ) ; /* Cutout arrowvals */ void RCREND_interp_CB ( MCW_arrowval * , XtPointer ) ; /* Precalc menu */ void RCREND_clip_CB ( MCW_arrowval * , XtPointer ) ; /* Clip arrowvals */ void RCREND_xhair_recv( int,int , int *, void * ) ; /* 29 Mar 1999 */ void RCREND_environ_CB( char * ) ; /* 20 Jun 2000 */ void RCREND_choose_av_CB ( MCW_arrowval * , XtPointer ) ; /* Sub-brick menus */ char * RCREND_choose_av_label_CB( MCW_arrowval * , XtPointer ) ; void RCREND_opacity_scale_CB ( MCW_arrowval * , XtPointer ) ; void RCREND_finalize_dset_CB( Widget , XtPointer , MCW_choose_cbs * ) ; /* dataset chosen */ void RCREND_reload_dataset(void) ; /* actual reloading work */ void RCREND_xhair_underlay(THD_3dim_dataset *); /* make xhairs in underlay */ void RCREND_xhair_overlay (THD_3dim_dataset *, MRI_IMAGE *) ; /* in overlay */ static PLUGIN_interface * plint = NULL ; /* what AFNI sees */ void RCREND_textact_CB( Widget , XtPointer , XtPointer ) ; /* press Enter in a textfield */ static float angle_fstep = 5.0 ; static float cutout_fstep = 5.0 ; #define RCREND_NUM_interp_modes 3 static char * interp_mode_strings[] = { "Neighbor" , "Twostep" , "Linear" } ; static int interp_mode[] = { CREN_NN,CREN_TWOSTEP,CREN_LINEAR } ; static int interp_ival = CREN_NN ; /*==========================================================================*/ #define ALLOW_INCROT /* 26 Apr 2002 - RWCox */ #ifdef ALLOW_INCROT /** prototypes for incremental rotation matrix/angles stuff **/ static void RCREND_inc_angles( int,float, float *,float *,float *) ; static THD_dmat33 RCREND_rotmatrix( int,double , int,double , int,double ) ; static void RCREND_rotmatrix_to_angles( THD_dmat33, double *, double *, double * ) ; /** toggle button for incremental rotations **/ static MCW_bbox *incrot_bbox ; /** callback for above toggle **/ static void RCREND_incrot_CB(Widget,XtPointer,XtPointer) ; /** prototype for what happens when user increments angles **/ static void RCREND_do_incrot( MCW_arrowval * ) ; #endif /* ALLOW_INCROT */ /*==========================================================================*/ /*********************************************************************** Set up the interface to the user. Note that we bypass the normal interface creation, and simply have the menu selection directly call the main function, which will create a custom set of interface widgets. ************************************************************************/ DEFINE_PLUGIN_PROTOTYPE PLUGIN_interface * PLUGIN_init( int ncall ) { char * env ; float val ; if( ncall > 0 ) return(NULL); /* only one interface */ plint = PLUTO_new_interface( "Render [new]" , NULL , NULL , PLUGIN_CALL_IMMEDIATELY , RCREND_main ) ; PLUTO_add_hint( plint , "Volume Rendering" ) ; PLUTO_set_sequence( plint , "A:graphics" ) ; /***----- retrieve relevant environment variables, if any -----***/ env = getenv("AFNI_RENDER_ANGLE_DELTA") ; if( env != NULL ){ val = strtod(env,NULL) ; if( val > 0.0 && val < 100.0 ) angle_fstep = val ; } PLUTO_register_environment_numeric( "AFNI_RENDER_ANGLE_DELTA" , "Angle stepsize in deg (volume renderer)" , 1,9,0,(int)angle_fstep, RCREND_environ_CB ); /*--*/ env = getenv("AFNI_RENDER_CUTOUT_DELTA") ; if( env != NULL ){ val = strtod(env,NULL) ; if( val > 0.0 && val < 100.0 ) cutout_fstep = val ; } PLUTO_register_environment_numeric( "AFNI_RENDER_CUTOUT_DELTA" , "Cutout stepsize in mm (volume renderer)" , 1,9,0,(int)cutout_fstep, RCREND_environ_CB ); #if 0 /* 2002 Mar 06 : showthru factor has moved to the plugin GUI */ env = getenv("AFNI_RENDER_SHOWTHRU_FAC") ; if( env != NULL ){ val = strtod(env,NULL) ; if( val < 0.0 || val > 1.0 ) val = 1.0 ; } else { val = 1.0 ; } PLUTO_register_environment_numeric( "AFNI_RENDER_SHOWTHRU_FAC" , "ShowThru mixing factor (volume renderer)", 30,100,2,(int)rint(100.0*val) , NULL ) ; #endif /*-- done --*/ return(plint); } /*************************************************************************** Internal data structures ****************************************************************************/ #define NO_DATASET_STRING "[No Dataset is Loaded]" /* Interface widgets */ static Widget shell=NULL , anat_rowcol , info_lab , choose_pb ; static Widget done_pb , help_pb , draw_pb , reload_pb ; static MCW_arrowval * roll_av , * pitch_av , * yaw_av , * interp_av ; static MCW_bbox * xhair_bbox , * dynamic_bbox , * accum_bbox ; static MCW_arrowval * choose_av , * opacity_scale_av ; /* 08 Mar 2001 - stuff for colored xhairs */ extern void RCREND_xhair_EV( Widget, XtPointer, XEvent *, Boolean * ) ; extern void RCREND_xhair_ovc_CB( Widget, XtPointer, MCW_choose_cbs * ) ; static int xhair_ovc = 0 ; /* 17 Jun 2005 - stuff for overlay labels */ extern void RCREND_accum_lab_EV( Widget, XtPointer, XEvent *, Boolean * ) ; extern void RCREND_accum_lab_CB( Widget, XtPointer, MCW_choose_cbs * ) ; static char accum_label[256] = "\0" ; static int accum_lab_replace = 0 ; static char * RCREND_dummy_av_label[2] = { "[Nothing At All]" , "[Nothing At All]" } ; static Widget top_rowcol , anat_frame ; #define CLIP_RANGE 32767 static Widget range_lab ; static MCW_arrowval * clipbot_av , * cliptop_av ; static float brickfac = 0.0 ; static Widget range_faclab , clipbot_faclab , cliptop_faclab ; void RCREND_graf_CB( MCW_graf * , void * ) ; static MCW_graf * opa_graf ; static MCW_graf * gry_graf ; static MCW_pasgraf * his_graf ; static char * xhair_bbox_label[1] = { "See Xhairs" } ; static char * dynamic_bbox_label[1] = { "DynaDraw" } ; static char * accum_bbox_label[1] = { "Accumulate" } ; /*----------------------------------------------------------------*/ /* rickr - cox rendering data */ typedef struct /* bigstuff v1.8 [rickr] */ { byte r[NPANE_BIG]; /* for ease of calling CREN_set_rgbmap() */ byte g[NPANE_BIG]; byte b[NPANE_BIG]; } CR_bigstuff; typedef struct { THD_fvec3 xp[2][2]; /* 12 points for the 6 xhair segments */ THD_fvec3 yp[2][2]; THD_fvec3 zp[2][2]; } CR_xhairs; typedef struct { void * rh; /* render handle */ float omap[GRAF_SIZE]; /* opacity map - graph data */ THD_3dim_dataset * dset_or; /* re-oriented dataset */ THD_3dim_dataset * fset_or; /* re-oriented func dataset */ THD_3dim_dataset * mset; /* master for producing fset_or */ FD_brick * fdm; /* FD_brick for matset dset */ THD_mat33 rotm; /* current rotation matrix */ CR_xhairs xhseg; /* 12 dicom crosshair segments */ CR_bigstuff bigstuff;/* for bigmode functional overlay*/ } CR_data; CR_data gcr; #define CRBM_IS_BLACK_INDEX( index ) \ ( ( wfunc_color_pbar->bigcolor[index].r == 0 ) && \ ( wfunc_color_pbar->bigcolor[index].g == 0 ) && \ ( wfunc_color_pbar->bigcolor[index].b == 0 ) ) /*----------------------------------------------------------------*/ /* debug stuff - rickr 2002.09.04 */ #define CR_MAX_DEBUG 2 #define CR_TEXT_LEN 512 typedef struct { THD_fvec3 xhairs; /* last cross hair coords */ int level; char text[CR_TEXT_LEN]; /* for debug sprintf functions */ } CR_debug; CR_debug gcr_debug; static int r_debug_check( CR_debug * d, char * str ); /*----------------------------------------------------------------*/ static int grcr_hist_low[256]; static int grcr_hist_high[256]; /* rickr - temp routines */ void rcr_disp_hist( unsigned char * im, int nvox, int b1, int cut, int b2 ); static void idisp_xhair_pts ( char * note, CR_xhairs * p ); /*----------------------------------------------------------------*/ /* new xhair routines */ #define BOUND_VAL(a,v,b) { if (v < (a)) v = (a); if (v > (b)) v = (b); } static int draw_xhairs_in_image( CR_xhairs * x, MRI_IMAGE * im ); static int draw_image_line( MRI_IMAGE * im, THD_fvec3 * p1, THD_fvec3 * p2, byte * rgb ); static int get_xhair_points( CR_xhairs * pts, THD_3dim_dataset * dset ); static int ovc_to_rgb_bytes( int ovc, byte * rgb, MCW_DCOV * ov ); static int rotate_xhair_points( CR_xhairs * xh, THD_mat33 * rotm ); static int xhairs_to_image_pts( CR_xhairs * xh, THD_3dim_dataset * dset ); /*----------------------------------------------------------------*/ /* Other data */ static MCW_DC * dc ; /* display context */ static Three_D_View * im3d ; /* AFNI controller */ static THD_3dim_dataset * dset ; /* rai orient dset */ static MCW_idcode dset_idc ; /* 31 Mar 1999 */ static int new_dset = 0 ; /* Is it new? */ static int new_fset = 0 ; /* Is func new? 28 June 2002 - rickr */ static int dset_ival = 0 ; /* Sub-brick index */ static char dset_title[THD_MAX_NAME] ; /* Title string */ static MRI_IMAGE * grim=NULL ; /* volumes to render */ static MRI_IMAGE * grim_showthru=NULL ; /* 07 Jan 2000 */ #define FREEIM(x) if( (x) != NULL ){ mri_free(x); (x)=NULL; } #define FREE_VOLUMES \ do{ FREEIM(grim) ; \ FREEIM(grim_showthru); } while(0) ; #define NEED_VOLUMES (grim == NULL) /* moved for more 'global' access 26 June 2002 - rickr */ # define IS_AXIAL_RAI(ds) ( ( (ds)->daxes->xxorient == ORI_R2L_TYPE ) && \ ( (ds)->daxes->yyorient == ORI_A2P_TYPE ) && \ ( (ds)->daxes->zzorient == ORI_I2S_TYPE ) ) # define IS_AXIAL_LPI(ds) ( ( (ds)->daxes->xxorient == ORI_L2R_TYPE ) && \ ( (ds)->daxes->yyorient == ORI_P2A_TYPE ) && \ ( (ds)->daxes->zzorient == ORI_I2S_TYPE ) ) /* rickr - 2002.09.03 */ #define FVEC_TIMES_MAT(v,A) \ ( tempA_fvec3.xyz[0] = (v).xyz[0] * (A).mat[0][0] \ +(v).xyz[1] * (A).mat[1][0] \ +(v).xyz[2] * (A).mat[2][0] ,\ tempA_fvec3.xyz[1] = (v).xyz[0] * (A).mat[0][1] \ +(v).xyz[1] * (A).mat[1][1] \ +(v).xyz[2] * (A).mat[2][1] ,\ tempA_fvec3.xyz[2] = (v).xyz[0] * (A).mat[0][2] \ +(v).xyz[1] * (A).mat[1][2] \ +(v).xyz[2] * (A).mat[2][2] , tempA_fvec3 ) /* rickr - 2002.09.24 */ #define DIV_FVEC3_BY_CONST(f,a) \ ( (f).xyz[0] = (f).xyz[0] / (a) , \ (f).xyz[1] = (f).xyz[1] / (a) , \ (f).xyz[2] = (f).xyz[2] / (a) ) static int dynamic_flag = 0 ; static int accum_flag = 0 ; static float angle_roll = 70.0 ; static float angle_pitch = 120.0 ; static float angle_yaw = 0.0 ; static int xhair_flag = 0 ; static int xhair_ixold = -666 ; /* remember the past */ static int xhair_jyold = -666 ; static int xhair_kzold = -666 ; static int xhair_omold = -666 ; /* 02 Jun 1999 */ static int xhair_recv = -1 ; /* 29 Mar 1999 */ #define CHECK_XHAIR_MOTION ( im3d->vinfo->i1 != xhair_ixold || \ im3d->vinfo->j2 != xhair_jyold || \ im3d->vinfo->k3 != xhair_kzold || \ im3d->vinfo->xhairs_orimask != xhair_omold ) static int new_data_loaded = 0 ; static int renderer_open = 0 ; static int npixels = 0 ; /*------------------ Stuff for the image display window -----------------*/ static MCW_imseq * imseq = NULL ; static MRI_IMARR * renderings = NULL ; void RCREND_open_imseq( void ) ; void RCREND_update_imseq( void ) ; void RCREND_destroy_imseq( void ) ; XtPointer RCREND_imseq_getim( int , int , XtPointer ) ; void RCREND_seq_send_CB( MCW_imseq * , XtPointer , ISQ_cbs * ) ; /*---------------- Stuff for Automate mode --------------*/ static MCW_bbox * automate_bbox ; static int automate_flag = 0 ; static char * automate_bbox_label[1] = { "Automate" } ; static MCW_arrowval * autoframe_av ; static Widget autocompute_pb , autocancel_pb ; void RCREND_autoflag_CB (Widget , XtPointer , XtPointer) ; /* Automate toggle */ void RCREND_autocompute_CB(Widget , XtPointer , XtPointer) ; /* Compute pushbutton */ void RCREND_autocancel_CB (Widget , XtPointer , XtPointer) ; /* Cancel pushbutton */ /*-------------------------- Stuff for cutout logic ----------------------*/ void RCREND_cutout_type_CB( MCW_arrowval * , XtPointer ) ; void RCREND_numcutout_CB ( MCW_arrowval * , XtPointer ) ; void RCREND_cutout_set_CB ( Widget , XtPointer , XtPointer ) ; typedef struct { /* widgets for a cutout */ Widget hrc , param_lab , set_pb ; MCW_arrowval * type_av , * param_av ; MCW_bbox * mustdo_bbox ; } RCREND_cutout ; RCREND_cutout * RCREND_make_cutout( int n ) ; /* makes the widgets */ #define MAX_CUTOUTS 9 static RCREND_cutout * cutouts[MAX_CUTOUTS] ; #define CUTOUT_OR 0 #define CUTOUT_AND 1 static char * cutout_logic_labels[] = { "OR" , "AND" } ; static char * cutout_mustdo_names[] = { "NO" , "YES" } ; static int num_cutouts = 0 ; static int logic_cutout = CUTOUT_OR ; MCW_arrowval * numcutout_av ; MCW_arrowval * logiccutout_av ; typedef struct { /* store the status */ int num , logic ; /* of the cutouts */ int type[MAX_CUTOUTS] , mustdo[MAX_CUTOUTS] ; float param[MAX_CUTOUTS] , opacity_scale ; char param_str[MAX_CUTOUTS][AV_MAXLEN+4] ; } CUTOUT_state ; #define MIN_OPACITY_SCALE 0.000 CUTOUT_state current_cutout_state , old_cutout_state ; void RCREND_load_cutout_state(void) ; /* load from widgets */ int RCREND_cutout_state_changed(void) ; /* has it changed? */ void RCREND_cutout_blobs(MRI_IMAGE *) ; /* actually do cutouts */ static char * mustdo_bbox_label[1] = { "Must Do" } ; /*----------------- stuff for evaluation of expressions ------------*/ static double atoz[26] ; /* values of 'a', 'b', ..., 'z' in expressions */ #define N_IND 13 /* 'n' */ #define T_IND 19 /* 't' */ #define X_IND 23 /* 'x' */ #define Y_IND 24 /* 'y' */ #define Z_IND 25 /* 'z' */ float RCREND_evaluate( MCW_arrowval * ) ; /*-------------------- Icon Pixmap for Image Window ----------------------*/ static Pixmap afni48ren_pixmap = XmUNSPECIFIED_PIXMAP ; #define afni48ren_width 48 #define afni48ren_height 48 /* from file afni48ren.xbm */ static unsigned char afni48ren_bits[] = { 0xff, 0xff, 0xc1, 0xc1, 0xff, 0xff, 0xff, 0x7f, 0x60, 0x00, 0xfe, 0xff, 0xff, 0x0f, 0x30, 0x10, 0xf0, 0xff, 0xff, 0x01, 0x37, 0xf0, 0x80, 0xff, 0x7f, 0xe0, 0x77, 0xe0, 0x07, 0xff, 0x7f, 0xfe, 0xe0, 0x00, 0x3f, 0xff, 0x7f, 0x1e, 0xc0, 0x03, 0x38, 0xff, 0x3f, 0x00, 0x87, 0xe7, 0x01, 0xff, 0x3f, 0xf0, 0x07, 0xe7, 0x0f, 0xfe, 0x3f, 0x7f, 0xc0, 0x04, 0x7e, 0xfe, 0x3f, 0x0f, 0xe6, 0x67, 0x70, 0xfe, 0x3f, 0xe0, 0xa7, 0xe7, 0x03, 0xfe, 0x1f, 0xfc, 0x21, 0x83, 0x3f, 0xfc, 0x9f, 0x1f, 0xe0, 0x00, 0xfc, 0xfc, 0x9f, 0x83, 0xc7, 0xe1, 0xc1, 0xfc, 0x1f, 0xf0, 0x87, 0xe7, 0x0f, 0xfc, 0x0f, 0x7f, 0x00, 0x07, 0x7e, 0xf8, 0xcf, 0x0f, 0xc0, 0x04, 0xf0, 0xf9, 0xcf, 0x80, 0xe7, 0xe7, 0x81, 0xf9, 0x0f, 0xf0, 0xa7, 0xe3, 0x1f, 0xf8, 0x0f, 0x7f, 0xe0, 0x00, 0xfe, 0xf9, 0xcf, 0x0f, 0xc0, 0x01, 0xe0, 0xf1, 0xc7, 0x00, 0x87, 0xe3, 0x00, 0xf0, 0x07, 0xf8, 0x07, 0xe7, 0x1f, 0xf0, 0x87, 0xff, 0xc0, 0x86, 0xff, 0xf1, 0xe7, 0x07, 0xe0, 0x03, 0xe0, 0xf3, 0x77, 0x00, 0xe0, 0x00, 0x00, 0xe6, 0x03, 0x00, 0xc0, 0x01, 0x00, 0xe0, 0x03, 0x00, 0x80, 0x03, 0x00, 0xe0, 0x03, 0xf8, 0x0f, 0xf8, 0x1f, 0xe0, 0x81, 0xff, 0x3f, 0xfc, 0xff, 0xc1, 0xff, 0xff, 0x7f, 0xfe, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x01, 0xf8, 0xff, 0xff, 0x1f, 0xc0, 0xff, 0x00, 0x00, 0x00, 0x00, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0x31, 0x74, 0x31, 0xc4, 0xe8, 0xc6, 0xad, 0x67, 0xad, 0xb7, 0xcd, 0xda, 0xad, 0x57, 0xad, 0xb7, 0xad, 0xfa, 0x31, 0x56, 0x2d, 0xc6, 0xad, 0xfa, 0xb5, 0x57, 0xad, 0xd7, 0xad, 0x8a, 0xad, 0x37, 0xad, 0xb7, 0x6d, 0xda, 0xad, 0x77, 0xad, 0xb7, 0xed, 0xda, 0x2d, 0x74, 0x31, 0xb4, 0xe8, 0xc6, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xab, 0xaa, 0xaa, 0xaa, 0xaa, 0xaa}; /*-------------------- Functional overlay stuff ----------------------*/ #define INVALIDATE_OVERLAY do{ FREEIM(ovim) ; } while(0) #define DO_OVERLAY ((func_dset != NULL && func_see_overlay) || \ func_see_ttatlas || (xhair_flag && xhair_ovc > 0) ) #define NEED_OVERLAY (DO_OVERLAY && ovim == NULL) #define NEED_RELOAD (NEED_VOLUMES || NEED_OVERLAY) #define TURNOFF_OVERLAY_WIDGETS \ do{ XmString xstr ; \ xstr = XmStringCreateLtoR( NO_DATASET_STRING , \ XmFONTLIST_DEFAULT_TAG ) ; \ XtVaSetValues( wfunc_info_lab , XmNlabelString,xstr , NULL ) ; \ XmStringFree(xstr) ; \ \ xstr = RCREND_range_label() ; \ XtVaSetValues( wfunc_range_label , XmNlabelString , xstr , NULL ) ; \ XmStringFree(xstr) ; \ \ xstr = RCREND_autorange_label() ; \ XtVaSetValues( wfunc_range_bbox->wbut[0], XmNlabelString,xstr, NULL ) ; \ XmStringFree(xstr) ; \ \ AV_SENSITIZE( wfunc_color_av , False ) ; \ AV_SENSITIZE( wfunc_thresh_av , False ) ; \ } while(0) void RCREND_func_widgets(void) ; void RCREND_init_cmap(void) ; void RCREND_reload_func_dset(void) ; void RCREND_reload_renderer(void) ; void RCREND_overlay_ttatlas(void) ; /* 12 Jul 2001 */ static Widget wfunc_open_pb ; void RCREND_open_func_CB( Widget , XtPointer , XtPointer ) ; static Widget wfunc_frame=NULL , wfunc_rowcol , wfunc_choose_pb , wfunc_uber_rowcol , wfunc_info_lab , wfunc_vsep ; static Widget wfunc_thr_rowcol , wfunc_thr_label , wfunc_thr_scale=NULL , wfunc_thr_pval_label ; static MCW_arrowval * wfunc_thr_top_av ; static Widget wfunc_color_rowcol , wfunc_color_label ; static MCW_pbar * wfunc_color_pbar=NULL ; static MCW_arrowval * wfunc_color_av , * wfunc_thresh_av , * wfunc_colornum_av ; static MCW_bbox * wfunc_color_bbox ; static Widget wfunc_choices_rowcol , wfunc_choices_label , wfunc_buck_frame , wfunc_buck_rowcol , wfunc_opacity_frame , wfunc_opacity_rowcol , wfunc_range_rowcol , wfunc_range_frame ; static Widget wfunc_range_label ; static MCW_arrowval * wfunc_opacity_av , * wfunc_ST_fac_av , * wfunc_range_av ; static MCW_arrowval * wfunc_range_rotate_av ; /* 30 Mar 2001 */ static MCW_bbox * wfunc_see_overlay_bbox , * wfunc_cut_overlay_bbox , * wfunc_kill_clusters_bbox , * wfunc_range_bbox , * wfunc_do_ST_bbox ; /* 2002 Mar 06 */ static MCW_arrowval * wfunc_clusters_rmm_av , * wfunc_clusters_vmul_av ; static MCW_bbox * wfunc_see_ttatlas_bbox ; /* 24 Jul 2001 */ static Widget wfunc_pbar_menu , wfunc_pbar_equalize_pb , wfunc_pbar_settop_pb ; static Widget wfunc_pbar_saveim_pb ; static MCW_arrowval * wfunc_pbar_palette_av ; static MCW_arrowval * wfunc_pbar_mixshade_av ; /* 21 Dec 1999 */ extern void RCREND_pbarmenu_CB( Widget , XtPointer , XtPointer ) ; extern void RCREND_pbarmenu_EV( Widget , XtPointer , XEvent * , Boolean * ) ; extern void RCREND_palette_av_CB( MCW_arrowval * , XtPointer ) ; extern void RCREND_mixshade_av_CB( MCW_arrowval * , XtPointer ) ; /* 21 Dec 1999 */ extern void RCREND_set_pbar_top_CB( Widget , XtPointer , MCW_choose_cbs * ) ; extern void RCREND_finalize_saveim_CB( Widget , XtPointer , MCW_choose_cbs * ) ; #define DEFAULT_FUNC_RANGE 10000.0 static int func_use_autorange = 1 ; static float func_threshold = 0.5 ; static float func_thresh_top = 1.0 ; static int func_use_thresh = 1 ; /* not currently alterable */ static float func_color_opacity = 0.5 ; static int func_see_overlay = 0 ; static int func_see_ttatlas = 0 ; /* 24 Jul 2001 */ static int func_cut_overlay = 0 ; static int func_kill_clusters = 0 ; static float func_clusters_rmm = 1.0 ; static float func_clusters_vmul = 200.0 ; static int func_posfunc = 0 ; static float func_range = DEFAULT_FUNC_RANGE ; static float func_autorange = DEFAULT_FUNC_RANGE ; static int func_computed = 0 ; #define FUNC_RANGE \ ((func_range==0.0 || func_use_autorange ) ? func_autorange : func_range) static int func_showthru = 0 ; /* 07 Jan 2000 */ static int func_showthru_pass = 0 ; static float func_showthru_fac = 0.75;/* 06 Mar 2002 */ #if 0 /* 07 Feb 2002 - removed for CREN */ static int func_showthru_dcue = 0 ; /* 11 Sep 2001 */ #endif #define NOSHADE 1 #define NOMIX 2 static int func_mixshade = 0 ; /* 20 Dec 1999 */ static THD_3dim_dataset * func_dset = NULL ; static MCW_idcode func_dset_idc ; /* 31 Mar 1999 */ static int func_color_ival = 0 ; static int func_thresh_ival = 0 ; static int func_cmap_set = 0 ; static MRI_IMAGE * ovim ; static char func_dset_title[THD_MAX_NAME] ; /* Title string */ char * RCREND_thresh_tlabel_CB( MCW_arrowval * , XtPointer ) ; void RCREND_setup_color_pbar(void) ; XmString RCREND_range_label(void) ; XmString RCREND_autorange_label(void) ; void RCREND_range_bbox_CB ( Widget , XtPointer , XtPointer ) ; void RCREND_color_bbox_CB ( Widget , XtPointer , XtPointer ) ; void RCREND_thr_scale_CB ( Widget , XtPointer , XtPointer ) ; void RCREND_thr_scale_drag_CB( Widget , XtPointer , XtPointer ) ; void RCREND_see_overlay_CB ( Widget , XtPointer , XtPointer ) ; void RCREND_cut_overlay_CB ( Widget , XtPointer , XtPointer ) ; void RCREND_kill_clusters_CB ( Widget , XtPointer , XtPointer ) ; void RCREND_finalize_func_CB ( Widget , XtPointer , MCW_choose_cbs * ) ; void RCREND_see_ttatlas_CB ( Widget , XtPointer , XtPointer ) ; void RCREND_do_ST_CB ( Widget , XtPointer , XtPointer ) ; void RCREND_ST_factor_CB ( MCW_arrowval * , XtPointer ) ; void RCREND_range_av_CB ( MCW_arrowval * , XtPointer ) ; void RCREND_thresh_top_CB ( MCW_arrowval * , XtPointer ) ; void RCREND_colornum_av_CB ( MCW_arrowval * , XtPointer ) ; void RCREND_color_opacity_CB( MCW_arrowval * , XtPointer ) ; void RCREND_clusters_av_CB ( MCW_arrowval * , XtPointer ) ; void RCREND_color_pbar_CB( MCW_pbar * , XtPointer , int ) ; void RCREND_set_thr_pval(void) ; static int reset_bigcolors( rgbyte * bcs ); /* v1.8 [rickr] */ #define COLSIZE 20 /* to modify optmenus */ #undef FIX_SCALE_SIZE #undef HIDE_SCALE #ifdef FIX_SCALE_SIZE_PROBLEM # define FIX_SCALE_SIZE \ do{ int sel_height ; XtPointer sel_ptr ; \ if( wfunc_thr_scale != NULL ){ \ XtVaGetValues( wfunc_thr_scale , \ XmNuserData , &sel_ptr , NULL ) ; \ sel_height = (int) sel_ptr ; \ XtVaSetValues( wfunc_thr_scale , \ XmNheight , sel_height , NULL ) ; \ XtManageChild(wfunc_thr_scale) ; \ } } while(0) # define HIDE_SCALE \ do{ if(wfunc_thr_scale != NULL) XtUnmanageChild(wfunc_thr_scale); } while(0) #else # define FIX_SCALE_SIZE /* nada */ # define HIDE_SCALE /* nada */ #endif /*-------------------------------------------------------------------*/ #define USE_SCRIPTING #ifdef USE_SCRIPTING static Widget script_menu , script_cbut , /* the menu */ script_save_this_pb , script_save_many_pb , script_read_exec_pb , script_read_this_pb ; static MCW_bbox * script_load_bbox , * script_brindex_bbox ; void RCREND_script_CB(Widget , XtPointer , XtPointer) ; void RCREND_script_menu( Widget ) ; void RCREND_script_load_CB( Widget , XtPointer , XtPointer ) ; void RCREND_script_brindex_CB( Widget , XtPointer , XtPointer ) ; void RCREND_save_this_CB( Widget , XtPointer , MCW_choose_cbs * ) ; void RCREND_save_many_CB( Widget , XtPointer , MCW_choose_cbs * ) ; void RCREND_read_this_CB( Widget , XtPointer , MCW_choose_cbs * ) ; void RCREND_read_this_finalize_CB( Widget , XtPointer , MCW_choose_cbs * ) ; void RCREND_read_exec_CB( Widget , XtPointer , MCW_choose_cbs * ) ; static int script_load = 0 ; static int script_load_last = -1 ; static int script_brindex = 0 ; static int script_dontdraw = 0 ; /* 24 Nov 2000 */ #define SCRIPT_GRAFS /* Bastille Day 1999 */ #ifdef SCRIPT_GRAFS typedef struct { int nh , spl , xh[MAX_GHANDS] , yh[MAX_GHANDS] ; } graf_state ; static int graf_states_equal( graf_state * g1 , graf_state * g2 ) { int ii ; if( g1->nh != g2->nh ) return 0 ; if( g1->spl != g2->spl ) return 0 ; for( ii=0 ; ii < g1->nh ; ii++ ){ if( g1->xh[ii] != g2->xh[ii] ) return 0 ; if( g1->yh[ii] != g2->yh[ii] ) return 0 ; } return 1 ; } static void graf_state_get( MCW_graf * gp , graf_state * gs ) { GRAF_get_setup( gp , &(gs->nh) , gs->xh , gs->yh , &(gs->spl) ) ; return ; } static void graf_state_put( MCW_graf * gp , graf_state * gs ) { GRAF_put_setup( gp , gs->nh , gs->xh , gs->yh , gs->spl ) ; return ; } static MCW_bbox * script_graf_bbox ; void RCREND_script_graf_CB( Widget , XtPointer , XtPointer ) ; static int script_graf = 0 ; #endif /* SCRIPT_GRAFS */ #define SCRIPT_DSETS #ifdef SCRIPT_DSETS /* 12 Apr 2000: stuff for changing datasets from script */ static MCW_bbox * script_dset_bbox ; void RCREND_script_dset_CB( Widget , XtPointer , XtPointer ) ; static int script_dsetchange = 0 ; #endif /* data structure to store the state of rendering operations */ typedef struct { char dset_name[THD_MAX_NAME] , func_dset_name[THD_MAX_NAME] ; MCW_idcode dset_idc , func_dset_idc ; int dset_ival , func_color_ival , func_thresh_ival ; int clipbot , cliptop ; float angle_roll , angle_pitch , angle_yaw ; int xhair_flag ; int xhair_ovc ; /* 08 Mar 2001 */ float func_threshold ; float func_thresh_top ; float func_color_opacity ; float func_showthru_fac ; /* 08 Mar 2002 */ int func_showthru ; /* 08 Mar 2002 */ int func_see_overlay ; int func_see_ttatlas ; /* 24 Jul 2001 */ int func_cut_overlay ; int func_kill_clusters ; float func_clusters_rmm ; float func_clusters_vmul ; int func_use_autorange ; float func_range ; int pbar_mode , pbar_npane ; float pbar_pval[NPANE_MAX+1] ; CUTOUT_state current_cutout_state ; #ifdef SCRIPT_GRAFS graf_state bright_graf_state , opacity_graf_state ; #endif } RENDER_state ; typedef struct { int num , nall ; RENDER_state ** rsarr ; } RENDER_state_array ; # define RSA_SUBSTATE(name,nn) ((name)->rsarr[(nn)]) # define RSA_COUNT(name) ((name)->num) # define INC_RSA 32 # define INIT_RSA(name) \ do{ int iq ; (name) = (RENDER_state_array *) malloc(sizeof(RENDER_state_array)) ; \ (name)->num = 0 ; (name)->nall = INC_RSA ; \ (name)->rsarr = (RENDER_state **)malloc(sizeof(RENDER_state *)*INC_RSA) ; \ for( iq=0 ; iq < INC_RSA ; iq++ ) (name)->rsarr[iq] = NULL ; \ break ; } while(0) # define ADDTO_RSA(name,imm) \ do{ int nn , iq ; \ if( (name)->num == (name)->nall ){ \ nn = (name)->nall = 1.1*(name)->nall + INC_RSA ; \ (name)->rsarr = realloc( (name)->rsarr,sizeof(RENDER_state *)*nn ); \ for( iq=(name)->num ; iq < (name)->nall ; iq++ ) (name)->rsarr[iq] = NULL ; } \ nn = (name)->num ; ((name)->num)++ ; \ (name)->rsarr[nn] = (imm) ; break ; } while(0) # define FREE_RSA(name) \ do{ if( (name) != NULL ){ \ free((name)->rsarr); free((name)); (name) = NULL; } break; } while(0) # define DESTROY_RSA(name) \ do{ int nn ; \ if( (name) != NULL ){ \ for( nn=0 ; nn < (name)->num ; nn++ ) free((name)->rsarr[nn]) ; \ free((name)->rsarr); free((name)); (name) = NULL; } break; } while(0) void RCREND_state_to_widgets( RENDER_state * ) ; void RCREND_widgets_to_state( RENDER_state * ) ; char * RCREND_save_state ( RENDER_state * , RENDER_state * ) ; RENDER_state_array * RCREND_read_states( char * , RENDER_state * ) ; static RENDER_state_array * renderings_state = NULL ; static RENDER_state * last_rendered_state = NULL ; #endif /* USE_SCRIPTING */ /*-------------------------------------------------------------------*/ /*************************************************************************** Will be called from AFNI when user selects from Plugins menu. ****************************************************************************/ char * RCREND_main( PLUGIN_interface * plint ) { XmString xstr ; /*-- sanity checks --*/ if( ! IM3D_OPEN(plint->im3d) ) return "AFNI Controller\nnot opened?!" ; if( renderer_open ){ XtMapWidget(shell) ; XRaiseWindow( XtDisplay(shell) , XtWindow(shell) ) ; return NULL ; } im3d = plint->im3d ; /* save for local use */ /*-- create widgets, first time through --*/ if( shell == NULL ){ dc = im3d->dc ; /* save this too */ RCREND_make_widgets() ; PLUTO_set_topshell( plint , shell ) ; /* 22 Sep 2000 */ RWC_visibilize_widget( shell ) ; /* 27 Sep 2000 */ } /*-- set titlebar --*/ { char ttl[PLUGIN_STRING_SIZE] ; sprintf( ttl , "AFNI C Renderer %s" , AFNI_controller_label(im3d) ) ; XtVaSetValues( shell , XmNtitle , ttl , NULL ) ; } /*-- set some widget values --*/ xstr = XmStringCreateLtoR( NO_DATASET_STRING , XmFONTLIST_DEFAULT_TAG ) ; XtVaSetValues( info_lab , XmNlabelString , xstr , NULL ) ; XmStringFree(xstr) ; xstr = XmStringCreateLtoR( "Min=?????? Max=??????" , XmFONTLIST_DEFAULT_TAG ) ; XtVaSetValues( range_lab , XmNlabelString , xstr , NULL ) ; XmStringFree(xstr) ; AV_assign_ival( clipbot_av , -CLIP_RANGE ) ; AV_assign_ival( cliptop_av , CLIP_RANGE ) ; brickfac = 0.0 ; XtUnmanageChild( range_faclab ) ; XtUnmanageChild( clipbot_faclab ) ; XtUnmanageChild( cliptop_faclab ) ; MCW_set_bbox( xhair_bbox , 0 ) ; xhair_flag = 0 ; xhair_ovc = 0 ; MCW_set_bbox( dynamic_bbox , 0 ) ; dynamic_flag = 0 ; MCW_set_bbox( accum_bbox , 0 ) ; accum_flag = 0 ; MCW_set_bbox( automate_bbox , 0 ) ; automate_flag = 0 ; XtSetSensitive( autocompute_pb , False ) ; AV_assign_ival( numcutout_av , 0 ) ; /* turn off cutouts */ RCREND_numcutout_CB( numcutout_av , NULL ) ; RCREND_load_cutout_state() ; old_cutout_state = current_cutout_state ; AV_SENSITIZE( choose_av , False ) ; /*--- some of the function widgets, too ---*/ if( wfunc_frame != NULL ){ TURNOFF_OVERLAY_WIDGETS ; } /*-- pop the widget up --*/ XtMapWidget(shell) ; PLUTO_cursorize(shell) ; /*-- misc initialization --*/ dset = NULL ; /* not rendering anything */ dset_ival = 0 ; /* if we were, it would be #0 */ renderer_open = 1 ; /* renderer is now open for business */ imseq = NULL ; /* no image window is open yet */ grim = NULL ; /* don't have volumes to render yet */ gcr.rh = NULL; /* no render handle yet */ gcr.dset_or = NULL; /* no reoriented underlay dataset yet */ gcr.fset_or = NULL; /* no reoriented overlay dataset yet */ gcr.mset = NULL; /* no reorientation master dset yet */ gcr.fdm = NULL; /* no master FD_brick yet */ ovim = NULL ; /* no overlay volume yet */ func_dset = NULL ; /* no functional dataset yet */ new_data_loaded = 0 ; /* not yet */ grim_showthru = NULL ; /* 07 Jan 2000 */ set_MCW_pasgraf( his_graf , NULL ) ; /* set histogram graph to 0's */ redraw_MCW_pasgraf( his_graf ) ; xhair_ixold = -666 ; xhair_jyold = -666 ; xhair_kzold = -666 ; memset( &gcr_debug, 0, sizeof(gcr_debug) ); /* init debug struct */ /* 29 Mar 1999: register to receive updates from AFNI */ #if 1 xhair_recv = AFNI_receive_init( im3d , RECEIVE_VIEWPOINT_MASK | RECEIVE_DRAWNOTICE_MASK | RECEIVE_DSETCHANGE_MASK | RECEIVE_TIMEINDEX_MASK /* 29 Jan 2003 */ , RCREND_xhair_recv , NULL , "RCREND_xhair_recv" ) ; #else xhair_recv = AFNI_receive_init( im3d , RECEIVE_VIEWPOINT_MASK , RCREND_xhair_recv , NULL , "RCREND_xhair_recv" ) ; #endif MPROBE ; return NULL ; } /*------------------------------------------------------------------------ Make the control popup for this thing --------------------------------------------------------------------------*/ /*-- structures defining action buttons (at bottom of popup) --*/ #define NACT 4 /* number of action buttons */ static MCW_action_item RCREND_actor[NACT] = { {"Help",RCREND_help_CB,NULL, "Displays more help" , "Displays more help",0} , {"Draw",RCREND_draw_CB,NULL, "(Re)Draw the image" , "(Re)Draw the image",0} , {"Reload",RCREND_reload_CB,NULL, "Reload dataset values" , "Reload dataset values",0} , {"done",RCREND_done_CB,NULL, "Close renderer\nand image." , "Close windows",1} } ; #define SEP_HOR(ww) XtVaCreateManagedWidget( \ "AFNI" , xmSeparatorWidgetClass , (ww) , \ XmNseparatorType , XmSINGLE_LINE , \ XmNinitialResourcesPersistent , False , \ NULL ) #define SEP_VER(ww) XtVaCreateManagedWidget( \ "AFNI" , xmSeparatorWidgetClass , (ww) , \ XmNseparatorType , XmDOUBLE_LINE , \ XmNorientation , XmVERTICAL , \ XmNinitialResourcesPersistent , False , \ NULL ) void RCREND_make_widgets(void) { XmString xstr ; Widget hrc , vrc ; int ii ; ENTRY( "RCREND_make_widgets" ); /***=============================================================*/ /*** top level shell for window manager ***/ shell = XtVaAppCreateShell( "AFNI" , "AFNI" , topLevelShellWidgetClass , dc->display , XmNtitle , "AFNI R" , /* top of window */ XmNiconName , "R" , /* label on icon */ XmNdeleteResponse , XmDO_NOTHING , /* deletion handled below */ XmNallowShellResize , True , /* let code resize shell? */ XmNmappedWhenManaged , False , /* must map it manually */ XmNinitialResourcesPersistent , False , NULL ) ; DC_yokify( shell , dc ) ; /* 14 Sep 1998 */ #ifndef DONT_INSTALL_ICONS if( afni48_good ) /* set icon pixmap */ XtVaSetValues( shell , XmNiconPixmap , afni48_pixmap , NULL ) ; #endif if( MCW_isitmwm(shell) ) /* remove some MWM functions */ XtVaSetValues( shell , XmNmwmFunctions , MWM_FUNC_MOVE | MWM_FUNC_CLOSE | MWM_FUNC_MINIMIZE , NULL ) ; XmAddWMProtocolCallback( /* make "Close" window menu work */ shell , XmInternAtom( dc->display , "WM_DELETE_WINDOW" , False ) , RCREND_done_CB , (XtPointer) plint ) ; /*** horizontal rowcol to hold ALL interface stuff ***/ top_rowcol = XtVaCreateWidget( "AFNI" , xmRowColumnWidgetClass , shell , XmNorientation , XmHORIZONTAL , XmNpacking , XmPACK_TIGHT , XmNadjustLast , False , XmNadjustMargin , False , XmNtraversalOn , True , XmNmarginWidth , 0 , XmNmarginHeight , 0 , XmNinitialResourcesPersistent , False , NULL ) ; /*** vertical rowcolumn widget to hold anat interface stuff ***/ anat_frame = XtVaCreateWidget( "AFNI" , xmFrameWidgetClass , top_rowcol , XmNshadowType , XmSHADOW_ETCHED_IN , XmNshadowThickness , 5 , XmNtraversalOn , True , XmNinitialResourcesPersistent , False , NULL ) ; anat_rowcol = XtVaCreateWidget( "AFNI" , xmRowColumnWidgetClass , anat_frame , XmNpacking , XmPACK_TIGHT , XmNorientation , XmVERTICAL , XmNadjustLast , False , XmNadjustMargin, False , XmNtraversalOn , True , XmNinitialResourcesPersistent , False , NULL ) ; /***=============================================================*/ /*** label at top to let user know who we are ***/ xstr = XmStringCreateLtoR( NO_DATASET_STRING , XmFONTLIST_DEFAULT_TAG ) ; info_lab = XtVaCreateManagedWidget( "AFNI" , xmLabelWidgetClass , anat_rowcol , XmNlabelString , xstr , XmNrecomputeSize , False , XmNinitialResourcesPersistent , False , NULL ) ; XmStringFree(xstr) ; MCW_register_help( info_lab , "Shows dataset being rendered" ) ; /***** top row of widgets to choose dataset and sub-brick *****/ SEP_HOR(anat_rowcol) ; /* separator widget */ hrc = XtVaCreateWidget( "AFNI" , xmRowColumnWidgetClass , anat_rowcol , XmNorientation , XmHORIZONTAL , XmNpacking , XmPACK_TIGHT , XmNadjustLast , False , XmNadjustMargin , False , XmNtraversalOn , True , XmNmarginWidth , 0 , XmNmarginHeight , 0 , XmNinitialResourcesPersistent , False , NULL ) ; /*** button to let user choose dataset to render ***/ xstr = XmStringCreateLtoR( "Choose Underlay Dataset" , XmFONTLIST_DEFAULT_TAG ) ; choose_pb = XtVaCreateManagedWidget( "AFNI" , xmPushButtonWidgetClass , hrc , XmNalignment , XmALIGNMENT_CENTER , XmNlabelString , xstr , XmNtraversalOn , True , XmNinitialResourcesPersistent , False , NULL ) ; XmStringFree(xstr) ; XtAddCallback( choose_pb, XmNactivateCallback, RCREND_choose_CB, NULL ) ; MCW_register_help( choose_pb , "Use this to popup a\n" "'chooser' that lets\n" "you select which\n" "dataset to render." ) ; /*** menu to let user choose sub-brick to deal with ***/ SEP_VER(hrc) ; choose_av = new_MCW_arrowval( hrc , /* parent Widget */ "Brick " , /* label */ MCW_AV_optmenu , /* option menu style */ 0 , /* first option */ 1 , /* last option */ 0 , /* initial selection */ MCW_AV_readtext , /* ignored but needed */ 0 , /* decimal shift */ RCREND_choose_av_CB , /* callback when changed */ NULL , /* data for above */ MCW_av_substring_CB , /* text creation routine */ RCREND_dummy_av_label /* data for above */ ) ; /*** button to open and close overlay panel ***/ SEP_VER(hrc) ; xstr = XmStringCreateLtoR( "Overlay" , XmFONTLIST_DEFAULT_TAG ) ; wfunc_open_pb = XtVaCreateManagedWidget( "AFNI" , xmPushButtonWidgetClass , hrc , XmNalignment , XmALIGNMENT_CENTER , XmNlabelString , xstr , XmNtraversalOn , True , XmNinitialResourcesPersistent , False , NULL ) ; XmStringFree(xstr) ; XtAddCallback( wfunc_open_pb, XmNactivateCallback, RCREND_open_func_CB, NULL ) ; XtManageChild(hrc) ; /***=============================================================*/ /*** horizontal rowcol for data value clipping ***/ SEP_HOR(anat_rowcol) ; /* separator */ hrc = XtVaCreateWidget( "AFNI" , xmRowColumnWidgetClass , anat_rowcol , XmNorientation , XmHORIZONTAL , XmNpacking , XmPACK_TIGHT , XmNadjustLast , False , XmNadjustMargin , False , XmNtraversalOn , True , XmNmarginWidth , 0 , XmNmarginHeight , 0 , XmNinitialResourcesPersistent , False , NULL ) ; /*** vertical rowcol for dataset range information labels ***/ vrc = XtVaCreateWidget( "AFNI" , xmRowColumnWidgetClass , hrc , XmNpacking , XmPACK_TIGHT , XmNorientation , XmVERTICAL , XmNadjustLast , False , XmNadjustMargin, False , XmNtraversalOn , True , XmNmarginWidth , 0 , XmNmarginHeight, 0 , XmNinitialResourcesPersistent , False , NULL ) ; /*** 1st label for dataset range information ***/ xstr = XmStringCreateLtoR( "Min=?????? Max=??????" , XmFONTLIST_DEFAULT_TAG ) ; range_lab = XtVaCreateManagedWidget( "AFNI" , xmLabelWidgetClass , vrc , XmNlabelString , xstr , XmNrecomputeSize , False , XmNinitialResourcesPersistent , False , NULL ) ; XmStringFree(xstr) ; MCW_register_help( range_lab , "Shows the range of the data stored\n" "in the brick voxels.\n" "\n" "N.B.: These values are NOT scaled\n" " by any floating point\n" " brick scaling factor." ) ; /*** 2nd label for scaled dataset range information ***/ xstr = XmStringCreateLtoR( "[123456789 123456789]" , XmFONTLIST_DEFAULT_TAG ) ; range_faclab = XtVaCreateWidget( "AFNI" , xmLabelWidgetClass , vrc , XmNlabelString , xstr , XmNrecomputeSize , False , XmNinitialResourcesPersistent , False , NULL ) ; XmStringFree(xstr) ; MCW_register_help( range_faclab , "Shows the range of data stored\n" "in the brick, this time multiplied\n" "by the brick's scaling factor." ) ; XtManageChild(vrc) ; SEP_VER(hrc) ; /*** arrowvals to get dataset clip levels ***/ /*** vertical rowcol for Bot arrowval ***/ vrc = XtVaCreateWidget( "AFNI" , xmRowColumnWidgetClass , hrc , XmNpacking , XmPACK_TIGHT , XmNorientation , XmVERTICAL , XmNadjustLast , False , XmNadjustMargin, False , XmNtraversalOn , True , XmNmarginWidth , 0 , XmNmarginHeight, 0 , XmNinitialResourcesPersistent , False , NULL ) ; clipbot_av = new_MCW_arrowval( vrc , "Bot " , MCW_AV_downup , -CLIP_RANGE,CLIP_RANGE,-CLIP_RANGE , MCW_AV_editext , 0 , RCREND_clip_CB , NULL , NULL,NULL ) ; MCW_reghelp_children( clipbot_av->wrowcol , "All (unscaled) voxel values below\n" "'Bot' will be increased to this\n" "value. The larger of 'Bot' and\n" "'Min' is the left edge of the\n" "brick graphs shown below." ) ; xstr = XmStringCreateLtoR( "[-> 123456789]" , XmFONTLIST_DEFAULT_TAG ) ; clipbot_faclab = XtVaCreateWidget( "AFNI" , xmLabelWidgetClass , vrc , XmNlabelString , xstr , XmNrecomputeSize , False , XmNinitialResourcesPersistent , False , NULL ) ; XmStringFree(xstr) ; MCW_register_help( clipbot_faclab , "Shows the scaled\nvalue of 'Bot'." ) ; XtManageChild(vrc) ; SEP_VER(hrc) ; /*** vertical rowcol for Top arrowval ***/ vrc = XtVaCreateWidget( "AFNI" , xmRowColumnWidgetClass , hrc , XmNpacking , XmPACK_TIGHT , XmNorientation , XmVERTICAL , XmNadjustLast , False , XmNadjustMargin, False , XmNtraversalOn , True , XmNmarginWidth , 0 , XmNmarginHeight, 0 , XmNinitialResourcesPersistent , False , NULL ) ; cliptop_av = new_MCW_arrowval( vrc , "Top " , MCW_AV_downup , -CLIP_RANGE,CLIP_RANGE, CLIP_RANGE , MCW_AV_editext , 0 , RCREND_clip_CB , NULL , NULL,NULL ) ; MCW_reghelp_children( cliptop_av->wrowcol , "All (unscaled) voxel values above\n" "'Top' will be decreased to this\n" "value. The smaller of 'Top' and\n" "'Max' is the right edge of the\n" "brick graphs shown below." ) ; xstr = XmStringCreateLtoR( "[-> 123456789]" , XmFONTLIST_DEFAULT_TAG ) ; cliptop_faclab = XtVaCreateWidget( "AFNI" , xmLabelWidgetClass , vrc , XmNlabelString , xstr , XmNrecomputeSize , False , XmNinitialResourcesPersistent , False , NULL ) ; XmStringFree(xstr) ; MCW_register_help( clipbot_faclab , "Shows the scaled\nvalue of 'Top'." ) ; XtManageChild(vrc) ; XtManageChild(hrc) ; /***=============================================================*/ /*** horizontal rowcol for graphs ***/ SEP_HOR(anat_rowcol) ; /* separator */ hrc = XtVaCreateWidget( "AFNI" , xmRowColumnWidgetClass , anat_rowcol , XmNorientation , XmHORIZONTAL , XmNpacking , XmPACK_TIGHT , XmNadjustLast , False , XmNadjustMargin, False , XmNtraversalOn , True , XmNmarginWidth , 0 , XmNmarginHeight, 0 , XmNinitialResourcesPersistent , False , NULL ) ; /*** graph to control grayscale ***/ gry_graf = new_MCW_graf( hrc , im3d->dc, "Brightness", RCREND_graf_CB, NULL ) ; MCW_reghelp_children( gry_graf->topform , "This graph controls the brightness (y-axis) of each\n" "voxel, as a function of input signal (x-axis).\n\n" "After you change this curve, you must press\n" "'Draw' to see the effect on the rendered image.\n\n" "* To change the curve, drag the square handles\n" " using mouse Button 1 or Button 3.\n" "* Dragging with Button 3 shows a label indicating\n" " the (x,y) coordinates of the handle.\n" "* Use the # button to add and remove handles.\n" "* Use the Crv button to use spline interpolation.\n" "* Use the Line button to reset the curve to y=x.\n" ) ; SEP_VER(hrc) ; /*** graph to control opacity ***/ opa_graf = new_MCW_graf( hrc , im3d->dc, "Opacity", RCREND_graf_CB, NULL ) ; MCW_reghelp_children( opa_graf->topform , "This graph controls the opacity (y-axis) of each\n" "voxel, as a function of input signal (x-axis).\n\n" "After you change this curve, you must press\n" "'Draw' to see the effect on the rendered image.\n\n" "* To change the curve, drag the square handles\n" " using mouse Button 1 or Button 3.\n" "* Dragging with Button 3 shows a label indicating\n" " the (x,y) coordinates of the handle.\n" "* Use the # button to add and remove handles.\n" "* Use the Crv button to use spline interpolation.\n" "* Use the Line button to reset the curve to y=x.\n" ) ; SEP_VER(hrc) ; /*** passive graph to show data distribution ***/ his_graf = new_MCW_pasgraf( hrc , im3d->dc , "Sqrt Histogram" ) ; his_graf->mode = PASGRAF_BAR ; MCW_reghelp_children( his_graf->topform , "The graph height is proportional to\n" "the square-root of the histogram of\n" "the input signal.\n" "\n" "* The histogram at 0 is not included\n" " in the scaling, since it tends to\n" " be huge. The square-root is graphed\n" " to enhance the range of the plot.\n" "* Press Button 3 in this window to see a\n" " popup label with the (x,y) coordinate.\n" ) ; XtManageChild(hrc) ; /***=============================================================*/ /*** horizontal rowcol to hold cutout controls ***/ SEP_HOR(anat_rowcol) ; /* separator */ hrc = XtVaCreateWidget( "AFNI" , xmRowColumnWidgetClass , anat_rowcol , XmNorientation , XmHORIZONTAL , XmNpacking , XmPACK_TIGHT , XmNadjustLast , False , XmNadjustMargin, False , XmNtraversalOn , True , XmNmarginWidth , 0 , XmNmarginHeight, 0 , XmNinitialResourcesPersistent , False , NULL ) ; /*** option menu to choose number of cutouts ***/ numcutout_av = new_MCW_optmenu( hrc , "Cutouts " , 0 , MAX_CUTOUTS , num_cutouts,0 , RCREND_numcutout_CB , NULL , NULL , NULL ) ; MCW_reghelp_children( numcutout_av->wrowcol , "Use this to choose the number of cutouts\n" "to apply before rendering. Controls for\n" "the number selected will be activated below." ) ; /*** option menu to choose cutout logic ***/ logiccutout_av = new_MCW_optmenu( hrc , "+" , 0 , 1 , logic_cutout,0 , NULL , NULL , MCW_av_substring_CB , cutout_logic_labels ) ; MCW_reghelp_children( logiccutout_av->wrowcol , "Use this to control the logic of how\n" "multiple cutouts are combined:\n\n" "OR = the union of all regions\n" "AND = the intersection of all regions" ) ; SEP_VER(hrc) ; /* separator */ /*** arrowval to select opacity reduction factor ***/ opacity_scale_av = new_MCW_arrowval( hrc , "Opacity Factor " , MCW_AV_downup , 0,10,10 , MCW_AV_noactext , 1 , RCREND_opacity_scale_CB , NULL , NULL,NULL ) ; XtAddCallback( opacity_scale_av->wtext, XmNactivateCallback, RCREND_textact_CB, opacity_scale_av ) ; /*** 07 July 1999: insert menu to control scripting actions ***/ #ifdef USE_SCRIPTING SEP_VER(hrc) ; RCREND_script_menu( hrc ) ; #endif XtManageChild(hrc) ; /*** Create the widgets for each cutout ***/ for( ii=0 ; ii < MAX_CUTOUTS ; ii++ ) cutouts[ii] = RCREND_make_cutout(ii) ; /***=============================================================*/ /*** horizontal rowcol to hold automation controls ***/ SEP_HOR(anat_rowcol) ; /* separator */ hrc = XtVaCreateWidget( "AFNI" , xmRowColumnWidgetClass , anat_rowcol , XmNorientation , XmHORIZONTAL , XmNpacking , XmPACK_TIGHT , XmNadjustLast , False , XmNadjustMargin, False , XmNtraversalOn , True , XmNmarginWidth , 0 , XmNmarginHeight, 0 , XmNinitialResourcesPersistent , False , NULL ) ; /*** button box to enable automation mode ***/ automate_bbox = new_MCW_bbox( hrc , 1 , automate_bbox_label , MCW_BB_check , MCW_BB_noframe , RCREND_autoflag_CB , NULL ) ; MCW_set_bbox( automate_bbox , automate_flag ) ; MCW_reghelp_children( automate_bbox->wrowcol , "IN: Enable automation of renderings\n" "OUT: Don't allow automated rendering" ) ; SEP_VER(hrc) ; /* separator */ /*** arrowval to control number of frames to compute */ autoframe_av = new_MCW_arrowval( hrc , "Frames " , MCW_AV_downup , 2,999,5 , MCW_AV_editext , 0 , NULL , NULL , NULL,NULL ) ; MCW_reghelp_children( autoframe_av->wrowcol , "Use this to set the number\n" "of frames that will be rendered\n" "when 'Compute' is activated." ) ; SEP_VER(hrc) ; /* separator */ /*** pushbutton to activate the automation ***/ xstr = XmStringCreateLtoR( "Compute" , XmFONTLIST_DEFAULT_TAG ) ; autocompute_pb = XtVaCreateManagedWidget( "AFNI" , xmPushButtonWidgetClass , hrc , XmNlabelString , xstr , XmNtraversalOn , True , XmNinitialResourcesPersistent , False , NULL ) ; XmStringFree(xstr) ; XtAddCallback( autocompute_pb, XmNactivateCallback, RCREND_autocompute_CB, NULL ) ; MCW_register_help( autocompute_pb , "Use this to start the\n" "automation of rendering" ) ; /*** pushbutton to cancel the automation [not managed now] ***/ xstr = XmStringCreateLtoR( " * CANCEL * " , XmFONTLIST_DEFAULT_TAG ) ; autocancel_pb = XtVaCreateWidget( "AFNI" , xmPushButtonWidgetClass , hrc , XmNlabelString , xstr , XmNtraversalOn , True , XmNinitialResourcesPersistent , False , NULL ) ; XmStringFree(xstr) ; XtAddCallback( autocancel_pb, XmNactivateCallback, RCREND_autocancel_CB, NULL ) ; XtManageChild(hrc) ; /***=============================================================*/ /*** horizontal rowcol to hold miscellaneous display controls ***/ SEP_HOR(anat_rowcol) ; /* separator */ hrc = XtVaCreateWidget( "AFNI" , xmRowColumnWidgetClass , anat_rowcol , XmNorientation , XmHORIZONTAL , XmNpacking , XmPACK_TIGHT , XmNadjustLast , False , XmNadjustMargin, False , XmNtraversalOn , True , XmNmarginWidth , 0 , XmNmarginHeight, 0 , XmNinitialResourcesPersistent , False , NULL ) ; /*** option menu to choose interpolation mode ***/ interp_av = new_MCW_optmenu( hrc , "Interp " , 0 , RCREND_NUM_interp_modes-1 , interp_ival,0 , RCREND_interp_CB , NULL , MCW_av_substring_CB , interp_mode_strings ) ; MCW_reghelp_children( interp_av->wrowcol , "Use this to set the interpolation mode. The\n" "computation time increases from Neighbor to Linear.\n" "\n" "Neighbor = choose the value of the closest voxel\n" "Twostep = like Neighbor, but medium-distance\n" " points use the average of both neighbors\n" "Linear = distance-weighted average of neighbors" ) ; SEP_VER(hrc) ; /* separator */ /*** button box to show AFNI crosshair location ***/ xhair_bbox = new_MCW_bbox( hrc , 1 , xhair_bbox_label , MCW_BB_check , MCW_BB_noframe , RCREND_xhair_CB , NULL ) ; /* 08 Mar 2001: Button3 popup to control xhair color */ XtInsertEventHandler( xhair_bbox->wbut[0] , 0 | ButtonPressMask /* button presses */ , FALSE , /* nonmaskable events? */ RCREND_xhair_EV , /* handler */ NULL , /* client data */ XtListTail /* last in queue */ ) ; MCW_set_bbox( xhair_bbox , xhair_flag ) ; MCW_reghelp_children( xhair_bbox->wrowcol , "IN: show AFNI crosshair location\n" "OUT: don't show AFNI crosshairs\n" "\n" "N.B.: Must press Reload to see the\n" " crosshair position updated\n" " if it is changed in AFNI." ) ; SEP_VER(hrc) ; /* separator */ /*** button box to do dynamic updates ***/ dynamic_bbox = new_MCW_bbox( hrc , 1 , dynamic_bbox_label , MCW_BB_check , MCW_BB_noframe , RCREND_dynamic_CB , NULL ) ; MCW_set_bbox( dynamic_bbox , dynamic_flag ) ; MCW_reghelp_children( dynamic_bbox->wrowcol , "IN: Redraw immediately upon changes\n" "OUT: Redraw only when commanded\n" "\n" "N.B.: Changes to the AFNI crosshair\n" " position are not detectable\n" " to force a dynamic redraw." ) ; SEP_VER(hrc) ; /* separator */ /*** button box to accumulate images ***/ accum_bbox = new_MCW_bbox( hrc , 1 , accum_bbox_label , MCW_BB_check , MCW_BB_noframe , RCREND_accum_CB , NULL ) ; MCW_set_bbox( accum_bbox , accum_flag ) ; MCW_reghelp_children( accum_bbox->wrowcol , "IN: Accumulate images for viewing\n" "OUT: Save only the latest images" ) ; /* 17 Jun 2005: Button3 popup to control overlay label */ XtInsertEventHandler( accum_bbox->wbut[0] , 0 | ButtonPressMask /* button presses */ , FALSE , /* nonmaskable events? */ RCREND_accum_lab_EV, /* handler */ NULL , /* client data */ XtListTail /* last in queue */ ) ; XtManageChild(hrc) ; /***=============================================================*/ /*** horizontal rowcol to hold angle arrows ***/ SEP_HOR(anat_rowcol) ; /* separator widget */ hrc = XtVaCreateWidget( "AFNI" , xmRowColumnWidgetClass , anat_rowcol , XmNorientation , XmHORIZONTAL , XmNpacking , XmPACK_TIGHT , XmNadjustLast , False , XmNadjustMargin, False , XmNtraversalOn , True , XmNmarginWidth , 0 , XmNmarginHeight, 0 , XmNinitialResourcesPersistent , False , NULL ) ; /*==========================================================================*/ #ifdef ALLOW_INCROT /* 26 Apr 2002 - RWCox */ { static char * incrot_bbox_label[1] = { "I" } ; incrot_bbox = new_MCW_bbox( hrc , 1 , incrot_bbox_label , MCW_BB_check , MCW_BB_noframe , RCREND_incrot_CB, NULL ) ; MCW_set_bbox( incrot_bbox , 1 ) ; MCW_reghelp_children( incrot_bbox->wrowcol , "OUT: angles increment globally\n" "IN: angles increment locally" ) ; MCW_reghint_children( incrot_bbox->wrowcol , "Incremental rotation?" ) ; SEP_VER(hrc) ; } #endif /* ALLOW_INCROT */ /** N.B.: removed trailing space from "Roll", "Pitch", "Yaw" labels for arrowvals below, to make space for the bbox created above **/ /*==========================================================================*/ /*** arrowvals to choose rotation angles ***/ roll_av = new_MCW_arrowval( hrc , "Roll" , MCW_AV_downup , -999999,999999,(int)(0.1*angle_roll) , MCW_AV_noactext , -1 , RCREND_angle_CB , NULL , NULL,NULL ) ; roll_av->fstep = angle_fstep ; MCW_reghelp_children( roll_av->wrowcol , "Use this to set the roll angle\n" "(about the I-S axis) for viewing,\n" "then press 'Draw'" ) ; XtAddCallback( roll_av->wtext, XmNactivateCallback, RCREND_textact_CB, roll_av ) ; SEP_VER(hrc) ; /* separator widget */ pitch_av = new_MCW_arrowval( hrc , "Pitch" , MCW_AV_downup , -999999,999999,(int)(0.1*angle_pitch) , MCW_AV_noactext , -1 , RCREND_angle_CB , NULL , NULL,NULL ) ; pitch_av->fstep = angle_fstep ; MCW_reghelp_children( pitch_av->wrowcol , "Use this to set the pitch angle\n" "(about the R-L axis) for viewing,\n" "then press 'Draw'" ) ; XtAddCallback( pitch_av->wtext, XmNactivateCallback, RCREND_textact_CB, pitch_av ) ; SEP_VER(hrc) ; /* separator widget */ yaw_av = new_MCW_arrowval( hrc , "Yaw" , MCW_AV_downup , -999999,999999,(int)(0.1*angle_yaw) , MCW_AV_noactext , -1 , RCREND_angle_CB , NULL , NULL,NULL ) ; yaw_av->fstep = angle_fstep ; MCW_reghelp_children( yaw_av->wrowcol , "Use this to set the yaw angle\n" "(about the A-P axis) for viewing,\n" "then press 'Draw'" ) ; XtAddCallback( yaw_av->wtext, XmNactivateCallback, RCREND_textact_CB, yaw_av ) ; /** 26 Apr 2002: add hints to these arrows as well **/ MCW_reghint_children( roll_av->wrowcol , "Angle about I-S axis" ) ; MCW_reghint_children( pitch_av->wrowcol , "Angle about R-L axis" ) ; MCW_reghint_children( yaw_av->wrowcol , "Angle about A-P axis" ) ; /*==========================================================================*/ #if 1 #ifdef ALLOW_INCROT /* 26 Apr 2002 - RWCox */ XtVaSetValues( roll_av->wtext , XmNcolumns , 8 , NULL ) ; XtVaSetValues( pitch_av->wtext , XmNcolumns , 8 , NULL ) ; XtVaSetValues( yaw_av->wtext , XmNcolumns , 8 , NULL ) ; #endif #endif /*==========================================================================*/ XtManageChild(hrc) ; /***=============================================================*/ /*** a set of action buttons below the line ***/ SEP_HOR(anat_rowcol) ; (void) MCW_action_area( anat_rowcol , RCREND_actor , NACT ) ; help_pb = (Widget) RCREND_actor[0].data ; draw_pb = (Widget) RCREND_actor[1].data ; reload_pb = (Widget) RCREND_actor[2].data ; done_pb = (Widget) RCREND_actor[3].data ; /***=============================================================*/ /*** that's all ***/ XtManageChild(anat_rowcol) ; XtManageChild(anat_frame) ; XtManageChild(top_rowcol) ; XtRealizeWidget(shell) ; /* will not be mapped */ WAIT_for_window(shell) ; POPUP_cursorize(xhair_bbox->wbut[0]) ; POPUP_cursorize(accum_bbox->wbut[0]) ; /*** 12 July 1999: make the overlay widgets now, instead of later ***/ RCREND_func_widgets() ; #if 0 XtVaSetValues( anat_rowcol , XmNresizeWidth , False , NULL ) ; #endif EXRETURN ; } /*------------------------------------------------------------------- Make a line of cutout widgets ---------------------------------------------------------------------*/ #define NUM_CUTOUT_TYPES 22 static char * cutout_type_labels[NUM_CUTOUT_TYPES] = { "No Cut" , "Right of" , "Left of" , "Anterior to" , "Posterior to" , "Inferior to" , "Superior to" , "Expr > 0" , "TT Ellipsoid " , "Behind AL-PR" , "Front AL-PR" , /* x+y > val , x+y < val */ "Front AR-PL" , "Behind AR-PL" , /* x-y > val , x-y < val */ "Above AS-PI" , "Below AS-PI" , /* y+z > val , y+z < val */ "Below AI-PS" , "Above AI-PS" , /* y-z > val , y-z < val */ "Above RS-LI" , "Below RS-LI" , /* x+z > val , x+z < val */ "Below RI-LS" , "Above RI-LS" , /* x-z > val , x-z < val */ "NonOverlay++" } ; static char * cutout_param_labels[NUM_CUTOUT_TYPES] = { "Parameter: " , "x(-R+L) [mm]:" , "x(-R+L) [mm]:" , "y(-A+P) [mm]:" , "y(-A+P) [mm]:" , "z(-I+S) [mm]:" , "z(-I+S) [mm]:" , "Expression: " , "Percentage: " , "Value [mm]: " , "Value [mm]: " , "Value [mm]: " , "Value [mm]: " , "Value [mm]: " , "Value [mm]: " , "Value [mm]: " , "Value [mm]: " , "Value [mm]: " , "Value [mm]: " , "Value [mm]: " , "Value [mm]: " , "Radius++[mm]:" } ; static char * cutout_type_names[NUM_CUTOUT_TYPES] = { "CUT_NONE" , "CUT_RIGHT_OF" , "CUT_LEFT_OF" , "CUT_ANTERIOR_TO" , "CUT_POSTERIOR_TO" , "CUT_INFERIOR_TO" , "CUT_SUPERIOR_TO" , "CUT_EXPRESSION" , "CUT_TT_ELLIPSOID" , "CUT_SLANT_XPY_GT" , "CUT_SLANT_XPY_LT" , "CUT_SLANT_XMY_GT" , "CUT_SLANT_XMY_LT" , "CUT_SLANT_YPZ_GT" , "CUT_SLANT_YPZ_LT" , "CUT_SLANT_YMZ_GT" , "CUT_SLANT_YMZ_LT" , "CUT_SLANT_XPZ_GT" , "CUT_SLANT_XPZ_LT" , "CUT_SLANT_XMZ_GT" , "CUT_SLANT_XMZ_LT" , "CUT_NONOVERLAY" } ; #define CUT_NONE 0 #define CUT_RIGHT_OF 1 #define CUT_LEFT_OF 2 #define CUT_ANTERIOR_TO 3 #define CUT_POSTERIOR_TO 4 #define CUT_INFERIOR_TO 5 #define CUT_SUPERIOR_TO 6 #define CUT_EXPRESSION 7 #define CUT_TT_ELLIPSOID 8 #define CUT_SLANT_XPY_GT 9 /* slant cuts added 17 Feb 1999 */ #define CUT_SLANT_XPY_LT 10 #define CUT_SLANT_XMY_GT 11 #define CUT_SLANT_XMY_LT 12 #define CUT_SLANT_YPZ_GT 13 #define CUT_SLANT_YPZ_LT 14 #define CUT_SLANT_YMZ_GT 15 #define CUT_SLANT_YMZ_LT 16 #define CUT_SLANT_XPZ_GT 17 #define CUT_SLANT_XPZ_LT 18 #define CUT_SLANT_XMZ_GT 19 #define CUT_SLANT_XMZ_LT 20 #define CUT_NONOVERLAY 21 #define CUT_SLANT_BASE 9 #define CUT_SLANT_NUM 12 #define SQ2 0.7071 static float cut_slant_normals[CUT_SLANT_NUM][3] = { { SQ2 , SQ2 , 0.0 } , {-SQ2 ,-SQ2 , 0.0 } , { SQ2 ,-SQ2 , 0.0 } , {-SQ2 , SQ2 , 0.0 } , { 0.0 , SQ2 , SQ2 } , { 0.0 ,-SQ2 ,-SQ2 } , { 0.0 , SQ2 ,-SQ2 } , { 0.0 ,-SQ2 ,+SQ2 } , { SQ2 , 0.0 , SQ2 } , {-SQ2 , 0.0 ,-SQ2 } , { SQ2 , 0.0 ,-SQ2 } , {-SQ2 , 0.0 , SQ2 } } ; #if 0 static int cut_slant_sign[CUT_SLANT_NUM] = { 1 , -1 , 1 , -1 , 1 , -1 , 1 , -1 , 1 , -1 , 1 , -1 } ; #endif RCREND_cutout * RCREND_make_cutout( int n ) { XmString xstr ; char str[64] ; RCREND_cutout * rc ; ENTRY( "RCREND_make_cutout" ); rc = myXtNew(RCREND_cutout) ; /* horizontal rowcol holds all that follows */ rc->hrc = XtVaCreateWidget( "AFNI" , xmRowColumnWidgetClass , anat_rowcol , XmNorientation , XmHORIZONTAL , XmNpacking , XmPACK_TIGHT , XmNadjustLast , False , XmNadjustMargin, False , XmNtraversalOn , True , XmNmarginWidth , 0 , XmNmarginHeight, 0 , XmNinitialResourcesPersistent , False , NULL ) ; /* menu to choose type of cutout */ sprintf(str,"#%d",n+1) ; rc->type_av = new_MCW_optmenu( rc->hrc , str , 0 , NUM_CUTOUT_TYPES-1 , CUT_NONE,0 , RCREND_cutout_type_CB , NULL , MCW_av_substring_CB , cutout_type_labels ) ; if( NUM_CUTOUT_TYPES >= COLSIZE ) AVOPT_columnize( rc->type_av , 1+(NUM_CUTOUT_TYPES+1)/COLSIZE ) ; MCW_reghelp_children( rc->type_av->wrowcol , "Use this to set the type of cutout\n" "controlled by this line of inputs." ) ; /* label to indicate parameter to enter */ xstr = XmStringCreateLtoR( cutout_param_labels[0] , XmFONTLIST_DEFAULT_TAG ) ; rc->param_lab = XtVaCreateWidget( "AFNI" , xmLabelWidgetClass , rc->hrc , XmNlabelString , xstr , XmNinitialResourcesPersistent , False , NULL ) ; XmStringFree(xstr) ; /* arrowval to enter parameter */ rc->param_av = new_MCW_arrowval( rc->hrc , NULL , MCW_AV_downup , -999999,999999,0 , MCW_AV_noactext , -1 , RCREND_param_CB , NULL , NULL,NULL ) ; rc->param_av->fstep = cutout_fstep ; XtAddCallback( rc->param_av->wtext, XmNactivateCallback, RCREND_textact_CB, rc->param_av ) ; XtUnmanageChild( rc->param_av->wrowcol ) ; /* button to "Get" parameter from AFNI */ xstr = XmStringCreateLtoR( "Get" , XmFONTLIST_DEFAULT_TAG ) ; rc->set_pb = XtVaCreateWidget( "AFNI" , xmPushButtonWidgetClass , rc->hrc , XmNlabelString , xstr , XmNtraversalOn , True , XmNinitialResourcesPersistent , False , NULL ) ; XmStringFree(xstr) ; XtAddCallback( rc->set_pb, XmNactivateCallback, RCREND_cutout_set_CB, NULL ) ; MCW_register_help( rc->set_pb , "Use this to get the parameter\n" "for this cutout from the current\n" "AFNI crosshair location." ) ; /* button box to allow "must do" status (overriding "AND") */ rc->mustdo_bbox = new_MCW_bbox( rc->hrc , 1 , mustdo_bbox_label , MCW_BB_check , MCW_BB_noframe , NULL , NULL ) ; MCW_set_bbox( rc->mustdo_bbox , 0 ) ; MCW_reghelp_children( rc->mustdo_bbox->wrowcol , "Use this to force the cutout\n" "to be performed, even if the\n" "chosen logic is 'AND'. If the\n" "logic is 'OR', this does nothing." ) ; XtUnmanageChild( rc->mustdo_bbox->wrowcol ) ; XtManageChild( rc->hrc ) ; RETURN( rc ); } /*------------------------------------------------------------------- Callback for done button ---------------------------------------------------------------------*/ static int quit_first = 1 ; void RCREND_done_timeout_CB( XtPointer client_data , XtIntervalId * id ) { MCW_set_widget_label( done_pb , "done" ) ; quit_first = 1 ; return ; } void RCREND_done_CB( Widget w, XtPointer client_data, XtPointer call_data ) { /** like AFNI itself, require two quick presses to exit **/ if( w == done_pb && quit_first && renderings != NULL ){ MCW_set_widget_label( done_pb , "DONE " ) ; quit_first = 0 ; (void) XtAppAddTimeOut( XtWidgetToApplicationContext(done_pb) , 5000 , RCREND_done_timeout_CB , NULL ) ; return ; } if( xhair_recv >= 0 ) /* 29 Mar 1999 */ AFNI_receive_control( im3d, xhair_recv,EVERYTHING_SHUTDOWN, NULL ) ; RCREND_destroy_imseq() ; /* destroy the image window */ DESTROY_IMARR(renderings) ; /* destroy the images */ #ifdef USE_SCRIPTING DESTROY_RSA(renderings_state) ; script_load_last = -1 ; #endif if( wfunc_frame != NULL && XtIsManaged(wfunc_frame) ) /* close overlay */ RCREND_open_func_CB(NULL,NULL,NULL) ; XtUnmapWidget( shell ) ; renderer_open = 0 ; imseq = NULL ; if( dset != NULL ) dset = NULL ; if( func_dset != NULL ) func_dset = NULL ; if( gcr.dset_or != NULL ) /* delete re-oriented copy */ { THD_delete_3dim_dataset( gcr.dset_or, FALSE ); gcr.dset_or = NULL; } if( gcr.fset_or != NULL ) { THD_delete_3dim_dataset( gcr.fset_or, FALSE ); gcr.fset_or = NULL; } if( gcr.mset != NULL ) gcr.mset = NULL; /* there is no new data here */ if( gcr.fdm != NULL ) { free(gcr.fdm); gcr.fdm = NULL; } if( gcr.rh != NULL ){ destroy_CREN_renderer(gcr.rh) ; gcr.rh = NULL ; func_cmap_set = 0 ; } FREE_VOLUMES ; INVALIDATE_OVERLAY ; MPROBE ; return ; } /*------------------------------------------------------------------- Load the data from the dataset into local arrays ---------------------------------------------------------------------*/ void RCREND_reload_dataset(void) { THD_3dim_dataset * local_dset; int sublist[2] = {1, 0}; /* sub-brick list for resampling */ int ii , nvox , vmin,vmax , cbot,ctop , ival,val , cutdone, btype ; float fac ; void * var ; byte * gar ; MRI_IMAGE * vim ; XmString xstr ; char str[64] ; #define HISTOGRAMATE /* 25 Jul 2001 */ #define NHIST 255 int vtop ; ENTRY( "RCREND_reload_dataset" ); MCW_invert_widget(reload_pb) ; /* flash a signal */ /* start by tossing any old data */ FREE_VOLUMES ; /* make sure the dataset is in memory */ DSET_load(dset) ; local_dset = dset; /* default - if we don't re-orient */ ival = dset_ival; /* unless we resample, then 0 */ /* make an oriented underlay, if needed 26 June 2002 - rickr */ if ( !IS_AXIAL_RAI( dset ) ) { if ( new_dset || gcr.dset_or == NULL ) /* we need a new one */ { if ( gcr.dset_or != NULL ) /* lose the old one */ { THD_delete_3dim_dataset( gcr.dset_or, FALSE ); gcr.dset_or = NULL; } /* resample only sub-brick dset_ival */ sublist[0] = 1; sublist[1] = dset_ival; fprintf(stderr, "++ reorienting underlay as rai..."); gcr.dset_or = r_new_resam_dset(dset, NULL, 0,0,0, "rai", RESAM_NN_TYPE, sublist); fprintf(stderr, " done\n"); } if (gcr.dset_or == NULL) XBell(dc->display,100); /* an error - keep local_dset as dset */ else { local_dset = gcr.dset_or; /* woohoo! we have our new dataset */ ival = 0; } } gcr.mset = local_dset; /* we have our rendering master */ /* reset fdm - delete and re-create */ if( gcr.fdm != NULL ) free(gcr.fdm); gcr.fdm = THD_oriented_brick( gcr.mset, "RAI" ); /* get mast FD_brick */ vim = DSET_BRICK(local_dset,ival) ; nvox = vim->nvox ; var = DSET_ARRAY(local_dset,ival) ; brickfac = DSET_BRICK_FACTOR(local_dset,ival) ; /* find data range, clip it, convert to bytes */ grim = mri_new_conforming( vim , MRI_byte ) ; /* new image data */ gar = MRI_BYTE_PTR(grim) ; btype = DSET_BRICK_TYPE(local_dset,ival); switch( btype ){ default:{ fprintf( stderr, "RCREND_reload_dataset: invalid brick type %d\n", btype ); EXRETURN; } case MRI_short:{ short * sar = (short *) var ; vmin = vmax = sar[0] ; for( ii=1 ; ii < nvox ; ii++ ){ /* find range of values */ val = sar[ii] ; if( vmin > val ) vmin = val ; else if( vmax < val ) vmax = val ; } #ifdef HISTOGRAMATE if( vmax > vmin && vmin >= 0 && new_dset ){ /* 25 Jul 2001: find 'good' upper value */ int hist[NHIST] , nhist,nh; nhist = (vmax-vmin > NHIST) ? NHIST : (vmax-vmin) ; mri_histogram( vim , vmin,vmax , 1,nhist , hist ) ; for( nh=ii=0 ; ii < nvox ; ii++ ) if( sar[ii] ) nh++ ; /* count nonzeros */ nh *= 0.005 ; /* find 99.5% point */ for( ii=nhist-1 ; ii > 1 && nh > 0 ; ii-- ) nh -= hist[ii] ; /* in histogram */ vtop = vmin + (ii+0.5)*(vmax-vmin)/(nhist-0.01) ; if( vtop > vmax || vtop <= vmin ) vtop = vmax ; } else { vtop = vmax ; } #else vtop = vmax ; #endif if( new_dset ){ AV_assign_ival( clipbot_av , vmin ) ; cbot = vmin ; AV_assign_ival( cliptop_av , vtop ) ; ctop = vtop ; /* 25 Jul 2001: vmax -> vtop */ } else { cbot = MAX( clipbot_av->ival , vmin ) ; ctop = MIN( cliptop_av->ival , vmax ) ; } fac = (ctop > cbot) ? 127.9/(ctop-cbot) : 1.0 ; for( ii=0 ; ii < nvox ; ii++ ){ val = sar[ii] ; ival = fac * (val-cbot) ; RANGE(ival,0,127) ; gar[ii] = ival ; } } break ; case MRI_byte:{ byte * bar = (byte *) var ; vmin = vmax = bar[0] ; for( ii=1 ; ii < nvox ; ii++ ){ /* find range of values */ val = bar[ii] ; if( vmin > val ) vmin = val ; else if( vmax < val ) vmax = val ; } #ifdef HISTOGRAMATE if( vmax > vmin && new_dset ){ /* 25 Jul 2001: find 'good' upper value */ int hist[256] , nhist=256,nh; mri_histobyte( vim , hist ) ; for( nh=0,ii=1 ; ii < nhist ; ii++ ) nh += hist[ii] ; /* count nonzeros */ nh *= 0.005 ; /* find 99.5% point */ for( ii=nhist-1 ; ii > 1 && nh > 0 ; ii-- ) nh -= hist[ii] ; /* in histogram */ vtop = ii ; if( vtop > vmax || vtop <= vmin ) vtop = vmax ; } else { vtop = vmax ; } #else vtop = vmax ; #endif if( new_dset ){ AV_assign_ival( clipbot_av , vmin ) ; cbot = vmin ; AV_assign_ival( cliptop_av , vtop ) ; ctop = vtop ; /* 25 Jul 2001: vmax -> vtop */ } else { cbot = MAX( clipbot_av->ival , vmin ) ; ctop = MIN( cliptop_av->ival , vmax ) ; } fac = (ctop > cbot) ? 127.9/(ctop-cbot) : 1.0 ; for( ii=0 ; ii < nvox ; ii++ ){ val = bar[ii] ; ival = fac * (val-cbot) ; RANGE(ival,0,127) ; gar[ii] = ival ; } } break ; } /* set label showing data range */ sprintf(str,"Min=%d Max=%d",vmin,vmax) ; xstr = XmStringCreateLtoR( str , XmFONTLIST_DEFAULT_TAG ) ; XtVaSetValues( range_lab , XmNlabelString , xstr , NULL ) ; XmStringFree(xstr) ; /* if brick is scaled, show the scaled labels */ HIDE_SCALE ; if( brickfac != 0.0 && brickfac != 1.0 ){ char minch[16] , maxch[16] ; AV_fval_to_char( vmin*brickfac , minch ) ; AV_fval_to_char( vmax*brickfac , maxch ) ; sprintf(str,"[%s %s]",minch,maxch) ; xstr = XmStringCreateLtoR( str , XmFONTLIST_DEFAULT_TAG ) ; XtVaSetValues( range_faclab , XmNlabelString , xstr , NULL ) ; XmStringFree(xstr) ; AV_fval_to_char( brickfac * clipbot_av->ival , minch ) ; sprintf(str,"[-> %s]",minch) ; xstr = XmStringCreateLtoR( str , XmFONTLIST_DEFAULT_TAG ) ; XtVaSetValues( clipbot_faclab , XmNlabelString , xstr , NULL ) ; XmStringFree(xstr) ; AV_fval_to_char( brickfac * cliptop_av->ival , maxch ) ; sprintf(str,"[-> %s]",maxch) ; xstr = XmStringCreateLtoR( str , XmFONTLIST_DEFAULT_TAG ) ; XtVaSetValues( cliptop_faclab , XmNlabelString , xstr , NULL ) ; XmStringFree(xstr) ; XtManageChild( range_faclab ) ; XtManageChild( clipbot_faclab ) ; XtManageChild( cliptop_faclab ) ; } else { XtUnmanageChild( range_faclab ) ; XtUnmanageChild( clipbot_faclab ) ; XtUnmanageChild( cliptop_faclab ) ; } FIX_SCALE_SIZE ; /* check graphs and set opacity map */ { int hist[256] , nvox = grim->nvox , htop , ii, max_index ; float ofac = current_cutout_state.opacity_scale ; /* do the histogram graph */ MCW_histo_bytes( nvox , MRI_BYTE_PTR(grim) , hist ) ; /* get value of the maximum non-zero index */ for( max_index=255; (max_index > 0) && !hist[max_index]; max_index-- ) ; if ( max_index >= GRAF_SIZE ) /* then fold 2 into 1 */ for( ii=0 ; ii < GRAF_SIZE ; ii++ ) hist[ii] = hist[2*ii] + hist[2*ii+1]; htop = 0 ; for( ii=0 ; ii < GRAF_SIZE ; ii++ ) if( ii > 0 && hist[ii] > htop ) htop = hist[ii] ; /* find max value */ if( htop == 0 ){ set_MCW_pasgraf( his_graf , NULL ) ; } else { float scl = (GRAF_SIZE-0.44) / sqrt((double)htop) ; byte bhis[GRAF_SIZE] ; for( ii=0 ; ii < GRAF_SIZE ; ii++ ) bhis[ii] = (hist[ii] > htop) ? GRAF_SIZE-1 : (byte)(scl*sqrt((double)hist[ii])+0.49) ; set_MCW_pasgraf( his_graf , bhis ) ; } redraw_MCW_pasgraf( his_graf ) ; /* modify the grayscale per the brightness graf */ if( ! gry_graf->yeqx ){ byte * bar=MRI_BYTE_PTR(grim) , * fun=gry_graf->func ; /* remember that the graph has 256 values, but gray data has 128 */ for( ii=0 ; ii < nvox ; ii++ ) bar[ii] = fun[ 2 * bar[ii] ] >> 1 ; } /* modify the opacity per the opacity graf */ for ( ii=0; ii < GRAF_SIZE; ii++ ) /* init */ gcr.omap[ii] = ii / (GRAF_SIZE - 1.0); if( !opa_graf->yeqx || ofac < 1.0 ){ byte * fun=opa_graf->func ; if( !opa_graf->yeqx ) for ( ii=0; ii < GRAF_SIZE; ii++ ) gcr.omap[ii] = fun[2*ii]/(2.0*GRAF_SIZE - 1); if( ofac < 1.0 ) for ( ii=0; ii < GRAF_SIZE; ii++ ) gcr.omap[ii] *= ofac; } CREN_set_opamap( gcr.rh, gcr.omap, func_color_opacity ); /** 02 Dec 2002 [RWCox]: set bot,top values for graphs (for Button3 popup labels) **/ GRAF_set_xyrange( gry_graf , (float)cbot, (float)ctop , 0.0,255.0 ) ; GRAF_set_xyrange( opa_graf , (float)cbot, (float)ctop , 0.0, 1.0 ) ; PASGRAF_set_xyrange( his_graf , (float)cbot, (float)ctop , 0.0, 0.0 ) ; } /*--- Now deal with overlay, if need be ---*/ func_computed = 0 ; /* at this point, data is for grayscale rendering */ cutdone = 0 ; /* cutouts not done yet */ if( DO_OVERLAY ){ byte * gar , * ovar ; int nvox = grim->nvox , ii ; RCREND_reload_func_dset() ; if( num_cutouts > 0 && !func_cut_overlay ){ /* do cutouts NOW if not */ RCREND_cutout_blobs(grim) ; /* done to overlay */ cutdone = 1 ; } ovar = MRI_BYTE_PTR(ovim) ; if( !func_showthru ){ /* the old code: embed color into volume */ gar = MRI_BYTE_PTR(grim) ; /* convert gar into index into the functional colormap */ for( ii=0 ; ii < nvox ; ii++ ) if( ovar[ii] != 0 ) gar[ii] = 128 + ovar[ii] ; /* color */ /* 127->128 11 Jan 2002 */ } else { /* 07 Jan 2000: make the showthru bricks instead */ byte * garst ; grim_showthru = mri_new_conforming( vim , MRI_byte ) ; garst = MRI_BYTE_PTR(grim_showthru) ; memset( garst , 0 , sizeof(byte)*nvox ) ; for( ii=0 ; ii < nvox ; ii++ ) /* load values only if overlay */ if( ovar[ii] != 0 ) garst[ii] = 128 + ovar[ii] ; /* color */ } func_computed = 1 ; /* data is now set for color rendering */ } /*--- Other piddling details ---*/ if( num_cutouts > 0 && !cutdone ){ /* if cutouts hit overlay */ RCREND_cutout_blobs(grim) ; if( func_showthru ) RCREND_cutout_blobs(grim_showthru) ; } #if 0 /* rcr - do not render crosshairs with data (for the moment) */ /* 2002.08.29 */ /* fill crosshair data into image volumes */ if( xhair_flag ) { if ( !ISVALID_DSET( gcr.mset ) ) XBell(dc->display,100); else if ( ! func_computed ) RCREND_xhair_underlay( gcr.mset ); /* underlay only */ else if ( func_showthru ) RCREND_xhair_overlay( gcr.mset, grim_showthru ); else RCREND_xhair_overlay( gcr.mset, grim ); /* overlay is in grim */ } #endif MCW_invert_widget(reload_pb) ; /* turn the signal off */ new_dset = 0 ; new_data_loaded = 1 ; FIX_SCALE_SIZE ; /* 09 May 2001 */ EXRETURN ; } /*------------------------------------------------------------------- Callback for the reload button ---------------------------------------------------------------------*/ void RCREND_reload_CB( Widget w, XtPointer client_data, XtPointer call_data ) { ENTRY( "RCREND_reload_CB" ); if( dset == NULL ){ XBell(dc->display,100) ; EXRETURN ; } RCREND_reload_dataset() ; /* load data again */ if( gcr.rh != NULL ) RCREND_draw_CB(NULL,NULL,NULL) ; /* draw */ EXRETURN ; } /*----------------------------------------------------------------------- Actually send the computed bricks to the renderer -------------------------------------------------------------------------*/ void RCREND_reload_renderer(void) { ENTRY( "RCREND_reload_renderer" ); if( gcr.rh == NULL ) EXRETURN ; /* error */ CREN_set_interp(gcr.rh, interp_ival); if( func_computed && func_showthru && func_showthru_pass ) { /* if we have a reoriented dataset, use it 27 June 2002 - rickr */ if ( gcr.fset_or != NULL ) CREN_dset_axes(gcr.rh, gcr.fset_or ); else CREN_dset_axes(gcr.rh, func_dset ); CREN_set_databytes(gcr.rh, grim_showthru->nx, grim_showthru->ny, grim_showthru->nz, MRI_BYTE_PTR(grim_showthru)); } else { /* if we have re-oriented the dataset, give that to the library */ if ( gcr.dset_or != NULL ) CREN_dset_axes(gcr.rh, gcr.dset_or ); else CREN_dset_axes(gcr.rh, dset ); CREN_set_databytes(gcr.rh, grim->nx, grim->ny, grim->nz, MRI_BYTE_PTR(grim)); } if( func_computed && !func_cmap_set ){ if ( wfunc_color_pbar->bigmode ) CREN_set_rgbmap( gcr.rh, NPANE_BIG, gcr.bigstuff.r, gcr.bigstuff.g, gcr.bigstuff.b ); else CREN_set_rgbmap( gcr.rh, MIN( dc->ovc->ncol_ov, GRAF_SIZE ), (dc)->ovc->r_ov, (dc)->ovc->g_ov, (dc)->ovc->b_ov ); func_cmap_set = 1 ; /* do we need to reset the colormap? */ } EXRETURN ; } /*------------------------------------------------------------------- Callback for draw button ---------------------------------------------------------------------*/ void RCREND_draw_CB( Widget w, XtPointer client_data, XtPointer call_data ) { MRI_IMAGE * rim ; ENTRY( "RCREND_draw_CB" ); #ifdef USE_SCRIPTING if( script_dontdraw ) EXRETURN ; /* 24 Nov 2000 */ #endif if( dset == NULL ){ XBell(dc->display,100) ; EXRETURN ; } MCW_invert_widget(draw_pb) ; /* if needed, create stuff for rendering */ /* rickr - get new renderer and set basic opacity map */ if ( gcr.rh == NULL ) { int ii; gcr.rh = new_CREN_renderer(); for ( ii = 0; ii < GRAF_SIZE; ii++ ) gcr.omap[ii] = ii/(GRAF_SIZE - 1.0); CREN_set_opamap( gcr.rh, gcr.omap, 1.0 ); } RCREND_load_cutout_state() ; /* load cutout data from widgets */ if( RCREND_cutout_state_changed() ){ /* if cutouts changed, must reload data */ FREE_VOLUMES ; if( func_cut_overlay ) INVALIDATE_OVERLAY ; old_cutout_state = current_cutout_state ; } if( xhair_flag && CHECK_XHAIR_MOTION ){ /* check for new crosshair position */ if( xhair_ovc > 0 && DO_OVERLAY ) INVALIDATE_OVERLAY ; else FREE_VOLUMES ; } if ( NEED_RELOAD ) RCREND_reload_dataset(); /* setup for viewing */ angle_roll = RCREND_evaluate( roll_av ) ; /* read angles from arrowvals */ angle_pitch = RCREND_evaluate( pitch_av ) ; angle_yaw = RCREND_evaluate( yaw_av ) ; /* create and display the rendered image */ if( new_data_loaded || /* new data was loaded here */ CREN_needs_data(gcr.rh) || /* renderer isn't ready yet */ (func_computed && func_showthru) ){ /* am doing ShowThru images */ func_showthru_pass = 0 ; /* always a good value */ RCREND_reload_renderer() ; /* load data from arrays here into renderer */ new_data_loaded = 0 ; } CREN_set_angles(gcr.rh, angle_yaw * PI / 180, angle_pitch * PI / 180, angle_roll * PI / 180 ); CREN_set_min_opacity(gcr.rh, 0.05 * current_cutout_state.opacity_scale); rim = CREN_render( gcr.rh, &gcr.rotm ); if( rim == NULL ){ (void) MCW_popup_message( draw_pb , "** Rendering fails, **\n" "** for unknown reasons **\n\n" "** Sorry -- RWCox **\n" , MCW_USER_KILL | MCW_TIMER_KILL ) ; XBell(dc->display,100) ; MCW_invert_widget(draw_pb) ; EXRETURN ; } /* 07 Jan 2000 - make the showthru image now, if needed */ #if 1 # define STCOM(x) (x) #else # define STCOM(x) (((x) + ((x)<<1))>>2) /* 0.75 * x */ #endif if( func_computed && func_showthru ){ MRI_IMAGE * cim ; float ccf=func_showthru_fac , ggf ; /* 10 Jan 2000: merger factors */ int ccm=0 ; if( ccf < 0.0 || ccf > 1.0 ) ccf = 1.0 ; ggf = 1.0 - ccf ; ccm = (ccf != 1.0) ; func_showthru_pass = 1 ; RCREND_reload_renderer() ; /* load showthru data */ #if 0 /* dcue not yet implemented in CREN 24 Jan 2002 */ if( func_showthru_dcue ) MREN_depth_cue( render_handle , 1 ) ; /* 11 Sep 2001 */ #endif cim = CREN_render( gcr.rh, &gcr.rotm ); /* render it */ #if 0 if( func_showthru_dcue ) MREN_depth_cue( render_handle , 0 ) ; /* 11 Sep 2001 */ #endif if( cim == NULL ){ (void) MCW_popup_message( draw_pb , "** ShowThru Rendering fails, **\n" "** for unknown reasons **\n\n" "** Sorry about that -- RWCox **\n" , MCW_USER_KILL | MCW_TIMER_KILL ) ; XBell(dc->display,100) ; } else { byte *rar=MRI_BYTE_PTR(rim), *car=MRI_RGB_PTR(cim); /* composite it */ int ii, i3; for( ii=0 ; ii < cim->nvox ; ii++ ){ i3 = ii * 3; if( car[i3] == 0 && car[i3+1] == 0 && car[i3+2] == 0 ){ car[i3] = car[i3+1] = car[i3+2] = STCOM( rar[i3] ); } else if( ccm ){ /* 10 Jan 2000 */ car[i3] = ccf*car[i3] + ggf*rar[i3] ; /* merge color */ car[i3+1] = ccf*car[i3+1] + ggf*rar[i3] ; /* & grayscale */ car[i3+2] = ccf*car[i3+2] + ggf*rar[i3] ; } } mri_free(rim) ; rim = cim ; } } if ( xhair_flag ) { get_xhair_points( &gcr.xhseg, gcr.mset ); xhairs_to_image_pts( &gcr.xhseg, gcr.mset ); /* now (0,0,z) means the center of the projected image */ rotate_xhair_points( &gcr.xhseg, &gcr.rotm ); draw_xhairs_in_image( &gcr.xhseg, rim ); } /* 20 Dec 1999 - restrict colors, if ordered and needed */ if( rim->kind == MRI_rgb && wfunc_color_pbar != NULL && func_mixshade >= NOSHADE ){ int ii ; byte * bp = MRI_RGB_PTR(rim) ; for( ii=0 ; ii < rim->nvox ; ii++ ) if( bp[3*ii] != 0 || bp[3*ii+1] != 0 || bp[3*ii+2] != 0 ) DC_rgb_to_ovrgb( dc , wfunc_color_pbar->num_panes , wfunc_color_pbar->ov_index , (func_mixshade == NOMIX) , bp+(3*ii) , bp+(3*ii+1) , bp+(3*ii+2) ) ; } #ifdef USE_SCRIPTING if( last_rendered_state == NULL ) last_rendered_state = (RENDER_state *) malloc(sizeof(RENDER_state)) ; RCREND_widgets_to_state( last_rendered_state ) ; #endif mri_add_name( accum_label , rim ) ; /* 17 Jun 2005 */ if( accum_flag || automate_flag ){ if( renderings == NULL ){ INIT_IMARR( renderings ) ; #ifdef USE_SCRIPTING INIT_RSA( renderings_state ) ; script_load_last = -1 ; #endif } ADDTO_IMARR( renderings , rim ) ; #ifdef USE_SCRIPTING { RENDER_state * rs = (RENDER_state *) malloc(sizeof(RENDER_state)) ; *rs = *last_rendered_state ; ADDTO_RSA( renderings_state , rs ) ; } #endif } else { DESTROY_IMARR( renderings ) ; INIT_IMARR( renderings ) ; ADDTO_IMARR( renderings , rim ) ; #ifdef USE_SCRIPTING { RENDER_state * rs = (RENDER_state *) malloc(sizeof(RENDER_state)) ; DESTROY_RSA( renderings_state ) ; INIT_RSA( renderings_state ) ; script_load_last = -1 ; *rs = *last_rendered_state ; ADDTO_RSA( renderings_state , rs ) ; } #endif } RCREND_update_imseq() ; MCW_invert_widget(draw_pb) ; FIX_SCALE_SIZE ; /* 09 May 2001 */ EXRETURN ; } /*------------------------------------------------------------------- Callback for help button ---------------------------------------------------------------------*/ void RCREND_help_CB( Widget w, XtPointer client_data, XtPointer call_data ) { ENTRY( "RCREND_help_CB" ); (void ) new_MCW_textwin( info_lab , "++++++++++++++++++ V O L U M E R E N D E R I N G ++++++++++++++++++\n" "\n" "This plugin is used to render one brick from a 3D dataset in grayscale\n" "(the underlay), possibly overlaid in color with another (functional)\n" "dataset. Although lengthy, this help is still rather terse. Some\n" "experimentation will be needed to get decent results, since there are\n" "many controls that affect the way the final images appear.\n" "\n" "General Notes:\n" "--------------\n" " * To be rendered, an underlay dataset must be stored as bytes or\n" " shorts (but may have a floating point scaling factor attached).\n" " The dataset must be stored as axial slices in the 'RAI' \n" " orientation (x axis is Right-to-Left, y axis is\n" " Anterior-to-Posterior, and z axis is Inferior-to-Superior).\n" " This orientation is how datasets are written out in the +acpc\n" " and +tlrc coordinates -- with axial slices.\n" /* #ifdef ONLY_AXIAL 25 June, 2002 */ " N.B.: Combining the 3ddup and 3dresample programs makes it\n" " possible to create an cubical-voxel axially-oriented\n" " copy of any dataset.\n" #ifndef ONLY_AXIAL " N.B.: The requirement that the dataset be stored in axial slices\n" " has been removed; however, the cutouts will not work\n" " properly. For example, a 'Superior to' cutout will remove\n" " voxels along the 3rd axis of a dataset; for a dataset made\n" " up of sagittal slices, this will result in a 'Left of' or\n" " a 'Right of' type of cutting.\n" #endif "\n" " * The program 3dIntracranial can be used to remove extra-cranial\n" " matter from an anatomical dataset.\n" "\n" " * Use the Draw button to render an image after making changes\n" " to the drawing parameters or after closing the image window.\n" "\n" " * The 'Reload' button is used to re-copy the dataset brick into\n" " the renderer. This can be used if you are altering the\n" " dataset interactively with the Draw Dataset plugin.\n" " Otherwise, you probably don't need this often, since the reload\n" " operation will be carried out as needed by the renderer.\n" "\n" " * The Interpolation mode determines the type of interpolation\n" " between two neighbors along a view line.\n" "\n" " Neighbor = use the value of the closer neighbor.\n" " Twostep = if close to a neighbor, use its value.\n" " Otherwise, use the average of both neighbors.\n" " Linear = use a distance-weighted average of neighbors.\n" "\n" " * If you depress 'See Xhairs', a 3D set of crosshairs\n" " corresponding to the AFNI focus position will be drawn.\n" " If you move the crosshairs in one of the AFNI image\n" " windows, the rendering window is not automatically updated,\n" " unless the 'DynaDraw' button is also depressed. Otherwise,\n" " the next time the rendering is redrawn for some other\n" " reason, the correct crosshair positions will be shown.\n" " 02 Jun 1999: The renderer will now draw partial crosshair sets,\n" " as indicated by the 'Xhairs' chooser in the AFNI control\n" " window from which the renderer was started.\n" " 08 Mar 2001: Right-clicking (mouse button 3) on this toggle will\n" " popup a color chooser. If you set the color to something\n" " besides 'none', AND if you are displaying a color overlay,\n" " then the crosshairs will be rendered in the overlay (you\n" " could still choose 'white' for the color, if you like).\n" " N.B.: If the color opacity is set to 'ShowThru', then the\n" " crosshairs will show through whatever underlay data\n" " may be in the way.\n" " N.B.: If you want only the crosshairs in color, then set the\n" " theshold on the overlay dataset so high that no actual\n" " data will show in color. The crosshair overlay will\n" " still be visible.\n" " N.B.: If you change the crosshair gap in the AFNI control panel,\n" " you will have to press 'Reload' in the renderer to force\n" " a redraw with the new crosshairs.\n" "\n" " * If you depress 'DynaDraw', then the image will be re-\n" " rendered immediately whenever certain actions are taken:\n" " + The 'See Xhairs' toggle state is changed.\n" " + A viewing angle is changed by pressing an arrow.\n" " + A cutout parameter value is changed by pressing an arrow or\n" " a 'Get' button.\n" " Changing one of these values by directly typing in the\n" " corresponding text entry field will NOT force a redraw\n" " even in DynaDraw mode -- you will have to press 'Draw'.\n" " Other changes (e.g., altering the opacity graph) will\n" " not force a dynamic redraw and also require the use\n" " of the 'Draw' button.\n" " N.B.: The data entry fields with which DynaDraw and Automate\n" " interact are displayed with a raised border, unlike\n" " other data entry fields (e.g., 'Bot').\n" " N.B.: The default stepsize for the angle and cutout variables\n" " when an arrow is pressed is 5.0. These values can be\n" " altered by setting the Unix environment variables\n" " AFNI_RENDER_ANGLE_DELTA and AFNI_RENDER_CUTOUT_DELTA\n" " prior to running AFNI. Once AFNI is started, these\n" " stepsizes can only be altered from the\n" " 'Datamode->Misc->Edit Environment' menu item.\n" " N.B.: Other circumstances that will invoke automatic redrawing\n" " when DynaDraw is depressed include:\n" " + The crosshairs are moved in an AFNI image window belonging\n" " to the same controller, AND the 'See Xhairs' button is\n" " depressed.\n" " + You are also editing the dataset using the 'Draw Dataset'\n" " plugin (invoked from the same controller), and you have\n" " changed the dataset with a drawing operation.\n" "\n" " * If you depress 'Accumulate' IN, then rendered images are\n" " saved as they are computed and can be re-viewed in the\n" " image display window. If Accumulate is OUT, then only\n" " the latest image is kept.\n" " N.B.: The image display window is like an AFNI slice viewing\n" " window, but the slider control simply lets you scroll\n" " back through past renderings, not through the spatial\n" " extent of the dataset in any way. Each additional\n" " accumulated image appears as the last image in the\n" " sequence, no matter where you are when 'Draw' activates.\n" #ifdef ALLOW_INCROT /* 26 Apr 2002 - RWCox */ "\n" " * The 'I' toggle left of the 'Roll' button lets you select\n" " incremental mode for angle changes made with the arrow\n" " buttons.\n" " + In incremental mode, the extra rotation implied by\n" " pressing the arrow button will be around the spatial\n" " axis corresponding to that button:\n" " Roll=I-S axis, Pitch=R-L axis, Yaw=A-P axis.\n" " + In non-incremental mode, the rotation angles are always\n" " applied in the order Yaw, Pitch, Roll; the result may\n" " not be intuitive since the SO(3) group is not Abelian.\n" " + In incremental mode, when you press an angle arrow button,\n" " new absolute spatial angles corresponding to the changed\n" " orientation are calculated and put into the Roll, Pitch,\n" " and Yaw text fields.\n" " + Incremental rotation mode does not combine with Automate.\n" #endif /* ALLOW_INCROT */ "\n" "Brightness and Opacity:\n" "-----------------------\n" " * The Min= and Max= values show the range of numbers stored\n" " in the dataset brick. The brick is copied into internal\n" " memory for rendering. You can use the 'Bot' and 'Top'\n" " controls to limit the range of the copied voxel data.\n" " Anything below 'Bot' will be set to the Bot value, and\n" " anything above 'Top' will be set to the Top value.\n" " (If Bot < Min, then Bot is effectively equal to Min;\n" " if Top > Max, then Top is effectively equal to Max.)\n" " In this way, you can eliminate the effect of a few extreme\n" " data values.\n" "\n" " * The 'Sqrt Histogram' graph displays the square root of the\n" " histogram of the dataset brick. (The square root is graphed\n" " so that small values will be somewhat enhanced in the display.\n" " Also, the 0 bin value is not used in selecting the scale factor\n" " for display, since it often is far larger than other bins.)\n" " The purpose of this histogram is to let you choose good\n" " values for Bot and Top. After altering Bot and Top, press\n" " the 'Reload' button to make the histogram graph be redrawn.\n" "\n" " * The 'Brightness' and 'Opacity' graphs are used to control\n" " the mappings from dataset signal level (the numbers stored\n" " in the voxels) to the grayscale and opacity levels.\n" " The abscissa represents the copied voxel values, ranging\n" " the larger of Min or Bot, to the smaller of Max or Top.\n" " A larger opacity makes a voxel less transparent; for\n" " example, a high opacity with a low brightness is like\n" " black barrier in the line of sight. Zero opacity\n" " means a voxel is transparent and does not contribute to\n" " the rendered image.\n" "\n" " * The ordinate on the 'Brightness' graph ranges from black\n" " at the bottom to white at the top. The ordinate on the\n" " 'Opacity' graph ranges from 0 (transparent) to 1 (opaque).\n" "\n" " * The 'Opacity Factor' control lets you scale the entire\n" " underlay opacity down by some constant factor. (However,\n" " this doesn't seem to be very useful.)\n" "\n" "Cutouts:\n" "--------\n" " * The 'Cutouts' menu lets you select the number of regions to\n" " be cut out of the volume prior to rendering (this is done\n" " by setting the voxel opacity inside each cutout to zero).\n" " Up to 9 cutouts can be combined at once. There are 21 types\n" " of cutouts, each of which is controlled by a single parameter.\n" " For example, the parameter for a 'Right of' cutout is an\n" " x-coordinate, and all the voxels to the right of this value\n" " will be included in the cutout.\n" " N.B.: Right (of midline) = negative x\n" " Left (of midline) = positive x\n" " Anterior (to the AC) = negative y\n" " Posterior (to the AC) = positive y\n" " Inferior (to the AC-PC) = negative z\n" " Superior (to the AC-PC) = positive z\n" "\n" " * The 'Expr > 0' cutout is a special (and slow) case. Instead\n" " of a number, you enter an expression (using the 3dcalc\n" " syntax) containing the symbols 'x', 'y', and 'z', which\n" " represent the spatial coordinates of each voxel. Voxels\n" " where the expression evaluates to a positive number will\n" " be cut out. For example, '900-x*x-y*y-z*z' will cut out\n" " the INTERIOR of a sphere of radius 30 mm, centered at the\n" " origin of coordinates; 'x*x+y*y+z*z-900' will remove the\n" " EXTERIOR of this sphere.\n" "\n" " * The 'TT Ellipsoid' cutout will remove all voxels exterior to\n" " an ellipsoid that approximates the outer contour of the\n" " Talairach-Tournoux atlas, when its 'Percentage' parameter\n" " is set to 100. Smaller percentages will shrink the ellipsoid\n" " in towards the center of the brain; large percentages will\n" " expand it outwards.\n" "\n" " * The 12 'Behind', 'Above', etc. cutouts are relative to planes at\n" " 45 degrees to the standard views. For example, 'Behind AL-PR'\n" " cuts out behind a slice that starts at Anterior-Left (AL) and\n" " ends up at Posterior-Right (PR) -- halfway between a coronal\n" " and a sagittal slice. The simplest way to set the value\n" " parameter for these (at least to start) is to use 'Get'.\n" "\n" " * The 'NonOverlay++' cutout will remove all voxels that would not\n" " get colored if the overlay were turned on. (If the parameter\n" " 'Radius++' is positive, then the region is dilated by that\n" " many mm in all directions.) This can be useful for seeing\n" " exactly the anatomy that is deemed to be 'active'.\n" " Notes:\n" " + If there is no overlay dataset loaded, then this type of cutout\n" " has no effect.\n" " + 'Get' does nothing for this type of cutout.\n" " + Viewing the color overlay with Radius++ set to a positive\n" " value may be confusing, since the colored voxels will be\n" " buried inside the visible tissue. The combination of bright\n" " colors with high color opacity and the use of a small\n" " underlay opacity factor can make it possible to see the\n" " color overlay through the translucent surrounding shell\n" " of thickness Radius++.\n" "\n" " * Cutouts can be combined with the 'OR' logic, which means that\n" " the union of all the specified cutout regions will be\n" " removed. They can also be combined with the 'AND' logic,\n" " which means that the intersection of the cutout regions\n" " will be removed.\n" " N.B.: If the 'AND' logic is selected, a cutout can still be\n" " forced to be removed in its entirety using the 'Must Do'\n" " control. (That is, 'AND' only applies to those that\n" " do NOT have 'Must Do' selected; 'OR' applies to those\n" " that DO have 'Must Do' selected.) For an example, try\n" " combining 'Right of', 'Anterior to', and 'Superior to'\n" " cutouts first with 'OR', then with 'AND', and finally\n" " with 'AND' but with the 'Superior to' cutout set to\n" " 'Must Do'.\n" "\n" "Automating the Calculation of Many Renderings:\n" "----------------------------------------------\n" " * If you depress 'Automate', then the automatic generation\n" " of renderings as some parameter varies is enabled.\n" " The 'Frames' field controls how many renderings will\n" " be made. To vary some parameter, you type an\n" " arithmetic expression in the variable 't' in the\n" " parameter control field. The parameters that can\n" " be so varied are the viewing angles and the cutout\n" " parameters (i.e., those whose data entry field is\n" " drawn with a raised border). For the first rendering\n" " t=0, for the second t=1, etc., up to t=N-1, where N is\n" " the number of frames ordered. (You can also use the\n" " variable 'N' in the parameter expressions.) Once the\n" " expressions are set up, press 'Compute' to begin\n" " rendering automation. (Then go have a pumpernickel\n" " bagel and a cup of lapsang souchong tea.)\n" "\n" " * Notes about Automate:\n" " 1) If none of the parameters has an expression involving\n" " 't', then each frame in the rendering will be identical,\n" " but the program won't detect that and you will waste a\n" " lot of CPU time and memory.\n" " 2) Use the same expression syntax as with program 3dcalc.\n" " An illegal expression (e.g., '2+*3t') will silently\n" " evaluate to zero.\n" " 3) It is legal to have more than one parameter depend on 't'.\n" " For example, combining cutouts\n" " Anterior To: -50+2*t\n" " Posterior To: -40+2*t\n" " with the OR logic produces a 10 mm thick coronal slice\n" " that slides backwards at 2 mm per frame.\n" " 4) If Accumulate is on, then the frames created by automated\n" " rendering will be added to the list of stored images. If\n" " Accumulate is off, the previously saved images will be\n" " discarded, and only the newly generated image sequence will\n" " be available for viewing.\n" " 5) There is no way to save an animation to disk as a unit.\n" " However, you could use the 'Save:bkg' button on the image\n" " viewer to save each image to disk in PNM format, then convert\n" " the collection of individual image files to some movie format,\n" " using software outside of the AFNI package.\n" " 6) Using an arrow to change a field with an expression\n" " entered will result in the destruction of the expression\n" " string and its replacement by a number.\n" " 7) At the end of an Automate run, you can convert a field\n" " with an expression to its final numerical value by\n" " clicking in the data field and then pressing the\n" " Enter (or Return) key. In combination with Accumulate,\n" " this makes it easy to chain together the results of\n" " multiple automated rendering computations, first varying\n" " one parameter and then another.\n" " 8) During an Automate run, a 'CANCEL' button to the right\n" " of 'Compute' becomes visible. If you press this, then\n" " the automation will be interrupted when it finishes the\n" " image on which it is working (you have to wait until\n" " that time -- pressing the button twice won't help!).\n" #ifdef USE_SCRIPTING " Z) The 'Scripts' method, described below, is an entirely\n" " separate method of generating multiple renderings at once.\n" #endif "\n" "Color Overlays\n" "--------------\n" "By pressing the 'Overlay' button, you can access the controls for\n" "displaying a second dataset in color on top of the underlay dataset.\n" "(Unlike the underlay dataset, the voxel data in the overlay may be\n" "stored as floats, as well as shorts or bytes.) The controls are\n" "designed to be similar to the controls in the 'Define Function'\n" "control panel in a main AFNI window, and so only the principal\n" "differences will be explained here.\n" "\n" " * One brick ('Color') is chosen to determine the colors shown, and\n" " another brick ('Thr') to determine which voxels from the overlay\n" " dataset will be shown at all. If you don't want thresholding\n" " applied, just set the threshold slider down to 0.\n" "\n" " * The 'Color Opacity' control determines how opaque each supra-\n" " threshold voxel will be. If it is set to 'Underlay', then\n" " the opacity of a colored voxel will be determined from the\n" " underlay opacity at that location.\n" "\n" " * 'ShowThru Mode' is used to toggle whether the color overlay shows\n" " through the grayscale underlay. This will make the color\n" " overlay show through the grayscale underlay no matter how far\n" " it is embedded inside the brain.\n" " The resulting image is made up of the chosen percentage of the\n" " overlay image, plus the remaining percentage of the underlay.\n" " This is done by doing 2 renderings, 1 with the underlay only\n" " and one with the overlay only. The resulting 2 images are\n" " then merged. The default merger is to use an overlay pixel\n" " if it is nonzero, otherwise use the corresponding underlay\n" " pixel. The ShowThru factor can be used to control the\n" " merging when the overlay image pixel is nonzero. Suppose\n" " that its value is denoted by c. Then the merging algorithm,\n" " at each image pixel, is\n" " if( overlay == 0 ) pixel = underlay;\n" " else pixel = c * overlay + (1-c) * underlay;\n" " I personally like the results with c=0.65.\n" "\n" " * 'See Overlay' is used to toggle the color overlay computations\n" " on and off - it should be pressed IN for the overlay to become\n" " visible.\n" " * 'TT Atlas' is used to toggle the overlay of regions from the\n" " Talairach Atlas on and off. This option only has effect if\n" " the underlay dataset being viewed in the the +tlrc coordinate\n" " system and has 1 mm cubical voxels (the default).\n" "\n" " * 'Cutout Overlay' determines if the cutout operations affect the\n" " overlaid voxels. If it is pressed IN, then cutouts will include\n" " the overlay; if OUT, then colored voxels will hang free in\n" " empty space where the underlay was cutout beneath them.\n" " N.B.: If 'Color Opacity' is set to 'Underlay', the cutouts will\n" " hit the overlay in all cases, since cutouts are implemented\n" " by setting the opacity of the underlay to zero in the chosen\n" " regions.\n" "\n" " * 'Remove Small Clusters', if pressed IN, will cause clusters\n" " of voxels below a given threshold volume to be excised before\n" " rendering. The parameters defining this cluster editing are\n" " determined by the controls immmediately beneath, which use the\n" " same conventions as program 3dclust.\n" "\n" " * None of the overlay controls are hooked up to 'DynaDraw' or to\n" " 'Automate'. You must manually press 'Draw' to see the effects\n" " of changes to these settings.\n" "\n" " * A slightly different colormap is used when rendering overlays than\n" " when only underlays are visible. The result is that the grayscale\n" " underlay will look a little different between the 'See Overlay'\n" " IN and OUT conditions. Also, the colormaps are rendered into\n" " 24 bit RGB images, which might not be faithfully displayed in\n" " the image window if your system is using an X11 PseudoColor visual\n" " (the most common display mode). If the 'Save:bkg' button is used\n" " to save a set of RGB images, they will be saved in their internal\n" " color resolution (in PPM format); they might appear slightly\n" " different when viewed outside AFNI (e.g., using program xv).\n" " N.B.: When viewing an RGB image, most of the image processing\n" " options available from the 'Disp' control panel do not\n" " function. 'Sharpen' still works, and is very useful.\n" "\n" " * The Button-3 popup menu under the 'Color' label above the pbar\n" " has an extra menu that lets you control the way in which colors\n" " are mixed in the final display.\n" " In the Volpack color computations, colors are composited along rays\n" " that pass through the volume. This can produce peculiar mixes of\n" " colors (e.g., a blue voxel behind a red voxel will produce a\n" " purple shade, which may not be on the color pbar).\n" " The 'Mixing' menu lets you excise these mixed colors:\n" " * 'Normal' gives the full range of Volpack generated colors.\n" " * 'NoShade' means the overlaid colors generated by Volpack\n" " will be remapped to the set shown on the pbar.\n" " * 'NoMix' means that the overlay colors rendered will be remapped\n" " to shades of the set shown on the pbar (that is, the hues will\n" " be the same, but the intensities will vary).\n" " N.B.: Volpack controls the color composition along each pixel ray.\n" " All that these options do is remap the composed colors back to\n" " a more limited set. If no good match is found, then a gray shade\n" " with the same intensity is chosen. The results of this remapping\n" " may or may not look good!\n" " N.B.: As discussed above, when shown on a PseudoColor display, shaded\n" " colors may look peculiar, but will be saved to a PPM file\n" " properly.\n" " N.B.: You may want to set 'Color Opacity' to 1.0 when using the 'NoMix'\n" " and 'NoShade' options.\n" #ifdef USE_SCRIPTING " N.B.: The setting of the 'Mixing' control is NOT saved or restored by\n" " any of the 'Scripts' options!\n" #endif "\n" #ifdef USE_SCRIPTING "Scripts: [July 1999]\n" "--------\n" " * This facility, controlled from the 'Scripts' menu button, lets you\n" " save rendering settings and recall them later. This is useful\n" " when you want to render multiple datasets in the same fashion.\n" "\n" " * 'Save This' will store the current state of the rendering settings\n" " to a file. Rendering settings are stored in files with suffix\n" " '.rset', and are ASCII files that can be edited -- with care!\n" "\n" " * 'Save Many' will store all the rendering settings used to create\n" " the currently accumulated images.\n" "\n" " * 'Read This' will read one rendering state from a .rset file and\n" " make the interface widgets reflect that state. Nothing will\n" " be rendered until you press the 'Draw' button. If the .rset\n" " file has more than one rendering state (e.g., from ''Save Many')\n" " you will be asked to choose which state you want to load.\n" "\n" " * 'Read & Exec' will read a set of rendering states from a .rset file\n" " and execute them, producing a new set of images. If more than\n" " one rendering is being computed, you can use the CANCEL button\n" " (as in Automate) to stop the rendering.\n" "\n" " * The toggle button 'Load Widgets', when activated, will cause the\n" " interface widgets to be loaded with the rendering state used to\n" " create the currently visible image. When you move the slider\n" " to see a different image from the accumulation, the widgets will\n" " change accordingly. This lets you recall how you created a\n" " particular image. If this button is deactivated, then the\n" " widgets will reflect the last image rendered, no matter which\n" " image is actually visible.\n" "\n" " * The toggle button 'Brick Index?' controls whether the sub-brick\n" " indexes are to be changed when new rendering values are loaded\n" " via one of the 'Read' buttons, or via the 'Load Widgets' toggle.\n" #ifdef SCRIPT_GRAFS "\n" " * The toggle button 'Alter Grafs?' controls whether the Brightness\n" " and Opacity interactive graphs are to be restored when new\n" " rendering values are loaded (via 'Read' or 'Load Widgets').\n" #endif #ifdef SCRIPT_DSETS "\n" " * The toggle button 'Alter Dsets?' controls whether the datasets\n" " will be changed (from the dataset ID codes) when new\n" " rendering values are loaded (via 'Read' or 'Load Widgets').\n" #endif "\n" " N.B.: When you render a new image, it always goes at the END of\n" " the accumulation, no matter which image you happen to be\n" " viewing when you press 'Draw', or otherwise cause rendering.\n" " N.B.: The format of a .rset file is described in the documentation\n" " file README.render_scripts. By editing such a file, you can\n" " create a script that can be used to create many renderings\n" " at once (via the 'Read & Exec' button).\n" "\n" #endif /* USE_SCRIPTING */ "Final Notes:\n" "------------\n" " * The images produced may look a little blurry. You can use the\n" " 'Sharpen' option on the image viewer 'Disp' control panel to\n" " make them look nicer.\n" "\n" " * When only one image is rendered (i.e., Accumulate is off), the\n" " image viewer window does not show the control widgets. The 'Disp'\n" " controls can be accessed by the combination keypress-mouseclick\n" " 'Shift-Button3' in the image window, and the 'Save' controls by\n" " 'Alt-Button3' (some systems don't allow this last combination\n" " to be detected by the application program; in such a case, you\n" " must have at least 2 images accumulated to be able to use 'Save').\n" "\n" " * This plugin is very CPU and memory intensive, and will not run\n" " at all decently on a computer with less than 128 MB of RAM.\n" "------------------------------------------------------------------------\n" "\n" "RW Cox, Milwaukee - February 1999 [first version]\n" " - July 1999 [Scripts]\n" " - April 2000 [Scripts can change datasets]\n" "\n" "RC Reynolds, NIH - " PLUG_CRENDER_VERSION "\n" , TEXT_READONLY ) ; EXRETURN ; } /*------------------------------------------------------------------- Callback for (underlay) choose button. Criteria for datasets that can be rendered: - must be in current session - must have actual bricks of bytes or shorts - bricks must have cubical grid - bricks must be in RAI orientation ---------------------------------------------------------------------*/ /*-- 26 Apr 1999: relax requirement that dataset be axial --*/ #ifdef ONLY_AXIAL # define IS_AXIAL(ds) ( IS_AXIAL_RAI(ds) || IS_AXIAL_LPI(ds) ) #else # define IS_AXIAL(ds) (1) #endif #define USEFUL_DSET(ds) \ (( ISVALID_DSET(ds) ) && \ ( DSET_INMEMORY(ds) ) && \ /* 11 Jan 2002: ignore: ( DSET_CUBICAL(ds) ) && */ \ ( DSET_BRICK_TYPE(ds,0) == MRI_short || \ DSET_BRICK_TYPE(ds,0) == MRI_byte || \ (DSET_BRICK_TYPE(ds,0) == MRI_float && float_ok)) \ /* 11 Jan 2002: ignore: && IS_AXIAL(ds) */ ) static int ndsl = 0 ; static PLUGIN_dataset_link * dsl = NULL ; void RCREND_load_dsl( THD_3dim_dataset * mset , int float_ok ) { THD_session * ss = im3d->ss_now ; /* current session */ int vv = im3d->vinfo->view_type ; /* view type */ THD_3dim_dataset * qset ; int id , nx,ny,nz ; ENTRY( "RCREND_load_dsl" ); ndsl = 0 ; /* initialize */ if( ISVALID_DSET(mset) ){ nx = DSET_NX(mset) ; ny = DSET_NY(mset) ; nz = DSET_NZ(mset) ; } else { nx = ny = nz = 0 ; } /* scan datasets */ for( id=0 ; id < ss->num_dsset ; id++ ){ qset = ss->dsset[id][vv] ; if( ! USEFUL_DSET(qset) ) continue ; /* skip this one */ #if 0 /* overlay grid no longer needs to match underlay rickr 2002.07.18 */ if( nx > 0 && DSET_NX(qset) != nx ) continue ; /* must match */ if( ny > 0 && DSET_NY(qset) != ny ) continue ; /* brick size */ if( nz > 0 && DSET_NZ(qset) != nz ) continue ; #endif ndsl++ ; dsl = (PLUGIN_dataset_link *) XtRealloc( (char *) dsl , sizeof(PLUGIN_dataset_link)*ndsl ) ; make_PLUGIN_dataset_link(qset, dsl + (ndsl-1)) ; /* cf. afni_plugin.c */ } EXRETURN ; } void RCREND_choose_CB( Widget w, XtPointer client_data, XtPointer call_data ) { int vv = im3d->vinfo->view_type ; /* view type */ THD_3dim_dataset * qset ; int id , ltop , llen , dofunc ; char qnam[THD_MAX_NAME] , label[THD_MAX_NAME] ; static char ** strlist = NULL ; int isl = -2 ; /* 03 Apr 1999 */ MCW_idcode midc ; ENTRY( "RCREND_choose_CB" ); /*-- decide if we want overlay (func) or underlay --*/ dofunc = (w == wfunc_choose_pb) ; if( dofunc && !ISVALID_DSET(dset) ){ (void) MCW_popup_message( w , "Can't choose overlay\nbefore underlay!" , MCW_USER_KILL | MCW_TIMER_KILL ) ; XBell(dc->display,100) ; EXRETURN ; } if( dofunc ) RCREND_load_dsl( dset , 1 ) ; else RCREND_load_dsl( NULL , 0 ) ; /* found nothing? exit */ if( ndsl < 1 ){ (void) MCW_popup_message( w , "Didn't find\nany datasets\nto render!" , MCW_USER_KILL | MCW_TIMER_KILL ) ; XBell(dc->display,100) ; EXRETURN ; } /*--- loop over dataset links and patch their titles to include an indicator of the dataset type ---*/ ltop = 4 ; for( id=0 ; id < ndsl ; id++ ){ llen = strlen(dsl[id].title) ; ltop = MAX(ltop,llen) ; } for( id=0 ; id < ndsl ; id++ ){ qset = PLUTO_find_dset( &(dsl[id].idcode) ) ; if( ! ISVALID_DSET(qset) ) continue ; if( ISANAT(qset) ){ if( ISANATBUCKET(qset) ) /* 30 Nov 1997 */ sprintf(qnam,"%-*s [%s:%d]" , ltop,dsl[id].title , ANAT_prefixstr[qset->func_type] , DSET_NVALS(qset) ) ; else if( DSET_NUM_TIMES(qset) == 1 ) sprintf(qnam,"%-*s [%s]" , ltop,dsl[id].title , ANAT_prefixstr[qset->func_type] ) ; else sprintf(qnam,"%-*s [%s:3D+t:%d]" , ltop,dsl[id].title , ANAT_prefixstr[qset->func_type] , DSET_NUM_TIMES(qset) ) ; } else { if( ISFUNCBUCKET(qset) ) /* 30 Nov 1997 */ sprintf(qnam,"%-*s [%s:%d]" , ltop,dsl[id].title , FUNC_prefixstr[qset->func_type] , DSET_NVALS(qset) ) ; else if( DSET_NUM_TIMES(qset) == 1 ) sprintf(qnam,"%-*s [%s]" , ltop,dsl[id].title , FUNC_prefixstr[qset->func_type] ) ; else sprintf(qnam,"%-*s [%s:3D+t:%d]" , ltop,dsl[id].title , FUNC_prefixstr[qset->func_type] , DSET_NVALS(qset) ) ; } if( DSET_COMPRESSED(qset) ) strcat(qnam,"z") ; strcpy( dsl[id].title , qnam ) ; } /*--- make a popup chooser for the user to browse ---*/ POPDOWN_strlist_chooser ; strlist = (char **) XtRealloc( (char *)strlist , sizeof(char *)*ndsl ) ; for( id=0 ; id < ndsl ; id++ ) strlist[id] = dsl[id].title ; /*-- 03 Apr 1999: set the initial selection in the chooser --*/ if( dofunc && func_dset != NULL ){ midc = func_dset_idc; isl = -1; } else if( !dofunc && dset != NULL ){ midc = dset_idc ; isl = -1; } if( isl == -1 ){ for( id=0 ; id < ndsl ; id++ ){ if( EQUIV_IDCODES(midc,dsl[id].idcode) ){ isl = id ; break ; } } } /*-- popup the chooser -- */ sprintf( label , "AFNI Dataset from\nthe %s" , VIEW_typestr[vv] ) ; MCW_choose_strlist( w , label , ndsl , isl , strlist , (dofunc) ? RCREND_finalize_func_CB : RCREND_finalize_dset_CB , NULL ) ; EXRETURN ; } void RCREND_finalize_dset_CB( Widget w, XtPointer fd, MCW_choose_cbs * cbs ) { int id = cbs->ival ; THD_3dim_dataset * qset ; XmString xstr ; char str[2*THD_MAX_NAME] ; float fac ; ENTRY( "RCREND_finalize_dset_CB" ); /* check for errors */ if( ! renderer_open ){ POPDOWN_strlist_chooser ; XBell(dc->display,100) ; EXRETURN ; } if( id < 0 || id >= ndsl ){ XBell(dc->display,100) ; EXRETURN ; } qset = PLUTO_find_dset( &(dsl[id].idcode) ) ; /* the new dataset? */ if( qset == NULL ){ XBell(dc->display,100) ; EXRETURN ; } /* if there was an existing renderer, kill it off */ if( gcr.rh != NULL ){ destroy_CREN_renderer(gcr.rh) ; gcr.rh = NULL ; func_cmap_set = 0 ; } FREE_VOLUMES ; INVALIDATE_OVERLAY ; /* accept this dataset */ dset = qset ; dset_idc = qset->idcode ; /* 31 Mar 1999 */ npixels = 256 ; /* size of image to render */ npixels = MAX( npixels , DSET_NX(dset) ) ; npixels = MAX( npixels , DSET_NY(dset) ) ; npixels = MAX( npixels , DSET_NZ(dset) ) ; /* refit the sub-brick selector menu */ if( dset_ival >= DSET_NVALS(dset) ) dset_ival = DSET_NVALS(dset)-1 ; refit_MCW_optmenu( choose_av , 0 , /* new minval */ DSET_NVALS(dset)-1 , /* new maxval */ dset_ival , /* new inival */ 0 , /* new decim? */ RCREND_choose_av_label_CB , /* text routine */ dset /* text data */ ) ; AV_SENSITIZE( choose_av , (DSET_NVALS(dset) > 1) ) ; /* write the informational label */ strcpy( dset_title , dsl[id].title ) ; fac = DSET_BRICK_FACTOR(dset,dset_ival) ; if( fac == 0.0 || fac == 1.0 ){ strcpy(str,dset_title) ; } else { char abuf[16] ; AV_fval_to_char( fac , abuf ) ; sprintf(str,"%s [* %s]", dset_title , abuf ) ; } xstr = XmStringCreateLtoR( str , XmFONTLIST_DEFAULT_TAG ) ; XtVaSetValues( info_lab , XmNlabelString , xstr , NULL ) ; XmStringFree(xstr) ; /* if the existing overlay dataset doesn't match dset, warn the user */ if( func_dset != NULL && ( DSET_NX(dset) != DSET_NX(func_dset) || DSET_NY(dset) != DSET_NY(func_dset) || DSET_NZ(dset) != DSET_NZ(func_dset) ) ){ #if 0 /* don't need to invalidate, just warn user rickr - 2002.07.18 */ INVALIDATE_OVERLAY ; func_dset = NULL ; TURNOFF_OVERLAY_WIDGETS ; #endif /* just warn user rickr 2002.07.18 */ (void) MCW_popup_message( choose_pb , " \n" "** Warning: **\n" "** new underlay dataset does **\n" "** not match dimensions of **\n" "** existing overlay dataset. **\n", MCW_USER_KILL | MCW_TIMER_KILL ) ; } if ( func_dset != NULL ) /* we may need to resample functional overlay */ new_fset = 1; /* read the new data */ new_dset = 1; /* flag it as new */ RCREND_reload_dataset() ; /* load the data */ EXRETURN ; } void RCREND_finalize_func_CB( Widget w, XtPointer fd, MCW_choose_cbs * cbs ) { int id = cbs->ival ; THD_3dim_dataset * qset , * oset ; XmString xstr ; char str[2*THD_MAX_NAME] ; float fac ; ENTRY( "RCREND_finalize_func_CB" ); /* check for errors */ if( ! renderer_open ){ POPDOWN_strlist_chooser ; XBell(dc->display,100) ; EXRETURN ; } if( id < 0 || id >= ndsl ){ XBell(dc->display,100) ; EXRETURN ; } qset = PLUTO_find_dset( &(dsl[id].idcode) ) ; /* the new dataset? */ if( qset == NULL ){ XBell(dc->display,100) ; EXRETURN ; } /* accept this dataset */ FREE_VOLUMES ; INVALIDATE_OVERLAY ; oset = func_dset ; func_dset = qset ; func_dset_idc = qset->idcode ; /* 31 Mar 1999 */ /* refit the sub-brick selector menus */ if( oset == NULL ){ func_color_ival = 0 ; func_thresh_ival = 1 ; } if( func_color_ival >= DSET_NVALS(func_dset) ) func_color_ival = DSET_NVALS(func_dset)-1; refit_MCW_optmenu( wfunc_color_av , 0 , /* new minval */ DSET_NVALS(func_dset)-1 , /* new maxval */ func_color_ival , /* new inival */ 0 , /* new decim? */ RCREND_choose_av_label_CB , /* text routine */ func_dset /* text data */ ) ; AV_SENSITIZE( wfunc_color_av , (DSET_NVALS(func_dset) > 1) ) ; if( func_thresh_ival >= DSET_NVALS(func_dset) ) func_thresh_ival = DSET_NVALS(func_dset)-1; refit_MCW_optmenu( wfunc_thresh_av , 0 , /* new minval */ DSET_NVALS(func_dset)-1 , /* new maxval */ func_thresh_ival , /* new inival */ 0 , /* new decim? */ RCREND_choose_av_label_CB , /* text routine */ func_dset /* text data */ ) ; AV_SENSITIZE( wfunc_thresh_av , (DSET_NVALS(func_dset) > 1) ) ; /* write the informational label */ strcpy( func_dset_title , dsl[id].title ) ; fac = DSET_BRICK_FACTOR(func_dset,func_color_ival) ; if( fac == 0.0 || fac == 1.0 ){ strcpy(str,func_dset_title) ; } else { char abuf[16] ; AV_fval_to_char( fac , abuf ) ; sprintf(str,"%s [* %s]", func_dset_title , abuf ) ; } xstr = XmStringCreateLtoR( str , XmFONTLIST_DEFAULT_TAG ) ; XtVaSetValues( wfunc_info_lab , XmNlabelString , xstr , NULL ) ; XmStringFree(xstr) ; /* fix the range labels */ xstr = RCREND_range_label() ; XtVaSetValues( wfunc_range_label , XmNlabelString , xstr , NULL ) ; XmStringFree(xstr) ; xstr = RCREND_autorange_label() ; XtVaSetValues( wfunc_range_bbox->wbut[0], XmNlabelString,xstr , NULL ) ; XmStringFree(xstr) ; /* fix the p-value label */ RCREND_set_thr_pval() ; /* read the new data */ new_fset = 1 ; /* flag it as new */ RCREND_reload_dataset() ; /* load the data */ AFNI_hintize_pbar( wfunc_color_pbar , FUNC_RANGE ) ; /* 30 Jul 2001 */ EXRETURN ; } /*------------------------------------------------------------------- Callback for interpolation menu ---------------------------------------------------------------------*/ void RCREND_interp_CB( MCW_arrowval * av , XtPointer cd ) { ENTRY( "RCREND_interp_CB" ); interp_ival = av->ival ; CREN_set_interp( gcr.rh, interp_ival ); EXRETURN ; } /*------------------------------------------------------------------- Callback for angle arrowvals ---------------------------------------------------------------------*/ void RCREND_angle_CB( MCW_arrowval * av , XtPointer cd ) { float na ; ENTRY( "RCREND_angle_CB" ); /*==========================================================================*/ #ifdef ALLOW_INCROT /* 26 Apr 2002 - RWCox */ if( cd == NULL && MCW_val_bbox(incrot_bbox) ){ /* increment mode */ RCREND_do_incrot( av ) ; /* ==> do work here */ EXRETURN ; } #endif /*==========================================================================*/ if( av == roll_av ){ na = angle_roll = av->fval ; if( na < 0.0 ) na += 360 ; else if( na >= 360.0 ) na -= 360.0 ; if( na != av->fval ){ AV_assign_fval( av , na ) ; angle_roll = na ; } } else if( av == pitch_av ){ na = angle_pitch = av->fval ; if( na < 0.0 ) na += 360 ; else if( na >= 360.0 ) na -= 360.0 ; if( na != av->fval ){ AV_assign_fval( av , na ) ; angle_pitch = na ; } } else if( av == yaw_av ){ na = angle_yaw = av->fval ; if( na < 0.0 ) na += 360 ; else if( na >= 360.0 ) na -= 360.0 ; if( na != av->fval ){ AV_assign_fval( av , na ) ; angle_yaw = na ; } } else { EXRETURN ; /* should never happen */ } if( cd == NULL && dynamic_flag && gcr.rh != NULL ) RCREND_draw_CB(NULL,NULL,NULL) ; EXRETURN ; } /*==========================================================================*/ #ifdef ALLOW_INCROT /* 26 Apr 2002 - RWCox */ void RCREND_incrot_CB( Widget w , XtPointer cld , XtPointer cad ) { ENTRY( "RCREND_incrot_CB" ); if( MCW_val_bbox(automate_bbox) ){ /* don't allow incrot */ MCW_set_bbox( incrot_bbox , 0 ) ; /* if Automate is set */ EXRETURN ; } /* if incrot is on, then force arrowvals back to numerical values in case they are now encoded as Automate expressions */ if( MCW_val_bbox(incrot_bbox) ){ RCREND_textact_CB( roll_av ->wtext , (XtPointer)roll_av , NULL ) ; RCREND_textact_CB( pitch_av->wtext , (XtPointer)pitch_av , NULL ) ; RCREND_textact_CB( yaw_av ->wtext , (XtPointer)yaw_av , NULL ) ; } } /*--------------------------------------------------------------------------*/ void RCREND_do_incrot( MCW_arrowval * av ) { int ax ; float th , roll,pitch,yaw ; ENTRY( "RCREND_do_incrot" ); /* read angles from arrowval's current status */ roll = roll_av ->fval ; pitch = pitch_av->fval ; yaw = yaw_av ->fval ; /* choose axis of rotation based on what was just clicked, and set current angle for that axis to last value (before click) */ if( av == roll_av ){ ax = 2; roll = av->old_fval; } else if( av == pitch_av ){ ax = 0; pitch = av->old_fval; } else if( av == yaw_av ){ ax = 1; yaw = av->old_fval; } else EXRETURN ; /* should never happen */ th = av->fval - av->old_fval ; /* angle increment */ roll *= (PI/180) ; /* convert to radians */ pitch *= (PI/180) ; yaw *= (PI/180) ; th *= (PI/180) ; /* compute new angles */ RCREND_inc_angles( ax, th, &yaw , &pitch, &roll ) ; roll = 0.001 * rint( (180000.0/PI)*roll ) ; /* convert to degrees */ pitch = 0.001 * rint( (180000.0/PI)*pitch ) ; /* (rounded to 1/1000) */ yaw = 0.001 * rint( (180000.0/PI)*yaw ) ; /* put back into arrowvals */ AV_assign_fval( roll_av , roll ) ; angle_roll = roll ; AV_assign_fval( yaw_av , yaw ) ; angle_yaw = yaw ; AV_assign_fval( pitch_av , pitch ) ; angle_pitch = pitch ; /* redraw if desirable */ if( dynamic_flag && gcr.rh != NULL ) RCREND_draw_CB(NULL,NULL,NULL) ; EXRETURN ; } #endif /* ALLOW_INCROT */ /*==========================================================================*/ /*----------------------------------------------------------------------- Callback for xhair toggle button -------------------------------------------------------------------------*/ #define CHECK_XHAIR_ERROR \ do{ if( xhair_flag && dset!=NULL && \ ! EQUIV_DATAXES(dset->daxes,im3d->wod_daxes) ){ \ MCW_set_bbox( xhair_bbox , 0 ) ; xhair_flag = 0 ; \ (void) MCW_popup_message( xhair_bbox->wrowcol , \ " Can't overlay AFNI crosshairs\n" \ "because dataset grid and AFNI\n" \ "viewing grid don't coincide." , \ MCW_USER_KILL | MCW_TIMER_KILL ) ; \ XBell(dc->display,100) ; EXRETURN ; \ } } while(0) /*-------------------------------------------------------------------------*/ void RCREND_xhair_CB( Widget w , XtPointer cd , XtPointer call_data ) { int old_xh = xhair_flag ; ENTRY( "RCREND_xhair_CB" ); xhair_flag = MCW_val_bbox( xhair_bbox ) ; if( old_xh == xhair_flag ) EXRETURN ; CHECK_XHAIR_ERROR ; FREE_VOLUMES ; INVALIDATE_OVERLAY ; xhair_ixold = -666 ; xhair_jyold = -666 ; xhair_kzold = -666 ; /* forget */ if( cd == NULL && dynamic_flag && gcr.rh != NULL ) RCREND_draw_CB(NULL,NULL,NULL) ; EXRETURN ; } /*------------------------------------------------------------------------ Event handler for Button #3 popup on xhair toggle -- 08 Mar 2001 --------------------------------------------------------------------------*/ void RCREND_xhair_EV( Widget w , XtPointer cd , XEvent * ev , Boolean * continue_to_dispatch ) { ENTRY( "RCREND_xhair_EV" ); switch( ev->type ){ case ButtonPress:{ XButtonEvent *event = (XButtonEvent *) ev ; if( event->button == Button3 || event->button == Button2 ){ MCW_choose_ovcolor( w,dc , xhair_ovc , RCREND_xhair_ovc_CB,NULL ) ; } } break ; } EXRETURN ; } /*-------------------------------------------------------------------------*/ void RCREND_xhair_ovc_CB( Widget w , XtPointer cd , MCW_choose_cbs * cbs ) { ENTRY( "RCREND_xhair_ovc_CB" ); xhair_ovc = cbs->ival ; INVALIDATE_OVERLAY ; FREE_VOLUMES ; EXRETURN; } /*------------------------------------------------------------------------- 29 Mar 1999: called by AFNI when the user changes the crosshair location 30 Mar 1999: or when the user draws into the controller image window ---------------------------------------------------------------------------*/ void RCREND_xhair_recv( int why , int np , int * ijk , void * junk ) { ENTRY( "RCREND_xhair_recv" ); switch( why ){ /*-- change of time index - 29 Jan 2003 --*/ case RECEIVE_TIMEINDEX:{ int ind , red=0 ; if( !dynamic_flag || !IM3D_OPEN(im3d) ) EXRETURN ; ind = im3d->vinfo->time_index ; if( dset != NULL && DSET_NVALS(dset) > 1 && ind != dset_ival && DSET_NVALS(dset) > ind ){ AV_assign_ival( choose_av , ind ) ; RCREND_choose_av_CB( choose_av , NULL ) ; red = 1 ; } if( func_dset != NULL && DSET_NVALS(func_dset) > 1 && ind != func_color_ival && DSET_NVALS(func_dset) > ind ){ AV_assign_ival( wfunc_color_av , ind ) ; RCREND_choose_av_CB( wfunc_color_av , NULL ) ; red = 1 ; } if( red ){ FREE_VOLUMES; RCREND_draw_CB(NULL,NULL,NULL); } } EXRETURN ; /*-- change of crosshair location --*/ case RECEIVE_VIEWPOINT:{ if( !xhair_flag || !dynamic_flag || gcr.rh == NULL ) EXRETURN ; CHECK_XHAIR_ERROR ; if( CHECK_XHAIR_MOTION ){ FREE_VOLUMES ; RCREND_draw_CB(NULL,NULL,NULL) ; } } EXRETURN ; /*-- user drew something --*/ case RECEIVE_DRAWNOTICE:{ /* 30 Mar 1999 */ int doit=0 ; new_dset = 1; /* we may need a reoriented underlay 28 June 2002 */ new_fset = 1; /* we may need a reoriented overlay 28 June 2002 */ if( EQUIV_DSETS(im3d->anat_now,dset) || /* can't tell if user */ EQUIV_DSETS(im3d->fim_now,dset) ){ /* is drawing on anat */ doit = 1 ; FREE_VOLUMES ; } if( EQUIV_DSETS(im3d->anat_now,func_dset) || /* or is drawing */ EQUIV_DSETS(im3d->fim_now,func_dset) ){ /* on the fim */ doit = 1 ; INVALIDATE_OVERLAY ; /* 15 Jun 1999: fix the range labels */ { XmString xstr ; xstr = RCREND_range_label() ; XtVaSetValues( wfunc_range_label , XmNlabelString , xstr , NULL ) ; XmStringFree(xstr) ; xstr = RCREND_autorange_label() ; XtVaSetValues( wfunc_range_bbox->wbut[0], XmNlabelString,xstr , NULL ) ; XmStringFree(xstr) ; } } if( doit && dynamic_flag && gcr.rh != NULL ) RCREND_draw_CB(NULL,NULL,NULL) ; } EXRETURN ; /*-- dataset pointers have changed --*/ case RECEIVE_DSETCHANGE:{ /* 31 Mar 1999 */ new_dset = 1; /* we may need a reoriented underlay 28 June 2002 */ new_fset = 1; /* we may need a reoriented overlay 28 June 2002 */ if( dset != NULL ) dset = PLUTO_find_dset( &dset_idc ) ; if( func_dset != NULL ) func_dset = PLUTO_find_dset( &func_dset_idc ) ; FREE_VOLUMES ; INVALIDATE_OVERLAY ; (void) MCW_popup_message( reload_pb , "********** NOTICE ***********\n" "* Session rescan has forced *\n" "* purge of dataset brick(s) *\n" "* from memory. *\n" "*****************************" , MCW_USER_KILL | MCW_TIMER_KILL ) ; } EXRETURN ; } /* end of switch on "why" */ EXRETURN ; } /*------------------------------------------------------------------------ Event handler for Button #3 popup on accum toggle - 17 Jun 2005 --------------------------------------------------------------------------*/ void RCREND_accum_lab_EV( Widget w , XtPointer cd , XEvent *ev , Boolean *continue_to_dispatch ) { ENTRY( "RCREND_accum_lab_EV" ); switch( ev->type ){ case ButtonPress:{ XButtonEvent *event = (XButtonEvent *) ev ; if( event->button == Button3 || event->button == Button2 ){ char *ttl ; accum_lab_replace = ( (event->state & ShiftMask) || (event->state & ControlMask) ) ; ttl = (accum_lab_replace) ? "Replacment Label" : "New Overlay Label" ; MCW_choose_string( w,ttl,accum_label , RCREND_accum_lab_CB,NULL ) ; } } break ; } EXRETURN ; } /*-------------------------------------------------------------------------*/ void RCREND_accum_lab_CB( Widget w , XtPointer fd , MCW_choose_cbs *cbs ) { if( cbs != NULL && cbs->reason == mcwCR_string && cbs->cval != NULL ){ MCW_strncpy( accum_label , cbs->cval , 255 ) ; if( accum_lab_replace && renderings != NULL && imseq != NULL ){ int nn=-1 ; MRI_IMAGE *rim ; drive_MCW_imseq( imseq , isqDR_getimnr , (XtPointer)&nn ) ; if( nn >= 0 && nn < IMARR_COUNT(renderings) ){ MRI_IMAGE *rim = IMARR_SUBIM(renderings,nn) ; mri_add_name( accum_label , rim ) ; drive_MCW_imseq( imseq , isqDR_display , (XtPointer)nn ) ; } } } return ; } /*------------------------------------------------------------------------- callbacks for other toggle buttons ---------------------------------------------------------------------------*/ void RCREND_dynamic_CB( Widget w , XtPointer client_data , XtPointer call_data ) { ENTRY( "RCREND_dynamic_CB" ); dynamic_flag = MCW_val_bbox( dynamic_bbox ) ; EXRETURN ; } void RCREND_accum_CB( Widget w , XtPointer client_data , XtPointer call_data ) { ENTRY( "RCREND_accum_CB" ); accum_flag = MCW_val_bbox( accum_bbox ) ; EXRETURN ; } /*----------------------------------------------------------------------- Overlay some white lines showing the crosshair location. Note that this function assumes that the current anat dataset in AFNI is defined on exactly the same grid as the rendering dataset. -------------------------------------------------------------------------*/ #define GR(i,j,k) gar[(i)+(j)*nx+(k)*nxy] #define OP(i,j,k) oar[(i)+(j)*nx+(k)*nxy] #define GXH_GRAY 127 /* 2002 Mar 06: 255->127 */ #define GXH_COLOR 127 #define OXH 255 void RCREND_xhair_underlay( THD_3dim_dataset * mset ) { THD_ivec3 ixyz; THD_fvec3 fxyz; int ix,jy,kz , nx,ny,nz,nxy , ii , gap , om ; float xi,yj,zk; byte * gar; byte gxh; ENTRY( "RCREND_xhair_underlay" ); if( grim == NULL ) EXRETURN ; /* error */ gxh = (xhair_ovc > 0) ? (128+xhair_ovc) : GXH_GRAY; CHECK_XHAIR_ERROR ; /* get Dicom mm coords */ xi = im3d->vinfo->xi; yj = im3d->vinfo->yj; zk = im3d->vinfo->zk; nx = grim->nx; ny = grim->ny; nxy = nx * ny; nz = grim->nz; if ( !ISVALID_DSET(mset) ) /* But, that cannot be! Aaaaaaghh! */ { XBell(dc->display,100); EXRETURN; } /* convert to ijk in mset rickr 2002.08.05 */ LOAD_FVEC3( fxyz, xi, yj, zk ); /* Dicom coords for dset */ fxyz = THD_dicomm_to_3dmm(mset, fxyz); /* mm coords for mset */ ixyz = THD_3dmm_to_3dind (mset, fxyz); /* ijk coords for mset */ UNLOAD_IVEC3( ixyz, ix, jy, kz ); om = im3d->vinfo->xhairs_orimask ; /* 02 Jun 1999 */ if( ix < 0 || ix >= nx ) EXRETURN ; /* error */ if( jy < 0 || jy >= ny ) EXRETURN ; /* error */ if( kz < 0 || kz >= nz ) EXRETURN ; /* error */ gap = im3d->vinfo->crosshair_gap ; gar = MRI_BYTE_PTR(grim) ; /* 02 Jun 1999: allow for partial crosshair drawing */ if( (om & ORIMASK_LR) != 0 ){ for( ii=0 ; ii < nx ; ii++ ){ if( abs(ii-ix) > gap ){ GR(ii,jy,kz) = gxh ; } } } if( (om & ORIMASK_AP) != 0 ){ for( ii=0 ; ii < ny ; ii++ ){ if( abs(ii-jy) > gap ){ GR(ix,ii,kz) = gxh ; } } } if( (om & ORIMASK_IS) != 0 ){ for( ii=0 ; ii < nz ; ii++ ){ if( abs(ii-kz) > gap ){ GR(ix,jy,ii) = gxh ; } } } xhair_ixold = ix ; xhair_jyold = jy ; xhair_kzold = kz ; /* memory */ xhair_omold = om ; /* 02 Jun 1999 */ EXRETURN ; } /*------------------------------------------------------------------ Called when the user changes a graph - just signals that we need to reload the data. --------------------------------------------------------------------*/ void RCREND_graf_CB( MCW_graf * gp , void * cd ) { ENTRY( "RCREND_graf_CB" ); FREE_VOLUMES ; /* free the volumes, will force reloading at redraw */ EXRETURN ; } /*------------------------------------------------------------------- Callback for clipping arrowvals ---------------------------------------------------------------------*/ void RCREND_clip_CB( MCW_arrowval * av , XtPointer cd ) { ENTRY( "RCREND_clip_CB" ); FREE_VOLUMES ; /* free the volumes, will force reloading */ if( clipbot_av->ival >= cliptop_av->ival ){ if( av == clipbot_av ) AV_assign_ival( clipbot_av , cliptop_av->ival - 1 ) ; else AV_assign_ival( cliptop_av , clipbot_av->ival + 1 ) ; } /* if brick is scaled, re-show the scaled labels */ if( brickfac != 0.0 && brickfac != 1.0 ){ char minch[16] , maxch[16] , str[64] ; XmString xstr ; if( av == clipbot_av ){ AV_fval_to_char( brickfac * clipbot_av->ival , minch ) ; sprintf(str,"[-> %s]",minch) ; xstr = XmStringCreateLtoR( str , XmFONTLIST_DEFAULT_TAG ) ; XtVaSetValues( clipbot_faclab , XmNlabelString , xstr , NULL ) ; XmStringFree(xstr) ; } else { AV_fval_to_char( brickfac * cliptop_av->ival , maxch ) ; sprintf(str,"[-> %s]",maxch) ; xstr = XmStringCreateLtoR( str , XmFONTLIST_DEFAULT_TAG ) ; XtVaSetValues( cliptop_faclab , XmNlabelString , xstr , NULL ) ; XmStringFree(xstr) ; } } EXRETURN ; } /*------------------------------------------------------------------- Callback for cutout type optmenu -- change the appearance of the widgets that follow it ---------------------------------------------------------------------*/ void RCREND_cutout_type_CB( MCW_arrowval * av , XtPointer cd ) { int iv , val ; XmString xstr ; ENTRY( "RCREND_cutout_type_CB" ); for( iv=0 ; iv < num_cutouts ; iv++ ) if( av == cutouts[iv]->type_av ) break ; if( iv == num_cutouts ) EXRETURN ; val = av->ival ; /* new type */ HIDE_SCALE ; if( val == CUT_NONE ){ XtUnmanageChild( cutouts[iv]->param_lab ) ; XtUnmanageChild( cutouts[iv]->param_av->wrowcol ) ; XtUnmanageChild( cutouts[iv]->set_pb ) ; XtUnmanageChild( cutouts[iv]->mustdo_bbox->wrowcol ) ; } else { xstr = XmStringCreateLtoR( cutout_param_labels[val], XmFONTLIST_DEFAULT_TAG ) ; XtVaSetValues( cutouts[iv]->param_lab , XmNlabelString , xstr , NULL ) ; XmStringFree(xstr) ; XtManageChild( cutouts[iv]->param_lab ) ; XtManageChild( cutouts[iv]->param_av->wrowcol ) ; XtManageChild( cutouts[iv]->set_pb ) ; XtManageChild( cutouts[iv]->mustdo_bbox->wrowcol ) ; #undef DESENS #ifdef DESENS { Boolean sens ; sens = (val != CUT_EXPRESSION) ; /* deactivate */ XtSetSensitive(cutouts[iv]->param_av->wup ,sens) ; /* if is an */ XtSetSensitive(cutouts[iv]->param_av->wdown,sens) ; /* Expr > 0 */ XtSetSensitive(cutouts[iv]->set_pb ,sens) ; /* cutout */ } #else if( val == CUT_EXPRESSION ){ /* if Expr > 0 */ XtUnmanageChild( cutouts[iv]->param_av->wup ); /* expand the */ XtUnmanageChild( cutouts[iv]->param_av->wdown ); /* size of the */ XtUnmanageChild( cutouts[iv]->set_pb ); /* text field */ XtVaSetValues( cutouts[iv]->param_av->wtext , XmNcolumns , AV_NCOL + 9 , NULL ) ; } else { /* shrink size */ XtVaSetValues( cutouts[iv]->param_av->wtext , /* of text field */ XmNcolumns , AV_NCOL , /* to normal for */ NULL ) ; /* other cutouts */ XtManageChild( cutouts[iv]->param_av->wup ); XtManageChild( cutouts[iv]->param_av->wdown ); XtManageChild( cutouts[iv]->set_pb ); } #endif } FIX_SCALE_SIZE ; EXRETURN ; } /*------------------------------------------------------------------- Callback for cutout number optmenu ---------------------------------------------------------------------*/ void RCREND_numcutout_CB( MCW_arrowval * av , XtPointer cd ) { int ii ; num_cutouts = av->ival ; ENTRY( "RCREND_numcutout_CB" ); HIDE_SCALE ; for( ii=0 ; ii < MAX_CUTOUTS ; ii++ ){ if( ii < num_cutouts ) XtManageChild( cutouts[ii]->hrc ) ; else XtUnmanageChild( cutouts[ii]->hrc ) ; } FIX_SCALE_SIZE ; EXRETURN ; } /*--------------------------------------------------------------------- Routines to load and compare cutout states -----------------------------------------------------------------------*/ void RCREND_load_cutout_state(void) { int ii ; char * str ; ENTRY( "RCREND_load_cutout_state" ); current_cutout_state.num = num_cutouts ; current_cutout_state.logic = logic_cutout = logiccutout_av->ival ; for( ii=0 ; ii < MAX_CUTOUTS ; ii++ ){ current_cutout_state.type[ii] = cutouts[ii]->type_av->ival ; current_cutout_state.mustdo[ii] = MCW_val_bbox( cutouts[ii]->mustdo_bbox ) ; current_cutout_state.param[ii] = RCREND_evaluate( cutouts[ii]->param_av ) ; if( current_cutout_state.type[ii] == CUT_EXPRESSION ){ str = XmTextFieldGetString( cutouts[ii]->param_av->wtext ) ; strcpy( current_cutout_state.param_str[ii] , str ) ; XtFree(str) ; } else { current_cutout_state.param_str[ii][0] = '\0' ; } } current_cutout_state.opacity_scale = RCREND_evaluate( opacity_scale_av ) ; current_cutout_state.opacity_scale = MAX( MIN_OPACITY_SCALE , current_cutout_state.opacity_scale ) ; current_cutout_state.opacity_scale = MIN( 1.000 , current_cutout_state.opacity_scale ) ; EXRETURN ; } int RCREND_cutout_state_changed(void) { int ii ; ENTRY( "RCREND_cutout_state_changed" ); if( current_cutout_state.opacity_scale != old_cutout_state.opacity_scale ) RETURN(1); if( current_cutout_state.num != old_cutout_state.num ) RETURN(1) ; if( current_cutout_state.num == 0 ) RETURN(0) ; if( current_cutout_state.num > 1 && (current_cutout_state.logic != old_cutout_state.logic) ) RETURN(1) ; for( ii=0 ; ii < current_cutout_state.num ; ii++ ){ if( current_cutout_state.type[ii] != old_cutout_state.type[ii] ) RETURN(1) ; if( current_cutout_state.type[ii] == CUT_NONE ) continue ; switch( current_cutout_state.type[ii] ){ default : if( current_cutout_state.param[ii] != old_cutout_state.param[ii] ) RETURN(1); break ; case CUT_EXPRESSION: if( strcmp( current_cutout_state.param_str[ii] , old_cutout_state.param_str[ii] ) != 0 ) RETURN(1) ; if( automate_flag && strchr(current_cutout_state.param_str[ii],'t') != NULL ) RETURN(1) ; break ; } if( current_cutout_state.logic != CUTOUT_OR && current_cutout_state.num > 1 && current_cutout_state.mustdo[ii] != old_cutout_state.mustdo[ii] ) RETURN(1) ; } RETURN(0) ; } /*------------------------------------------------------------------------------- When the user presses a cutout "Get" button ---------------------------------------------------------------------------------*/ #define TT_XMID 0.0 /* centroid */ #define TT_YMID 16.0 #define TT_ZMID 5.0 #define TT_XSEMI 68.0 /* semi-axes */ #define TT_YSEMI 86.0 #define TT_ZSEMI 69.0 void RCREND_cutout_set_CB( Widget w, XtPointer client_data, XtPointer call_data ) { int iv , typ ; float val ; ENTRY( "RCREND_cutout_set_CB" ); for( iv=0 ; iv < num_cutouts ; iv++ ) if( w == cutouts[iv]->set_pb ) break ; if( iv == num_cutouts ) EXRETURN ; typ = cutouts[iv]->type_av->ival ; switch( typ ){ default: XBell(dc->display,100) ; EXRETURN ; /* action is not defined */ case CUT_RIGHT_OF: case CUT_LEFT_OF: val = im3d->vinfo->xi ; break ; case CUT_ANTERIOR_TO: case CUT_POSTERIOR_TO: val = im3d->vinfo->yj ; break ; case CUT_INFERIOR_TO: case CUT_SUPERIOR_TO: val = im3d->vinfo->zk ; break ; case CUT_TT_ELLIPSOID:{ float x = im3d->vinfo->xi , y = im3d->vinfo->yj , z = im3d->vinfo->zk ; val = (x-TT_XMID) * (x-TT_XMID) / (TT_XSEMI * TT_XSEMI) + (y-TT_YMID) * (y-TT_YMID) / (TT_YSEMI * TT_YSEMI) + (z-TT_ZMID) * (z-TT_ZMID) / (TT_ZSEMI * TT_ZSEMI) ; val = 0.1 * rint( 1000.0 * sqrt(val) ) ; /* round to 1 decimal place */ } break ; case CUT_SLANT_XPY_GT: case CUT_SLANT_XPY_LT: case CUT_SLANT_XMY_GT: case CUT_SLANT_XMY_LT: case CUT_SLANT_YPZ_GT: case CUT_SLANT_YPZ_LT: case CUT_SLANT_YMZ_GT: case CUT_SLANT_YMZ_LT: case CUT_SLANT_XPZ_GT: case CUT_SLANT_XPZ_LT: case CUT_SLANT_XMZ_GT: case CUT_SLANT_XMZ_LT:{ float x = im3d->vinfo->xi , y = im3d->vinfo->yj , z = im3d->vinfo->zk ; int isl = typ - CUT_SLANT_BASE ; val = cut_slant_normals[isl][0] * x + cut_slant_normals[isl][1] * y + cut_slant_normals[isl][2] * z ; val = 0.1 * rint( 10.0 * val ) ; /* round to 0.1 mm */ } break ; } AV_assign_fval( cutouts[iv]->param_av , val ) ; if( dynamic_flag && gcr.rh != NULL ) RCREND_draw_CB(NULL,NULL,NULL) ; EXRETURN ; } /*-------------------------------------------------------------------- Actually do the cutouts in a brick. 11 Jan 2000: modified to pass opacity image in, rather than use the global opim ----------------------------------------------------------------------*/ void RCREND_cutout_blobs( MRI_IMAGE * oppim ) { THD_3dim_dataset * local_dset; int ii,jj,kk , nx,ny,nz,nxy,nxyz , cc , typ , ncc,logic,nmust,mus ; int ibot,itop , jbot,jtop , kbot,ktop ; float par ; float dx,dy,dz , xorg,yorg,zorg , xx,yy,zz ; byte * oar , * gar ; byte ncdone = 0 ; ENTRY( "RCREND_cutout_blobs" ); /* if we have a reoriented dataset, use it 26 June 2002 - rickr */ if ( gcr.dset_or != NULL ) local_dset = gcr.dset_or; else local_dset = dset; ncc = current_cutout_state.num ; logic = current_cutout_state.logic ; if( ncc < 1 || oppim == NULL ) EXRETURN ; /* error */ /* find out if the logic is effectively "OR" */ if( ncc == 1 ){ logic = CUTOUT_OR ; } else { for( nmust=cc=0 ; cc < ncc ; cc++ ) if( current_cutout_state.mustdo[cc] ) nmust++ ; if( nmust >= ncc-1 ) logic = CUTOUT_OR ; } /* initialize */ oar = MRI_BYTE_PTR(oppim) ; if( oar == NULL ) EXRETURN ; nx = oppim->nx ; ny = oppim->ny ; nxy = nx * ny ; nz = oppim->nz ; nxyz = nxy * nz ; if( logic == CUTOUT_AND ){ gar = (byte *) malloc( sizeof(byte) * nxyz ) ; /* counts of hits */ memset( gar , 0 , sizeof(byte) * nxyz ) ; } /* now use the local_dset 26 June 2002 - rickr */ dx = local_dset->daxes->xxdel; dy = local_dset->daxes->yydel; dz = local_dset->daxes->zzdel; xorg = local_dset->daxes->xxorg; yorg = local_dset->daxes->yyorg; zorg = local_dset->daxes->zzorg; for( cc=0 ; cc < ncc ; cc++ ){ /* loop over cutouts */ typ = current_cutout_state.type[cc] ; mus = current_cutout_state.mustdo[cc] ; par = current_cutout_state.param[cc] ; if( typ == CUT_NONE ) continue ; /* error */ switch( typ ){ /*............................................................*/ case CUT_RIGHT_OF: case CUT_LEFT_OF: case CUT_ANTERIOR_TO: case CUT_POSTERIOR_TO: case CUT_INFERIOR_TO: case CUT_SUPERIOR_TO:{ /* a rectangular region */ int q ; ibot = 0 ; itop = nx-1 ; /* everything */ jbot = 0 ; jtop = ny-1 ; kbot = 0 ; ktop = nz-1 ; switch( typ ){ case CUT_RIGHT_OF: q = (int)( (par-xorg)/dx - 0.499 ); RANGE(q,0,itop); itop = q; break ; case CUT_LEFT_OF: q = (int)( (par-xorg)/dx + 1.499 ); RANGE(q,0,itop); ibot = q; break ; case CUT_ANTERIOR_TO: q = (int)( (par-yorg)/dy - 0.499 ); RANGE(q,0,jtop); jtop = q; break ; case CUT_POSTERIOR_TO: q = (int)( (par-yorg)/dy + 1.499 ); RANGE(q,0,jtop); jbot = q; break ; case CUT_INFERIOR_TO: q = (int)( (par-zorg)/dz - 0.499 ); RANGE(q,0,ktop); ktop = q; break ; case CUT_SUPERIOR_TO: q = (int)( (par-zorg)/dz + 1.499 ); RANGE(q,0,ktop); kbot = q; break ; } if( logic == CUTOUT_AND && ! mus ){ /* count hits */ ncdone++ ; for( kk=kbot ; kk <= ktop ; kk++ ) for( jj=jbot ; jj <= jtop ; jj++ ) for( ii=ibot ; ii <= itop ; ii++ ) GR(ii,jj,kk)++ ; } else { /* just blast it */ for( kk=kbot ; kk <= ktop ; kk++ ) for( jj=jbot ; jj <= jtop ; jj++ ) for( ii=ibot ; ii <= itop ; ii++ ) OP(ii,jj,kk) = 0 ; } } break ; /* end of rectangular region cutout */ /*............................................................*/ case CUT_TT_ELLIPSOID:{ /* an ellipsoid */ float dxa=dx/TT_XSEMI , dya=dy/TT_YSEMI , dza=dz/TT_ZSEMI ; float xga=(xorg-TT_XMID)/TT_XSEMI , yga=(yorg-TT_YMID)/TT_YSEMI , zga=(zorg-TT_ZMID)/TT_ZSEMI , ebot=0.0001*par*par ; if( logic == CUTOUT_AND && ! mus ){ /* count hits */ ncdone++ ; for( kk=0 ; kk < nz ; kk++ ){ zz = zga + kk*dza ; zz = zz*zz ; for( jj=0 ; jj < ny ; jj++ ){ yy = yga + jj*dya ; yy = yy*yy + zz ; if( yy < ebot ){ for( ii=0 ; ii < nx ; ii++ ){ xx = xga + ii*dxa ; xx = xx*xx + yy ; if( xx > ebot ) GR(ii,jj,kk)++ ; } } else { for( ii=0 ; ii < nx ; ii++ ) GR(ii,jj,kk)++ ; } }} } else { /* blast it */ for( kk=0 ; kk < nz ; kk++ ){ zz = zga + kk*dza ; zz = zz*zz ; for( jj=0 ; jj < ny ; jj++ ){ yy = yga + jj*dya ; yy = yy*yy + zz ; if( yy < ebot ){ for( ii=0 ; ii < nx ; ii++ ){ xx = xga + ii*dxa ; xx = xx*xx + yy ; if( xx > ebot ) OP(ii,jj,kk) = 0 ; } } else { for( ii=0 ; ii < nx ; ii++ ) OP(ii,jj,kk) = 0 ; } }} } } break ; /* end of ellipsoid cutout */ /*............................................................*/ #define VSIZE nx case CUT_EXPRESSION:{ /* expression > 0 */ PARSER_code * pcode ; double * abc[26] , * temp ; /* parse the expression */ pcode = PARSER_generate_code( current_cutout_state.param_str[cc] ) ; if( pcode == NULL ) break ; /* skip this cutout */ /* create the evaluation workspaces */ temp = (double *) malloc( sizeof(double) * VSIZE ) ; for( jj=0 ; jj < 26 ; jj++ ) abc[jj] = (double *) malloc( sizeof(double) * VSIZE ) ; for( jj=0 ; jj < 23 ; jj++ ) /* load zeros for */ for( ii=0 ; ii < VSIZE; ii++ ) /* 'a' ... 'w' */ abc[jj][ii] = 0.0 ; for( ii=0 ; ii < VSIZE ; ii++ ){ /* load 't' and 'n' */ abc[N_IND][ii] = atoz[N_IND] ; abc[T_IND][ii] = atoz[T_IND] ; } /* loop over rows of voxels and evaluate expressions */ for( kk=0 ; kk < nz ; kk++ ){ zz = zorg + kk*dz ; for( jj=0 ; jj < ny ; jj++ ){ yy = yorg + jj*dy ; for( ii=0 ; ii < nx ; ii++ ){ /* load row */ abc[X_IND][ii] = xorg + ii*dx ; abc[Y_IND][ii] = yy ; abc[Z_IND][ii] = zz ; } /* evaluate the expression */ PARSER_evaluate_vector(pcode, abc, VSIZE, temp); /* cut cut cut */ if( logic == CUTOUT_AND && ! mus ){ /* count hits */ for( ii=0 ; ii < nx ; ii++ ) if( temp[ii] > 0.0 ) GR(ii,jj,kk)++ ; } else { /* blast'em */ for( ii=0 ; ii < nx ; ii++ ) if( temp[ii] > 0.0 ) OP(ii,jj,kk) = 0 ; } }} /* end of loops over jj,kk (y,z) */ if( logic == CUTOUT_AND && ! mus ) ncdone++ ; /* free workspaces */ for( jj=0 ; jj < 26 ; jj++ ) free(abc[jj]) ; free(temp) ; free(pcode) ; } break ; /* end of expression cutout */ /*............................................................*/ case CUT_SLANT_XPY_GT: /* the slanted cut planes */ case CUT_SLANT_XPY_LT: case CUT_SLANT_XMY_GT: case CUT_SLANT_XMY_LT: case CUT_SLANT_YPZ_GT: case CUT_SLANT_YPZ_LT: case CUT_SLANT_YMZ_GT: case CUT_SLANT_YMZ_LT: case CUT_SLANT_XPZ_GT: case CUT_SLANT_XPZ_LT: case CUT_SLANT_XMZ_GT: case CUT_SLANT_XMZ_LT:{ int isl = typ - CUT_SLANT_BASE ; float xn = cut_slant_normals[isl][0] , dxn = dx * xn , yn = cut_slant_normals[isl][1] , dyn = dy * yn , zn = cut_slant_normals[isl][2] , dzn = dz * zn , pval ; pval = par - xn*xorg - yn*yorg - zn*zorg ; if( logic == CUTOUT_AND && ! mus ){ /* count hits */ ncdone++ ; for( kk=0,zz=-pval ; kk < nz ; kk++,zz+=dzn ){ /* zz = kk * dzn - pval ; */ for( jj=0,yy=zz ; jj < ny ; jj++,yy+=dyn ){ /* yy = jj * dyn + zz ; */ for( ii=0,xx=yy ; ii < nx ; ii++,xx+=dxn ){ /* xx = ii * dxn + yy ; */ if( xx > 0.0 ) GR(ii,jj,kk)++ ; } }} } else { /* blast it */ for( kk=0,zz=-pval ; kk < nz ; kk++,zz+=dzn ){ /* zz = kk * dzn - pval ; */ for( jj=0,yy=zz ; jj < ny ; jj++,yy+=dyn ){ /* yy = jj * dyn + zz ; */ for( ii=0,xx=yy ; ii < nx ; ii++,xx+=dxn ){ /* xx = ii * dxn + yy ; */ if( xx > 0.0 ) OP(ii,jj,kk) = 0 ; } }} } } break ; /* end of slant cutout */ /*............................................................*/ #define OVAR(i,j,k) ovar[(i)+(j)*nx+(k)*nxy] #define KEEP(i,j,k) keep[(i)+(j)*nx+(k)*nxy] case CUT_NONOVERLAY: if( DO_OVERLAY ){ /* 24 Jul 2001: changed condition from func_dset != NULL */ byte * ovar ; float adx=fabs(dx) , ady=fabs(dy) , adz=fabs(dz) ; if( ovim == NULL ) RCREND_reload_func_dset() ; /* get the global */ ovar = MRI_BYTE_PTR(ovim) ; /* overlay image */ if( par < adx && par < ady && par < adz ){ /* no dilation */ if( logic == CUTOUT_AND && ! mus ){ /* count hits */ ncdone++ ; for( ii=0 ; ii < nxyz ; ii++ ) if( ovar[ii] == 0 ) gar[ii]++ ; } else { /* nuke'em */ for( ii=0 ; ii < nxyz ; ii++ ) if( ovar[ii] == 0 ) oar[ii] = 0 ; } } else { /* dilation */ MCW_cluster * mask = MCW_build_mask( adx,ady,adz, par ) ; int mnum = mask->num_pt , pp,ip,jp,kp ; short * mi = mask->i , * mj = mask->j , * mk = mask->k ; byte * keep = calloc(nxyz,sizeof(byte)) ; for( kk=0 ; kk < nz ; kk++ ) /* make list of points to keep */ for( jj=0 ; jj < ny ; jj++ ) for( ii=0 ; ii < nx ; ii++ ) if( OVAR(ii,jj,kk) != 0 ){ /* keep nbhd of this point */ KEEP(ii,jj,kk) = 1 ; for( pp=0 ; pp < mnum ; pp++ ){ ip = ii + mi[pp]; jp = jj + mj[pp]; kp = kk + mk[pp]; if( ip >= 0 && ip < nx && jp >= 0 && jp < ny && kp >= 0 && kp < nz ) KEEP(ip,jp,kp) = 1 ; } } KILL_CLUSTER(mask) ; /* toss the trash */ /* now do the cutting */ if( logic == CUTOUT_AND && ! mus ){ /* count hits */ ncdone++ ; for( ii=0 ; ii < nxyz ; ii++ ) if( keep[ii] == 0 ) gar[ii]++ ; } else { /* nuke'em */ for( ii=0 ; ii < nxyz ; ii++ ) if( keep[ii] == 0 ) oar[ii] = 0 ; } free(keep) ; /* toss the trash */ } } break ; /* end of nonoverlay cutout */ } /* end of switch over type of cutout */ } /* end of loop over cutouts */ /* with AND, blast only those that were hit every time */ if( logic == CUTOUT_AND && ncdone > 0 ){ for( ii=0 ; ii < nxyz ; ii++ ) if( gar[ii] == ncdone ) oar[ii] = 0 ; free(gar) ; } EXRETURN ; } /*----------------------------------------------------------------------- Callback for cutout parameter arrowval -- only action is to redraw if dynamic mode is on -------------------------------------------------------------------------*/ void RCREND_param_CB( MCW_arrowval * av , XtPointer cd ) { ENTRY( "RCREND_param_CB" ); if( cd == NULL && dynamic_flag && gcr.rh != NULL ) RCREND_draw_CB(NULL,NULL,NULL) ; EXRETURN; } /*----------------------------------------------------------------------- Evaluate an arrowval string that may be a number or may be a more complicated arithmetic expression. Input variables are stored in the global array atoz[] ; -------------------------------------------------------------------------*/ float RCREND_evaluate( MCW_arrowval * av ) { PARSER_code * pcode ; char * str , * cpt ; float val ; ENTRY( "RCREND_evaluate" ); /* get the string to convert */ if( av == NULL ) RETURN(0.0) ; /* these cases should */ if( av->wtext == NULL ) RETURN(av->fval) ; /* never happen */ str = XmTextFieldGetString( av->wtext ) ; if( str == NULL || str[0] == '\0' ){ XtFree(str) ; RETURN(0.0) ; } /* rcr - until I make a button for this... */ if ( r_debug_check( &gcr_debug, str ) ) /* if this is a debug action */ { XtFree(str); /* free the string, and */ RETURN(av->fval); /* return the previous value */ } /* try a regular numerical conversion */ val = strtod( str , &cpt ) ; for( ; *cpt != '\0' && isspace(*cpt) ; cpt++ ) ; /* skip blanks */ if( *cpt == '\0' ){ XtFree(str); AV_assign_fval(av,val); RETURN(val); } /* try to parse an expression */ pcode = PARSER_generate_code( str ) ; if( pcode == NULL ){ XtFree(str) ; RETURN(0.0) ; } val = PARSER_evaluate_one( pcode , atoz ) ; free(pcode) ; XtFree(str) ; RETURN(val); } /*------------------------------------------------------------------------- When the user presses Enter in a textfield that might have an expression entered, evaluate the expression numerically and load that number back into the textfield. ---------------------------------------------------------------------------*/ void RCREND_textact_CB( Widget wtex, XtPointer client_data, XtPointer call_data ) { MCW_arrowval * av = (MCW_arrowval *) client_data ; float sval ; int iv ; ENTRY( "RCREND_textact_CB" ); for( iv=0 ; iv < num_cutouts ; iv++ ) /* skip if is an Expr > 0 cutout */ if( av == cutouts[iv]->param_av && cutouts[iv]->type_av->ival == CUT_EXPRESSION ) EXRETURN ; MCW_invert_widget(wtex) ; sval = RCREND_evaluate( av ) ; AV_assign_fval( av , sval ) ; MCW_invert_widget(wtex) ; EXRETURN ; } /********************************************************************** Stuff to deal with the image display window ***********************************************************************/ /*----------------------------------------------------------------------- Open an image display window. -------------------------------------------------------------------------*/ static int any_rgb_images ; void RCREND_open_imseq( void ) { int ntot , ii ; ENTRY( "RCREND_open_imseq" ); if( imseq != NULL || renderings == NULL || IMARR_COUNT(renderings) == 0 ) EXRETURN ; ntot = IMARR_COUNT(renderings) ; any_rgb_images = 0 ; for( ii=0 ; ii < ntot ; ii++ ){ if( IMARR_SUBIMAGE(renderings,ii) != NULL && IMARR_SUBIMAGE(renderings,ii)->kind == MRI_rgb ){ any_rgb_images = 1 ; break ; } } imseq = open_MCW_imseq( dc , RCREND_imseq_getim , NULL ) ; drive_MCW_imseq( imseq , isqDR_clearstat , NULL ) ; { ISQ_options opt ; /* change some options from the defaults */ ISQ_DEFAULT_OPT(opt) ; opt.save_one = False ; /* change to Save:bkg */ opt.save_pnm = False ; opt.save_filter = -1 ; /* 27 Jun 2001 */ drive_MCW_imseq( imseq , isqDR_options , (XtPointer) &opt ) ; drive_MCW_imseq( imseq , isqDR_periodicmont , (XtPointer) 0 ) ; drive_MCW_imseq( imseq , isqDR_penbbox , (XtPointer) 0 ) ; } /* make it popup */ drive_MCW_imseq( imseq , isqDR_realize, NULL ) ; NORMAL_cursorize( imseq->wimage ) ; /* 07 Dec 2001 */ drive_MCW_imseq( imseq , isqDR_title, "AFNI Renderings" ) ; if( ntot == 1 ) drive_MCW_imseq( imseq , isqDR_onoffwid , (XtPointer) isqDR_offwid ) ; else { drive_MCW_imseq( imseq , isqDR_onoffwid , (XtPointer) isqDR_onwid ) ; drive_MCW_imseq( imseq , isqDR_opacitybut , (XtPointer) 0 ) ; /* 07 Mar 2001 */ } drive_MCW_imseq( imseq , isqDR_reimage , (XtPointer) (ntot-1) ) ; #ifndef DONT_INSTALL_ICONS if( afni48_good && afni48ren_pixmap == XmUNSPECIFIED_PIXMAP ){ Pixel bg_pix , fg_pix ; XtVaGetValues( info_lab , XmNforeground , &fg_pix , XmNbackground , &bg_pix , NULL ) ; afni48ren_pixmap = XCreatePixmapFromBitmapData( XtDisplay(shell) , RootWindowOfScreen(XtScreen(shell)) , afni48ren_bits , afni48ren_width , afni48ren_height , bg_pix , fg_pix , DefaultDepthOfScreen(XtScreen(shell)) ) ; } if( afni48_good ) drive_MCW_imseq( imseq,isqDR_icon , (XtPointer) afni48ren_pixmap ) ; #endif EXRETURN ; } void RCREND_update_imseq( void ) { int ntot , ii ; ENTRY( "RCREND_update_imseq" ); if( imseq == NULL ){ RCREND_open_imseq() ; EXRETURN ; } if( renderings == NULL || IMARR_COUNT(renderings) == 0 ) EXRETURN ; ntot = IMARR_COUNT(renderings) ; any_rgb_images = 0 ; for( ii=0 ; ii < ntot ; ii++ ){ if( IMARR_SUBIMAGE(renderings,ii) != NULL && IMARR_SUBIMAGE(renderings,ii)->kind == MRI_rgb ){ any_rgb_images = 1 ; break ; } } drive_MCW_imseq( imseq , isqDR_newseq , NULL ) ; if( ntot == 1 ) drive_MCW_imseq( imseq , isqDR_onoffwid , (XtPointer) isqDR_offwid ) ; else { drive_MCW_imseq( imseq , isqDR_onoffwid , (XtPointer) isqDR_onwid ) ; drive_MCW_imseq( imseq , isqDR_opacitybut , (XtPointer) 0 ) ; /* 07 Mar 2001 */ } drive_MCW_imseq( imseq , isqDR_reimage , (XtPointer)(ntot-1) ) ; EXRETURN ; } void RCREND_destroy_imseq( void ) { ENTRY( "RCREND_destroy_imseq" ); if( imseq == NULL ) EXRETURN ; drive_MCW_imseq( imseq , isqDR_destroy , NULL ) ; EXRETURN ; } /*------------------------------------------------------------------ Routine to provide data to the imseq. Just returns the control information, or the selected image. --------------------------------------------------------------------*/ XtPointer RCREND_imseq_getim( int n , int type , XtPointer handle ) { int ntot = 0 ; ENTRY( "RCREND_imseq_getim" ); if( renderings != NULL ) ntot = IMARR_COUNT(renderings) ; if( ntot < 1 ) ntot = 1 ; /*--- send control info ---*/ if( type == isqCR_getstatus ){ MCW_imseq_status *stat = myXtNew( MCW_imseq_status ); /* will be free-d */ /* when imseq is */ /* destroyed */ stat->num_total = ntot ; stat->num_series = stat->num_total ; stat->send_CB = RCREND_seq_send_CB ; stat->parent = NULL ; stat->aux = NULL ; stat->transforms0D = &(GLOBAL_library.registered_0D) ; stat->transforms2D = &(GLOBAL_library.registered_2D) ; RETURN((XtPointer)stat); } /*--- no overlay, never ---*/ if( type == isqCR_getoverlay ) RETURN(NULL) ; /*--- label string ---*/ if( type == isqCR_getlabel ){ char *lab=NULL ; MRI_IMAGE *im ; if( renderings != NULL ){ if( n < 0 ) n = 0 ; else if( n >= ntot ) n = ntot-1 ; im = IMARR_SUBIMAGE(renderings,n) ; if( accum_lab_replace ) mri_add_name( accum_label , im ) ; if( im->name != NULL ) lab = strdup(im->name) ; } RETURN(lab) ; } /*--- return a copy of a rendered image (since the imseq will delete it when it is done) ---*/ if( type == isqCR_getimage || type == isqCR_getqimage ){ MRI_IMAGE *im=NULL , *rim ; if( renderings != NULL ){ if( n < 0 ) n = 0 ; else if( n >= ntot ) n = ntot-1 ; rim = IMARR_SUBIMAGE(renderings,n) ; if( any_rgb_images ) im = mri_to_rgb( rim ) ; else im = mri_to_mri( rim->kind , rim ) ; #ifdef USE_SCRIPTING if( renderings_state != NULL && n < RSA_COUNT(renderings_state) && ! automate_flag && script_load && script_load_last != n ){ RCREND_state_to_widgets( RSA_SUBSTATE(renderings_state,n) ) ; script_load_last = n ; } #endif } RETURN((XtPointer)im); } RETURN(NULL); /* should not occur, but who knows? */ } /*--------------------------------------------------------------------------- Routine called when the imseq wants to send a message. In this case, all we need to handle is the destroy message, so that we can free some memory. -----------------------------------------------------------------------------*/ void RCREND_seq_send_CB( MCW_imseq * seq , XtPointer handle , ISQ_cbs * cbs ) { ENTRY( "RCREND_seq_send_CB" ); switch( cbs->reason ){ case isqCR_destroy:{ myXtFree(imseq->status) ; myXtFree(imseq) ; imseq = NULL ; } break ; } EXRETURN ; } /*---------------------------------------------------------------------------- Automation callbacks ------------------------------------------------------------------------------*/ void RCREND_autoflag_CB( Widget w , XtPointer client_data , XtPointer call_data ) { int flag = MCW_val_bbox( automate_bbox ) ; ENTRY( "RCREND_autoflag_CB" ); XtSetSensitive( autocompute_pb , (Boolean) flag ) ; #ifdef ALLOW_INCROT /* 26 Apr 2002 - RWCox */ if( flag ) MCW_set_bbox( incrot_bbox , 0 ) ; #endif EXRETURN ; } /*-------------------------------------------------------------------------- Drive the automatic computation ----------------------------------------------------------------------------*/ static int autokill ; void RCREND_autocompute_CB( Widget w, XtPointer client_data, XtPointer call_data ) { int it , ntime = autoframe_av->ival ; float scl = 100.0/ntime ; Widget autometer ; ENTRY( "RCREND_autocompute_CB" ); automate_flag = 1 ; AFNI_block_rescan(1) ; if( ! accum_flag ){ DESTROY_IMARR(renderings) ; #ifdef USE_SCRIPTING DESTROY_RSA(renderings_state) ; #endif } atoz[N_IND] = ntime ; autometer = MCW_popup_meter( shell , METER_TOP_WIDE ) ; XtManageChild( autocancel_pb ) ; AFNI_add_interruptable( autocancel_pb ) ; autokill = 0 ; for( it=0 ; it < ntime ; it++ ){ atoz[T_IND] = it ; AV_assign_ival( autoframe_av , it+1 ) ; RCREND_draw_CB(NULL,NULL,NULL) ; if( it < ntime-1 ){ AFNI_process_interrupts(autocancel_pb) ; if( autokill ) break ; } MCW_set_meter( autometer , (int)(scl*(it+1)) ) ; } MCW_popdown_meter( autometer ) ; /*-- done: turn off automation --*/ MCW_set_bbox( automate_bbox , 0 ) ; XtSetSensitive( autocompute_pb , False ) ; XtUnmanageChild( autocancel_pb ) ; AFNI_add_interruptable(NULL) ; automate_flag = 0 ; AFNI_block_rescan(0) ; EXRETURN ; } /*-------------------------------------------------------------------------- Set a flag to cancel the automatic computation ----------------------------------------------------------------------------*/ void RCREND_autocancel_CB( Widget w, XtPointer client_data, XtPointer call_data ) { ENTRY( "RCREND_autocancel_CB" ); if( autokill ){ XBell(dc->display,100) ; EXRETURN ; } autokill = 1 ; EXRETURN; } /*-------------------------------------------------------------------------- What happens when the user selects a sub-brick from a menu ----------------------------------------------------------------------------*/ void RCREND_choose_av_CB( MCW_arrowval * av , XtPointer cd ) { XmString xstr ; char str[2*THD_MAX_NAME] ; ENTRY( "RCREND_choose_av_CB" ); /*--- selection of an underlay sub-brick ---*/ if( av == choose_av && dset != NULL && av->ival < DSET_NVALS(dset) ){ float fac = DSET_BRICK_FACTOR(dset,av->ival) ; if( fac == 0.0 || fac == 1.0 ){ /* rewrite the informational label */ strcpy(str,dset_title) ; } else { char abuf[16] ; AV_fval_to_char( fac , abuf ) ; sprintf(str,"%s [* %s]", dset_title , abuf ) ; } xstr = XmStringCreateLtoR( str , XmFONTLIST_DEFAULT_TAG ) ; XtVaSetValues( info_lab , XmNlabelString , xstr , NULL ) ; XmStringFree(xstr) ; dset_ival = av->ival ; /* read this sub-brick */ new_dset = 1 ; /* flag it as new */ FREE_VOLUMES ; /* free the internal data */ RCREND_reload_dataset() ; /* load the data */ if( gcr.rh != NULL ) RCREND_draw_CB(NULL,NULL,NULL) ; /* draw */ /*--- selection of overlay color sub-brick ---*/ } else if( av == wfunc_color_av && func_dset != NULL && av->ival < DSET_NVALS(func_dset) ){ float fac = DSET_BRICK_FACTOR(func_dset,av->ival) ; if( fac == 0.0 || fac == 1.0 ){ /* rewrite the informational label */ strcpy(str,func_dset_title) ; } else { char abuf[16] ; AV_fval_to_char( fac , abuf ) ; sprintf(str,"%s [* %s]", func_dset_title , abuf ) ; } xstr = XmStringCreateLtoR( str , XmFONTLIST_DEFAULT_TAG ) ; XtVaSetValues( wfunc_info_lab , XmNlabelString , xstr , NULL ) ; XmStringFree(xstr) ; func_color_ival = av->ival ; /* fix the range labels */ xstr = RCREND_range_label() ; XtVaSetValues( wfunc_range_label , XmNlabelString , xstr , NULL ) ; XmStringFree(xstr) ; xstr = RCREND_autorange_label() ; XtVaSetValues( wfunc_range_bbox->wbut[0], XmNlabelString,xstr , NULL ) ; XmStringFree(xstr) ; new_fset = 1 ; /* flag it as new - may resample new sub-brick */ INVALIDATE_OVERLAY ; AFNI_hintize_pbar( wfunc_color_pbar , FUNC_RANGE ) ; /* 30 Jul 2001 */ /*--- selection of overlay threshold sub-brick ---*/ } else if( av == wfunc_thresh_av && func_dset != NULL && av->ival < DSET_NVALS(func_dset) ){ func_thresh_ival = av->ival ; /* fix the range label */ xstr = RCREND_range_label() ; XtVaSetValues( wfunc_range_label , XmNlabelString , xstr , NULL ) ; XmStringFree(xstr) ; /* fix the p-value label */ RCREND_set_thr_pval() ; new_fset = 1 ; /* flag it as new - may resample new sub-brick */ INVALIDATE_OVERLAY ; } EXRETURN ; } /*--------------------------------------------------------------------------- Make a label for a sub-brick selector menu -----------------------------------------------------------------------------*/ char * RCREND_choose_av_label_CB( MCW_arrowval * av , XtPointer cd ) { static char blab[32] ; THD_3dim_dataset * dset = (THD_3dim_dataset *) cd ; static char * lfmt[3] = { "#%1d %-14.14s" , "#%2d %-14.14s" , "#%3d %-14.14s" } ; static char * rfmt[3] = { "%-14.14s #%1d" , "%-14.14s #%2d" , "%-14.14s #%3d" } ; ENTRY( "RCREND_choose_av_label_CB" ); if( ISVALID_3DIM_DATASET(dset) ){ #ifdef USE_RIGHT_BUCK_LABELS if( DSET_NVALS(dset) < 10 ) sprintf(blab, rfmt[0] , DSET_BRICK_LABEL(dset,av->ival) , av->ival ) ; else if( DSET_NVALS(dset) < 100 ) sprintf(blab, rfmt[1] , DSET_BRICK_LABEL(dset,av->ival) , av->ival ) ; else sprintf(blab, rfmt[2] , DSET_BRICK_LABEL(dset,av->ival) , av->ival ) ; #else if( DSET_NVALS(dset) < 10 ) sprintf(blab, lfmt[0] , av->ival , DSET_BRICK_LABEL(dset,av->ival) ) ; else if( DSET_NVALS(dset) < 100 ) sprintf(blab, lfmt[1] , av->ival , DSET_BRICK_LABEL(dset,av->ival) ) ; else sprintf(blab, lfmt[2] , av->ival , DSET_BRICK_LABEL(dset,av->ival) ) ; #endif } else sprintf(blab," #%d ",av->ival) ; /* should not happen! */ RETURN(blab) ; } /*---------------------------------------------------------------------------------- Make the widgets for the functional overlay ------------------------------------------------------------------------------------*/ void RCREND_func_widgets(void) { XmString xstr ; Widget wqqq ; int sel_height ; ENTRY( "RCREND_func_widgets" ); /* top level managers */ wfunc_vsep = SEP_VER(top_rowcol) ; wfunc_frame = XtVaCreateWidget( "AFNI" , xmFrameWidgetClass , top_rowcol , XmNshadowType , XmSHADOW_ETCHED_IN , XmNshadowThickness , 5 , XmNtraversalOn , True , XmNinitialResourcesPersistent , False , NULL ) ; wfunc_uber_rowcol = XtVaCreateWidget( "AFNI" , xmRowColumnWidgetClass , wfunc_frame , XmNorientation , XmVERTICAL , XmNpacking , XmPACK_TIGHT , XmNtraversalOn , True , XmNinitialResourcesPersistent , False , NULL ) ; xstr = XmStringCreateLtoR( NO_DATASET_STRING , XmFONTLIST_DEFAULT_TAG ) ; wfunc_info_lab = XtVaCreateManagedWidget( "AFNI" , xmLabelWidgetClass , wfunc_uber_rowcol , XmNlabelString , xstr , XmNrecomputeSize , False , XmNinitialResourcesPersistent , False , NULL ) ; XmStringFree(xstr) ; SEP_HOR(wfunc_uber_rowcol) ; xstr = XmStringCreateLtoR( "Choose Overlay Dataset" , XmFONTLIST_DEFAULT_TAG ) ; wfunc_choose_pb = XtVaCreateManagedWidget( "AFNI" , xmPushButtonWidgetClass , wfunc_uber_rowcol , XmNalignment , XmALIGNMENT_CENTER , XmNlabelString , xstr , XmNtraversalOn , True , XmNinitialResourcesPersistent , False , NULL ) ; XmStringFree(xstr) ; XtAddCallback( wfunc_choose_pb, XmNactivateCallback, RCREND_choose_CB, NULL ) ; SEP_HOR(wfunc_uber_rowcol) ; wfunc_rowcol = XtVaCreateWidget( "AFNI" , xmRowColumnWidgetClass , wfunc_uber_rowcol , XmNorientation , XmHORIZONTAL , XmNpacking , XmPACK_TIGHT , XmNtraversalOn , True , XmNinitialResourcesPersistent , False , NULL ) ; /*---------------------------- 1st column: threshold stuff ----------------------------*/ wfunc_thr_rowcol = XtVaCreateWidget( "AFNI" , xmRowColumnWidgetClass , wfunc_rowcol , XmNorientation , XmVERTICAL , XmNpacking , XmPACK_TIGHT , XmNmarginHeight, 0 , XmNmarginWidth , 0 , XmNtraversalOn , True , XmNinitialResourcesPersistent , False , NULL ) ; xstr = XmStringCreateLtoR( "Thresh" , XmFONTLIST_DEFAULT_TAG ) ; wfunc_thr_label = XtVaCreateManagedWidget( "AFNI" , xmLabelWidgetClass , wfunc_thr_rowcol , XmNlabelString , xstr , XmNrecomputeSize , False , XmNinitialResourcesPersistent , False , NULL ) ; XmStringFree(xstr) ; { int smax , stop , decim , sstep ; /* 30 Nov 1997: */ decim = THR_TOP_EXPON ; /* compute parameters */ smax = (int)( pow(10.0,decim) + 0.001 ) ; /* for scale display. */ stop = smax - 1 ; sstep = smax / 1000 ; if( sstep < 1 ) sstep = 1 ; { char *eee = getenv("AFNI_THRESH_BIGSTEP") ; /* 09 May 2003 */ if( eee != NULL ){ int iq=strtol(eee,NULL,10); if(iq > 0) sstep=iq; } } #ifdef BOXUP_SCALE wqqq = XtVaCreateManagedWidget( "AFNI" , xmFrameWidgetClass , wfunc_thr_rowcol , XmNshadowType , XmSHADOW_ETCHED_IN , XmNtraversalOn , True , XmNinitialResourcesPersistent , False , NULL ) ; #else wqqq = wfunc_thr_rowcol ; #endif #if 1 MCW_widget_geom( anat_frame , &sel_height , NULL,NULL,NULL ) ; sel_height -= (74 + 24*MAX_CUTOUTS) ; /* shorter allows for widgets below */ #else sel_height = 290 ; /* a hardwired approach */ #endif wfunc_thr_scale = XtVaCreateManagedWidget( "scale" , xmScaleWidgetClass , wqqq , XmNminimum , 0 , XmNmaximum , stop , XmNscaleMultiple , sstep , XmNdecimalPoints , decim , XmNshowValue , True , XmNvalue , (int)(smax*func_threshold) , XmNorientation , XmVERTICAL , XmNheight , sel_height , XmNborderWidth , 0 , XmNtraversalOn , True , XmNinitialResourcesPersistent , False , NULL ) ; } #ifdef FIX_SCALE_SIZE_PROBLEM XtVaSetValues( wfunc_thr_scale , XmNuserData , (XtPointer) sel_height , NULL ) ; #endif XtAddCallback( wfunc_thr_scale , XmNvalueChangedCallback , RCREND_thr_scale_CB , NULL ) ; XtAddCallback( wfunc_thr_scale , XmNdragCallback , RCREND_thr_scale_drag_CB , NULL ) ; /** label for computed p-value, under scale **/ xstr = XmStringCreateLtoR( THR_PVAL_LABEL_NONE , XmFONTLIST_DEFAULT_TAG ) ; wfunc_thr_pval_label = XtVaCreateManagedWidget( "AFNI" , xmLabelWidgetClass , wfunc_thr_rowcol , XmNlabelString , xstr , XmNrecomputeSize , False , XmNinitialResourcesPersistent , False , NULL ) ; XmStringFree(xstr) ; /** optmenu to choose top value for scale **/ wfunc_thr_top_av = new_MCW_arrowval( wfunc_thr_rowcol , "**" , MCW_AV_optmenu , 0,THR_TOP_EXPON,0 , MCW_AV_notext , 0 , RCREND_thresh_top_CB , NULL , RCREND_thresh_tlabel_CB , NULL ) ; XtManageChild(wfunc_thr_rowcol) ; /*--------------- column 2: color chooser stuff ------------------------------*/ wfunc_color_rowcol = XtVaCreateWidget( "AFNI" , xmRowColumnWidgetClass , wfunc_rowcol , XmNorientation , XmVERTICAL , XmNmarginHeight, 0 , XmNmarginWidth , 0 , XmNpacking , XmPACK_TIGHT , XmNtraversalOn , True , XmNinitialResourcesPersistent , False , NULL ) ; xstr = XmStringCreateLtoR( "Color" , XmFONTLIST_DEFAULT_TAG ) ; wfunc_color_label = XtVaCreateManagedWidget( "AFNI" , xmLabelWidgetClass , wfunc_color_rowcol , XmNlabelString , xstr , XmNinitialResourcesPersistent , False , NULL ) ; XmStringFree(xstr) ; /**-- Popup menu to control some facets of the pbar --**/ wfunc_pbar_menu = XmCreatePopupMenu( wfunc_color_label , "menu" , NULL , 0 ) ; SAVEUNDERIZE(XtParent(wfunc_pbar_menu)) ; /* 27 Feb 2001 */ VISIBILIZE_WHEN_MAPPED(wfunc_pbar_menu) ; if( !AFNI_yesenv("AFNI_DISABLE_TEAROFF") ) TEAROFFIZE(wfunc_pbar_menu) ; XtInsertEventHandler( wfunc_color_label , /* handle events in label */ 0 | ButtonPressMask /* button presses */ , FALSE , /* nonmaskable events? */ RCREND_pbarmenu_EV , /* handler */ NULL , /* client data */ XtListTail /* last in queue */ ) ; (void) XtVaCreateManagedWidget( "dialog" , xmLabelWidgetClass , wfunc_pbar_menu , LABEL_ARG("-- Cancel --") , XmNrecomputeSize , False , XmNinitialResourcesPersistent , False , NULL ) ; (void) XtVaCreateManagedWidget( "dialog" , xmSeparatorWidgetClass , wfunc_pbar_menu , XmNseparatorType , XmSINGLE_LINE , NULL ) ; wfunc_pbar_equalize_pb = XtVaCreateManagedWidget( "dialog" , xmPushButtonWidgetClass , wfunc_pbar_menu , LABEL_ARG("Equalize Spacing") , XmNmarginHeight , 0 , XmNtraversalOn , True , XmNinitialResourcesPersistent , False , NULL ) ; XtAddCallback( wfunc_pbar_equalize_pb , XmNactivateCallback , RCREND_pbarmenu_CB , im3d ) ; wfunc_pbar_settop_pb = XtVaCreateManagedWidget( "dialog" , xmPushButtonWidgetClass , wfunc_pbar_menu , LABEL_ARG("Set Top Value") , XmNmarginHeight , 0 , XmNtraversalOn , True , XmNinitialResourcesPersistent , False , NULL ) ; XtAddCallback( wfunc_pbar_settop_pb , XmNactivateCallback , RCREND_pbarmenu_CB , im3d ) ; /* 15 Jun 2000: image save button */ wfunc_pbar_saveim_pb = XtVaCreateManagedWidget( "dialog" , xmPushButtonWidgetClass , wfunc_pbar_menu , LABEL_ARG("Save to PPM") , XmNmarginHeight , 0 , XmNtraversalOn , True , XmNinitialResourcesPersistent , False , NULL ) ; MCW_register_hint( wfunc_pbar_saveim_pb , "Write out as image file" ); XtAddCallback( wfunc_pbar_saveim_pb , XmNactivateCallback , RCREND_pbarmenu_CB , im3d ) ; (void) XtVaCreateManagedWidget( "dialog" , xmSeparatorWidgetClass , wfunc_pbar_menu , XmNseparatorType , XmSINGLE_LINE , NULL ) ; { static char * pb_dum_label[2] = { "Dummy" , "Dummy" } ; wfunc_pbar_palette_av = new_MCW_arrowval( wfunc_pbar_menu , /* parent Widget */ "Set Pal " , /* label */ MCW_AV_optmenu , /* option menu style */ 0 , /* first option */ 1 , /* last option */ 0 , /* initial selection */ MCW_AV_readtext , /* ignored but needed */ 0 , /* ditto */ RCREND_palette_av_CB , /* callback when changed */ NULL , /* data for above */ MCW_av_substring_CB , /* text creation routine */ pb_dum_label /* data for above */ ) ; } (void) XtVaCreateManagedWidget( "dialog" , xmSeparatorWidgetClass , wfunc_pbar_menu , XmNseparatorType , XmSINGLE_LINE , NULL ) ; { static char * pb_dum_label[3] = { "Normal" , "NoShade" , "NoMix" } ; /* 21 Dec 1999 */ wfunc_pbar_mixshade_av = new_MCW_arrowval( wfunc_pbar_menu , /* parent Widget */ "Mixing " , /* label */ MCW_AV_optmenu , /* option menu style */ 0 , /* first option */ 2 , /* last option */ 0 , /* initial selection */ MCW_AV_readtext , /* ignored but needed */ 0 , /* ditto */ RCREND_mixshade_av_CB , /* callback when changed */ NULL , /* data for above */ MCW_av_substring_CB , /* text creation routine */ pb_dum_label /* data for above */ ) ; } if( GPT != NULL && PALTAB_NUM(GPT) > 0 ){ refit_MCW_optmenu( wfunc_pbar_palette_av , 0 , /* new minval */ PALTAB_NUM(GPT)-1 , /* new maxval */ 0 , /* new inival */ 0 , /* new decim? */ AFNI_palette_label_CB , /* text routine */ NULL /* text data */ ) ; } else { XtUnmanageChild( wfunc_pbar_palette_av->wrowcol ) ; } /**-- Color pbar to control intensity-to-color mapping --**/ { float pmin=-1.0 , pmax=1.0 ; int npane = INIT_panes_sgn ; /* from afni.h */ /* 22->15 v1.8 [rickr] */ sel_height -= 15 ; /* a little shorter than the scale */ wfunc_color_pbar = new_MCW_pbar( wfunc_color_rowcol , /* parent */ dc , /* display */ npane , /* number panes */ sel_height / npane , /* init pane height */ pmin , pmax , /* value range */ RCREND_color_pbar_CB , /* callback */ NULL ) ; /* callback data */ wfunc_color_pbar->parent = NULL ; wfunc_color_pbar->mode = 0 ; wfunc_color_pbar->bigmode = 1 ; /* v1.8 [rickr] */ wfunc_color_pbar->npan_save[0] = INIT_panes_sgn ; /* from afni.h */ wfunc_color_pbar->npan_save[1] = INIT_panes_pos ; wfunc_color_pbar->hide_changes = INIT_panes_hide ; RCREND_setup_color_pbar() ; /* other setup stuff */ (void) XtVaCreateManagedWidget( "AFNI" , xmSeparatorWidgetClass , wfunc_color_rowcol , XmNseparatorType , XmSINGLE_LINE , NULL ) ; wfunc_colornum_av = new_MCW_arrowval( wfunc_color_rowcol , "#" , MCW_AV_optmenu , NPANE_MIN , NPANE_MAX+1 , wfunc_color_pbar->bigmode ? NPANE_MAX+1 : npane , MCW_AV_notext , 0 , RCREND_colornum_av_CB , NULL , AFNI_inten_av_texter,NULL ) ; PBAR_set_bigmode( wfunc_color_pbar , 1 , pmin,pmax ) ; /* v1.8 [rickr] */ if( NPANE_MAX >= COLSIZE ) AVOPT_columnize( wfunc_colornum_av , 1+(NPANE_MAX+1)/COLSIZE ) ; } /*--- toggle button to control posfunc option for pbar ---*/ { char * color_bbox_label[1] = { "Pos?" } ; wfunc_color_bbox = new_MCW_bbox( wfunc_color_rowcol , 1 , color_bbox_label , MCW_BB_check , MCW_BB_noframe , RCREND_color_bbox_CB , NULL ) ; } XtManageChild(wfunc_color_rowcol) ; /*---------- Column 3: Choices controls -------------------------------*/ wfunc_choices_rowcol = XtVaCreateWidget( "AFNI" , xmRowColumnWidgetClass , wfunc_rowcol , XmNorientation , XmVERTICAL , XmNpacking , XmPACK_TIGHT , XmNmarginHeight, 0 , XmNmarginWidth , 0 , XmNtraversalOn , True , XmNinitialResourcesPersistent , False , NULL ) ; #if 0 /* 2002 Mar 05: rickr - make space for showthru factor */ xstr = XmStringCreateLtoR( "Choices" , XmFONTLIST_DEFAULT_TAG ) ; wfunc_choices_label = XtVaCreateManagedWidget( "AFNI" , xmLabelWidgetClass , wfunc_choices_rowcol , XmNlabelString , xstr , XmNinitialResourcesPersistent , False , NULL ) ; XmStringFree(xstr) ; #endif /*--- 30 Nov 1997: sub-brick menus ---*/ wfunc_buck_frame = XtVaCreateWidget( "AFNI" , xmFrameWidgetClass , wfunc_choices_rowcol , XmNshadowType , XmSHADOW_ETCHED_IN , XmNtraversalOn , True , XmNinitialResourcesPersistent , False , NULL ) ; wfunc_buck_rowcol = XtVaCreateWidget( "AFNI" , xmRowColumnWidgetClass , wfunc_buck_frame , XmNorientation , XmVERTICAL , XmNpacking , XmPACK_TIGHT , XmNtraversalOn , True , XmNinitialResourcesPersistent , False , NULL ) ; /*--- Sub-brick selectors for Color & Threshold ---*/ /* (Actual labels are set when used) */ wfunc_color_av = new_MCW_arrowval( wfunc_buck_rowcol , /* parent Widget */ "Color" , /* label */ MCW_AV_optmenu , /* option menu style */ 0 , /* first option */ 1 , /* last option */ 0 , /* initial selection */ MCW_AV_readtext , /* ignored but needed */ 0 , /* decimal shift */ RCREND_choose_av_CB , /* callback when changed */ NULL , /* data for above */ MCW_av_substring_CB , /* text creation routine */ RCREND_dummy_av_label /* data for above */ ) ; wfunc_thresh_av = new_MCW_arrowval( wfunc_buck_rowcol , /* parent Widget */ "Thr " , /* label */ MCW_AV_optmenu , /* option menu style */ 0 , /* first option */ 1 , /* last option */ 0 , /* initial selection */ MCW_AV_readtext , /* ignored but needed */ 0 , /* decimal shift */ RCREND_choose_av_CB , /* callback when changed */ NULL , /* data for above */ MCW_av_substring_CB , /* text creation routine */ RCREND_dummy_av_label /* data for above */ ) ; XtManageChild( wfunc_buck_rowcol ) ; XtManageChild( wfunc_buck_frame ) ; AV_SENSITIZE( wfunc_color_av , False ) ; /* turn these off for now */ AV_SENSITIZE( wfunc_thresh_av , False ) ; /*--- menus to control opacity of color ---*/ wfunc_opacity_frame = XtVaCreateWidget( "AFNI" , xmFrameWidgetClass , wfunc_choices_rowcol , XmNshadowType , XmSHADOW_ETCHED_IN , XmNtraversalOn , True , XmNinitialResourcesPersistent , False , NULL ) ; wfunc_opacity_rowcol = XtVaCreateWidget( "AFNI" , xmRowColumnWidgetClass , wfunc_opacity_frame , XmNorientation , XmVERTICAL , XmNpacking , XmPACK_TIGHT , XmNtraversalOn , True , XmNinitialResourcesPersistent , False , NULL ) ; { static char * func_opacity_labels[11] = { /* 11 Sep 2001: add ST+Dcue */ "Underlay" , /* 07 Feb 2002: remove both STs */ " 0.1" , " 0.2" , " 0.3" , " 0.4" , " 0.5" , " 0.6" , " 0.7" , " 0.8" , " 0.9" , " 1.0" } ; wfunc_opacity_av = new_MCW_arrowval( wfunc_opacity_rowcol , /* parent Widget */ "Color Opacity " , /* label */ MCW_AV_optmenu , /* option menu style */ 0 , /* first option */ 10 , /* last option - was 12 */ 5 , /* initial selection */ MCW_AV_readtext , /* ignored but needed */ 0 , /* decimal shift */ RCREND_color_opacity_CB , /* callback when changed */ NULL , /* data for above */ MCW_av_substring_CB , /* text creation routine */ func_opacity_labels /* data for above */ ) ; } /*--- ShowThru toggle button and ST factor pulldown menu */ { /* 2002 Mar 06 */ static char * ST_factor_labels[21] = { " 0%", " 5%", " 10%", " 15%", " 20%", " 25%", " 30%", " 35%", " 40%", " 45%", " 50%", " 55%", " 60%", " 65%", " 70%", " 75%", " 80%", " 85%", " 90%", " 95%", "100%" }; char * color_ST_label[1] = { "ShowThru Mode " }; Widget wrc; wrc = XtVaCreateWidget( "AFNI", xmRowColumnWidgetClass, wfunc_opacity_rowcol, XmNorientation, XmHORIZONTAL, XmNpacking, XmPACK_TIGHT, XmNmarginHeight, 0, XmNmarginWidth, 0, XmNtraversalOn, True , XmNinitialResourcesPersistent, False, NULL ); wfunc_do_ST_bbox = new_MCW_bbox( wrc, 1, color_ST_label, MCW_BB_check, MCW_BB_noframe, RCREND_do_ST_CB, NULL ); wfunc_ST_fac_av = new_MCW_arrowval( wrc, /* parent Widget */ "@", /* label */ MCW_AV_optmenu, /* option menu style */ 0, /* first option */ 20, /* last option */ 15, /* initial selection */ MCW_AV_readtext, /* ignored but needed */ 0, /* decimal shift */ RCREND_ST_factor_CB, /* callback when changed */ NULL, /* data for above */ MCW_av_substring_CB, /* text creation routine */ ST_factor_labels /* data for above */ ); MCW_reghint_children( wfunc_do_ST_bbox->wrowcol, "render ShowThru images - see BHelp" ); MCW_reghelp_children( wfunc_do_ST_bbox->wrowcol, "Use this button to toggle the ShowThru option.\n" "In ShowThru mode, the overlay image is rendered\n" "separately from the underlay (and each with its\n" "respective opacity factor). Then the two are\n" "mixed according to the ShowThru percentage." ) ; MCW_reghint_children( wfunc_ST_fac_av->wrowcol, "set ShowThru factor" ); MCW_reghelp_children( wfunc_ST_fac_av->wrowcol, "This selects the percentage of the overlay image\n" "(and resulting percentage of the underlay image)\n" "used in producing the ShowThru image." ) ; XtSetSensitive( wfunc_ST_fac_av->wrowcol, False ); XtManageChild(wrc); } /*--- toggle switches to control if we see function ---*/ { char * see_overlay_label[1] = { "See Overlay" } ; char * cut_overlay_label[1] = { "Cutout Overlay" } ; char * kill_clusters_label[1] = { "Remove Small Clusters" } ; char * see_ttatlas_label[1] = { "TT Atlas" } ; /* 24 Jul 2001 */ Widget wrc ; wrc = XtVaCreateWidget( /* 24 Jul 2001 */ "AFNI" , xmRowColumnWidgetClass , wfunc_opacity_rowcol , XmNorientation , XmHORIZONTAL , XmNpacking , XmPACK_TIGHT , XmNmarginHeight, 0 , XmNmarginWidth , 0 , XmNtraversalOn , True , XmNinitialResourcesPersistent , False , NULL ) ; wfunc_see_overlay_bbox = new_MCW_bbox( wrc , 1 , see_overlay_label , MCW_BB_check , MCW_BB_noframe , RCREND_see_overlay_CB , NULL ) ; wfunc_see_ttatlas_bbox = new_MCW_bbox( wrc , /* 24 Jul 2001 */ 1 , see_ttatlas_label , MCW_BB_check , MCW_BB_noframe , RCREND_see_ttatlas_CB , NULL ) ; if( TT_retrieve_atlas() == NULL ) XtSetSensitive( wfunc_see_ttatlas_bbox->wrowcol , False ) ; XtManageChild(wrc) ; /* 24 Jul 2001 */ wfunc_cut_overlay_bbox = new_MCW_bbox( wfunc_opacity_rowcol , 1 , cut_overlay_label , MCW_BB_check , MCW_BB_noframe , RCREND_cut_overlay_CB , NULL ) ; wfunc_kill_clusters_bbox = new_MCW_bbox( wfunc_opacity_rowcol , 1 , kill_clusters_label , MCW_BB_check , MCW_BB_noframe , RCREND_kill_clusters_CB , NULL ) ; wfunc_clusters_rmm_av = new_MCW_arrowval( wfunc_opacity_rowcol , " rmm " , MCW_AV_downup , 0 , 99 , (int)(10*func_clusters_rmm) , MCW_AV_edittext , 1 , RCREND_clusters_av_CB,NULL,NULL,NULL ) ; wfunc_clusters_vmul_av = new_MCW_arrowval( wfunc_opacity_rowcol , " vmul " , MCW_AV_downup , 0 , 9999 , (int)(0.1*func_clusters_vmul), MCW_AV_edittext , -1 , RCREND_clusters_av_CB,NULL,NULL,NULL ) ; AV_SENSITIZE( wfunc_clusters_rmm_av , False ) ; AV_SENSITIZE( wfunc_clusters_vmul_av, False ) ; } XtManageChild( wfunc_opacity_rowcol ) ; XtManageChild( wfunc_opacity_frame ) ; /*--- range controls ---*/ wfunc_range_frame = XtVaCreateManagedWidget( "AFNI" , xmFrameWidgetClass , wfunc_choices_rowcol , XmNshadowType , XmSHADOW_ETCHED_IN , XmNtraversalOn , True , XmNinitialResourcesPersistent , False , NULL ) ; wfunc_range_rowcol = XtVaCreateWidget( "AFNI" , xmRowColumnWidgetClass , wfunc_range_frame , XmNorientation , XmVERTICAL , XmNpacking , XmPACK_TIGHT , XmNtraversalOn , True , XmNinitialResourcesPersistent , False , NULL ) ; /*--- label to show the data ranges from the overlay dataset ---*/ xstr = RCREND_range_label() ; /* make a dummy label */ wfunc_range_label = XtVaCreateManagedWidget( "AFNI" , xmLabelWidgetClass , wfunc_range_rowcol , XmNrecomputeSize , False , XmNlabelString , xstr , XmNtraversalOn , True , XmNinitialResourcesPersistent , False , NULL ) ; XmStringFree(xstr) ; /*--- toggle button to control automatic range scaling for pbar ---*/ { char * range_bbox_label[1] = { "autoRange:xxxxxxxxx" } ; wfunc_range_bbox = new_MCW_bbox( wfunc_range_rowcol , 1 , range_bbox_label , MCW_BB_check , MCW_BB_noframe , RCREND_range_bbox_CB , NULL ) ; MCW_set_bbox( wfunc_range_bbox , 1 ) ; xstr = RCREND_autorange_label() ; XtVaSetValues( wfunc_range_bbox->wbut[0], XmNlabelString,xstr , NULL ) ; XmStringFree(xstr) ; } /*--- 30 Mar 2001: put next 2 things in a horizontal rowcol ---*/ wqqq = XtVaCreateWidget( "dialog" , xmRowColumnWidgetClass , wfunc_range_rowcol , XmNorientation , XmHORIZONTAL , XmNpacking , XmPACK_TIGHT , XmNtraversalOn , True , XmNinitialResourcesPersistent , False , NULL ) ; /*--- arrowval to provide user control for pbar scaling ---*/ wfunc_range_av = new_MCW_arrowval( wqqq , /* parent */ NULL , /* label */ MCW_AV_downup , /* arrow directions */ 0 , /* min value */ 9999999 , /* max value */ (int)(func_range) , /* init value */ MCW_AV_editext , /* input/output text display */ 0 , /* decimal shift */ RCREND_range_av_CB , /* routine to call when button */ NULL , /* is pressed, and its data */ NULL,NULL /* no special display */ ) ; AV_SENSITIZE( wfunc_range_av , False ) ; /*--- 30 Mar 2001: rotate pbar ---*/ wfunc_range_rotate_av = new_MCW_arrowval( wqqq , "Rota" , MCW_AV_downup , 0,0,0 , MCW_AV_notext , 0 , AFNI_range_rotate_av_CB , (XtPointer) wfunc_color_pbar , NULL,NULL ) ; XtManageChild( wqqq ) ; XtManageChild( wfunc_range_rowcol ) ; XtManageChild( wfunc_range_frame ) ; XtManageChild( wfunc_choices_rowcol ) ; XtManageChild( wfunc_rowcol ) ; XtManageChild( wfunc_uber_rowcol ) ; #if 0 XtVaSetValues( wfunc_uber_rowcol , XmNresizeWidth , False , NULL ) ; #endif EXRETURN ; } /*--------------------------------------------------------------------------------- Initialize the functional colormap -----------------------------------------------------------------------------------*/ void RCREND_init_cmap(void) { ENTRY( "RCREND_init_cmap" ); reset_bigcolors( wfunc_color_pbar->bigcolor ); /* bigmode v1.8 [rickr] */ if ( wfunc_color_pbar->bigmode ) CREN_set_rgbmap( gcr.rh, NPANE_BIG, gcr.bigstuff.r, gcr.bigstuff.g, gcr.bigstuff.b ); else CREN_set_rgbmap( gcr.rh, MIN( dc->ovc->ncol_ov, GRAF_SIZE ), (dc)->ovc->r_ov, (dc)->ovc->g_ov, (dc)->ovc->b_ov ); EXRETURN ; } /*------------------------------------------------------------------------ 09 May 2001: fix a Solaris stupidity, where the scale is resized improperly when the Define Function panel is opened! --------------------------------------------------------------------------*/ #if defined(SOLARIS) && defined(FIX_SCALE_SIZE_PROBLEM) static void fixscale( XtPointer client_data , XtIntervalId * id ) { ENTRY( "fixscale" ); FIX_SCALE_SIZE ; EXRETURN; } #endif /*--------------------------------------------------------------------------- Open or close the overlay control panel -----------------------------------------------------------------------------*/ void RCREND_open_func_CB( Widget w, XtPointer client_data, XtPointer call_data ) { ENTRY( "RCREND_open_func_CB" ); if( wfunc_frame == NULL ) RCREND_func_widgets() ; /* need to make them */ if( XtIsManaged(wfunc_frame) ){ /* if open, close */ XtUnmanageChild(wfunc_vsep ) ; XtUnmanageChild(wfunc_frame) ; } else { /* if closed, open */ HIDE_SCALE ; XtManageChild(wfunc_vsep ) ; XtManageChild(wfunc_frame) ; update_MCW_pbar( wfunc_color_pbar ) ; /* may need to be redrawn */ FIX_SCALE_SIZE ; #if defined(SOLARIS) && defined(FIX_SCALE_SIZE_PROBLEM) (void) XtAppAddTimeOut( XtWidgetToApplicationContext(wfunc_frame), 50,fixscale,NULL ) ; /* 09 May 2001 */ #endif RCREND_init_cmap() ; /* setup the colormap */ POPUP_cursorize(wfunc_color_label) ; if( wfunc_color_pbar->bigmode ) POPUP_cursorize( wfunc_color_pbar->panew ) ; /* 08 Apr 2005 */ else NORMAL_cursorize( wfunc_color_pbar->panew ) ; /* 08 Apr 2005 */ } MCW_invert_widget(wfunc_open_pb) ; /* a flag */ EXRETURN ; } /*--------------------------------------------------------------------------- Makes the labels for the thresh_top menu (narrower than the default) -----------------------------------------------------------------------------*/ char * RCREND_thresh_tlabel_CB( MCW_arrowval * av , XtPointer junk ) { static char tlab[8] ; ENTRY( "RCREND_thresh_tlabel_CB" ); sprintf(tlab,"%d",av->ival) ; RETURN(tlab); } /*--------------------------------------------------------------------------- Initialize the memory of the color selector -----------------------------------------------------------------------------*/ void RCREND_setup_color_pbar(void) { MCW_pbar * pbar = wfunc_color_pbar ; int np , i , jm , lcol ; ENTRY( "RCREND_setup_color_pbar" ); reset_bigcolors( pbar->bigcolor ); jm = pbar->mode ; lcol = dc->ovc->ncol_ov - 1 ; /** load the 'save' values for all possible pane counts **/ for( np=NPANE_MIN ; np <= NPANE_MAX ; np++ ){ for( i=0 ; i <= np ; i++ ){ pbar->pval_save[np][i][0] = INIT_pval_sgn[np][i] ; /* from afni.h */ pbar->pval_save[np][i][1] = INIT_pval_pos[np][i] ; } for( i=0 ; i < np ; i++ ){ pbar->ovin_save[np][i][0] = MIN( lcol , INIT_ovin_sgn[np][i] ) ; pbar->ovin_save[np][i][1] = MIN( lcol , INIT_ovin_pos[np][i] ) ; } } /** load the values for the current pane count **/ np = pbar->num_panes ; jm = pbar->mode ; for( i=0 ; i <= np ; i++ ) pbar->pval[i] = pbar->pval_save[np][i][jm] ; for( i=0 ; i < np ; i++ ) pbar->ov_index[i] = pbar->ovin_save[np][i][jm] ; pbar->update_me = 1 ; EXRETURN ; } /*------------------------------------------------------------------------- Create label for range display in function control panel ---------------------------------------------------------------------------*/ XmString RCREND_range_label(void) { char fim_minch[10] = " --------" , fim_maxch[10] = " --------" , thr_minch[10] = " --------" , thr_maxch[10] = " --------" ; char buf[256] , qbuf[16] ; XmString xstr ; int iv ; ENTRY( "RCREND_range_label" ); if( ISVALID_DSET(func_dset) && ISVALID_STATISTIC(func_dset->stats) ){ iv = func_color_ival ; if( DSET_VALID_BSTAT(func_dset,iv) ){ AV_fval_to_char( func_dset->stats->bstat[iv].min , qbuf ) ; sprintf( fim_minch , "%9.9s" , qbuf ) ; AV_fval_to_char( func_dset->stats->bstat[iv].max , qbuf ) ; sprintf( fim_maxch , "%9.9s" , qbuf ) ; } iv = func_thresh_ival ; if( DSET_VALID_BSTAT(func_dset,iv) ){ AV_fval_to_char( func_dset->stats->bstat[iv].min , qbuf ) ; sprintf( thr_minch , "%9.9s" , qbuf ) ; AV_fval_to_char( func_dset->stats->bstat[iv].max , qbuf ) ; sprintf( thr_maxch , "%9.9s" , qbuf ) ; } } sprintf( buf , "Color %s:%s\nThr %s:%s" , fim_minch,fim_maxch, thr_minch,thr_maxch ) ; xstr = XmStringCreateLtoR( buf , XmFONTLIST_DEFAULT_TAG ) ; RETURN(xstr) ; } /*------------------------------------------------------------------------ Find the autorange and make a label for the autorange control widget --------------------------------------------------------------------------*/ XmString RCREND_autorange_label(void) { XmString xstr ; float rrr = DEFAULT_FUNC_RANGE ; char buf[32] , qbuf[16] ; ENTRY( "RCREND_autorange_label" ); if( ISVALID_DSET(func_dset) ){ RELOAD_STATS(func_dset) ; if( ISVALID_STATISTIC(func_dset->stats) ){ float s1 , s2 ; int iv ; iv = func_color_ival ; if( DSET_VALID_BSTAT(func_dset,iv) ){ s1 = fabs(func_dset->stats->bstat[iv].min) , s2 = fabs(func_dset->stats->bstat[iv].max) ; rrr = (s1func_type , func_dset->stat_aux ) ; else pval = -1.0 ; #else pval = THD_stat_to_pval( thresh , DSET_BRICK_STATCODE(func_dset,func_thresh_ival) , DSET_BRICK_STATAUX (func_dset,func_thresh_ival) ) ; #endif if( pval < 0.0 ){ strcpy( buf , THR_PVAL_LABEL_NONE ) ; } else { if( pval == 0.0 ){ strcpy( buf , "p = 0" ) ; } else if( pval >= 0.9999 ){ strcpy( buf , "p = 1" ) ; } else if( pval >= 0.0010 ){ char qbuf[16] ; sprintf( qbuf , "%5.4f" , pval ) ; strcpy( buf , qbuf+1 ) ; } else { int dec = (int)(0.999 - log10(pval)) ; pval = pval * pow( 10.0 , (double) dec ) ; /* between 1 and 10 */ if( dec < 10 ) sprintf( buf , "%3.1f-%1d" , pval, dec ) ; else sprintf( buf , "%1d.-%2d" , (int)pval, dec ) ; } } MCW_set_widget_label( wfunc_thr_pval_label , buf ) ; EXRETURN ; } /*------------------------------------------------------------------------------ When the user alters the threshold [at the end] --------------------------------------------------------------------------------*/ void RCREND_thr_scale_CB( Widget w, XtPointer client_data, XtPointer call_data ) { XmScaleCallbackStruct * cbs = (XmScaleCallbackStruct *) call_data ; float fff ; ENTRY( "RCREND_thr_scale_CB" ); fff = THR_FACTOR * cbs->value ; /* between 0 and 1 now */ if( fff >= 0.0 && fff <= 1.0 ) func_threshold = fff ; else EXRETURN ; RCREND_set_thr_pval() ; MCW_discard_events_all( w , ButtonPressMask ) ; /* 20 Mar 2007 */ INVALIDATE_OVERLAY ; FIX_SCALE_SIZE ; /* 09 May 2001 */ EXRETURN ; } /*------------------------------------------------------------------------------- When the user drags the slider (just update the p-value) ---------------------------------------------------------------------------------*/ void RCREND_thr_scale_drag_CB( Widget w, XtPointer client_data, XtPointer call_data ) { XmScaleCallbackStruct * cbs = (XmScaleCallbackStruct *) call_data ; float fff ; ENTRY( "RCREND_thr_scale_drag_CB" ); fff = THR_FACTOR * cbs->value ; /* between 0 and 1 now */ if( fff >= 0.0 && fff <= 1.0 ) func_threshold = fff ; else EXRETURN ; RCREND_set_thr_pval() ; EXRETURN ; } /*------------------------------------------------------------------------------- Called when the user toggles the autorange button ---------------------------------------------------------------------------------*/ void RCREND_range_bbox_CB( Widget w, XtPointer cd, XtPointer cb) { int newauto = MCW_val_bbox(wfunc_range_bbox) ; ENTRY( "RCREND_range_bbox_CB" ); if( newauto == func_use_autorange ) EXRETURN ; /* no change? */ func_use_autorange = newauto ; func_range = (newauto) ? (func_autorange) : (wfunc_range_av->fval) ; AFNI_hintize_pbar( wfunc_color_pbar , FUNC_RANGE ) ; /* 30 Mar 2001 */ AV_SENSITIZE( wfunc_range_av , ! newauto ) ; INVALIDATE_OVERLAY ; EXRETURN ; } /*------------------------------------------------------------------------------ Called when the user changes the function range --------------------------------------------------------------------------------*/ void RCREND_range_av_CB( MCW_arrowval * av , XtPointer cd ) { ENTRY( "RCREND_range_av_CB" ); func_range = av->fval ; AFNI_hintize_pbar( wfunc_color_pbar , FUNC_RANGE ) ; /* 30 Mar 2001 */ INVALIDATE_OVERLAY ; EXRETURN ; } /*------------------------------------------------------------------------------ Called to change the scaling on the threshold slider --------------------------------------------------------------------------------*/ void RCREND_thresh_top_CB( MCW_arrowval * av , XtPointer cd ) { static float dval[9] = { 1.0 , 10.0 , 100.0 , 1000.0 , 10000.0 , 100000.0 , 1000000.0 , 10000000.0 , 100000000.0 } ; int decim ; float tval ; ENTRY( "RCREND_thresh_top_CB" ); tval = dval[av->ival] ; if( tval <= 0.0 ) tval = 1.0 ; decim = (2*THR_TOP_EXPON) - (int)(THR_TOP_EXPON + 0.01 + log10(tval)) ; if( decim < 0 ) decim = 0 ; XtVaSetValues( wfunc_thr_scale, XmNdecimalPoints, decim, NULL ) ; func_thresh_top = tval ; RCREND_set_thr_pval() ; INVALIDATE_OVERLAY ; EXRETURN ; } /*------------------------------------------------------------------------------- Called by the pbar routines when the pbar is altered. ---------------------------------------------------------------------------------*/ void RCREND_color_pbar_CB( MCW_pbar * pbar , XtPointer cd , int reason ) { ENTRY( "RCREND_color_pbar_CB" ); FIX_SCALE_SIZE ; INVALIDATE_OVERLAY ; /* to be sure v1.8 [rickr] */ reset_bigcolors( wfunc_color_pbar->bigcolor ); AFNI_hintize_pbar( wfunc_color_pbar , FUNC_RANGE ) ; /* 30 Mar 2001 */ EXRETURN ; } /*------------------------------------------------------------------------------ Called to change the number of panels in the pbar --------------------------------------------------------------------------------*/ void RCREND_colornum_av_CB( MCW_arrowval * av , XtPointer cd ) { ENTRY( "RCREND_colornum_av_CB" ); HIDE_SCALE ; if( av->ival > NPANE_MAX ){ int npane=wfunc_color_pbar->num_panes , jm=wfunc_color_pbar->mode ; float pmax=wfunc_color_pbar->pval_save[npane][0][jm] , pmin=wfunc_color_pbar->pval_save[npane][npane][jm] ; PBAR_set_bigmode( wfunc_color_pbar , 1 , pmin,pmax ) ; RCREND_color_pbar_CB( wfunc_color_pbar, im3d, 0 ) ; POPUP_cursorize( wfunc_color_pbar->panew ) ; /* 08 Apr 2005 */ } else { wfunc_color_pbar->bigmode = 0 ; alter_MCW_pbar( wfunc_color_pbar , av->ival , NULL ) ; NORMAL_cursorize( wfunc_color_pbar->panew ) ; /* 08 Apr 2005 */ } FIX_SCALE_SIZE ; INVALIDATE_OVERLAY ; EXRETURN ; } /*------------------------------------------------------------------------------ Called when the user toggles the posfunc button --------------------------------------------------------------------------------*/ void RCREND_color_bbox_CB( Widget w, XtPointer cd, XtPointer cb) { int jm , newpos=MCW_val_bbox(wfunc_color_bbox) ; ENTRY( "RCREND_color_bbox_CB" ); if( newpos == func_posfunc ) EXRETURN ; /* no change? */ func_posfunc = newpos ; jm = wfunc_color_pbar->mode = (newpos) ? 1 : 0 ; /* pbar mode */ HIDE_SCALE ; if( wfunc_color_pbar->bigmode ){ /* 30 Jan 2003 */ int npane=wfunc_color_pbar->num_panes ; float pmax=wfunc_color_pbar->pval_save[npane][0][jm] , pmin=wfunc_color_pbar->pval_save[npane][npane][jm] ; wfunc_color_pbar->bigset = 0 ; PBAR_set_bigmode( wfunc_color_pbar , 1 , pmin,pmax ) ; AFNI_inten_pbar_CB( wfunc_color_pbar , im3d , 0 ) ; POPUP_cursorize( wfunc_color_pbar->panew ) ; /* 08 Apr 2005 */ } else { alter_MCW_pbar( wfunc_color_pbar, wfunc_color_pbar->npan_save[jm], NULL ); NORMAL_cursorize( wfunc_color_pbar->panew ) ; /* 08 Apr 2005 */ } FIX_SCALE_SIZE ; /* set the count on the wfunc_color_pbar control arrowval to match */ if ( wfunc_color_pbar->bigmode ) AV_assign_ival( wfunc_colornum_av , NPANE_MAX+1 ) ; else AV_assign_ival( wfunc_colornum_av , wfunc_color_pbar->npan_save[jm] ) ; INVALIDATE_OVERLAY ; EXRETURN ; } /*------------------------------------------------------------------------------ Called to set the overlay opacity factor --------------------------------------------------------------------------------*/ void RCREND_color_opacity_CB( MCW_arrowval * av , XtPointer cd ) { ENTRY( "RCREND_color_opacity_CB" ); func_color_opacity = 0.1 * av->ival; func_color_opacity = MIN(func_color_opacity, 1.0); INVALIDATE_OVERLAY ; EXRETURN ; } /*------------------------------------------------------------------------------ Called to toggle visibility of the overlay --------------------------------------------------------------------------------*/ void RCREND_see_overlay_CB( Widget w, XtPointer cd, XtPointer cb) { int newsee = MCW_val_bbox(wfunc_see_overlay_bbox) ; ENTRY( "RCREND_see_overlay_CB" ); if( newsee == func_see_overlay ) EXRETURN ; func_see_overlay = newsee ; INVALIDATE_OVERLAY ; FREE_VOLUMES ; EXRETURN ; } /*------------------------------------------------------------------------------ Called to toggle visibility of the TT atlas -- 24 Jul 2001 --------------------------------------------------------------------------------*/ void RCREND_see_ttatlas_CB( Widget w, XtPointer cd, XtPointer cb) { int newsee = MCW_val_bbox(wfunc_see_ttatlas_bbox) ; ENTRY( "RCREND_see_ttatlas_CB" ); if( newsee == func_see_ttatlas ) EXRETURN ; func_see_ttatlas = newsee ; INVALIDATE_OVERLAY ; FREE_VOLUMES ; EXRETURN ; } /*----------------------------------------------------------------------------- Called to toggle the ShowThru option for overlay -- 2002 Mar 06 --------------------------------------------------------------------------------*/ void RCREND_do_ST_CB( Widget w, XtPointer cd, XtPointer cb ) { int newsee = MCW_val_bbox(wfunc_do_ST_bbox); ENTRY( "RCREND_do_ST_CB" ); if( newsee == func_showthru ) EXRETURN; func_showthru = newsee; INVALIDATE_OVERLAY; FREE_VOLUMES; if ( func_showthru ) XtSetSensitive( wfunc_ST_fac_av->wrowcol, True ); else XtSetSensitive( wfunc_ST_fac_av->wrowcol, False ); EXRETURN; } /*------------------------------------------------------------------------------ Called to set the ShowThru factor -- 2002 Mar 06 --------------------------------------------------------------------------------*/ void RCREND_ST_factor_CB( MCW_arrowval * av , XtPointer cd ) { float osf = func_showthru_fac; ENTRY( "RCREND_ST_factor_CB" ); func_showthru_fac = av->ival * 0.05; if ( osf != func_showthru_fac ) INVALIDATE_OVERLAY ; EXRETURN ; } /*------------------------------------------------------------------------------ Called to toggle whether or not cutouts affect the overlay --------------------------------------------------------------------------------*/ void RCREND_cut_overlay_CB( Widget w, XtPointer cd, XtPointer cb) { int newcut = MCW_val_bbox(wfunc_cut_overlay_bbox) ; ENTRY( "RCREND_cut_overlay_CB" ); if( newcut == func_cut_overlay ) EXRETURN ; func_cut_overlay = newcut ; if( num_cutouts > 0 ){ INVALIDATE_OVERLAY ; } EXRETURN ; } /*------------------------------------------------------------------------------ Called to toggle whether or not to kill clusters --------------------------------------------------------------------------------*/ void RCREND_kill_clusters_CB( Widget w, XtPointer cd, XtPointer cb) { int cc , newkill = MCW_val_bbox(wfunc_kill_clusters_bbox) ; ENTRY( "RCREND_kill_clusters_CB" ); if( newkill == func_kill_clusters ) EXRETURN ; func_kill_clusters = newkill ; AV_SENSITIZE( wfunc_clusters_rmm_av , newkill ) ; AV_SENSITIZE( wfunc_clusters_vmul_av, newkill ) ; INVALIDATE_OVERLAY ; for( cc=0 ; cc < current_cutout_state.num ; cc++ ) if( current_cutout_state.type[cc] == CUT_NONOVERLAY ){ FREE_VOLUMES ; break ; } EXRETURN ; } void RCREND_clusters_av_CB( MCW_arrowval * av , XtPointer cd ) { int cc ; ENTRY( "RCREND_clusters_av_CB" ); INVALIDATE_OVERLAY ; for( cc=0 ; cc < current_cutout_state.num ; cc++ ) if( current_cutout_state.type[cc] == CUT_NONOVERLAY ){ FREE_VOLUMES ; break ; } EXRETURN ; } /*------------------------------------------------------------------------------- Event handler to find #3 button press for pbar popup, and popup the menu ---------------------------------------------------------------------------------*/ void RCREND_pbarmenu_EV( Widget w , XtPointer cd , XEvent * ev , Boolean * continue_to_dispatch ) { static int old_paltab_num = 0 ; ENTRY( "RCREND_pbarmenu_EV" ); switch( ev->type ){ case ButtonPress:{ XButtonEvent *event = (XButtonEvent *) ev ; if( event->button == Button3 || event->button == Button1 ){ /* in case the user read in any new palette, add them to menu */ if( GPT != NULL && PALTAB_NUM(GPT) > old_paltab_num ){ refit_MCW_optmenu( wfunc_pbar_palette_av , 0 , /* new minval */ PALTAB_NUM(GPT)-1 , /* new maxval */ 0 , /* new inival */ 0 , /* new decim? */ AFNI_palette_label_CB , /* text routine */ NULL /* text data */ ) ; XtManageChild( wfunc_pbar_palette_av->wrowcol ) ; old_paltab_num = PALTAB_NUM(GPT) ; } XmMenuPosition( wfunc_pbar_menu , event ) ; /* where */ XtManageChild ( wfunc_pbar_menu ) ; /* popup */ } } break ; } EXRETURN ; } /*-------------------------------------------------------------------------------- Callbacks for all actions in the pbar popup ----------------------------------------------------------------------------------*/ void RCREND_pbarmenu_CB( Widget w , XtPointer cd , XtPointer cbs ) { MCW_pbar * pbar ; int npane , jm , ii ; double pmax , pmin ; float pval[NPANE_MAX+1] ; ENTRY( "RCREND_pbarmenu_CB" ); pbar = wfunc_color_pbar ; npane = pbar->num_panes ; jm = pbar->mode ; pmax = pbar->pval_save[npane][0][jm] ; pmin = pbar->pval_save[npane][npane][jm] ; /*--- Equalize spacings ---*/ if( w == wfunc_pbar_equalize_pb ){ for( ii=0 ; ii <= npane ; ii++ ) pval[ii] = pmax - ii * (pmax-pmin)/npane ; HIDE_SCALE ; alter_MCW_pbar( pbar , 0 , pval ) ; FIX_SCALE_SIZE ; INVALIDATE_OVERLAY ; } /*--- Set top value ---*/ else if( w == wfunc_pbar_settop_pb ){ MCW_choose_integer( wfunc_choices_label , "Pbar Top" , 0 , 99999 , 1 , RCREND_set_pbar_top_CB , NULL ) ; } /*--- Save pbar into image file ---*/ else if( w == wfunc_pbar_saveim_pb ){ MCW_choose_string( wfunc_choices_label, "PPM file prefix" , NULL , RCREND_finalize_saveim_CB , cd ) ; } EXRETURN ; } void RCREND_palette_av_CB( MCW_arrowval * av , XtPointer cd ) { ENTRY( "RCREND_palette_av_CB" ); if( GPT == NULL || av->ival < 0 || av->ival >= PALTAB_NUM(GPT) ) EXRETURN ; HIDE_SCALE ; load_PBAR_palette_array( wfunc_color_pbar , /* cf. afni_setup.c */ PALTAB_ARR(GPT,av->ival) , 0 ) ; FIX_SCALE_SIZE ; INVALIDATE_OVERLAY ; EXRETURN ; } void RCREND_mixshade_av_CB( MCW_arrowval * av , XtPointer cd ) /* 21 Dec 1999 */ { ENTRY( "RCREND_mixshade_av_CB" ); func_mixshade = av->ival ; EXRETURN ; } void RCREND_set_pbar_top_CB( Widget w , XtPointer cd , MCW_choose_cbs * cbs ) { MCW_pbar * pbar ; float pval[NPANE_MAX+1] ; double pmax , fac ; int ii ; ENTRY( "RCREND_set_pbar_top_CB" ); if( ! renderer_open ){ POPDOWN_integer_chooser; XBell(dc->display,100); EXRETURN; } pmax = cbs->fval ; if( pmax <= 0.0 ) EXRETURN ; /* illegal */ pbar = wfunc_color_pbar ; fac = pmax / pbar->pval[0] ; if( fac == 1.0 ) EXRETURN ; /* no change */ for( ii=0 ; ii <= pbar->num_panes ; ii++ ) pval[ii] = fac * pbar->pval[ii] ; HIDE_SCALE ; alter_MCW_pbar( pbar , 0 , pval ) ; FIX_SCALE_SIZE ; INVALIDATE_OVERLAY ; EXRETURN ; } void RCREND_finalize_saveim_CB( Widget wcaller, XtPointer cd, MCW_choose_cbs * cbs ) { char *fname , *ptr ; int ll , nx=20 , ny=256 ; MRI_IMAGE * im ; ENTRY( "RCREND_finalize_saveim_CB" ); if( !renderer_open || cbs->reason != mcwCR_string || cbs->cval == NULL || (ll=strlen(cbs->cval)) == 0 ) EXRETURN; fname = (char *) malloc( sizeof(char) * (ll+8) ) ; strcpy( fname , cbs->cval ) ; if( ll > 240 || ! THD_filename_ok(fname) ){free(fname); EXRETURN;} ptr = strstr(fname,".ppm") ; if( ptr == NULL ) ptr = strstr(fname,".pnm") ; if( ptr == NULL ) ptr = strstr(fname,".jpg") ; if( ptr == NULL ) strcat(fname,".ppm") ; fprintf(stderr,"Writing palette image to %s\n",fname) ; ptr = getenv( "AFNI_PBAR_IMXY" ); if( ptr != NULL ){ ll = sscanf( ptr , "%dx%d" , &nx , &ny ) ; if( ll < 2 || nx < 1 || ny < 32 ){ nx=20; ny=256; } } im = MCW_pbar_to_mri( wfunc_color_pbar , nx,ny ) ; mri_write_pnm( fname , im ) ; POPDOWN_string_chooser; mri_free(im); free(fname); EXRETURN; } /*----------------------------------------------------------------------------------- Load color index data from the functional dataset into the local overlay array (heavily adapted from AFNI_func_overlay in afni_func.c). -------------------------------------------------------------------------------------*/ void RCREND_reload_func_dset(void) { THD_3dim_dataset * local_dset; MRI_IMAGE * cim , * tim ; int sublist[3] = {2, 0, 1}; /* sub-brick list for resampling */ int ival_func, ival_thr; void * car , * tar ; float cfac , tfac ; float bbot, btop, bdelta; int ii , nvox , num_lp , lp , bindex ; byte * ovar ; MCW_pbar * pbar = wfunc_color_pbar ; byte fim_ovc[NPANE_MAX+1] ; float fim_thr[NPANE_MAX] , scale_factor , thresh ; ENTRY( "RCREND_reload_func_dset" ); INVALIDATE_OVERLAY ; /* toss old overlay, if any */ if ( ! ISVALID_DSET( gcr.mset ) ) /* just continue with given func_dset */ { fprintf( stderr, "failure: no master for functional re-orientation" ); XBell( dc->display, 100 ); } /* 24 Jul 2001: if not seeing function, make empty ovim */ if( !func_see_overlay || func_dset == NULL ){ ovim = mri_new_conforming( DSET_BRICK(gcr.mset,dset_ival) , MRI_byte ) ; ovar = MRI_BYTE_PTR(ovim) ; memset( ovar , 0 , DSET_NVOX(gcr.mset) ) ; goto EndOfFuncOverlay ; /* AHA! */ } if ( pbar->bigmode ) /* v1.8 [rickr] */ CREN_set_rgbmap( gcr.rh, NPANE_BIG, gcr.bigstuff.r, gcr.bigstuff.g, gcr.bigstuff.b ); else CREN_set_rgbmap( gcr.rh, MIN( dc->ovc->ncol_ov, GRAF_SIZE ), (dc)->ovc->r_ov, (dc)->ovc->g_ov, (dc)->ovc->b_ov ); DSET_load(func_dset) ; /* make sure is in memory */ local_dset = func_dset; ival_func = func_color_ival; /* assign defaults, may resample to 0, 1 */ ival_thr = func_thresh_ival; /* if (!IS_AXIAL_RAI(func_dset)) - no longer our test for re-orientation */ if ( ! EQUIV_DATAXES( gcr.mset->daxes, func_dset->daxes ) ) { if ( new_fset || gcr.fset_or == NULL ) /* we need a new one */ { fprintf(stderr, "++ resampling overlay to master grid..."); if ( gcr.fset_or != NULL ) /* lose the old one */ THD_delete_3dim_dataset( gcr.fset_or, FALSE ); sublist[1] = func_color_ival; sublist[2] = func_thresh_ival; /* maybe we don't need to resample 2 sub-bricks */ if ( func_color_ival == func_thresh_ival ) sublist[0] = 1; else sublist[0] = 2; /* normal case, get 2 bricks */ gcr.fset_or = r_new_resam_dset(func_dset, gcr.mset, 0,0,0, NULL, RESAM_NN_TYPE, sublist); fprintf(stderr, " done\n"); } if (gcr.fset_or == NULL) XBell(dc->display,100); /* an error - keep local_dset as func_dset */ else { local_dset = gcr.fset_or; /* woohoo! we have our new dataset */ ival_func = 0; if ( func_color_ival == func_thresh_ival ) /* only need 1 brick? */ ival_thr = 0; else ival_thr = 1; } } /* color brick */ cim = DSET_BRICK(local_dset,ival_func) ; nvox = cim->nvox ; car = DSET_ARRAY(local_dset,ival_func) ; cfac = DSET_BRICK_FACTOR(local_dset,ival_func) ; if( cfac == 0.0 ) cfac = 1.0 ; tim = DSET_BRICK(local_dset,ival_thr) ; /* thresh brick */ tar = DSET_ARRAY(local_dset,ival_thr) ; tfac = DSET_BRICK_FACTOR(local_dset,ival_thr) ; if( tfac == 0.0 ) tfac = 1.0 ; ovim = mri_new_conforming( cim , MRI_byte ) ; /* new overlay */ ovar = MRI_BYTE_PTR(ovim) ; scale_factor = FUNC_RANGE ; /* for map from pbar to data value range */ num_lp = pbar->num_panes ; /* top to bottom */ for( lp=0 ; lp < num_lp ; lp++ ) fim_ovc[lp] = pbar->ov_index[lp] ; /* off the bottom */ fim_ovc[num_lp] = (func_posfunc) ? (0) : (fim_ovc[num_lp-1]) ; /* threshold in tar[] scale */ thresh = func_threshold * func_thresh_top / tfac ; /*--- Load the overlay image with the color overlay index ---*/ /* bigmode stuff - for computing color index [v1.8 rickr] */ /* if NPANE_BIG > 128, we still need to break this into [v1.10 rickr] */ /* 128 pieces, as only 128 colors are still being */ /* mapped in CREN_set_rgbmap() */ if ( pbar->bigmode ) { btop = scale_factor / cfac; bbot = (func_posfunc) ? (0) : -btop; bdelta = (btop - bbot) / 128; /* was NPANE_BIG 4 Apr 2006 [rickr] */ } else { for( lp=0 ; lp < num_lp ; lp++ ) fim_thr[lp] = scale_factor * pbar->pval[lp+1] / cfac ; } /*--- no thresholding needed ---*/ if( (thresh < 1.0 && cim->kind != MRI_float) || !func_use_thresh ){ switch( cim->kind ){ default: { fprintf( stderr, "RCREND_reload_func_dset: image kind %d is not" "supported here\n", cim->kind ); EXRETURN; } case MRI_short:{ short * sar = (short *) car ; for( ii=0 ; ii < nvox ; ii++ ){ if( sar[ii] == 0 ){ ovar[ii] = 0 ; } else if( func_posfunc && sar[ii] < 0 ){ ovar[ii] = 0 ; } else if ( pbar->bigmode ) { /* since 0 is considered blank, use 1 through 127 */ /* big_scale is used for NPANE_BIG > 128, to map a large color range down to [1,127] 4 Apr 2006 [rickr] */ bindex = (int)( (btop - sar[ii])/bdelta + 1); RANGE(bindex,1,127); /* was NPANE_BIG - 1 */ if ( CRBM_IS_BLACK_INDEX(bindex) ) ovar[ii] = 0; else ovar[ii] = bindex; } else { for( lp=0 ; lp < num_lp && sar[ii] < fim_thr[lp] ; lp++ ) ; /*nada*/ ovar[ii] = fim_ovc[lp] ; } } } break ; case MRI_float:{ float * sar = (float *) car ; for( ii=0 ; ii < nvox ; ii++ ){ if( sar[ii] == 0.0 ){ ovar[ii] = 0 ; } else if( func_posfunc && sar[ii] < 0.0 ){ ovar[ii] = 0 ; } else if ( pbar->bigmode ) { /* since 0 is considered blank, use 1 through 127 */ bindex = (int)( (btop - sar[ii])/bdelta + 1); RANGE(bindex,1,127); /* was NPANE_BIG - 1 */ if ( CRBM_IS_BLACK_INDEX(bindex) ) ovar[ii] = 0; else ovar[ii] = bindex; } else { for( lp=0 ; lp < num_lp && sar[ii] < fim_thr[lp] ; lp++ ) ; /*nada*/ ovar[ii] = fim_ovc[lp] ; } } } break ; case MRI_byte:{ byte * sar = (byte *) car ; for( lp=0 ; lp < num_lp ; lp++ ) if( pbar->pval[lp+1] <= 0.0 ) fim_thr[lp] = 0 ; for( ii=0 ; ii < nvox ; ii++ ){ if( sar[ii] == 0 ){ ovar[ii] = 0 ; } else if ( pbar->bigmode ) { /* since 0 is considered blank, use 1 through 127 */ bindex = (int)( (btop - sar[ii])/bdelta + 1); RANGE(bindex,1,127); /* was NPANE_BIG - 1 */ if ( CRBM_IS_BLACK_INDEX(bindex) ) ovar[ii] = 0; else ovar[ii] = bindex; } else { for( lp=0 ; lp < num_lp && sar[ii] < fim_thr[lp] ; lp++ ) ; /*nada*/ ovar[ii] = fim_ovc[lp] ; } } } break ; } /*--- end of no thresholding case ---*/ } else { /*--- start of thresholding ---*/ switch( cim->kind ){ default: { fprintf( stderr, "RCREND_reload_func_dset (2): image kind %d is not" "supported here\n", cim->kind ); EXRETURN; } case MRI_short:{ short * sar = (short *) car ; short * qar = (short *) tar ; int thr = (int) thresh ; for( ii=0 ; ii < nvox ; ii++ ){ if( (qar[ii] > -thr && qar[ii] < thr) || sar[ii] == 0 ){ ovar[ii] = 0 ; } else if( func_posfunc && sar[ii] < 0 ){ ovar[ii] = 0 ; /* apply Pos button 16 Jan 2007 [rickr] */ } else if ( pbar->bigmode ) { /* since 0 is considered blank, use 1 through 127 */ bindex = (int)( (btop - sar[ii])/bdelta + 1); RANGE(bindex,1,127); /* was NPANE_BIG - 1 */ if ( CRBM_IS_BLACK_INDEX(bindex) ) ovar[ii] = 0; else ovar[ii] = bindex; } else { for( lp=0 ; lp < num_lp && sar[ii] < fim_thr[lp] ; lp++ ) ; /*nada*/ ovar[ii] = fim_ovc[lp] ; } } } break ; case MRI_float:{ float * sar = (float *) car ; float * qar = (float *) tar ; float thr = thresh ; for( ii=0 ; ii < nvox ; ii++ ){ if( (qar[ii] > -thr && qar[ii] < thr) || sar[ii] == 0.0 ){ ovar[ii] = 0 ; } else if( func_posfunc && sar[ii] < 0.0 ){ ovar[ii] = 0 ; } else if ( pbar->bigmode ) { /* since 0 is considered blank, use 1 through 127 */ bindex = (int)( (btop - sar[ii])/bdelta + 1); RANGE(bindex,1,127); /* was NPANE_BIG - 1 */ if ( CRBM_IS_BLACK_INDEX(bindex) ) ovar[ii] = 0; else ovar[ii] = bindex; } else { for( lp=0 ; lp < num_lp && sar[ii] < fim_thr[lp] ; lp++ ) ; /*nada*/ ovar[ii] = fim_ovc[lp] ; } } } break ; case MRI_byte:{ byte * sar = (byte *) car ; byte * qar = (byte *) tar ; int thr = (int) thresh ; for( lp=0 ; lp < num_lp ; lp++ ) if( pbar->pval[lp+1] <= 0.0 ) fim_thr[lp] = 0 ; for( ii=0 ; ii < nvox ; ii++ ){ if( qar[ii] < thr || sar[ii] == 0 ){ ovar[ii] = 0 ; } else if ( pbar->bigmode ) { /* since 0 is considered blank, use 1 through 127 */ bindex = (int)( (btop - sar[ii])/bdelta + 1); RANGE(bindex,1,127); /* was NPANE_BIG - 1 */ if ( CRBM_IS_BLACK_INDEX(bindex) ) ovar[ii] = 0; else ovar[ii] = bindex; } else { for( lp=0 ; lp < num_lp && sar[ii] < fim_thr[lp] ; lp++ ) ; /*nada*/ ovar[ii] = fim_ovc[lp] ; } } } break ; } } /*----- if ordered, remove clusters -----*/ if( func_kill_clusters ){ int nx=ovim->nx , ny=ovim->ny , nz=ovim->nz , ptmin,iclu ; float dx = fabs(local_dset->daxes->xxdel) , dy = fabs(local_dset->daxes->yydel) , dz = fabs(local_dset->daxes->zzdel) ; float rmm = wfunc_clusters_rmm_av->fval , vmul = wfunc_clusters_vmul_av->fval ; MCW_cluster_array * clar ; MCW_cluster * cl ; if( (rmm >= dx || rmm >= dy || rmm >= dz) && vmul > (dx*dy*dz) ){ ptmin = vmul / (dx*dy*dz) + 0.99 ; clar = MCW_find_clusters( nx,ny,nz , dx,dy,dz , MRI_byte,ovar , rmm ) ; if( clar != NULL ){ for( iclu=0 ; iclu < clar->num_clu ; iclu++ ){ cl = clar->clar[iclu] ; if( cl->num_pt >= ptmin ) /* put back into array */ MCW_cluster_to_vol( nx,ny,nz , MRI_byte,ovar , cl ) ; } DESTROY_CLARR(clar) ; } } } /* end of cluster removal */ /*----- other overlay stuff -----*/ EndOfFuncOverlay: if( func_see_ttatlas ) RCREND_overlay_ttatlas() ; /* 12 July 2001 */ new_fset = 0; /* 28 June 2002 */ EXRETURN ; } /*----------------------------------------------------------------------- Overlay regions from the Talairach Daemon database, if possible -------------------------------------------------------------------------*/ #define HEMX 80 /* 1/2 the brain, in the x-direction */ #define ALLX (2*HEMX+1) /* all the brain, in the x-direction */ void RCREND_overlay_ttatlas(void) { TTRR_params *ttp ; THD_3dim_dataset *dseTT ; byte *b0 , *b1 , *ovar ; int nvox , ii,jj , xx ; int fwin , gwin , nreg , hemi,hbot ; byte *brik , *val , *ovc , g_ov , a_ov , final_ov ; ENTRY( "RCREND_overlay_ttatlas" ); /* sanity checks and setup */ if( ovim == NULL ) EXRETURN ; nvox = ovim->nvox ; #if 0 # define RET(s) do{fprintf(stderr,s);return;}while(0) #else # define RET(s) EXRETURN #endif /* 01 Aug 2001: retrieve Atlas dataset depending on size of brick */ #if 1 dseTT = TT_retrieve_atlas_nz(ovim->nz) ; if( dseTT == NULL ) RET("no dataset\n") ; #else dseTT = TT_retrieve_atlas() ; if( dseTT == NULL ) RET("no dataset\n") ; #endif if( DSET_NVOX(dseTT) != nvox ) RET("dataset mismatch\n"); ttp = TTRR_get_params() ; if( ttp == NULL ) RET("no ttp\n") ; DSET_load(dseTT) ; b0 = DSET_ARRAY(dseTT,0) ; b1 = DSET_ARRAY(dseTT,1) ; if( b0 == NULL || b1 == NULL ) RET("no bricks\n") ; ovar = MRI_BYTE_PTR(ovim) ; fwin = (ttp->meth == TTRR_METH_FGA) || (ttp->meth == TTRR_METH_FAG) ; gwin = (ttp->meth == TTRR_METH_FGA) || (ttp->meth == TTRR_METH_GAF) ; nreg = ttp->num ; brik = ttp->ttbrik ; val = ttp->ttval ; ovc = ttp->ttovc ; hemi = ttp->hemi ; switch( hemi ){ case TTRR_HEMI_LEFT: hbot=HEMX+1 ; break ; case TTRR_HEMI_RIGHT: hbot= 0 ; break ; case TTRR_HEMI_BOTH: hbot= 0 ; break ; } /* ready to do something */ for( xx=0,ii=hbot ; ii < nvox ; ii++ ){ if( hemi != TTRR_HEMI_BOTH ){ if( xx == HEMX ){ xx = 0 ; ii += HEMX ; continue ; /* skip ahead 1/2 row */ } xx++ ; } if( ovar[ii] && fwin ) continue ; /* function wins */ /* check atlas dataset for hits */ g_ov = a_ov = 0 ; for( jj=0 ; (g_ov==0 || a_ov==0) && jjvinfo->xi; yj = im3d->vinfo->yj; zk = im3d->vinfo->zk; nx = xovim->nx; ny = xovim->ny; nxy = nx * ny; nz = xovim->nz; /* convert to ijk in mset rickr 2002.08.05 */ LOAD_FVEC3( fxyz, xi, yj, zk ); /* Dicom coords for dset */ fxyz = THD_dicomm_to_3dmm(mset, fxyz); /* mm coords for mset */ ixyz = THD_3dmm_to_3dind (mset, fxyz); /* ijk coords for mset */ UNLOAD_IVEC3( ixyz, ix, jy, kz ); om = im3d->vinfo->xhairs_orimask ; /* 02 Jun 1999 */ if( ix < 0 || ix >= nx ) EXRETURN ; /* error */ if( jy < 0 || jy >= ny ) EXRETURN ; /* error */ if( kz < 0 || kz >= nz ) EXRETURN ; /* error */ gap = im3d->vinfo->crosshair_gap ; ovar = MRI_BYTE_PTR(xovim) ; /* 02 Jun 1999: allow for partial crosshair drawing */ if( (om & ORIMASK_LR) != 0 ){ for( ii=0 ; ii < nx ; ii++ ){ if( abs(ii-ix) > gap ){ OV(ii,jy,kz) = gxh ; } } } if( (om & ORIMASK_AP) != 0 ){ for( ii=0 ; ii < ny ; ii++ ){ if( abs(ii-jy) > gap ){ OV(ix,ii,kz) = gxh ; } } } if( (om & ORIMASK_IS) != 0 ){ for( ii=0 ; ii < nz ; ii++ ){ if( abs(ii-kz) > gap ){ OV(ix,jy,ii) = gxh ; } } } xhair_ixold = ix ; xhair_jyold = jy ; xhair_kzold = kz ; /* memory */ xhair_omold = om ; /* 02 Jun 1999 */ EXRETURN ; } /*--------------------------------------------------------------------------- Callback for opacity scale factor selection -----------------------------------------------------------------------------*/ void RCREND_opacity_scale_CB( MCW_arrowval * av , XtPointer cd ) { ENTRY( "RCREND_opacity_scale_CB" ); if( av->fval < MIN_OPACITY_SCALE ) AV_assign_fval(av,MIN_OPACITY_SCALE) ; if( cd == NULL && dynamic_flag && gcr.rh != NULL ) RCREND_draw_CB(NULL,NULL,NULL) ; EXRETURN ; } /***************************************************************************** Functions for saving the internal rendering state to a script, reading it back in, et cetera. ******************************************************************************/ #ifdef USE_SCRIPTING /*-------------------------------------------------------------------- 07 July 1999: Create the widgets for the script control menu ----------------------------------------------------------------------*/ void RCREND_script_menu( Widget parent ) { Widget rc , mbar ; static char * load_bbox_label[1] = { "Load Widgets" } ; static char * brindex_bbox_label[1] = { "Brick Index?" } ; #ifdef SCRIPT_GRAFS static char * graf_bbox_label[1] = { "Alter Grafs?" } ; #endif static char * dset_bbox_label[1] = { "Alter Dsets?" } ; ENTRY( "RCREND_script_menu" ); rc = XtVaCreateWidget( "dialog" , xmRowColumnWidgetClass , parent , XmNorientation , XmHORIZONTAL , XmNpacking , XmPACK_TIGHT , XmNtraversalOn , True , XmNinitialResourcesPersistent , False , NULL ) ; mbar = XmCreateMenuBar( rc , "dialog" , NULL,0 ) ; XtVaSetValues( mbar , XmNmarginWidth , 0 , XmNmarginHeight , 0 , XmNspacing , 3 , XmNborderWidth , 0 , XmNborderColor , 0 , XmNtraversalOn , True , XmNbackground , im3d->dc->ovc->pixov_brightest , NULL ) ; XtManageChild( mbar ) ; script_menu = XmCreatePulldownMenu( mbar , "menu" , NULL,0 ) ; VISIBILIZE_WHEN_MAPPED(script_menu) ; TEAROFFIZE(script_menu) ; script_cbut = XtVaCreateManagedWidget( "dialog" , xmCascadeButtonWidgetClass , mbar , LABEL_ARG("Scripts") , XmNsubMenuId , script_menu , XmNmarginWidth , 0 , XmNmarginHeight , 0 , XmNmarginBottom , 0 , XmNmarginTop , 0 , XmNmarginRight , 0 , XmNmarginLeft , 0 , XmNtraversalOn , True , XmNinitialResourcesPersistent , False , NULL ) ; #undef MENU_SLINE #define MENU_SLINE \ (void) XtVaCreateManagedWidget( \ "dialog" , xmSeparatorWidgetClass , script_menu , \ XmNseparatorType , XmSINGLE_LINE , NULL ) /* macro to create a new script menu button */ #define SCRIPT_MENU_BUT(wname,label) \ wname = \ XtVaCreateManagedWidget( \ "dialog" , xmPushButtonWidgetClass , script_menu , \ LABEL_ARG( label ) , \ XmNmarginHeight , 0 , \ XmNtraversalOn , True , \ XmNinitialResourcesPersistent , False , \ NULL ) ; \ XtAddCallback( wname , XmNactivateCallback , \ RCREND_script_CB , NULL ) ; /*** top of menu = a label to click on that does nothing at all ***/ (void) XtVaCreateManagedWidget( "dialog" , xmLabelWidgetClass , script_menu , LABEL_ARG("-- Cancel --") , XmNrecomputeSize , False , XmNinitialResourcesPersistent , False , NULL ) ; MENU_SLINE ; SCRIPT_MENU_BUT( script_save_this_pb , "Save This" ) ; SCRIPT_MENU_BUT( script_save_many_pb , "Save Many" ) ; MENU_SLINE ; SCRIPT_MENU_BUT( script_read_this_pb , "Read This" ) ; SCRIPT_MENU_BUT( script_read_exec_pb , "Read & Exec" ) ; MENU_SLINE ; script_load_bbox = new_MCW_bbox( script_menu , 1 , load_bbox_label , MCW_BB_check , MCW_BB_noframe , RCREND_script_load_CB , NULL ) ; MCW_reghint_children( script_load_bbox->wrowcol , "Recall settings from images" ) ; script_brindex_bbox = new_MCW_bbox( script_menu , 1 , brindex_bbox_label , MCW_BB_check , MCW_BB_noframe , RCREND_script_brindex_CB , NULL ) ; MCW_reghint_children( script_brindex_bbox->wrowcol , "Set brick index when loading widgets?" ) ; #ifdef SCRIPT_GRAFS script_graf_bbox = new_MCW_bbox( script_menu , 1 , graf_bbox_label , MCW_BB_check , MCW_BB_noframe , RCREND_script_graf_CB , NULL ) ; MCW_reghint_children( script_graf_bbox->wrowcol , "Set grafs when loading widgets?" ) ; #endif #ifdef SCRIPT_DSETS /* 12 Apr 2000 - toggle button for dataset changing via scripts */ script_dset_bbox = new_MCW_bbox( script_menu , 1 , dset_bbox_label , MCW_BB_check , MCW_BB_noframe , RCREND_script_dset_CB , NULL ) ; MCW_reghint_children( script_dset_bbox->wrowcol , "Change datasets when loading widgets?" ) ; #endif XtManageChild( rc ) ; EXRETURN ; } #ifdef SCRIPT_DSETS /*----------------------------------------------------------------------- Callback when the "Alter Dsets?" button is toggled -------------------------------------------------------------------------*/ void RCREND_script_dset_CB( Widget w , XtPointer cd , XtPointer cbs ) { ENTRY( "RCREND_script_dset_CB" ); script_dsetchange = MCW_val_bbox( script_dset_bbox ) ; EXRETURN ; } #endif #ifdef SCRIPT_GRAFS /*----------------------------------------------------------------------- Callback when the "Alter Grafs?" button is toggled -------------------------------------------------------------------------*/ void RCREND_script_graf_CB( Widget w , XtPointer cd , XtPointer cbs ) { ENTRY( "RCREND_script_graf_CB" ); script_graf = MCW_val_bbox( script_graf_bbox ) ; EXRETURN ; } #endif /*----------------------------------------------------------------------- Callback when the "Brick Index?" button is toggled -------------------------------------------------------------------------*/ void RCREND_script_brindex_CB( Widget w , XtPointer cd , XtPointer cbs ) { ENTRY( "RCREND_script_brindex_CB" ); script_brindex = MCW_val_bbox( script_brindex_bbox ) ; EXRETURN ; } /*----------------------------------------------------------------------- Callback when the "Load Widgets" button is toggled -------------------------------------------------------------------------*/ void RCREND_script_load_CB( Widget w , XtPointer cd , XtPointer cbs ) { int sl = MCW_val_bbox( script_load_bbox ) ; ENTRY( "RCREND_script_load_CB" ); if( sl == script_load ) EXRETURN ; /* no change? */ script_load = sl ; script_load_last = -1 ; if( script_load && imseq != NULL && renderings_state != NULL ){ int nn ; drive_MCW_imseq( imseq , isqDR_getimnr , (XtPointer) &nn ) ; if( nn >= 0 && nn < RSA_COUNT(renderings_state) ){ RCREND_state_to_widgets( RSA_SUBSTATE(renderings_state,nn) ) ; script_load_last = nn ; } } else if( !script_load && last_rendered_state != NULL ){ RCREND_state_to_widgets( last_rendered_state ) ; } EXRETURN ; } /*---------------------------------------------------------------------- Callback when a script menu button is pressed ------------------------------------------------------------------------*/ static char script_read_fname[THD_MAX_NAME] = "\0" ; void RCREND_script_CB( Widget w , XtPointer cd , XtPointer cbs ) { ENTRY( "RCREND_script_CB" ); if( w == script_save_this_pb ){ MCW_choose_string( w , "[Save This] Filename prefix:" , NULL , RCREND_save_this_CB , NULL ) ; EXRETURN ; } if( w == script_read_this_pb ){ MCW_choose_string( w , "[Read This] Filename prefix:" , script_read_fname , RCREND_read_this_CB , NULL ) ; EXRETURN ; } if( w == script_save_many_pb ){ if( renderings_state == NULL || RSA_COUNT(renderings_state) < 1 ){ (void) MCW_popup_message( script_cbut , " \n" "** No rendering states\n" "** available to save!\n" , MCW_USER_KILL | MCW_TIMER_KILL ) ; PLUTO_beep() ; EXRETURN ; } MCW_choose_string( w , "[Save Many] Filename prefix:" , NULL , RCREND_save_many_CB , NULL ) ; EXRETURN ; } if( w == script_read_exec_pb ){ #ifdef SCRIPT_DSETS if( dset == NULL && script_dsetchange == 0 ) #else if( dset == NULL ) #endif { (void) MCW_popup_message( script_cbut , " \n" "** No dataset loaded\n" "** for rendering!\n" , MCW_USER_KILL | MCW_TIMER_KILL ) ; PLUTO_beep() ; EXRETURN ; } MCW_choose_string( w , "[Read & Exec] Filename prefix:" , script_read_fname , RCREND_read_exec_CB , NULL ) ; EXRETURN ; } /*-- should never be reached --*/ PLUTO_beep() ; EXRETURN ; } /*---------------------------------------------------------------------- Called when the "Save This" filename chooser is activated ------------------------------------------------------------------------*/ void RCREND_save_this_CB( Widget w , XtPointer cd , MCW_choose_cbs * cbs ) { int ll ; char * fname , buf[256] , * sbuf ; RENDER_state rs ; FILE * fp ; ENTRY( "RCREND_save_this_CB" ); if( !renderer_open ){ POPDOWN_string_chooser ; EXRETURN ; } if( cbs->reason != mcwCR_string || cbs->cval == NULL || (ll = strlen(cbs->cval)) == 0 ){ PLUTO_beep() ; EXRETURN ; } fname = malloc( sizeof(char) * (ll+8) ) ; strcpy(fname,cbs->cval) ; if( strstr(fname,".rset") == NULL ){ if( fname[ll-1] != '.' ){ fname[ll++] = '.'; fname[ll] = '\0'; } strcat(fname,"rset") ; } if( !THD_filename_ok(fname) ){ sprintf(buf," \n" "** Filename %s is illegal!\n" "** Try something different.\n" , fname ) ; (void) MCW_popup_message( script_cbut , buf , MCW_USER_KILL | MCW_TIMER_KILL ) ; free(fname) ; PLUTO_beep() ; EXRETURN ; } if( THD_is_file(fname) ){ sprintf(buf," \n" "** File %s already exists!\n" "** AFNI won't overwrite it.\n" , fname ) ; (void) MCW_popup_message( script_cbut , buf , MCW_USER_KILL | MCW_TIMER_KILL ) ; free(fname) ; PLUTO_beep() ; EXRETURN ; } RCREND_widgets_to_state( &rs ) ; sbuf = RCREND_save_state( &rs , NULL ) ; if( sbuf == NULL ){ (void) MCW_popup_message( script_cbut , "\n" "** Some error occured when\n" "** trying to save the state!\n" , MCW_USER_KILL | MCW_TIMER_KILL ) ; free(fname) ; PLUTO_beep() ; EXRETURN ; } fp = fopen( fname , "w" ) ; if( fp == NULL ){ (void) MCW_popup_message( script_cbut , "\n" "** Some error occured when\n" "** trying to open the file!\n" , MCW_USER_KILL | MCW_TIMER_KILL ) ; free(sbuf) ; free(fname) ; PLUTO_beep() ; EXRETURN ; } POPDOWN_string_chooser ; fwrite( sbuf , 1 , strlen(sbuf) , fp ) ; fclose( fp ) ; free( sbuf ) ; free(fname) ; EXRETURN ; } /*---------------------------------------------------------------------- Called when the "Read This" filename chooser is activated ------------------------------------------------------------------------*/ void RCREND_read_this_CB( Widget w , XtPointer cd , MCW_choose_cbs * cbs ) { int ll ; char * fname , buf[256] ; RENDER_state rs ; RENDER_state_array * rsa ; ENTRY( "RCREND_read_this_CB" ); if( !renderer_open ){ POPDOWN_string_chooser ; EXRETURN ; } if( cbs->reason != mcwCR_string || cbs->cval == NULL || (ll = strlen(cbs->cval)) == 0 ){ PLUTO_beep() ; EXRETURN ; } fname = malloc( sizeof(char) * (ll+8) ) ; strcpy(fname,cbs->cval) ; strcpy(script_read_fname,fname) ; if( strstr(fname,".rset") == NULL ){ if( fname[ll-1] != '.' ){ fname[ll++] = '.'; fname[ll] = '\0'; } strcat(fname,"rset") ; } RCREND_widgets_to_state( &rs ) ; rsa = RCREND_read_states( fname , &rs ) ; if( rsa == NULL || RSA_COUNT(rsa) < 1 ){ sprintf(buf, "\n" "** Some error occured when\n" "** trying to read file %s\n" , fname ) ; (void) MCW_popup_message( script_cbut , buf , MCW_USER_KILL | MCW_TIMER_KILL ) ; free(fname) ; PLUTO_beep() ; EXRETURN ; } free(fname) ; POPDOWN_string_chooser ; if( RSA_COUNT(rsa) == 1 ){ MCW_choose_cbs cbs ; cbs.ival = 0 ; cbs.reason = mcwCR_integer ; RCREND_read_this_finalize_CB( NULL , (XtPointer) rsa , &cbs ) ; } else { MCW_choose_integer( w , "[Read This] State Index" , 0 , RSA_COUNT(rsa)-1 , 0 , RCREND_read_this_finalize_CB , (XtPointer) rsa ) ; } EXRETURN ; } void RCREND_read_this_finalize_CB( Widget w , XtPointer cd , MCW_choose_cbs * cbs ) { RENDER_state_array * rsa = (RENDER_state_array *) cd ; ENTRY( "RCREND_read_this_finalize_CB" ); POPDOWN_integer_chooser ; if( cbs->reason != mcwCR_integer || cbs->ival < 0 || cbs->ival >= RSA_COUNT(rsa) ){ PLUTO_beep() ; EXRETURN ; } RCREND_state_to_widgets( RSA_SUBSTATE(rsa,cbs->ival) ) ; DESTROY_RSA(rsa) ; EXRETURN ; } /*---------------------------------------------------------------------- Called when the "Save Many" filename chooser is activated ------------------------------------------------------------------------*/ void RCREND_save_many_CB( Widget w , XtPointer cd , MCW_choose_cbs * cbs ) { int ll , ii ; char * fname , buf[256] , * sbuf ; RENDER_state * rs ; FILE * fp ; ENTRY( "RCREND_save_many_CB" ); if( !renderer_open || renderings_state == NULL || RSA_COUNT(renderings_state) < 1 ){ POPDOWN_string_chooser ; EXRETURN ; } if( cbs->reason != mcwCR_string || cbs->cval == NULL || (ll = strlen(cbs->cval)) == 0 ){ PLUTO_beep() ; EXRETURN ; } fname = malloc( sizeof(char) * (ll+8) ) ; strcpy(fname,cbs->cval) ; if( strstr(fname,".rset") == NULL ){ if( fname[ll-1] != '.' ){ fname[ll++] = '.'; fname[ll] = '\0'; } strcat(fname,"rset") ; } if( !THD_filename_ok(fname) ){ sprintf(buf," \n" "** Filename %s is illegal!\n" "** Try something different.\n" , fname ) ; (void) MCW_popup_message( script_cbut , buf , MCW_USER_KILL | MCW_TIMER_KILL ) ; free(fname) ; PLUTO_beep() ; EXRETURN ; } if( THD_is_file(fname) ){ sprintf(buf," \n" "** File %s already exists!\n" "** AFNI won't overwrite it.\n" , fname ) ; (void) MCW_popup_message( script_cbut , buf , MCW_USER_KILL | MCW_TIMER_KILL ) ; free(fname) ; PLUTO_beep() ; EXRETURN ; } fp = fopen( fname , "w" ) ; if( fp == NULL ){ sprintf(buf, " \n" "** Some error occured when\n" "** trying to open file %s\n" , fname ) ; (void) MCW_popup_message( script_cbut , buf , MCW_USER_KILL | MCW_TIMER_KILL ) ; free(fname) ; PLUTO_beep() ; EXRETURN ; } free(fname) ; POPDOWN_string_chooser ; rs = NULL ; for( ii=0 ; ii < RSA_COUNT(renderings_state) ; ii++ ){ sbuf = RCREND_save_state( RSA_SUBSTATE(renderings_state,ii) , rs ) ; fwrite( sbuf , 1 , strlen(sbuf) , fp ) ; free(sbuf) ; rs = RSA_SUBSTATE(renderings_state,ii) ; } fclose( fp ) ; EXRETURN; } /*---------------------------------------------------------------------- Called when the "Read & Exec" filename chooser is activated ------------------------------------------------------------------------*/ void RCREND_read_exec_CB( Widget w , XtPointer cd , MCW_choose_cbs * cbs ) { int ll , it , ntime ; char * fname , buf[256] ; RENDER_state rs ; RENDER_state_array * rsa ; float scl ; Widget autometer ; ENTRY( "RCREND_read_exec_CB" ); if( !renderer_open ){ POPDOWN_string_chooser ; EXRETURN ; } if( cbs->reason != mcwCR_string || cbs->cval == NULL || (ll = strlen(cbs->cval)) == 0 ){ PLUTO_beep() ; EXRETURN ; } fname = malloc( sizeof(char) * (ll+8) ) ; strcpy(fname,cbs->cval) ; strcpy(script_read_fname,fname) ; if( strstr(fname,".rset") == NULL ){ if( fname[ll-1] != '.' ){ fname[ll++] = '.'; fname[ll] = '\0'; } strcat(fname,"rset") ; } RCREND_widgets_to_state( &rs ) ; rsa = RCREND_read_states( fname , &rs ) ; if( rsa == NULL || RSA_COUNT(rsa) < 1 ){ sprintf(buf, "\n" "** Some error occured when\n" "** trying to read file %s\n" , fname ) ; (void) MCW_popup_message( script_cbut , buf , MCW_USER_KILL | MCW_TIMER_KILL ) ; free(fname) ; PLUTO_beep() ; EXRETURN ; } free(fname) ; POPDOWN_string_chooser ; /*-- now execute the renderings (a la 'Automate' )--*/ automate_flag = 1 ; if( ! accum_flag ){ DESTROY_IMARR(renderings) ; DESTROY_RSA(renderings_state) ; } ntime = RSA_COUNT(rsa) ; if( ntime > 1 ){ autometer = MCW_popup_meter( shell , METER_TOP_WIDE ) ; XtManageChild( autocancel_pb ) ; AFNI_add_interruptable( autocancel_pb ) ; autokill = 0 ; scl = 100.0/ntime ; } for( it=0 ; it < ntime ; it++ ){ RCREND_state_to_widgets( RSA_SUBSTATE(rsa,it) ) ; if( dset == NULL ) break ; /* some error */ RCREND_draw_CB(NULL,NULL,NULL) ; if( it < ntime-1 ){ AFNI_process_interrupts(autocancel_pb) ; if( autokill ) break ; } if( ntime > 1 ) MCW_set_meter( autometer , (int)(scl*(it+1)) ) ; } /*-- done: cleanup time --*/ DESTROY_RSA(rsa) ; if( ntime > 1 ){ MCW_popdown_meter( autometer ) ; XtUnmanageChild( autocancel_pb ) ; AFNI_add_interruptable(NULL) ; } automate_flag = 0 ; EXRETURN ; } /*-------------------------------------------------------------------------- Read a file and return an array of rendering states. Code is adapted from afni_setup.c ----------------------------------------------------------------------------*/ #define ISTARRED(s) ( (s)[0]=='*' && (s)[1]=='*' && (s)[2]=='*' ) #define EOLSKIP \ do{ for( ; fptr[0] != '\n' && fptr[0] != '\0' ; fptr++ ) ; /* nada */ \ if( fptr[0] == '\0' ) goto Finished ; \ fptr++ ; } while(0) #define GETSSS \ do{ int nu=0,qq; \ if( fptr-fbuf >= nbuf || fptr[0] == '\0' ) goto Finished ; \ str[0]='\0'; qq=sscanf(fptr,"%127s%n",str,&nu); nused+=nu;fptr+=nu; \ if( str[0]=='\0' || qq==0 || nu==0 ) goto Finished ; \ } while(0) #define GETSTR \ do{ GETSSS ; \ while(str[0]=='!' || (str[0]=='/' && str[1]=='/') || \ (str[0]=='#' && str[1]=='\0') ){EOLSKIP; GETSSS;} \ } while(0) #define GETEQN \ do{ GETSTR ; if(ISTARRED(str)) goto SkipSection ; \ strcpy(left,str) ; \ GETSTR ; if(ISTARRED(str)) goto SkipSection ; \ strcpy(middle,str) ; \ GETSTR ; if(ISTARRED(str)) goto SkipSection ; \ strcpy(right,str) ; } while(0) #define NSBUF 256 /*--------------------------------------------------------------------------*/ RENDER_state_array * RCREND_read_states( char * fname , RENDER_state * rsbase ) { int nbuf , nused ; char * fbuf , * fptr ; char str[NSBUF] , left[NSBUF] , middle[NSBUF] , right[NSBUF] ; int ival ; float fval ; RENDER_state * rs ; RENDER_state_array * rsa = NULL ; ENTRY( "RCREND_read_states" ); /* setup & sanity checks */ fbuf = AFNI_suck_file( fname ) ; if( fbuf == NULL ) RETURN(NULL); nbuf = strlen(fbuf) ; fptr = fbuf ; nused = 0 ; /** scan for section strings, which start with "***" **/ str[0] = '\0' ; /* initialize string */ /**----------------------------------------**/ /**-- skip ahead to next section keyword --**/ SkipSection: while( ! ISTARRED(str) ){ GETSTR; } if( strcmp(str,"***END") == 0 ) goto Finished ; /*-- the only thing we like are ***RENDER sections --*/ if( strcmp(str,"***RENDER") != 0 ) goto SkipSection ; if( rsa == NULL ){ INIT_RSA(rsa) ; } /* create the output array */ rs = (RENDER_state *) calloc( 1,sizeof(RENDER_state) ) ; /* create the new state */ if( RSA_COUNT(rsa) == 0 && rsbase != NULL ){ *rs = *rsbase ; /* copy base state */ } else if( RSA_COUNT(rsa) > 0 ){ *rs = *(RSA_SUBSTATE(rsa,RSA_COUNT(rsa)-1)) ; /* copy previous state */ } ADDTO_RSA(rsa,rs) ; /* put new state in output array */ /*--- Scan for rendering variable assignments ---*/ #undef ASS_IVAL #define ASS_IVAL(a,b,c) { if( ival >= b && ival <= c ) a = ival ; } #undef ASS_FVAL #define ASS_FVAL(a,b,c) { if( fval >= b && fval <= c ) a = fval ; } while(1){ /* loop, looking for 'name = value' */ GETEQN ; /* loop exits when this fails */ /*-- dataset stuff --*/ if( strcmp(left,"dset_name") == 0 ){ MCW_strncpy(rs->dset_name,right,THD_MAX_NAME) ; #if 0 ZERO_IDCODE(rs->dset_idc) ; #endif continue ; /* the while(1) loop */ } if( strcmp(left,"func_dset_name") == 0 ){ MCW_strncpy(rs->func_dset_name,right,THD_MAX_NAME) ; #if 0 ZERO_IDCODE(rs->func_dset_idc) ; #endif continue ; } if( strcmp(left,"dset_idc") == 0 ){ MCW_strncpy(rs->dset_idc.str,right,MCW_IDSIZE) ; #if 0 rs->dset_name[0] = '\0' ; #endif continue ; } if( strcmp(left,"func_dset_idc") == 0 ){ MCW_strncpy(rs->func_dset_idc.str,right,MCW_IDSIZE) ; #if 0 rs->func_dset_name[0] = '\0' ; #endif continue ; } /*-- cutout stuff --*/ if( strcmp(left,"cutout_num") == 0 ){ ival = strtol(right,NULL,10) ; ASS_IVAL( rs->current_cutout_state.num , 0 , MAX_CUTOUTS ) ; continue ; } if( strcmp(left,"cutout_logic") == 0 ){ if( strcmp(right,"AND")==0 || strcmp(right,"and")==0 || strcmp(right,"And")==0 ) rs->current_cutout_state.logic = CUTOUT_AND ; else if( strcmp(right,"OR")==0 || strcmp(right,"or")==0 || strcmp(right,"Or")==0 ) rs->current_cutout_state.logic = CUTOUT_OR ; continue ; } if( strcmp(left,"opacity_scale") == 0 ){ fval = strtod(right,NULL) ; ASS_FVAL( rs->current_cutout_state.opacity_scale , MIN_OPACITY_SCALE , 1.0 ) ; continue ; } #define ASS_CUT_TYPE(nnn) \ if( strcmp(right,#nnn) == 0 ){ rs->current_cutout_state.type[iii] = nnn; continue;} if( strncmp(left,"cutout_type",strlen("cutout_type")) == 0 ){ char * srb = strstr(left,"[") ; if( srb != NULL ){ int iii = strtol(srb+1,NULL,10) ; if( iii >= 0 && iii < MAX_CUTOUTS ){ if( isdigit(right[0]) ){ ival = strtol(right,NULL,10) ; if( ival >= 0 && ival < NUM_CUTOUT_TYPES && ival != CUT_EXPRESSION ) rs->current_cutout_state.type[iii] = ival ; } else { ASS_CUT_TYPE(CUT_NONE) ; ASS_CUT_TYPE(CUT_RIGHT_OF) ; ASS_CUT_TYPE(CUT_LEFT_OF) ; ASS_CUT_TYPE(CUT_ANTERIOR_TO) ; ASS_CUT_TYPE(CUT_POSTERIOR_TO) ; ASS_CUT_TYPE(CUT_INFERIOR_TO) ; ASS_CUT_TYPE(CUT_SUPERIOR_TO) ; ASS_CUT_TYPE(CUT_TT_ELLIPSOID) ; ASS_CUT_TYPE(CUT_SLANT_XPY_GT) ; ASS_CUT_TYPE(CUT_SLANT_XPY_LT) ; ASS_CUT_TYPE(CUT_SLANT_XMY_GT) ; ASS_CUT_TYPE(CUT_SLANT_XMY_LT) ; ASS_CUT_TYPE(CUT_SLANT_YPZ_GT) ; ASS_CUT_TYPE(CUT_SLANT_YPZ_LT) ; ASS_CUT_TYPE(CUT_SLANT_YMZ_GT) ; ASS_CUT_TYPE(CUT_SLANT_YMZ_LT) ; ASS_CUT_TYPE(CUT_SLANT_XPZ_GT) ; ASS_CUT_TYPE(CUT_SLANT_XPZ_LT) ; ASS_CUT_TYPE(CUT_SLANT_XMZ_GT) ; ASS_CUT_TYPE(CUT_SLANT_XMZ_LT) ; ASS_CUT_TYPE(CUT_NONOVERLAY) ; } } } continue ; } if( strncmp(left,"cutout_mustdo",strlen("cutout_mustdo")) == 0 ){ char * srb = strstr(left,"[") ; if( srb != NULL ){ int iii = strtol(srb+1,NULL,10) ; if( iii >= 0 && iii < MAX_CUTOUTS ){ if( strcmp(right,"TRUE") == 0 || strcmp(right,"true") == 0 || strcmp(right,"True") == 0 || strcmp(right,"YES") == 0 || strcmp(right,"yes") == 0 || strcmp(right,"Yes") == 0 || strcmp(right,"1") == 0 ) rs->current_cutout_state.mustdo[iii] = 1 ; else if( strcmp(right,"FALSE") == 0 || strcmp(right,"false") == 0 || strcmp(right,"False") == 0 || strcmp(right,"NO") == 0 || strcmp(right,"no") == 0 || strcmp(right,"No") == 0 || strcmp(right,"0") == 0 ) rs->current_cutout_state.mustdo[iii] = 0 ; } } continue ; } if( strncmp(left,"cutout_param",strlen("cutout_param")) == 0 ){ char * srb = strstr(left,"[") ; if( srb != NULL ){ int iii = strtol(srb+1,NULL,10) ; if( iii >= 0 && iii < MAX_CUTOUTS ){ rs->current_cutout_state.param[iii] = strtod(right,NULL) ; } } continue ; } /*-- all other desiderata --*/ #define ASS_INT(nnn) if( strcmp(left,#nnn) == 0 ){ \ rs -> nnn = strtol(right,NULL,10) ; \ continue ; \ } #define ASS_FLOAT(nnn) if( strcmp(left,#nnn) == 0 ){ \ rs -> nnn = strtod(right,NULL) ; \ continue ; \ } #define ASS_FLOAT_SUB(nnn,mmm) \ if( strncmp(left,#nnn,strlen(#nnn)) == 0 ){ \ char * srb = strstr(left,"[") ; \ if( srb != NULL ){ \ int iii = strtol(srb+1,NULL,10) ; \ if( iii >= 0 && iii < mmm ) rs->nnn[iii] = strtod(right,NULL) ; \ } \ continue ; \ } ASS_INT(dset_ival) ; ASS_INT(func_color_ival) ; ASS_INT(func_thresh_ival) ; ASS_INT(clipbot) ; ASS_INT(cliptop) ; ASS_FLOAT(angle_roll) ; ASS_FLOAT(angle_pitch) ; ASS_FLOAT(angle_yaw) ; ASS_INT(xhair_flag) ; ASS_INT(xhair_ovc) ; /* 08 Mar 2001 */ ASS_INT( func_use_autorange ) ; ASS_FLOAT( func_threshold ) ; ASS_FLOAT( func_thresh_top ) ; ASS_FLOAT( func_color_opacity ) ; ASS_FLOAT( func_showthru_fac ) ; /* 08 Mar 2002 */ ASS_INT( func_showthru ) ; /* 08 Mar 2002 */ ASS_INT( func_see_overlay ) ; ASS_INT( func_see_ttatlas ) ; /* 24 Jul 2001 */ ASS_INT( func_cut_overlay ) ; ASS_INT( func_kill_clusters ) ; ASS_FLOAT( func_clusters_rmm ) ; ASS_FLOAT( func_clusters_vmul ) ; ASS_FLOAT( func_range ) ; ASS_INT( pbar_mode ) ; ASS_INT( pbar_npane ) ; ASS_FLOAT_SUB(pbar_pval,NPANE_MAX+1) ; #ifdef SCRIPT_GRAFS /*-- read graf stuff --*/ if( strcmp(left,"bright_nhands") == 0 ){ ival = strtol(right,NULL,10) ; if( ival > 1 && ival <= MAX_GHANDS ) rs->bright_graf_state.nh = ival ; continue ; } if( strcmp(left,"bright_spline") == 0 ){ rs->bright_graf_state.spl = strtol(right,NULL,10) ; continue ; } if( strncmp(left,"bright_handx",strlen("bright_handx")) == 0 ){ char * srb = strstr(left,"[") ; if( srb != NULL ){ int iii = strtol(srb+1,NULL,10) ; if( iii >= 0 && iii < MAX_GHANDS ){ rs->bright_graf_state.xh[iii] = strtol(right,NULL,10) ; } } continue ; } if( strncmp(left,"bright_handy",strlen("bright_handy")) == 0 ){ char * srb = strstr(left,"[") ; if( srb != NULL ){ int iii = strtol(srb+1,NULL,10) ; if( iii >= 0 && iii < MAX_GHANDS ){ rs->bright_graf_state.yh[iii] = strtol(right,NULL,10) ; } } continue ; } if( strcmp(left,"opacity_nhands") == 0 ){ ival = strtol(right,NULL,10) ; if( ival > 1 && ival <= MAX_GHANDS ) rs->opacity_graf_state.nh = ival ; continue ; } if( strcmp(left,"opacity_spline") == 0 ){ rs->opacity_graf_state.spl = strtol(right,NULL,10) ; continue ; } if( strncmp(left,"opacity_handx",strlen("opacity_handx")) == 0 ){ char * srb = strstr(left,"[") ; if( srb != NULL ){ int iii = strtol(srb+1,NULL,10) ; if( iii >= 0 && iii < MAX_GHANDS ){ rs->opacity_graf_state.xh[iii] = strtol(right,NULL,10) ; } } continue ; } if( strncmp(left,"opacity_handy",strlen("opacity_handy")) == 0 ){ char * srb = strstr(left,"[") ; if( srb != NULL ){ int iii = strtol(srb+1,NULL,10) ; if( iii >= 0 && iii < MAX_GHANDS ){ rs->opacity_graf_state.yh[iii] = strtol(right,NULL,10) ; } } continue ; } #endif /* SCRIPT_GRAFS */ } /* end of loop over "equations" in the ***RENDER section */ /* loop exits when GETEQN fails miserably (i.e., at EOF) */ Finished: free(fbuf) ; if( rsa != NULL && RSA_COUNT(rsa) == 0 ){ FREE_RSA(rsa) ; } RETURN(rsa); } /*------------------------------------------------------------------------------- Write a rendering state into a character buffer. (Free the return value when done with it.) ---------------------------------------------------------------------------------*/ #define RSDIFF_STR(nnn) (rsbase == NULL || strcmp(rsbase->nnn,rs->nnn) != 0 ) #define RSDIFF_NUM(nnn) (rsbase == NULL || rsbase->nnn != rs->nnn) #define RSDIFF_CUTNUM(nnn) \ (rsbase == NULL || rsbase->current_cutout_state.nnn != rs->current_cutout_state.nnn) #define RSP_STR(nnn) \ if( rs->nnn[0] != '\0' && RSDIFF_STR(nnn) ) \ sss = THD_zzprintf( sss , " " #nnn " = %s\n" , rs->nnn ) #define RSP_INT(nnn) \ if( RSDIFF_NUM(nnn) ) sss = THD_zzprintf( sss , " " #nnn " = %d\n" , rs->nnn ) #define RSP_F2C AV_format_fval /* could also be MV_format_fval */ #define RSP_FLOAT(nnn) \ if( RSDIFF_NUM(nnn) ) sss = THD_zzprintf( sss , " " #nnn " = %s\n" , RSP_F2C(rs->nnn) ) char * RCREND_save_state( RENDER_state * rs , RENDER_state * rsbase ) { char * sss ; int ii ; ENTRY( "RCREND_save_state" ); if( rs == NULL ) RETURN(NULL); sss = (char *) malloc( sizeof(char) * 32 ) ; strcpy(sss,"\n***RENDER\n") ; /* write dataset names */ RSP_STR(dset_name) ; RSP_STR(func_dset_name) ; /* write dataset ID codes [12 Apr 2000] */ if( rsbase == NULL || !EQUIV_IDCODES(rsbase->dset_idc,rs->dset_idc) ) if( !ISZERO_IDCODE(rs->dset_idc) ) sss = THD_zzprintf( sss , " dset_idc = %s\n" , rs->dset_idc.str ) ; if( rsbase == NULL || !EQUIV_IDCODES(rsbase->func_dset_idc,rs->func_dset_idc) ) if( !ISZERO_IDCODE(rs->func_dset_idc) ) sss = THD_zzprintf( sss , " func_dset_idc = %s\n" , rs->func_dset_idc.str ) ; /* scalar values */ RSP_INT(dset_ival) ; RSP_INT(func_color_ival) ; RSP_INT(func_thresh_ival) ; RSP_INT(clipbot) ; RSP_INT(cliptop) ; RSP_FLOAT(angle_roll) ; RSP_FLOAT(angle_pitch) ; RSP_FLOAT(angle_yaw) ; RSP_INT(xhair_flag) ; RSP_INT(xhair_ovc) ; /* 08 Mar 2001 */ RSP_INT( func_use_autorange ) ; RSP_FLOAT( func_threshold ) ; RSP_FLOAT( func_thresh_top ) ; RSP_FLOAT( func_color_opacity ) ; RSP_INT( func_see_overlay ) ; RSP_FLOAT( func_showthru_fac ) ; RSP_INT( func_showthru ) ; /* 08 Mar 2002 */ RSP_INT( func_cut_overlay ) ; RSP_INT( func_kill_clusters ) ; RSP_FLOAT( func_clusters_rmm ) ; RSP_FLOAT( func_clusters_vmul ) ; RSP_FLOAT( func_range ) ; RSP_INT( func_see_ttatlas ) ; /* 24 Jul 2001 */ /* pbar values [all of them if number or mode changed] */ if( rsbase == NULL || rsbase->pbar_mode != rs->pbar_mode || rsbase->pbar_npane != rs->pbar_npane ){ sss = THD_zzprintf( sss , " // new pbar values\n" ) ; sss = THD_zzprintf( sss , " pbar_mode = %d\n",rs->pbar_mode ) ; sss = THD_zzprintf( sss , " pbar_npane = %d\n",rs->pbar_npane ) ; for( ii=0 ; ii <= rs->pbar_npane ; ii++ ) sss = THD_zzprintf( sss , " pbar_pval[%d] = %s\n" , ii , RSP_F2C(rs->pbar_pval[ii]) ) ; } else { for( ii=0 ; ii <= rs->pbar_npane ; ii++ ) if( rsbase->pbar_pval[ii] != rs->pbar_pval[ii] ) sss = THD_zzprintf( sss , " pbar_pval[%d] = %s\n" , ii , RSP_F2C(rs->pbar_pval[ii]) ) ; } /* cutout stuff */ if( RSDIFF_NUM(current_cutout_state.opacity_scale) ) sss = THD_zzprintf(sss," opacity_scale = %s\n", RSP_F2C(rs->current_cutout_state.opacity_scale) ) ; /* all cutout parameters if number or global logic changed */ if( RSDIFF_NUM(current_cutout_state.num) || RSDIFF_NUM(current_cutout_state.logic) ){ sss = THD_zzprintf( sss , " // new cutout values\n" ) ; sss = THD_zzprintf( sss , " cutout_num = %d\n" , rs->current_cutout_state.num ) ; sss = THD_zzprintf( sss , " cutout_logic = %s\n" , cutout_logic_labels[rs->current_cutout_state.logic]) ; for( ii=0 ; ii < rs->current_cutout_state.num ; ii++ ){ sss = THD_zzprintf( sss , " cutout_type[%d] = %s\n" , ii , cutout_type_names[rs->current_cutout_state.type[ii]] ) ; sss = THD_zzprintf( sss , " cutout_mustdo[%d] = %s\n" , ii , cutout_mustdo_names[rs->current_cutout_state.mustdo[ii]] ) ; sss = THD_zzprintf( sss , " cutout_param[%d] = %s\n" , ii , RSP_F2C(rs->current_cutout_state.param[ii]) ) ; } } else { for( ii=0 ; ii < rs->current_cutout_state.num ; ii++ ){ if( RSDIFF_NUM(current_cutout_state.type[ii]) ) sss = THD_zzprintf( sss , " cutout_type[%d] = %s\n" , ii , cutout_type_names[rs->current_cutout_state.type[ii]] ) ; if( RSDIFF_NUM(current_cutout_state.mustdo[ii]) ) sss = THD_zzprintf( sss , " cutout_mustdo[%d] = %s\n" , ii , cutout_mustdo_names[rs->current_cutout_state.mustdo[ii]] ) ; if( RSDIFF_NUM(current_cutout_state.param[ii]) ) sss = THD_zzprintf( sss , " cutout_param[%d] = %s\n" , ii , RSP_F2C(rs->current_cutout_state.param[ii]) ) ; } } #ifdef SCRIPT_GRAFS /*-- write graf stuff --*/ if( rsbase == NULL || !graf_states_equal(&(rsbase->bright_graf_state),&(rs->bright_graf_state)) ){ sss = THD_zzprintf( sss , " // new bright graf values\n" ) ; sss = THD_zzprintf( sss , " bright_nhands = %d\n" , rs->bright_graf_state.nh ) ; sss = THD_zzprintf( sss , " bright_spline = %d\n" , rs->bright_graf_state.spl) ; for( ii=0 ; ii < rs->bright_graf_state.nh ; ii++ ){ sss = THD_zzprintf( sss , " bright_handx[%d] = %d\n" , ii , rs->bright_graf_state.xh[ii] ) ; sss = THD_zzprintf( sss , " bright_handy[%d] = %d\n" , ii , rs->bright_graf_state.yh[ii] ) ; } } if( rsbase == NULL || !graf_states_equal(&(rsbase->opacity_graf_state),&(rs->opacity_graf_state)) ){ sss = THD_zzprintf( sss , " // new opacity graf values\n" ) ; sss = THD_zzprintf( sss , " opacity_nhands = %d\n" , rs->opacity_graf_state.nh ) ; sss = THD_zzprintf( sss , " opacity_spline = %d\n" , rs->opacity_graf_state.spl) ; for( ii=0 ; ii < rs->opacity_graf_state.nh ; ii++ ){ sss = THD_zzprintf( sss , " opacity_handx[%d] = %d\n" , ii , rs->opacity_graf_state.xh[ii] ) ; sss = THD_zzprintf( sss , " opacity_handy[%d] = %d\n" , ii , rs->opacity_graf_state.yh[ii] ) ; } } #endif /* SCRIPT_GRAFS */ sss = THD_zzprintf( sss , "\n" ) ; RETURN(sss); } /*------------------------------------------------------------------------------ Copy the internal rendering state to a structure --------------------------------------------------------------------------------*/ #define TO_RS(nnn) (rs->nnn = nnn) void RCREND_widgets_to_state( RENDER_state * rs ) { int ii ; ENTRY( "RCREND_widgets_to_state" ); if( rs == NULL ) EXRETURN ; /* dataset stuff */ if( dset != NULL ){ strcpy( rs->dset_name , DSET_HEADNAME(dset) ) ; rs->dset_idc = dset->idcode ; } else { rs->dset_name[0] = '\0' ; ZERO_IDCODE(rs->dset_idc) ; } if( func_dset != NULL ){ strcpy( rs->func_dset_name , DSET_HEADNAME(func_dset) ) ; rs->func_dset_idc = func_dset->idcode ; } else { rs->func_dset_name[0] = '\0' ; ZERO_IDCODE(rs->func_dset_idc) ; } /* other scalars */ TO_RS(dset_ival) ; TO_RS(func_color_ival) ; TO_RS(func_thresh_ival) ; rs->clipbot = clipbot_av->ival ; rs->cliptop = cliptop_av->ival ; TO_RS(angle_roll) ; TO_RS(angle_pitch) ; TO_RS(angle_yaw) ; TO_RS(xhair_flag) ; TO_RS(xhair_ovc) ; /* 08 Mar 2001 */ if( wfunc_frame != NULL ){ TO_RS(func_use_autorange) ; TO_RS(func_threshold) ; TO_RS(func_thresh_top) ; TO_RS(func_color_opacity) ; TO_RS(func_see_overlay) ; TO_RS(func_showthru ) ; TO_RS(func_showthru_fac) ; /* 08 Mar 2002 */ TO_RS(func_cut_overlay) ; TO_RS(func_kill_clusters) ; TO_RS(func_clusters_rmm) ; TO_RS(func_clusters_vmul) ; TO_RS(func_range) ; TO_RS(func_see_ttatlas) ; /* 24 Jul 2001 */ /* pbar stuff */ rs->pbar_mode = wfunc_color_pbar->mode ; rs->pbar_npane = wfunc_color_pbar->num_panes ; for( ii=0 ; ii <= rs->pbar_npane ; ii++ ) rs->pbar_pval[ii] = wfunc_color_pbar->pval[ii] ; } /* cutout stuff */ RCREND_load_cutout_state() ; /* save current widget state into cutout state */ TO_RS(current_cutout_state.opacity_scale) ; TO_RS(current_cutout_state.num) ; TO_RS(current_cutout_state.logic) ; for( ii=0 ; ii < current_cutout_state.num ; ii++ ){ TO_RS( current_cutout_state.type[ii] ) ; TO_RS( current_cutout_state.mustdo[ii] ) ; TO_RS( current_cutout_state.param[ii] ) ; } #ifdef SCRIPT_GRAFS graf_state_get( gry_graf , &(rs->bright_graf_state) ) ; graf_state_get( opa_graf , &(rs->opacity_graf_state) ) ; #endif EXRETURN ; } /*------------------------------------------------------------------------------ Copy the structure values to the internal rendering state. Also must change the visible state of the interface widgets. In most cases, the widget internal values are set, then their callback routines are invoked to set the internal rendering state. --------------------------------------------------------------------------------*/ #define RSOK(nnn,bb,tt) (rs->nnn != nnn && rs->nnn >= bb && rs->nnn <= tt) #define DBI(nnn) fprintf(stderr,#nnn ": rs=%d wid=%d\n",rs->nnn,nnn) void RCREND_state_to_widgets( RENDER_state * rs ) { int ii , flag ; static XtPointer xpt = (XtPointer) "Mr Tambourine Man" ; ENTRY( "RCREND_state_to_widgets" ); if( rs == NULL ) EXRETURN ; script_dontdraw = 1 ; /* 24 Nov 2000 */ #ifdef SCRIPT_DSETS /* 12 Apr 2000: allow change of dataset! */ if( script_dsetchange ){ THD_3dim_dataset * qset ; MCW_choose_cbs cbs ; char serr[256] ; /*- underlay -*/ if( !ISZERO_IDCODE(rs->dset_idc) ){ if( dset == NULL || !EQUIV_IDCODES(rs->dset_idc,dset->idcode) ){ fprintf(stderr,"++ Changing underlay dataset to %s\n",rs->dset_idc.str) ; qset = PLUTO_find_dset( &(rs->dset_idc) ) ; if( !ISVALID_DSET(qset) ){ sprintf(serr, " \n" "** Can't find desired\n" "** underlay dataset:\n" "** %s\n" , rs->dset_idc.str ) ; (void) MCW_popup_message( script_cbut , serr , MCW_USER_KILL | MCW_TIMER_KILL ) ; PLUTO_beep() ; fprintf(stderr,"** Couldn't find new underlay dataset!\n") ; } else { ndsl = 1 ; dsl = (PLUGIN_dataset_link *) XtRealloc( (char *)dsl, sizeof(PLUGIN_dataset_link)*ndsl ); make_PLUGIN_dataset_link( qset , dsl ) ; cbs.ival = 0 ; RCREND_finalize_dset_CB( NULL , NULL , &cbs ) ; } } } /*- overlay -*/ if( !ISZERO_IDCODE(rs->func_dset_idc) && dset != NULL ){ if( func_dset == NULL || !EQUIV_IDCODES(rs->func_dset_idc,func_dset->idcode) ){ fprintf(stderr,"++ Changing overlay dataset to %s\n",rs->func_dset_idc.str) ; qset = PLUTO_find_dset( &(rs->func_dset_idc) ) ; if( !ISVALID_DSET(qset) ){ sprintf(serr, " \n" "** Can't find desired\n" "** overlay dataset:\n" "** %s\n" , rs->func_dset_idc.str ) ; (void) MCW_popup_message( script_cbut , serr , MCW_USER_KILL | MCW_TIMER_KILL ) ; PLUTO_beep() ; fprintf(stderr,"** Couldn't find new overlay dataset!\n") ; } else if( DSET_NX(dset) != DSET_NX(qset) || DSET_NY(dset) != DSET_NY(qset) || DSET_NZ(dset) != DSET_NZ(qset) ){ sprintf(serr," \n" "** Desired overlay dataset:\n" "** %s\n" "** doesn't match underlay\n" "** dataset's dimensions!\n", rs->func_dset_idc.str ); (void) MCW_popup_message( script_cbut , serr , MCW_USER_KILL | MCW_TIMER_KILL ) ; PLUTO_beep() ; fprintf(stderr,"** New overlay dataset doesn't match underlay dimensions!\n") ; } else { ndsl = 1 ; dsl = (PLUGIN_dataset_link *) XtRealloc( (char *)dsl,sizeof(PLUGIN_dataset_link)*ndsl ); make_PLUGIN_dataset_link( qset , dsl ) ; cbs.ival = 0 ; if( wfunc_frame == NULL || !XtIsManaged(wfunc_frame) ) RCREND_open_func_CB(NULL,NULL,NULL) ; RCREND_finalize_func_CB( NULL , NULL , &cbs ) ; } } } } #endif /* SCRIPT_DSETS */ /* change the sub-values in the dataset (maybe?) */ if( script_brindex ){ if( dset != NULL && RSOK(dset_ival,0,DSET_NVALS(dset)-1) ){ AV_assign_ival( choose_av , rs->dset_ival ) ; RCREND_choose_av_CB( choose_av , xpt ) ; } if( func_dset != NULL && RSOK(func_color_ival,0,DSET_NVALS(func_dset)-1) ){ AV_assign_ival( wfunc_color_av , rs->func_color_ival ) ; RCREND_choose_av_CB( wfunc_color_av , xpt ) ; } if( func_dset != NULL && RSOK(func_thresh_ival,0,DSET_NVALS(func_dset)-1) ){ AV_assign_ival( wfunc_thresh_av , rs->func_thresh_ival ) ; RCREND_choose_av_CB( wfunc_thresh_av , xpt ) ; } } #ifdef SCRIPT_GRAFS /* change grafs (maybe?) */ if( script_graf ){ graf_state gs ; graf_state_get( gry_graf , &gs ) ; if( ! graf_states_equal( &(rs->bright_graf_state) , &gs ) ){ FREE_VOLUMES ; graf_state_put( gry_graf , &(rs->bright_graf_state) ) ; } graf_state_get( opa_graf , &gs ) ; if( ! graf_states_equal( &(rs->opacity_graf_state) , &gs ) ){ FREE_VOLUMES ; graf_state_put( opa_graf , &(rs->opacity_graf_state) ) ; } } #endif /* SCRIPT_GRAFS */ /* change clipping values */ if( rs->clipbot != clipbot_av->ival ){ AV_assign_ival( clipbot_av , rs->clipbot ) ; RCREND_clip_CB( clipbot_av , NULL ) ; } if( rs->cliptop != cliptop_av->ival ){ AV_assign_ival( cliptop_av , rs->cliptop ) ; RCREND_clip_CB( cliptop_av , NULL ) ; } /* change angles */ if( RSOK(angle_roll,-359.9,719.9) ){ AV_assign_fval( roll_av , rs->angle_roll ) ; RCREND_angle_CB ( roll_av , xpt ) ; /* will set angle_roll */ } if( RSOK(angle_pitch,-359.9,719.9) ){ AV_assign_fval( pitch_av , rs->angle_pitch ) ; RCREND_angle_CB ( pitch_av , xpt ) ; /* will set angle_pitch */ } if( RSOK(angle_yaw,-359.9,719.9) ){ AV_assign_fval( yaw_av , rs->angle_yaw ) ; RCREND_angle_CB ( yaw_av , xpt ) ; /* will set angle_yaw */ } /* change xhair mode */ if( RSOK(xhair_flag,0,1) ){ xhair_flag = rs->xhair_flag ; MCW_set_bbox( xhair_bbox , xhair_flag ) ; } if( RSOK(xhair_ovc,0,dc->ovc->ncol_ov) ){ /* 08 Mar 2001 */ xhair_ovc = rs->xhair_ovc ; } /* change function stuff, if the functional widgets exist */ if( wfunc_frame != NULL ){ { static float dval[9] = { 1.0 , 10.0 , 100.0 , 1000.0 , 10000.0 , 100000.0 , 1000000.0 , 10000000.0 , 100000000.0 } ; if( RSOK(func_thresh_top,1.0,dval[THR_TOP_EXPON]) ){ for( ii=THR_TOP_EXPON ; ii > 0 ; ii-- ) if( rs->func_thresh_top >= dval[ii] ) break ; AV_assign_ival( wfunc_thr_top_av , ii ) ; RCREND_thresh_top_CB( wfunc_thr_top_av , NULL ) ; } } if( RSOK(func_threshold,0.0,0.9999) ){ XmScaleCallbackStruct cbs ; cbs.value = (int)( rs->func_threshold / THR_FACTOR + 0.01 ) ; RCREND_thr_scale_CB( NULL,NULL , &cbs ) ; XmScaleSetValue( wfunc_thr_scale , cbs.value ) ; /* oops, forgot this! 12 Apr 2000 */ } /* 06 Mar 2002: no showthru */ if( RSOK(func_color_opacity,0.0,1.001) ){ /* 11 Sep 2001: add ST+Dcue=12 */ ii = (int)(rs->func_color_opacity * 10.0 + 0.01) ; AV_assign_ival( wfunc_opacity_av , ii ) ; RCREND_color_opacity_CB( wfunc_opacity_av , NULL ) ; } if( RSOK(func_showthru,0,1) ){ /* 08 Mar 2002 */ MCW_set_bbox( wfunc_do_ST_bbox, rs->func_showthru ); RCREND_do_ST_CB( NULL,NULL,NULL ); } if( RSOK(func_showthru_fac,0.0,1.001) ){ /* 08 Mar 2002 */ ii = (int)(rs->func_showthru_fac * 20.0 + 0.01); AV_assign_ival( wfunc_ST_fac_av , ii ) ; RCREND_ST_factor_CB( wfunc_ST_fac_av , NULL ) ; } if( RSOK(func_see_overlay,0,1) ){ MCW_set_bbox( wfunc_see_overlay_bbox , rs->func_see_overlay ) ; RCREND_see_overlay_CB(NULL,NULL,NULL) ; } if( RSOK(func_see_ttatlas,0,1) ){ /* 24 Jul 2001 */ MCW_set_bbox( wfunc_see_ttatlas_bbox , rs->func_see_ttatlas ) ; RCREND_see_ttatlas_CB(NULL,NULL,NULL) ; } if( RSOK(func_cut_overlay,0,1) ){ MCW_set_bbox( wfunc_cut_overlay_bbox , rs->func_cut_overlay ) ; RCREND_cut_overlay_CB(NULL,NULL,NULL) ; } if( RSOK(func_kill_clusters,0,1) ){ MCW_set_bbox( wfunc_kill_clusters_bbox , rs->func_kill_clusters ) ; RCREND_kill_clusters_CB(NULL,NULL,NULL) ; } if( RSOK(func_clusters_rmm,0,99) ){ AV_assign_fval( wfunc_clusters_rmm_av , rs->func_clusters_rmm ) ; RCREND_clusters_av_CB(wfunc_clusters_rmm_av,xpt) ; } if( RSOK(func_clusters_vmul,0,9999) ){ AV_assign_fval( wfunc_clusters_vmul_av , rs->func_clusters_vmul ) ; RCREND_clusters_av_CB(wfunc_clusters_vmul_av,xpt) ; } if( RSOK(func_use_autorange,0,1) ){ MCW_set_bbox( wfunc_range_bbox , rs->func_use_autorange ) ; RCREND_range_bbox_CB(NULL,NULL,NULL) ; } if( RSOK(func_range,0,9999999) ){ AV_assign_fval( wfunc_range_av , rs->func_range ) ; RCREND_range_av_CB(wfunc_range_av,xpt) ; } /* pbar stuff */ if( rs->pbar_mode != wfunc_color_pbar->mode ){ MCW_set_bbox( wfunc_color_bbox , rs->pbar_mode ) ; RCREND_color_bbox_CB(NULL,NULL,NULL) ; } if( rs->pbar_npane != wfunc_color_pbar->num_panes ){ AV_assign_ival( wfunc_colornum_av , rs->pbar_npane ) ; RCREND_colornum_av_CB( wfunc_colornum_av , NULL ) ; } for( flag=ii=0 ; ii <= rs->pbar_npane ; ii++ ){ if( rs->pbar_pval[ii] != wfunc_color_pbar->pval[ii] ) flag++ ; } if( flag ){ alter_MCW_pbar( wfunc_color_pbar , 0 , rs->pbar_pval ) ; INVALIDATE_OVERLAY ; } } /* load cutout stuff into widgets */ RCREND_load_cutout_state() ; /* save current widget state into cutout state */ if( RSOK(current_cutout_state.opacity_scale,0.0,1.0) ){ AV_assign_fval( opacity_scale_av , rs->current_cutout_state.opacity_scale ) ; RCREND_opacity_scale_CB( opacity_scale_av , xpt ) ; } if( RSOK(current_cutout_state.num,0,MAX_CUTOUTS) ){ AV_assign_ival( numcutout_av , rs->current_cutout_state.num ) ; RCREND_numcutout_CB( numcutout_av , xpt ) ; } if( RSOK(current_cutout_state.logic,0,1) ){ AV_assign_ival( logiccutout_av , rs->current_cutout_state.logic ) ; FREE_VOLUMES ; } for( ii=0 ; ii < num_cutouts ; ii++ ){ if( RSOK(current_cutout_state.type[ii],0,NUM_CUTOUT_TYPES-1) ){ AV_assign_ival( cutouts[ii]->type_av , rs->current_cutout_state.type[ii] ) ; RCREND_cutout_type_CB( cutouts[ii]->type_av , xpt ) ; } if( RSOK(current_cutout_state.mustdo[ii],0,1) ){ MCW_set_bbox( cutouts[ii]->mustdo_bbox , rs->current_cutout_state.mustdo[ii] ) ; } if( RSOK(current_cutout_state.param[ii],-999999,999999) ){ AV_assign_fval( cutouts[ii]->param_av , rs->current_cutout_state.param[ii] ) ; } } RCREND_load_cutout_state() ; /* save current widget state into cutout state */ script_dontdraw = 0 ; /* 24 Nov 2000 */ EXRETURN ; } #endif /* USE_SCRIPTING */ /*------------------------------------------------------------------------- When a registered environment variable is changed, this function will be called - 20 Jun 2000! ---------------------------------------------------------------------------*/ void RCREND_environ_CB( char * ename ) { char * ept ; float val ; ENTRY( "RCREND_environ_CB" ); /* sanity checks */ if( ename == NULL ) EXRETURN ; ept = getenv(ename) ; if( ept == NULL ) EXRETURN ; /*---*/ if( strcmp(ename,"AFNI_RENDER_ANGLE_DELTA") == 0 ){ val = strtod(ept,NULL) ; if( val > 0.0 && val < 100.0 ){ angle_fstep = val ; if( shell != NULL ) roll_av->fstep = pitch_av->fstep = yaw_av->fstep = val ; } } /*---*/ else if( strcmp(ename,"AFNI_RENDER_CUTOUT_DELTA") == 0 ){ val = strtod(ept,NULL) ; if( val > 0.0 && val < 100.0 ){ int ii ; cutout_fstep = val ; if( shell != NULL ){ for( ii=0 ; ii < MAX_CUTOUTS ; ii++ ) cutouts[ii]->param_av->fstep = val ; } } } /*---*/ EXRETURN ; } void rcr_disp_hist( unsigned char * im, int nvox, int b1, int cut, int b2 ) { unsigned char * tmpi = im; unsigned char max = 0; int c1, s1, s2, cur; ENTRY( "rcr_disp_hist" ); if ( ( b1 > 256 ) || ( b2 > 256 ) || ( im == NULL ) ) { fprintf( stderr, "*** incorrect parameters to rcr_disp_hist\n" ); EXRETURN; } memset( grcr_hist_high, 0, 256 * sizeof(int) ); memset( grcr_hist_low, 0, 256 * sizeof(int) ); for ( c1 = 0, tmpi = im; c1 < nvox; c1++, tmpi++ ) if ( *tmpi > max ) max = *tmpi; s1 = (b1 <= 0) ? 1 : (cut+b1-1) / b1; s2 = (b2 <= 0) ? 1 : (max - cut + b2) / b2; /* m+1-c+b2-1 */ for ( c1 = 0, tmpi = im; c1 < nvox; c1++, tmpi++ ) { if ( *tmpi >= cut ) grcr_hist_high[(*tmpi - cut) / s2]++; else grcr_hist_low[*tmpi / s1]++; } printf( "nvox = %d, max = %d\n", nvox, max ); cur = 0; if ( cut && b1 ) { printf( "--------- lower buckets ---------\n" ); for (c1 = 0; c1 < b1; c1++) { printf( "[%d,%d] : %d\n", cur, cur + s1 - 1, grcr_hist_low[c1] ); cur += s1; } } cur = cut; printf( "--------- upper buckets ---------\n" ); for (c1 = 0; c1 < b2; c1++) { printf( "[%d,%d] : %d\n", cur, cur + s2 - 1, grcr_hist_high[c1] ); cur += s2; } EXRETURN; } /*==========================================================================*/ #ifdef ALLOW_INCROT /* 26 Apr 2002 - RWCox */ /*--------------------------------------------------------------------------*/ /*! Compute the changes in the rotation angles if we add an incremental rotation about axis ax (0,1,2) of size th. ----------------------------------------------------------------------------*/ static void RCREND_inc_angles( int ax, float th, float *yaw, float *pitch, float *roll ) { double a,b,c ; THD_dmat33 qq , rr , pp ; ENTRY( "RCREND_inc_angles" ); a = *yaw ; b = *pitch ; c = *roll ; /* fetch input angles */ qq = RCREND_rotmatrix( 1,a , 0,b , 2,c ) ; /* compute matrix from angles */ LOAD_ROT_MAT(rr,th,ax) ; /* incremental rotation */ pp = DMAT_MUL(rr,qq) ; /* total rotation matrix */ RCREND_rotmatrix_to_angles( pp , &a,&b,&c ) ; /* get angles from this */ *yaw = a ; *pitch = b ; *roll = c ; /* store angles */ EXRETURN ; } /*--------------------------------------------------------------------------*/ /*! Compute a rotation matrix. */ static THD_dmat33 RCREND_rotmatrix( int ax1,double th1 , int ax2,double th2 , int ax3,double th3 ) { THD_dmat33 q , p ; ENTRY( "RCREND_rotmatrix" ); LOAD_ROT_MAT( q , th1 , ax1 ) ; LOAD_ROT_MAT( p , th2 , ax2 ) ; q = DMAT_MUL( p , q ) ; LOAD_ROT_MAT( p , th3 , ax3 ) ; q = DMAT_MUL( p , q ) ; RETURN(q); } /*-----------------------------------------------------------------------*/ /*! Compute yaw=a, pitch=b, roll=c, given rotation matrix in form below: [cc ca + sc sb sa sc cb -cc sa + sc sb ca] [ ] Rz(c) * Rx(b) * Ry(a) = [-sc ca + cc sb sa cc cb sc sa + cc sb ca ] [ ] [ cb sa -sb cb ca ] - pitch will be between PI/2 and 3*PI/2. - this function only works if ax1=1, ax2=0, ax3=2 (the defaults) -------------------------------------------------------------------------*/ static void RCREND_rotmatrix_to_angles( THD_dmat33 q, double *yaw, double *pitch, double *roll ) { double a,b,c ; double sb,cb , sc,cc ; ENTRY( "RCREND_rotmatrix_to_angles" ); sb = -q.mat[2][1] ; b = PI-asin(sb) ; cb = cos(b) ; if( fabs(cb) < 0.001 ){ /* singular case */ a = 0 ; cc = q.mat[0][0] ; sc = q.mat[0][2] ; if( sb < 0.0 ) sc = -sc ; c = atan2( sc , cc ) ; } else { a = atan2( -q.mat[2][0] , -q.mat[2][2] ) ; c = atan2( -q.mat[0][1] , -q.mat[1][1] ) ; } if( a < 0 ) a += 2.0*PI ; if( c < 0 ) c += 2.0*PI ; *yaw = a ; *pitch = b ; *roll = c ; EXRETURN ; EXRETURN; } #endif /* ALLOW_INCROT */ /*==========================================================================*/ /*-------------------------------------------------------------------------- * compute spacial crosshair points, given axis grid *-------------------------------------------------------------------------- */ static int get_xhair_points( CR_xhairs * pts, THD_3dim_dataset * dset ) { THD_dataxes * dax = dset->daxes; float xi, yj, zk; float fa, fb; int om, gap; ENTRY( "get_xhair_points" ); if ( ! pts || ! dset ) RETURN(-1); xi = im3d->vinfo->xi; /* get Dicomm mm coords */ yj = im3d->vinfo->yj; zk = im3d->vinfo->zk; om = im3d->vinfo->xhairs_orimask; /* get xhair orient mask */ gap = im3d->vinfo->crosshair_gap; /* get gap radius */ /* set the 2 x-direction xhair segments */ if ( om & ORIMASK_LR ) { /* first pair */ fa = dax->xxorg; fb = xi - gap * dax->xxdel; LOAD_FVEC3( pts->xp[0][0], fa, yj, zk ); LOAD_FVEC3( pts->xp[0][1], fb, yj, zk ); if ( fb < fa ) /* stick to single endpoint */ pts->xp[0][1] = pts->xp[0][0]; /* second pair */ fa = xi + gap * dax->xxdel; fb = dax->xxorg + dax->xxdel * (dax->nxx - 1); LOAD_FVEC3( pts->xp[1][0], fa, yj, zk ); LOAD_FVEC3( pts->xp[1][1], fb, yj, zk ); if ( fb < fa ) /* stick to single endpoint */ pts->xp[1][0] = pts->xp[1][1]; } /* set the 2 y-direction xhair segments */ if ( om & ORIMASK_AP ) { /* first pair */ fa = dax->yyorg; fb = yj - gap * dax->yydel; LOAD_FVEC3( pts->yp[0][0], xi, fa, zk ); LOAD_FVEC3( pts->yp[0][1], xi, fb, zk ); if ( fb < fa ) /* stick to single endpoint */ pts->yp[0][1] = pts->yp[0][0]; /* second pair */ fa = yj + gap * dax->yydel; fb = dax->yyorg + dax->yydel * (dax->nyy - 1); LOAD_FVEC3( pts->yp[1][0], xi, fa, zk ); LOAD_FVEC3( pts->yp[1][1], xi, fb, zk ); if ( fb < fa ) /* stick to single endpoint */ pts->yp[1][0] = pts->yp[1][1]; } /* set the 2 z-direction xhair segments */ if ( om & ORIMASK_IS ) { /* first pair */ fa = dax->zzorg; fb = zk - gap * dax->zzdel; LOAD_FVEC3( pts->zp[0][0], xi, yj, fa ); LOAD_FVEC3( pts->zp[0][1], xi, yj, fb ); if ( fb < fa ) /* stick to single endpoint */ pts->zp[0][1] = pts->zp[0][0]; /* second pair */ fa = zk + gap * dax->zzdel; fb = dax->zzorg + dax->zzdel * (dax->nzz - 1); LOAD_FVEC3( pts->zp[1][0], xi, yj, fa ); LOAD_FVEC3( pts->zp[1][1], xi, yj, fb ); if ( fb < fa ) /* stick to single endpoint */ pts->zp[1][0] = pts->zp[1][1]; } /* debug stuff */ LOAD_FVEC3( gcr_debug.xhairs, xi, yj, zk ); /* save for later */ if ( gcr_debug.level > 0 ) r_idisp_vec3f( "-- xhair center : ", gcr_debug.xhairs.xyz ); RETURN(0); } /*-------------------------------------------------------------------------- * Subtract dataset center from all points, and divide by min delta, * this sets image origin to (0,0,0), and the voxel size to cubic units. *-------------------------------------------------------------------------- */ static int xhairs_to_image_pts( CR_xhairs * xh, THD_3dim_dataset * dset ) { THD_dataxes * dax = dset->daxes; THD_fvec3 org; float dx, dy, dz, min; ENTRY( "xhairs_to_image_pts" ); dx = dax->xxdel; /* set deltas */ dy = dax->yydel; dz = dax->zzdel; if ( dx <= 0.0 || dx <= 0.0 || dz <= 0.0 ) { fprintf( stderr, "failure: xhairs_to_image_pts - bad deltas!\n" ); RETURN(-1); /* leave image as is */ } if ( gcr_debug.level > 1 ) idisp_xhair_pts( "-- xh, pre im : ", xh ); /* set the origins to be the center of the 3-D volume */ org.xyz[0] = dax->xxorg + (dax->nxx - 1) * dx / 2.0; org.xyz[1] = dax->yyorg + (dax->nyy - 1) * dy / 2.0; org.xyz[2] = dax->zzorg + (dax->nzz - 1) * dz / 2.0; if ( gcr_debug.level > 0 ) { r_idisp_vec3f( "-- xhair point shift : ", org.xyz ); printf( "-- unitizing points by (%f,%f,%f)\n", dx, dy, dz ); } /* we have dicom grid, scale to max voxel mm grid */ min = dy < dx ? dy : dx; min = dz < min ? dz : min; /* 4 x-points */ xh->xp[0][0] = SUB_FVEC3(xh->xp[0][0],org); DIV_FVEC3_BY_CONST(xh->xp[0][0], min); xh->xp[0][1] = SUB_FVEC3(xh->xp[0][1],org); DIV_FVEC3_BY_CONST(xh->xp[0][1], min); xh->xp[1][0] = SUB_FVEC3(xh->xp[1][0],org); DIV_FVEC3_BY_CONST(xh->xp[1][0], min); xh->xp[1][1] = SUB_FVEC3(xh->xp[1][1],org); DIV_FVEC3_BY_CONST(xh->xp[1][1], min); /* 4 y-points */ xh->yp[0][0] = SUB_FVEC3(xh->yp[0][0],org); DIV_FVEC3_BY_CONST(xh->yp[0][0], min); xh->yp[0][1] = SUB_FVEC3(xh->yp[0][1],org); DIV_FVEC3_BY_CONST(xh->yp[0][1], min); xh->yp[1][0] = SUB_FVEC3(xh->yp[1][0],org); DIV_FVEC3_BY_CONST(xh->yp[1][0], min); xh->yp[1][1] = SUB_FVEC3(xh->yp[1][1],org); DIV_FVEC3_BY_CONST(xh->yp[1][1], min); /* 4 z-points */ xh->zp[0][0] = SUB_FVEC3(xh->zp[0][0],org); DIV_FVEC3_BY_CONST(xh->zp[0][0], min); xh->zp[0][1] = SUB_FVEC3(xh->zp[0][1],org); DIV_FVEC3_BY_CONST(xh->zp[0][1], min); xh->zp[1][0] = SUB_FVEC3(xh->zp[1][0],org); DIV_FVEC3_BY_CONST(xh->zp[1][0], min); xh->zp[1][1] = SUB_FVEC3(xh->zp[1][1],org); DIV_FVEC3_BY_CONST(xh->zp[1][1], min); /* apply shift to debug xhairs */ gcr_debug.xhairs = SUB_FVEC3(gcr_debug.xhairs,org); DIV_FVEC3_BY_CONST(gcr_debug.xhairs, min); if ( gcr_debug.level > 0 ) r_idisp_vec3f( "-- shifted xhair center : ", gcr_debug.xhairs.xyz ); RETURN(0); } static int rotate_xhair_points( CR_xhairs * xh, THD_mat33 * rotm ) { ENTRY( "rotate_xhair_points" ); if ( gcr_debug.level > 1 ) idisp_xhair_pts( "-- xh, pre rot : ", xh ); /* 4 x-direction points */ xh->xp[0][0] = FVEC_TIMES_MAT(xh->xp[0][0],*rotm); xh->xp[0][1] = FVEC_TIMES_MAT(xh->xp[0][1],*rotm); xh->xp[1][0] = FVEC_TIMES_MAT(xh->xp[1][0],*rotm); xh->xp[1][1] = FVEC_TIMES_MAT(xh->xp[1][1],*rotm); /* 4 y-direction points */ xh->yp[0][0] = FVEC_TIMES_MAT(xh->yp[0][0],*rotm); xh->yp[0][1] = FVEC_TIMES_MAT(xh->yp[0][1],*rotm); xh->yp[1][0] = FVEC_TIMES_MAT(xh->yp[1][0],*rotm); xh->yp[1][1] = FVEC_TIMES_MAT(xh->yp[1][1],*rotm); /* 4 z-direction points */ xh->zp[0][0] = FVEC_TIMES_MAT(xh->zp[0][0],*rotm); xh->zp[0][1] = FVEC_TIMES_MAT(xh->zp[0][1],*rotm); xh->zp[1][0] = FVEC_TIMES_MAT(xh->zp[1][0],*rotm); xh->zp[1][1] = FVEC_TIMES_MAT(xh->zp[1][1],*rotm); /* rotate debug xhairs */ gcr_debug.xhairs = FVEC_TIMES_MAT(gcr_debug.xhairs,*rotm); if ( gcr_debug.level > 1 ) { r_idisp_mat33f( "-- rotm : ", rotm->mat ); idisp_xhair_pts( "-- xh, post rot : ", xh ); } if ( gcr_debug.level > 0 ) r_idisp_vec3f( "-- rotated xhairs : ", gcr_debug.xhairs.xyz ); RETURN(0); } static void idisp_xhair_pts ( char * note, CR_xhairs * p ) { if ( ! p ) { fputs( "idisp_xhair_pts: p == NULL!\n", stderr ); return; } r_idisp_vec3f( note, p->xp[0][0].xyz ); r_idisp_vec3f( note, p->xp[0][1].xyz ); r_idisp_vec3f( note, p->xp[1][0].xyz ); r_idisp_vec3f( note, p->xp[1][1].xyz ); r_idisp_vec3f( note, p->yp[0][0].xyz ); r_idisp_vec3f( note, p->yp[0][1].xyz ); r_idisp_vec3f( note, p->yp[1][0].xyz ); r_idisp_vec3f( note, p->yp[1][1].xyz ); r_idisp_vec3f( note, p->zp[0][0].xyz ); r_idisp_vec3f( note, p->zp[0][1].xyz ); r_idisp_vec3f( note, p->zp[1][0].xyz ); r_idisp_vec3f( note, p->zp[1][1].xyz ); } static int draw_xhairs_in_image( CR_xhairs * x, MRI_IMAGE * im ) { byte rgb[3] = {0,0,0}; ENTRY( "draw_xhairs_in_image" ); if ( !x || !im ) RETURN(-1); ovc_to_rgb_bytes( xhair_ovc, rgb, dc->ovc ); draw_image_line( im, &x->xp[0][0], &x->xp[0][1], rgb ); draw_image_line( im, &x->xp[1][0], &x->xp[1][1], rgb ); draw_image_line( im, &x->yp[0][0], &x->yp[0][1], rgb ); draw_image_line( im, &x->yp[1][0], &x->yp[1][1], rgb ); draw_image_line( im, &x->zp[0][0], &x->zp[0][1], rgb ); draw_image_line( im, &x->zp[1][0], &x->zp[1][1], rgb ); RETURN(0); } /*---------------------------------------------------------------------- * Draw a colored line into the image. * Voxels are the units, with the origin in the center. *---------------------------------------------------------------------- */ static int draw_image_line( MRI_IMAGE * im, THD_fvec3 * p1, THD_fvec3 * p2, byte * rgb ) { byte * bp = MRI_RGB_PTR(im); int x, y; /* computed positions to plot */ int x1, y1, x2, y2; /* start and end positions */ int xtot, ytot; /* cumulative directional steps */ int xdir, ydir; /* direction of steps */ int xsteps, ysteps, points; /* total steps to make */ int index; /* image index */ int pc; /* point counter */ ENTRY( "draw_image_line" ); x1 = (int)( p1->xyz[0] + im->nx / 2 + 0.001 ); y1 = (int)( p1->xyz[1] + im->ny / 2 + 0.001 ); x2 = (int)( p2->xyz[0] + im->nx / 2 + 0.001 ); y2 = (int)( p2->xyz[1] + im->ny / 2 + 0.001 ); if ( x1 == x2 && y1 == y2 ) RETURN(0); BOUND_VAL( 0, x1, im->nx-1 ); /* just to be safe */ BOUND_VAL( 0, y1, im->ny-1 ); BOUND_VAL( 0, x2, im->nx-1 ); BOUND_VAL( 0, y2, im->ny-1 ); if ( x2 > x1 ) xdir = 1; else xdir = -1; if ( y2 > y1 ) ydir = 1; else ydir = -1; xsteps = abs(x2 - x1) + 1; ysteps = abs(y2 - y1) + 1; points = (xsteps >= ysteps) ? xsteps : ysteps; /* points to plot */ xtot = ytot = 0; for ( pc = 0; pc < points; pc++ ) { x = x1 + xdir * (xtot/points); /* hmmmm, that's a lot of work to */ y = y1 + ydir * (ytot/points); /* allow arbitrary direction, alas */ index = 3 * (x + y * im->nx); bp[index] = rgb[0]; bp[index+1] = rgb[1]; bp[index+2] = rgb[2]; xtot += xsteps; ytot += ysteps; } if ( gcr_debug.level > 0 ) { printf( "++ drawing line from (%f,%f) to (%f,%f)\n", p1->xyz[0], p1->xyz[1], p2->xyz[0], p2->xyz[1] ); printf( "-- as line from (%d,%d) to (%d,%d)\n", x1, y1, x2, y2 ); } RETURN(0); } #define RD_CHOICE_NONE 0x00 #define RD_CHOICE_HELP 0x01 #define RD_CHOICE_HIST 0x02 #define RD_CHOICE_DISP_COLORS 0x12 /* v1.8 */ #define RD_CHOICE_DISP_DSET 0x13 #define RD_CHOICE_DISP_IM 0x14 #define RD_CHOICE_DISP_XHAIRS 0x15 #define RD_CHOICE_SET_LEVEL 0x20 static int rd_debug_choice ( char ** str ); static int rd_disp_debug_help( char * str, CR_debug * d ); static int rd_disp_color_info ( char * str, CR_debug * d, CR_data * crd ); static int rd_disp_dset_info ( char * str, CR_debug * d, CR_data * crd ); static int rd_disp_mri_image ( char * str, CR_debug * d, MRI_IMARR * r ); static int rd_disp_xhairs ( char * str, CR_debug * d ); static int rd_set_debug_level( char * str, CR_debug * d ); /* this is the current interface for setting debug parameters */ static int r_debug_check( CR_debug * d, char * str ) { char * sp = str; int choice; ENTRY( "r_debug_check" ); /* check to see if we have a valid debug request */ if ( ! d ) RETURN(0); if ( ! sp ) RETURN(0); if ( isspace(*sp) ) sp++; /* allow a single leading space */ if ( *sp++ != 'd' ) RETURN(0); choice = rd_debug_choice( &sp ); /* so get past choice character(s) */ switch ( choice ) { case RD_CHOICE_HELP: rd_disp_debug_help(sp,d); break; case RD_CHOICE_HIST: fputs(g_cren_hist, stderr); break; case RD_CHOICE_DISP_COLORS: rd_disp_color_info(sp,d,&gcr); break; case RD_CHOICE_DISP_DSET: rd_disp_dset_info (sp,d,&gcr); break; case RD_CHOICE_DISP_IM: rd_disp_mri_image (sp,d,renderings); break; case RD_CHOICE_SET_LEVEL: rd_set_debug_level(sp,d); break; case RD_CHOICE_DISP_XHAIRS: rd_disp_xhairs (sp,d); break; default: printf( "error: invalid debug command: %5s\n", str ); break; } fflush(stdout); RETURN(1); /* we did something */ } static int rd_disp_color_info ( char * str, CR_debug * d, CR_data * crd ) { MCW_pbar * fcb = wfunc_color_pbar; int c, incr, half = NPANE_BIG/2; /* 4 Apr 2006 [rickr] */ if ( str && isdigit(*str) ) incr = abs(atoi(str)); else incr = 8; /* default increment for color value display */ fprintf(stderr,"-- debug color increment: %d\n", incr ); fprintf(stderr,"-- bigstuff: r g b +64 r g b\n" " --- --- --- --- --- ---\n"); for ( c = 0; c < half; c += incr ) fprintf(stderr, " %3d/%3d: %3d %3d %3d %3d %3d %3d\n", c, c+half, crd->bigstuff.r[c ], crd->bigstuff.g[c ], crd->bigstuff.b[c ], crd->bigstuff.r[c+half], crd->bigstuff.g[c+half], crd->bigstuff.b[c+half]); fprintf(stderr,"-- fcb: mode, bigmode, num_panes = %d,%d,%d\n", fcb->mode, fcb->bigmode, fcb->num_panes ); return 0; } static int rd_disp_dset_info ( char * str, CR_debug * d, CR_data * crd ) { THD_3dim_dataset * ds; if ( str[0] == 'd' && str[1] == 'd' ) ds = dset; else if ( str[0] == 'd' && str[1] == 'o' ) ds = crd->dset_or; else if ( str[0] == 'f' && str[1] == 'd' ) ds = func_dset; else if ( str[0] == 'f' && str[1] == 'o' ) ds = crd->fset_or; else if ( str[0] == 'm' && str[1] == 'o' ) ds = crd->mset; else { printf( "error: see 'd?' for list of valid control characters\n" ); return 0; } sprintf( d->text, "-- debug '%2s' : ", str ); r_idisp_thd_3dim_dataset( d->text, ds ); if ( ISVALID_DSET(ds) ) { r_idisp_thd_dataxes ( d->text, ds->daxes ); r_idisp_thd_datablock( d->text, ds->dblk ); } return 1; } static int rd_debug_choice( char ** str ) { int rv = RD_CHOICE_NONE; if ( **str == '?' ) rv = RD_CHOICE_HELP; else if ( **str == 'h' ) rv = RD_CHOICE_HIST; else if ( **str == 'c' ) rv = RD_CHOICE_DISP_COLORS; else if ( **str == 'd' ) rv = RD_CHOICE_DISP_DSET; else if ( **str == 'i' ) rv = RD_CHOICE_DISP_IM; else if ( **str == 'l' ) rv = RD_CHOICE_SET_LEVEL; else if ( **str == 'x' ) rv = RD_CHOICE_DISP_XHAIRS; (*str)++; /* all cases currently move past one 'choice' character */ return rv; } /* display debug commands */ static int rd_disp_debug_help( char * str, CR_debug * d ) { printf ( "------------------------------------------------------------------\n" "debugging commands:\n" "\n" " d? - debug help : display this menu\n" " dh - history : display plugin history\n" " dcN - display color info : bigmode color info (with step N)\n" " di - display image : display last mri image strcture\n" " dlN - debug level : set debug level to N, {0,1,2}\n" " ddX - display dataset info : display dataset info for \n" " dset, fset, or mset\n" " X is one of {dd,do,fd,fo,mo}\n" " dx - display crosshairs : display rotated crosshair center\n" "------------------------------------------------------------------\n" ); return 1; } /* display last image in the global 'renderings' array */ static int rd_disp_mri_image( char * str, CR_debug * d, MRI_IMARR * r ) { if ( r->num <= 0 ) return 1; /* valid attempt, but no data, so return 1 */ sprintf( d->text, "-- debug : imarr[%d] : ", r->num-1 ); r_idisp_mri_image( d->text, r->imarr[r->num-1] ); return 1; } static int rd_set_debug_level( char * str, CR_debug * d ) { int level = *str - '0'; if ( level < 0 || level > CR_MAX_DEBUG ) { printf( "error: valid debug levels are in [0,%d]\n", CR_MAX_DEBUG ); return 0; } d->level = level; printf( "-- debug: new level = %d\n", d->level ); return 1; } static int rd_disp_xhairs( char * str, CR_debug * d ) { r_idisp_vec3f( "-- debug: rotated xhairs : ", d->xhairs.xyz ); return 1; } static int ovc_to_rgb_bytes( int ovc, byte * rgb, MCW_DCOV * ov ) { BOUND_VAL( 0, ovc, ov->ncol_ov ); if ( ovc == 0 ) /* then use white */ { rgb[0] = 255; rgb[1] = 255; rgb[2] = 255; } else { /* get the intensities from the color map */ rgb[0] = ov->r_ov[ovc]; rgb[1] = ov->g_ov[ovc]; rgb[2] = ov->b_ov[ovc]; } if ( gcr_debug.level > 0 ) printf( "-- rgb vals are %d, %d, %d\n", rgb[0], rgb[1], rgb[2] ); return 1; } /* set color arrays to handle "bigmode" case [v1.8 rickr] */ static int reset_bigcolors( rgbyte * bcs ) { int i; if ( gcr_debug.level > 0 ) fprintf(stderr,"-- reset_bigcolors()\n"); for ( i = 0; i < NPANE_BIG; i++ ) { gcr.bigstuff.r[i] = bcs[i].r; gcr.bigstuff.g[i] = bcs[i].g; gcr.bigstuff.b[i] = bcs[i].b; } return 0; }