#include "SUMA_suma.h" extern SUMA_CommonFields *SUMAg_CF; extern int SUMAg_N_DOv; extern SUMA_DO *SUMAg_DOv; extern SUMA_SurfaceViewer *SUMAg_SVv; extern int SUMAg_N_SVv; /*! SUMA's viewing paramters are tuned to show human brains in mm units. When a human surface is entered in cm units, some scaling needs to be done so that the surface appears nice. For example, in the cm vs mm case, I got nasty shading errors because the viewing distance was too far. A similar problem might arise if the surface is very small, like a rat's for example. So there should be some scaling needed for other species. */ float SUMA_DimSclFac(char *units, char *specie) { static char FuncName[]={"SUMA_DimSclFac"}; float scm=1.0; SUMA_ENTRY; scm = 1.0; if (!units) units = SUMA_EnvVal("SUMA_NodeCoordsUnits"); /* scaling for units */ if (units) { if (!strcasecmp(units,"mm")) { scm = 1.0; } else if (!strcasecmp(units,"cm")) { scm = 10.0; } else { SUMA_S_Warnv( "Bad value of '%s' for units variable or\n" " ENV 'SUMA_NodeCoordsUnits'. Using 'mm'\n", units); } } /* scaling for species - not sure what to do yet. Wait until it is needed */ /* You'll need to set the factor be scm = scm * AVERAGE_HUMAN_BRAIN_SIZE/AVERAGE_SPECIE_SIZE */ /* might also want an automatic fix that guess at what to do */ SUMA_RETURN(scm); } /*! \brief a function that returns the first viewer that is in momentum mode isv = SUMA_WhichViewerInMomentum( SVv, N_SV, sv); \param SVv (SUMA_SurfaceViewer *) vector of surface viewers \param N_SV (int ) number of surface viewer structures in SVv \param sv (SUMA_SurfaceViewer *) if !NULL then the function returns the index of a surface viewer OTHER THAN sv that is in momentum mode. Otherwise the first surface viewer in momentum mode is returned \return isv (int) the index (into SVv) of the first viewer that is in momentum mode. If sv is not NULL then it is the index of the first viewer OTHER THAN sv that is in momentum mode If none are found, ans = -1 - To turn the index into the viewer's label use: if (isv >= 0) sprintf(slabel,"[%c] SUMA", 65+isv); else sprintf(slabel,"[DOH] SUMA"); */ int SUMA_WhichViewerInMomentum(SUMA_SurfaceViewer *SVv, int N_SV, SUMA_SurfaceViewer *sv) { static char FuncName[]={"SUMA_WhichViewerInMomentum"}; int ii = -1; SUMA_ENTRY; if (!SVv) SUMA_RETURN(-1); for (ii=0; ii < SUMAg_N_SVv; ++ii) { if (SVv[ii].GVS[SVv[ii].StdView].ApplyMomentum) { if (!sv) { /* don't care which one */ SUMA_RETURN(ii); } else if (&(SUMAg_SVv[ii]) != sv) { /* other than sv */ SUMA_RETURN(ii); } } } SUMA_RETURN(-1); } /*! return an appropriate fov setting */ float SUMA_sv_auto_fov(SUMA_SurfaceViewer *sv) { static char FuncName[]={"SUMA_sv_auto_fov"}; float mxdim = -1.0, fov = FOV_INITIAL, maxv[3]={-1.0, -1.0, -1.0}, *xyzr, minv[3]={1000000.0, 10000000.0, 1000000.0}, avgdim=30.0; int i, N_sel=0, *Vis_IDs=NULL, k=0; double CurrentDistance; SUMA_DSET *dset=NULL; SUMA_SurfaceObject *SO=NULL; SUMA_Boolean LocalHead = NOPE; SUMA_ENTRY; if (sv->iState < 0) { SUMA_S_Errv("Can't do this with no state set, returning %f!", FOV_INITIAL); SUMA_RETURN(FOV_INITIAL); } if (sv->auto_FOV_val[sv->iState] > 0.0) SUMA_RETURN(sv->auto_FOV_val[sv->iState]); /* automatic determination */ Vis_IDs = (int *)SUMA_malloc(sizeof(int)*SUMAg_N_DOv); minv[0] = minv[1] = minv[2] = 1; maxv[0] = maxv[1] = maxv[2] = 0; N_sel = SUMA_Selectable_ADOs (sv, SUMAg_DOv, Vis_IDs); if (!N_sel) { SUMA_LH("Nothing selectable!"); SUMA_free(Vis_IDs); Vis_IDs= NULL; SUMA_RETURN(FOV_INITIAL); } else { for (i=0; i maxv[0]) { /* init */ for (k=0; k<3; ++k) { minv[k] = SO->MinDims[k]; maxv[k] = SO->MaxDims[k]; } } else { for (k=0; k<3; ++k) { if (SO->MinDims[k] < minv[k]) minv[k] = SO->MinDims[k]; if (SO->MaxDims[k] > maxv[k]) maxv[k] = SO->MaxDims[k]; } } /* Note that prior to Aug. 2013 k only went up to 1 (k<2), both below and above. Despite that, avgdim was dividing by 3, rather than the expected '2'. Not sure why that was, but for now, we're sticking with saner situation */ avgdim = 0.0; for (k=0;k<3;++k) { if (maxv[k] - minv[k] > mxdim) mxdim = maxv[k] - minv[k]; avgdim += maxv[k] - minv[k]; } avgdim /= 3.0; break; case GRAPH_LINK_type: if(!(dset=SUMA_find_GLDO_Dset( (SUMA_GraphLinkDO *)(SUMAg_DOv[Vis_IDs[i]].OP)))){ SUMA_S_Err("Gilda"); break; } if (!SUMA_IS_REAL_VARIANT(iDO_variant(Vis_IDs[i]))) break; xyzr = SUMA_GDSET_XYZ_Range(dset, iDO_variant(Vis_IDs[i]), NULL); if (minv[0] > maxv[0]) { /* init */ for (k=0; k<3; ++k) { minv[k] = xyzr[2*k]; maxv[k] = xyzr[2*k+1]; } } else { for (k=0; k<3; ++k) { if (xyzr[2*k] < minv[k]) minv[k] = xyzr[2*k]; if (xyzr[2*k+1] > maxv[k]) maxv[k] = xyzr[2*k+1]; } } if (!strcmp(iDO_variant(Vis_IDs[i]), "GMATRIX")) { avgdim = 0.0; /* X Y dims only */ for (k=0;k<2;++k) { if (maxv[k] - minv[k] > mxdim) mxdim = maxv[k] - minv[k]; avgdim += maxv[k] - minv[k]; } avgdim /= 2.0; /* increase dims a little to allow for text placement */ avgdim *= 1.15; mxdim *= 1.15; } else { avgdim = 0.0; for (k=0;k<3;++k) { if (maxv[k] - minv[k] > mxdim) mxdim = maxv[k] - minv[k]; avgdim += maxv[k] - minv[k]; } avgdim /= 3.0; } break; case TRACT_type: { SUMA_TractDO *tdo=(SUMA_TractDO *)SUMAg_DOv[Vis_IDs[i]].OP; xyzr = SUMA_TDO_XYZ_Range(tdo, NULL); if (minv[0] > maxv[0]) { /* init */ for (k=0; k<3; ++k) { minv[k] = xyzr[2*k]; maxv[k] = xyzr[2*k+1]; } } else { for (k=0; k<3; ++k) { if (xyzr[2*k] < minv[k]) minv[k] = xyzr[2*k]; if (xyzr[2*k+1] > maxv[k]) maxv[k] = xyzr[2*k+1]; } } avgdim = 0.0; for (k=0;k<3;++k) { if (maxv[k] - minv[k] > mxdim) mxdim = maxv[k] - minv[k]; avgdim += maxv[k] - minv[k]; } avgdim /= 3.0; break; } case VO_type: { SUMA_VolumeObject *vo= (SUMA_VolumeObject *)SUMAg_DOv[Vis_IDs[i]].OP; xyzr = SUMA_VO_XYZ_Range(vo, NULL); if (minv[0] > maxv[0]) { /* init */ for (k=0; k<3; ++k) { minv[k] = xyzr[2*k]; maxv[k] = xyzr[2*k+1]; } } else { for (k=0; k<3; ++k) { if (xyzr[2*k] < minv[k]) minv[k] = xyzr[2*k]; if (xyzr[2*k+1] > maxv[k]) maxv[k] = xyzr[2*k+1]; } } avgdim = 0.0; for (k=0;k<3;++k) { if (maxv[k] - minv[k] > mxdim) mxdim = maxv[k] - minv[k]; avgdim += maxv[k] - minv[k]; } avgdim /= 3.0; break; } case CDOM_type: { xyzr = SUMA_CIFTI_DO_XYZ_Range( (SUMA_CIFTI_DO*)SUMAg_DOv[Vis_IDs[i]].OP, NULL); if (minv[0] > maxv[0]) { /* init */ for (k=0; k<3; ++k) { minv[k] = xyzr[2*k]; maxv[k] = xyzr[2*k+1]; } } else { for (k=0; k<3; ++k) { if (xyzr[2*k] < minv[k]) minv[k] = xyzr[2*k]; if (xyzr[2*k+1] > maxv[k]) maxv[k] = xyzr[2*k+1]; } } avgdim = 0.0; for (k=0;k<3;++k) { if (maxv[k] - minv[k] > mxdim) mxdim = maxv[k] - minv[k]; avgdim += maxv[k] - minv[k]; } avgdim /= 3.0; break; } case MASK_type: { SUMA_MaskDO *mo= (SUMA_MaskDO *)SUMAg_DOv[Vis_IDs[i]].OP; xyzr = SUMA_MDO_XYZ_Range(mo, NULL); if (minv[0] > maxv[0]) { /* init */ for (k=0; k<3; ++k) { minv[k] = xyzr[2*k]; maxv[k] = xyzr[2*k+1]; } } else { for (k=0; k<3; ++k) { if (xyzr[2*k] < minv[k]) minv[k] = xyzr[2*k]; if (xyzr[2*k+1] > maxv[k]) maxv[k] = xyzr[2*k+1]; } } avgdim = 0.0; for (k=0;k<3;++k) { if (maxv[k] - minv[k] > mxdim) mxdim = maxv[k] - minv[k]; avgdim += maxv[k] - minv[k]; } avgdim /= 3.0; break; } default: SUMA_LHv("Ignoring dims of %s\n", iDO_label(Vis_IDs[i])); avgdim = 0.0; for (k=0;k<3;++k) { if (maxv[k] - minv[k] > mxdim) mxdim = maxv[k] - minv[k]; avgdim += maxv[k] - minv[k]; } avgdim /= 3.0; break; } } } SUMA_free(Vis_IDs); Vis_IDs= NULL; SUMA_LHv("Have mxdim = %f, avgdim = %f (rat %.2f)\n", mxdim, avgdim, mxdim / avgdim); /* Current avg. distance from camera */ CurrentDistance = sqrt((sv->GVS[sv->StdView].ViewFrom[0] - sv->GVS[sv->StdView].ViewCenter[0]) * (sv->GVS[sv->StdView].ViewFrom[0] - sv->GVS[sv->StdView].ViewCenter[0]) + (sv->GVS[sv->StdView].ViewFrom[1] - sv->GVS[sv->StdView].ViewCenter[1]) * (sv->GVS[sv->StdView].ViewFrom[1] - sv->GVS[sv->StdView].ViewCenter[1]) + (sv->GVS[sv->StdView].ViewFrom[2] - sv->GVS[sv->StdView].ViewCenter[2]) * (sv->GVS[sv->StdView].ViewFrom[2] - sv->GVS[sv->StdView].ViewCenter[2])); if (mxdim > 0 && mxdim < 1000) { SUMA_LHv("sv %p mxdim=%f, avgdim=%f, rat=%f, DimSclFac = %f, " "CurrentDist = %f\n", sv, mxdim, avgdim, mxdim / avgdim, sv->GVS[sv->StdView].DimSclFac, CurrentDistance); if (mxdim / avgdim > 2.2) { /* crazy aspect, adopt average size */ mxdim = avgdim; } /* fov is the double of the angle at the camera of the square triangle with depth = CurrentDistance and width = half the biggest distance Not sure about the DimSclFac usage here, need to retest it at some point.*/ /* make mxdim a little larger for some framing */ fov = 2.0*atan((double)(mxdim *1.1* sv->GVS[sv->StdView].DimSclFac) / 2.0 / CurrentDistance)*180.0/SUMA_PI; SUMA_LHv("Computed fov to be %f degrees\n", fov); } else { static int nwarn=0; fov = FOV_INITIAL; if (!(nwarn % SUMA_MAX_SURF_VIEWERS)) { SUMA_S_Errv("max dim too strange (%f)\nUsing default (%f).\n" "Similar messages will be decimated.\n", mxdim, fov); } ++nwarn; } /* institutional memory */ sv->auto_FOV_val[sv->iState] = fov; SUMA_RETURN(fov); } /* This is used to hold the functions that manipulate SV, Surface Viewer Structures */ /*! \brief returns a string corresponding to the link type SUMA_Boolean SUMA_LockEnum_LockType (SUMA_LINK_TYPES i, char *Name); \param i (SUMA_LINK_TYPES) see enum type in SUMA_define.h \param Name (char *) a preallocated character string (no more than 50 chars) \return YUP/NOPE OK, error */ SUMA_Boolean SUMA_LockEnum_LockType (SUMA_LINK_TYPES i, char *Name) { static char FuncName[]={"SUMA_LockEnum_LockType"}; SUMA_ENTRY; switch (i) { case SUMA_No_Lock: sprintf (Name, "No Lock"); break; case SUMA_I_Lock: sprintf (Name, "Index Lock"); break; case SUMA_XYZ_Lock: sprintf (Name, "XYZ Lock"); break; default: sprintf (Name, "?"); SUMA_RETURN (NOPE); } SUMA_RETURN (YUP); } /* Use this function to capture GVS as last displayed */ SUMA_Boolean SUMA_CopyGeomViewStruct(SUMA_GEOMVIEW_STRUCT *gvs1, SUMA_GEOMVIEW_STRUCT *gvs2) { static char FuncName[]={"SUMA_CopyGeomViewStruct"}; SUMA_ENTRY; if (!gvs1 || !gvs2) SUMA_RETURN(NOPE); /* This assumes that SUMA_GEOMVIEW_STRUCT has no dynamic allocations */ memcpy(gvs2, gvs1, sizeof(SUMA_GEOMVIEW_STRUCT)); SUMA_RETURN(YUP); } /* overzealous but prudent */ SUMA_Boolean SUMA_DiffGeomViewStruct(SUMA_GEOMVIEW_STRUCT gvs1, SUMA_GEOMVIEW_STRUCT gvs2, int level) { static char FuncName[]={"SUMA_DiffGeomViewStruct"}; SUMA_ENTRY; if (level <= 1) { /* big deal, should change position of object in space */ if (SUMA_VEC_DIFF3(gvs1.ViewFrom, gvs2.ViewFrom)) SUMA_RETURN(2); if (SUMA_VEC_DIFF3(gvs1.ViewCenter, gvs2.ViewCenter)) SUMA_RETURN(4); if (SUMA_VEC_DIFF3(gvs1.ViewCamUp, gvs2.ViewCamUp)) SUMA_RETURN(6); if (gvs1.ViewDistance != gvs2.ViewDistance) SUMA_RETURN(7); if (SUMA_VEC_DIFF3(gvs1.translateVec, gvs2.translateVec)) SUMA_RETURN(15); if (SUMA_VEC_DIFF3(gvs1.RotaCenter, gvs2.RotaCenter)) SUMA_RETURN(16); if (SUMA_VEC_DIFF4(gvs1.deltaQuat, gvs2.deltaQuat)) SUMA_RETURN(24); if (SUMA_VEC_DIFF4(gvs1.currentQuat, gvs2.currentQuat)) SUMA_RETURN(25); if (gvs1.ApplyMomentum != gvs2.ApplyMomentum) SUMA_RETURN(26); } if (level > 1) { /* any change */ if (gvs1.DimSclFac != gvs2.DimSclFac) SUMA_RETURN(1); if (SUMA_VEC_DIFF3(gvs1.ViewFromOrig, gvs2.ViewFromOrig)) SUMA_RETURN(3); if (SUMA_VEC_DIFF3(gvs1.ViewCenterOrig, gvs2.ViewCenterOrig)) SUMA_RETURN(5); if (gvs1.translateBeginX != gvs2.translateBeginX) SUMA_RETURN(8); if (gvs1.translateBeginY != gvs2.translateBeginY) SUMA_RETURN(9); if (gvs1.translateDeltaX != gvs2.translateDeltaX) SUMA_RETURN(10); if (gvs1.translateDeltaY != gvs2.translateDeltaY) SUMA_RETURN(11); if (gvs1.TranslateGain != gvs2.TranslateGain) SUMA_RETURN(12); if (gvs1.ArrowtranslateDeltaX != gvs2.ArrowtranslateDeltaX) SUMA_RETURN(13); if (gvs1.ArrowtranslateDeltaY != gvs2.ArrowtranslateDeltaY) SUMA_RETURN(14); if (gvs1.zoomDelta != gvs2.zoomDelta) SUMA_RETURN(17); if (gvs1.zoomBegin != gvs2.zoomBegin) SUMA_RETURN(18); if (gvs1.spinDeltaX != gvs2.spinDeltaX) SUMA_RETURN(19); if (gvs1.spinDeltaY != gvs2.spinDeltaY) SUMA_RETURN(20); if (gvs1.spinBeginX != gvs2.spinBeginX) SUMA_RETURN(21); if (gvs1.spinBeginY != gvs2.spinBeginY) SUMA_RETURN(22); if (gvs1.MinIdleDelta != gvs2.MinIdleDelta) SUMA_RETURN(23); if (gvs1.vLHpry[0] != gvs2.vLHpry[0]) SUMA_RETURN(27); if (gvs1.vLHpry0[0] != gvs2.vLHpry0[0]) SUMA_RETURN(28); if (gvs1.vLHpry[1] != gvs2.vLHpry[1]) SUMA_RETURN(29); if (gvs1.vLHpry0[1] != gvs2.vLHpry0[1]) SUMA_RETURN(30); if (gvs1.vLHpry[2] != gvs2.vLHpry[2]) SUMA_RETURN(31); if (gvs1.vLHpry0[2] != gvs2.vLHpry0[2]) SUMA_RETURN(32); if (gvs1.LHlol != gvs2.LHlol) SUMA_RETURN(33); } /* if (gvs1. != gvs2.) SUMA_RETURN(); */ SUMA_RETURN(0); /* no difference */ } int SUMA_VerifyRenderOrder(char *ord, void *unused) { static char FuncName[]={"SUMA_VerifyRenderOrder"}; if (SUMA_SetObjectDisplayOrder(ord, NULL) < 0) return(0); return(1); } int SUMA_SetObjectDisplayOrder(char *ord, int *otseq) { static char FuncName[]={"SUMA_SetObjectDisplayOrder"}; NI_str_array *sar = NULL; int iii, cnt = 0, bad = 0, dummy[N_DO_TYPES], used[N_DO_TYPES]; SUMA_Boolean LocalHead = NOPE; SUMA_ENTRY; cnt = -1; if (!otseq) otseq = dummy; if (!ord || !strcmp(ord,"DEFAULT")) { otseq[0] = VO_type; otseq[1] = SO_type; otseq[2] = GRAPH_LINK_type; cnt = 3; RETURN(cnt); } memset(used, 0, N_DO_TYPES*sizeof(int)); deblank_name(ord); if ((sar = SUMA_NI_decode_string_list( ord , ",;"))) { cnt = 0; for (iii=0; iiinum; ++iii) { if (sar->str[iii]) deblank_name(sar->str[iii]); if (sar->str[iii]) { bad = 0; if (!strncasecmp(sar->str[iii],"sur", 3)) { if (!used[SO_type]) { otseq[cnt++] = SO_type; used[SO_type] = 1; } else { SUMA_S_Err("Duplicate reference to surfaces in %s", sar->str[iii]); bad = 1; } } else if (!strncasecmp(sar->str[iii],"vol", 3)) { if (!used[VO_type]) { otseq[cnt++] = VO_type; used[VO_type] = 1; } else { SUMA_S_Err("Duplicate reference to volumes in %s", sar->str[iii]); bad = 1; } } else if (!strncasecmp(sar->str[iii],"gra", 3)) { if (!used[GRAPH_LINK_type]) { otseq[cnt++] = GRAPH_LINK_type; used[GRAPH_LINK_type] = 1; } else { SUMA_S_Err("Duplicate reference to graphs in %s", sar->str[iii]); bad = 1; } } else if (!strcmp(sar->str[iii],"S")) { if (!used[SO_type]) { otseq[cnt++] = SO_type; used[SO_type] = 1; } else { SUMA_S_Err("Duplicate reference to surfaces in %s", sar->str[iii]); bad = 1; } } else if (!strcmp(sar->str[iii],"V")) { if (!used[VO_type]) { otseq[cnt++] = VO_type;; used[VO_type] = 1; } else { SUMA_S_Err("Duplicate reference to volumes in %s", sar->str[iii]); bad = 1; } } else if (!strcmp(sar->str[iii],"G")) { if (!used[GRAPH_LINK_type]) { otseq[cnt++] = GRAPH_LINK_type; used[GRAPH_LINK_type] = 1; } else { SUMA_S_Err("Duplicate reference to graphs in %s", sar->str[iii]); bad = 1; } } else if (sar->num == 1) { int OK=0, jjj; char *ss=sar->str[0]; for (jjj=0; jjjstr[0]); bad = 1; } break; case 'V': if (!used[VO_type]) { otseq[cnt++] = VO_type; used[VO_type] = 1; } else { SUMA_S_Err("Duplicate reference to volumes in %s", sar->str[0]); bad = 1; } break; case 'G': if (!used[GRAPH_LINK_type]) { otseq[cnt++] = GRAPH_LINK_type; used[GRAPH_LINK_type] = 1; } else { SUMA_S_Err("Duplicate reference to graphs in %s", sar->str[0]); bad = 1; } break; } } } else { bad = 2; } if (bad) { if (bad == 2 && otseq != dummy) { SUMA_S_Warn("Object type %s in env SUMA_ObjectDisplayOrder '%s'\n" "was not recognized. For now only 'surface'\n" "'volume', and 'graph' are allowed.\n", sar->str[iii], ord); } if (otseq == dummy) SUMA_RETURN(-1); } } } sar = SUMA_free_NI_str_array(sar); } if (LocalHead && otseq != dummy) { SUMA_LH("Sequence now:"); for (iii=0; iiiC_mode = SUMA_CONV_NONE; SV->C_filter = SUMA_C_newfilter(1, 1); SV->LoadedTextures[0] = -1; /* plug */ SV->N_GVS = SUMA_N_STANDARD_VIEWS; SV->GVS = (SUMA_GEOMVIEW_STRUCT *) SUMA_calloc(SV->N_GVS, sizeof(SUMA_GEOMVIEW_STRUCT)); SV->GVS_last_PickMode = (SUMA_GEOMVIEW_STRUCT *) SUMA_calloc(SV->N_GVS, sizeof(SUMA_GEOMVIEW_STRUCT)); if (!SV->GVS) { fprintf(SUMA_STDERR, "Error %s: Could not allocate for N_GVS.\n", FuncName); SUMA_RETURN (NULL); } SV->StdView = SUMA_3D; /* default */ /* set the standards for all viewing modes here */ SV->verbose = 1; { char *eee = getenv("SUMA_AdjustMouseMotionWithZoom"); if (eee) { if (strcasecmp (eee, "YES") == 0) SV->ZoomCompensate = 1.0; else SV->ZoomCompensate = 0.0; } else { SV->ZoomCompensate = 1.0; } } { char *eee = getenv("SUMA_FreezeFOVAcrossStates"); if (eee) { if (strcasecmp (eee, "YES") == 0) SV->FreezeZoomXstates = 1; else SV->FreezeZoomXstates = 0; } else { SV->FreezeZoomXstates = 1; } } { char *eee = getenv("SUMA_ViewOrthographicProjection"); if (eee) { if (strcasecmp (eee, "YES") == 0) SV->ortho = 1; else SV->ortho = 0; } else { SV->ortho = 0; } } SV->ShowLabelAtXhair = 1; if (!SUMA_isEnv("SUMA_ShowLabelsAtCrossHair","YES")) { SV->ShowLabelAtXhair = 0; } SV->ShowSelectedDatum = 1; SV->ShowSelectedFaceSet = 1; SV->ShowSelectedEdge = 1; SV->SelAdo = (DList *)SUMA_calloc(1,sizeof(DList)); dlist_init (SV->SelAdo, SUMA_Free_SelectedDO_Datum); SV->LastSel_ado_idcode_str = NULL; SV->Aspect = 1.0; SV->FOV = NULL; SV->auto_FOV_val = NULL; SV->FOV_last_PickMode = NULL; for (j=0; j < SV->N_GVS; ++j) { memset(&(SV->GVS[j]), 0, sizeof(SUMA_GEOMVIEW_STRUCT)); SV->GVS[j].DimSclFac = SUMA_DimSclFac(NULL, NULL); switch (j) { case SUMA_2D_Z0: case SUMA_2D_Z0L: /* Default top view, rotate by nothing */ SUMA_HOME_QUAT(j, SV->GVS[j].currentQuat); SV->GVS[j].ApplyMomentum = False; SV->GVS[j].vLHpry[0] = 0.0; SV->GVS[j].vLHpry0[0] = 0.0; SV->GVS[j].vLHpry[1] = 0.0; SV->GVS[j].vLHpry0[1] = 0.0; SV->GVS[j].vLHpry[2] = 0.0; SV->GVS[j].vLHpry0[2] = 0.0; SV->GVS[j].LHlol = 0; SV->GVS[j].MinIdleDelta = 1; SV->GVS[j].TranslateGain = TRANSLATE_GAIN; SV->GVS[j].ArrowtranslateDeltaX = ARROW_TRANSLATE_DELTAX; SV->GVS[j].ArrowtranslateDeltaY = ARROW_TRANSLATE_DELTAY; SV->GVS[j].ViewCamUp[0] = 0.0; SV->GVS[j].ViewCamUp[1] = 1.0; SV->GVS[j].ViewCamUp[2] = 0.0; SV->GVS[j].ViewFrom[0] = 0.0; SV->GVS[j].ViewFrom[1] = 0.0; SV->GVS[j].ViewFrom[2] = SUMA_DEFAULT_VIEW_FROM; SV->GVS[j].ViewCenter[0] = 0.0; SV->GVS[j].ViewCenter[1] = 0.0; SV->GVS[j].ViewCenter[2] = 0.0; SV->GVS[j].RotaCenter[0] = 0.0; SV->GVS[j].RotaCenter[1] = 0.0; SV->GVS[j].RotaCenter[2] = 0.0; break; case SUMA_3D_Z0: case SUMA_3D: SUMA_HOME_QUAT(j, SV->GVS[j].currentQuat); SV->GVS[j].ApplyMomentum = False; SV->GVS[j].vLHpry[0] = 0.0; SV->GVS[j].vLHpry0[0] = 0.0; SV->GVS[j].vLHpry[1] = 0.0; SV->GVS[j].vLHpry0[1] = 0.0; SV->GVS[j].vLHpry[2] = 0.0; SV->GVS[j].vLHpry0[2] = 0.0; SV->GVS[j].LHlol = 0; SV->GVS[j].MinIdleDelta = 1; SV->GVS[j].TranslateGain = TRANSLATE_GAIN; SV->GVS[j].ArrowtranslateDeltaX = ARROW_TRANSLATE_DELTAX; SV->GVS[j].ArrowtranslateDeltaY = ARROW_TRANSLATE_DELTAY; SV->GVS[j].ViewCamUp[0] = 0.0; SV->GVS[j].ViewCamUp[1] = 1.0; SV->GVS[j].ViewCamUp[2] = 0.0; SV->GVS[j].ViewFrom[0] = 0.0; SV->GVS[j].ViewFrom[1] = 0.0; SV->GVS[j].ViewFrom[2] = 0.0; SV->GVS[j].ViewCenter[0] = 0.0; SV->GVS[j].ViewCenter[1] = 0.0; SV->GVS[j].ViewCenter[2] = 0.0; SV->GVS[j].RotaCenter[0] = 0.0; SV->GVS[j].RotaCenter[1] = 0.0; SV->GVS[j].RotaCenter[2] = 0.0; SV->GVS[j].translateVec[0] = 0.0; SV->GVS[j].translateVec[1] = 0.0; break; default: fprintf(SUMA_STDERR, "Error %s: Undefined viewing mode.\n", FuncName); SUMA_RETURN (NULL); } } SV->light0_position[0] = 0.0; SV->light0_position[1] = 0.0; SV->light0_position[2] = 1.0 * SUMA_INTITIAL_LIGHT0_SWITCH; SV->lit_for = SUMA_INTITIAL_LIGHT0_SWITCH; SV->light0_position[3] = 0.0; lightstr = getenv("SUMA_Light0Position"); if (lightstr) { if (SUMA_StringToNum (lightstr, (void *)lightpos, 3, 1) == 3) { SV->light0_position[0] = lightpos[0]; SV->light0_position[1] = lightpos[1]; SV->light0_position[2] = lightpos[2]; } } SV->light1_position[0] = 1.0; SV->light1_position[1] = 1.0; SV->light1_position[2] = 1.0; SV->light1_position[3] = 0.0; lightstr = getenv("SUMA_Light1Position"); if (lightstr) { if (SUMA_StringToNum (lightstr, (void *)lightpos, 3, 1) == 3) { SV->light1_position[0] = lightpos[0]; SV->light1_position[1] = lightpos[1]; SV->light1_position[2] = lightpos[2]; } } SV->dim_spe = 1.0; SV->dim_dif = 1.0; SV->dim_emi = 1.0; SV->dim_amb = 1.0; { static SUMA_Boolean err = NOPE; float fv3[3]; char *eee = getenv("SUMA_Light0Color"); if (eee && !err) { if (SUMA_StringToNum (eee, (void *)fv3, 3, 1) != 3) { err = YUP; SUMA_SL_Err("Syntax error in environment\n" "variable SUMA_Light0Color"); SV->light0_color[0] = SUMA_LIGHT0_COLOR_R; SV->light0_color[1] = SUMA_LIGHT0_COLOR_G; SV->light0_color[2] = SUMA_LIGHT0_COLOR_B; SV->light0_color[3] = SUMA_LIGHT0_COLOR_A; }else { SV->light0_color[0] = fv3[0]; SV->light0_color[1] = fv3[1]; SV->light0_color[2] = fv3[2]; SV->light0_color[3] = SUMA_LIGHT0_COLOR_A; } }else { SV->light0_color[0] = SUMA_LIGHT0_COLOR_R; SV->light0_color[1] = SUMA_LIGHT0_COLOR_G; SV->light0_color[2] = SUMA_LIGHT0_COLOR_B; SV->light0_color[3] = SUMA_LIGHT0_COLOR_A; } } { static SUMA_Boolean err = NOPE; float fv3[3]; char *eee = getenv("SUMA_AmbientLight"); if (eee && !err) { if (SUMA_StringToNum (eee, (void *)fv3, 3, 1) != 3) { err = YUP; SUMA_SL_Err("Syntax error in environment\n" "variable SUMA_AmbientLight"); SV->lmodel_ambient[0] = SUMA_LMODEL_AMBIENT_COLOR_R; SV->lmodel_ambient[1] = SUMA_LMODEL_AMBIENT_COLOR_G; SV->lmodel_ambient[2] = SUMA_LMODEL_AMBIENT_COLOR_B; SV->lmodel_ambient[3] = SUMA_LMODEL_AMBIENT_COLOR_A; }else { SV->lmodel_ambient[0] = fv3[0]; SV->lmodel_ambient[1] = fv3[1]; SV->lmodel_ambient[2] = fv3[2]; SV->lmodel_ambient[3] = SUMA_LMODEL_AMBIENT_COLOR_A; } }else { SV->lmodel_ambient[0] = SUMA_LMODEL_AMBIENT_COLOR_R; SV->lmodel_ambient[1] = SUMA_LMODEL_AMBIENT_COLOR_G; SV->lmodel_ambient[2] = SUMA_LMODEL_AMBIENT_COLOR_B; SV->lmodel_ambient[3] = SUMA_LMODEL_AMBIENT_COLOR_A; } } { static SUMA_Boolean err = NOPE; float fv3[3]; char *eee = getenv("SUMA_BackgroundColor"); if (eee && !err) { if (SUMA_StringToNum (eee, (void *)fv3, 3,1) != 3) { err = YUP; SUMA_SL_Err("Syntax error in environment\n" "variable SUMA_BackgroundColor"); SV->clear_color[0] = SUMA_CLEAR_COLOR_R; SV->clear_color[1] = SUMA_CLEAR_COLOR_G; SV->clear_color[2] = SUMA_CLEAR_COLOR_B; SV->clear_color[3] = SUMA_CLEAR_COLOR_A; }else { SV->clear_color[0] = fv3[0]; SV->clear_color[1] = fv3[1]; SV->clear_color[2] = fv3[2]; SV->clear_color[3] = SUMA_CLEAR_COLOR_A; } }else { SV->clear_color[0] = SUMA_CLEAR_COLOR_R; SV->clear_color[1] = SUMA_CLEAR_COLOR_G; SV->clear_color[2] = SUMA_CLEAR_COLOR_B; SV->clear_color[3] = SUMA_CLEAR_COLOR_A; } } SV->wWindWidth = 350; SV->wWindHeight = 350; SV->DrawAreaHeightOffset = -1; /* Not known yet */ SV->DrawAreaWidthOffset = -1; /* Not known yet */ { char *eee = getenv("SUMA_ArrowRotAngle"); if (eee) { float rotval = strtod(eee, NULL); if (rotval > 0.0 && rotval < 360.0) SV->ArrowRotationAngle = SUMA_PI * rotval / 180.0; else SV->ArrowRotationAngle = SUMA_PI * ARROW_ROTATION_ANGLE_DEG / 180.0; } else SV->ArrowRotationAngle = SUMA_PI * ARROW_ROTATION_ANGLE_DEG / 180.0; } { char *eee = getenv("SUMA_KeyZoomGain"); if (eee) { float rotval = strtod(eee, NULL); if (rotval > 0.0 && rotval < 50.0) SV->KeyZoomGain = rotval/100.0; else SV->KeyZoomGain = 0.05; } else SV->KeyZoomGain = 0.05; } { char *eee = getenv("SUMA_KeyNodeJump"); if (eee) { int KeyNodeJump = (int)strtod(eee, NULL); if (KeyNodeJump > 0 && KeyNodeJump <= 10) SV->KeyNodeJump = KeyNodeJump; else SV->KeyNodeJump = 1; } else SV->KeyNodeJump = 1; } { char *eee = getenv("SUMA_ContourThickness"); if (eee) { float th = strtod(eee, NULL); if (th > 0 && th <= 100) SV->ContThick = th; else SV->ContThick = 1; } else SV->ContThick = 1; } { char *eee = getenv("SUMA_FOV_Original"); if (eee) { float fovinit = strtod(eee, NULL); if (fovinit > 1.0 && fovinit < 100.0) SV->FOV_original = fovinit; else if (fovinit < 0) SV->FOV_original = -1; else SV->FOV_original = FOV_INITIAL; } else SV->FOV_original = FOV_INITIAL; } SV->Open = NOPE; SV->RegistDO = (SUMA_DO_LOCATOR *) SUMA_calloc( SUMA_MAX_DISPLAYABLE_OBJECTS, sizeof(SUMA_DO_LOCATOR)); if (SV->RegistDO == NULL) { SUMA_S_Err("Failed to SUMA_malloc SV->RegistDO"); SUMA_RETURN (NULL); } SV->N_DO = 0; /* Nothing is registered with the viewer yet */ SV->ColList = (SUMA_COLORLIST_STRUCT **) SUMA_calloc(SUMA_MAX_DISPLAYABLE_OBJECTS,sizeof(SUMA_COLORLIST_STRUCT*)); SV->N_ColList = 0; /* this number reflects the number of surfaces that have colorlist structures in SV */ SV->ShowEyeAxis = 0; SV->ShowMeshAxis = 0; /* Turned off Oct 15 04 in favor of WorldAxis */ SV->ShowWorldAxis = SUMA_NO_WAX; SV->WAx = SUMA_Alloc_Axis ("Viewer World Axis", AO_type); if (SV->WAx == NULL) { fprintf(SUMA_STDERR,"Error %s: Error Allocating axis\n", FuncName); SUMA_RETURN(NULL); } SV->WAx->atype = SUMA_SCALE_BOX; SV->Ch = SUMA_Alloc_CrossHair (); if (SV->Ch == NULL) { SUMA_S_Err("Failed in SUMA_Alloc_CrossHair"); SUMA_RETURN (NULL); } else SV->ShowCrossHair = 1; SV->X = (SUMA_X *)SUMA_calloc(1,sizeof(SUMA_X)); if (SV->X == NULL) { SUMA_S_Err("Failed to SUMA_malloc SV->X"); SUMA_RETURN (NULL); } SV->X->Title = NULL; SV->X->LookAt_prmpt = NULL; SV->X->SetRot_prmpt = NULL; SV->X->JumpIndex_prmpt = NULL; SV->X->JumpXYZ_prmpt = NULL; SV->X->JumpFocusNode_prmpt = NULL; SV->X->JumpFocusFace_prmpt = NULL; SV->X->HighlightBox_prmpt = NULL; SV->X->SetRenderOrder_prmpt = NULL; SV->X->TOPLEVEL = NULL; SV->X->MOMENTUMID = 0; SV->X->REDISPLAYPENDING = 0; SV->X->DOUBLEBUFFER = True; SV->X->aWIDTH = SV->X->aHEIGHT = 300; /* if you change this, make sure you do so for fallbackResources in SUMA_display */ SV->X->ViewCont = SUMA_CreateViewContStruct(); SV->X->DPY = NULL; SV->X->FORM = SV->X->FRAME = SV->X->GLXAREA = NULL; SV->X->VISINFO = NULL; SV->X->REDISPLAYID = SV->X->MOMENTUMID = 0; SV->X->CMAP = 0; SV->X->GLXCONTEXT=NULL; SV->X->CrappyDrawable = 0; SV->X->gc=NULL; SV->X->ToggleCrossHair_View_tglbtn=NULL; SV->X->ToolsMenu = SUMA_Alloc_Menu_Widget(SW_N_Tools); SV->X->FileMenu = SUMA_Alloc_Menu_Widget(SW_N_File); SV->X->ViewMenu = SUMA_Alloc_Menu_Widget(SW_N_View); SV->X->HelpMenu = SUMA_Alloc_Menu_Widget(SW_N_Help); SV->Focus_DO_ID = -1; SV->State = NULL; SV->iState = -1; SV->VSv = NULL; SV->N_VSv = 0; SV->LastNonMapStateID = -1; SV->iCurGroup = -1; SV->CurGroupName = NULL; SUMA_AdoptGroup(SV, NULL); /* Set to the default so we have something */ SV->PolyMode = SRM_Fill; SV->TransMode = STM_0; SV->MouseMode = SUMA_DEF_MMODE; SV->MouseMode_ado_idcode_str = NULL; SV->DO_DrawMask = SDODM_All; #if SUMA_BACKFACE_CULL SV->BF_Cull = YUP; #else SV->BF_Cull = NOPE; #endif SV->ShowForeground = YUP; SV->ShowBackground = YUP; { char *eee = getenv("SUMA_CenterOnPatch"); if (eee) { if (strcasecmp (eee, "YES") == 0) SV->UsePatchDims = YUP; else SV->UsePatchDims = NOPE; } else { SV->UsePatchDims = NOPE; } } SV->Back_Modfact = SUMA_BACKGROUND_MODULATION_FACTOR; SV->isShaded = NOPE; SV->LinkAfniCrossHair = YUP; SV->ResetGLStateVariables = YUP; SV->NewGeom = NOPE; SV->BS = NULL; SV->ShowRight = YUP; SV->ShowLeft = YUP; SV->Record = NOPE; SV->rdc = SUMA_RDC_NOT_SET; SV->Blend_Mode = SUMA_NO_BLEND; SV->Do_3Drender = 0; memset(&(SV->SER), 0, sizeof(SUMA_EnablingRecord)); SV->DO_PickMode = 0; SV->pick_colid_list = NULL; SV->PickPix[0] = SV->PickPix[1] = -1; SV->pickrenpix4 = NULL; /* Squence of types to be displayed. Anything not in the list gets displayed first */ SV->N_otseq = SUMA_SetObjectDisplayOrder("DEFAULT", SV->otseq); { char *eee = getenv("SUMA_ObjectDisplayOrder"); if (eee) { if ((SV->N_otseq = SUMA_SetObjectDisplayOrder(eee, SV->otseq))<0){ SUMA_S_Warn("Failed to parse %s, reverting to default order", eee); SV->N_otseq = SUMA_SetObjectDisplayOrder("DEFAULT", SV->otseq); } } } SV->PryAx = 3; } SUMA_RETURN (SVv); } SUMA_PICK_RESULT *SUMA_Get_From_PickResult_List(SUMA_SurfaceViewer *sv, SUMA_ALL_DO *ado, char *variant) { static char FuncName[]={"SUMA_Get_From_PickResult_List"}; DListElmt *el = NULL; SUMA_SEL_ADO_DATUM *ss = NULL; SUMA_PICK_RESULT *PR = NULL; SUMA_Boolean LocalHead = NOPE; SUMA_ENTRY; if (!sv || !sv->SelAdo || !ado) SUMA_RETURN(NULL); if (!variant) variant = "none"; if (!dlist_size(sv->SelAdo)) SUMA_RETURN(NULL); SUMA_LH("Get %s", ADO_LABEL(ado)); el = NULL; do { if (!el) el = dlist_head(sv->SelAdo); else el = dlist_next(el); ss = (SUMA_SEL_ADO_DATUM *)el->data; if (!strcmp(ss->ado_idcode_str, ADO_ID(ado)) && !strcmp(ss->variant, variant)) { dlist_remove(sv->SelAdo, el, (void **)&ss); PR = ss->PR; ss->PR = NULL; SUMA_Free_SelectedDO_Datum(ss); ss = NULL; SUMA_RETURN(PR); } } while (el != dlist_tail(sv->SelAdo)); SUMA_RETURN(NULL); } SUMA_Boolean SUMA_Add_To_PickResult_List(SUMA_SurfaceViewer *sv, SUMA_ALL_DO *ado, char *variant, SUMA_PICK_RESULT **PR) { static char FuncName[]={"SUMA_Add_To_PickResult_List"}; DListElmt *el = NULL; SUMA_SEL_ADO_DATUM *ss = NULL; SUMA_Boolean LocalHead = NOPE; SUMA_ENTRY; if (!sv || !sv->SelAdo) SUMA_RETURN(NOPE); if (!variant) variant = "none"; if (ado) { /* Add */ SUMA_LH("Add %s", ADO_LABEL(ado)); ss = (SUMA_SEL_ADO_DATUM *) SUMA_calloc(1,sizeof(SUMA_SEL_ADO_DATUM)); ss->ado_idcode_str = SUMA_copy_string(ADO_ID(ado)); ss->variant = SUMA_copy_string(variant); if (PR) { ss->PR = *PR; *PR = NULL; } dlist_ins_next(sv->SelAdo, dlist_tail(sv->SelAdo), (void *)ss); } else if (!strcmp(variant,"TERSUM")) { /* empty entire list */ SUMA_LH("Cleanup your act %d entries", dlist_size(sv->SelAdo)); if (dlist_size(sv->SelAdo)) { while (el = dlist_head(sv->SelAdo)) { dlist_remove(sv->SelAdo, el, (void **)&ss); SUMA_Free_SelectedDO_Datum(ss); } } } else { SUMA_LH("Nothing done (%p, %s)", ado, variant); SUMA_DUMP_TRACE("Whodunnit?"); } SUMA_RETURN(YUP); } void SUMA_Show_PickList(DList *SelAdo, int detail, char *headstring, FILE *out) { static char FuncName[]={"SUMA_Show_PickList"}; char *s; SUMA_ENTRY; if (!out) out = SUMA_STDERR; if (headstring) fprintf(out, "%s", headstring); s = SUMA_PickList_Info(SelAdo, detail); if (s) fprintf(out, "%s", s); SUMA_ifree(s); SUMA_RETURNe; } char *SUMA_PickList_Info(DList *SelAdo, int detail) { static char FuncName[]={"SUMA_Show_PickList_Info"}; SUMA_STRING *SS = NULL; DListElmt *el = NULL; SUMA_SEL_ADO_DATUM *ss = NULL; char *s=NULL; SUMA_ALL_DO *ado=NULL; int ii; SUMA_Boolean LocalHead = NOPE; SUMA_ENTRY; SS = SUMA_StringAppend (NULL, NULL); if (!SelAdo) { SS = SUMA_StringAppend(SS, "NULL SelAdo"); goto OUT; } SS = SUMA_StringAppend_va(SS, "SelAdo list of %d entries\n", dlist_size(SelAdo)); el = NULL; ii = 0; do { if (!el) el = dlist_head(SelAdo); else el = dlist_next(el); if (!(ss = (SUMA_SEL_ADO_DATUM *)el->data)) { SS = SUMA_StringAppend_va(SS,"%d: NULL data!!!\n", ii); } else { if ((ado = SUMA_whichADOg(ss->ado_idcode_str))) { SS = SUMA_StringAppend_va(SS,"%d: ADO %s, variant %s\n", ii, ADO_LABEL(ado), ss->variant); }else { SS = SUMA_StringAppend_va(SS,"%d: ADO NULL!!!\n",ii); } } ++ii; } while (el != dlist_tail(SelAdo)); OUT: SS = SUMA_StringAppend (SS, NULL); s = SS->s; SUMA_free(SS); SUMA_RETURN(s); } SUMA_Boolean SUMA_Process_Selected_ADO(SUMA_SurfaceViewer *sv, int deepfirst) { static char FuncName[]={"SUMA_Process_Selected_ADO"}; DListElmt *el = NULL; SUMA_SEL_ADO_DATUM *ss = NULL; SUMA_ALL_DO *ado=NULL; int ii, nn, *isrt; SUMA_SEL_ADO_DATUM **ssv=NULL; float *Pxyz, *Sxyz; SUMA_Boolean LocalHead = NOPE; SUMA_ENTRY; if (!sv || !sv->SelAdo) SUMA_RETURN(NOPE); if (!(nn=dlist_size(sv->SelAdo))) { SUMA_LH("Empty list"); SUMA_RETURN(YUP); } if (LocalHead) { SUMA_Show_PickList(sv->SelAdo, 1, "Before sorting:\n", NULL); } if (nn > 1) { Pxyz = (float *)SUMA_calloc(3*nn, sizeof(float)); Sxyz = (float *)SUMA_calloc(3*nn, sizeof(float)); ssv = (SUMA_SEL_ADO_DATUM **)SUMA_calloc(nn, sizeof(SUMA_SEL_ADO_DATUM *)); el = NULL; ii = 0; do { if (!el) el = dlist_head(sv->SelAdo); else el = dlist_next(el); if (!(ss = (SUMA_SEL_ADO_DATUM *)el->data)) { SUMA_S_Err("NULL data!!!"); SUMA_RETURN(NOPE); } if ((ado = SUMA_whichADOg(ss->ado_idcode_str))) { /* Collect the PickXYZs */ Pxyz[3*ii] = ss->PR->PickXYZ[0]; Pxyz[3*ii+1] = ss->PR->PickXYZ[1]; Pxyz[3*ii+2] = ss->PR->PickXYZ[2]; ssv[ii] = ss; /* Keep a copy */ ++ii; } else { SUMA_LH("NULL ado?"); } } while (el != dlist_tail(sv->SelAdo)); nn = ii; /* in case we had null nonesense */ if (!nn) { SUMA_LH("Nothing good in here"); SUMA_ifree(ssv); SUMA_ifree(Pxyz); SUMA_ifree(Sxyz); SUMA_RETURN(YUP); } /* Now change all dems coords to eyeball land */ SUMA_World2ScreenCoordsF(sv, nn, Pxyz, Sxyz, NULL, YUP, NOPE); /* Sort by depth */ for (ii=0; iiSelAdo, dlist_head(sv->SelAdo), (void **)&ss); /* Do not delete ss, we have a copy in ssv */ } while (dlist_size(sv->SelAdo)); /* Now recreate in ordered fashion */ if (!deepfirst) { for (ii=0; iiSelAdo, dlist_head(sv->SelAdo), ssv[isrt[ii]]); ssv[isrt[ii]]=NULL; } } else { for (ii=0; iiSelAdo, dlist_head(sv->SelAdo), ssv[isrt[ii]]); ssv[isrt[ii]]=NULL; } } SUMA_ifree(isrt); SUMA_ifree(ssv); SUMA_ifree(Pxyz); SUMA_ifree(Sxyz); } if (LocalHead) { /* Show me the list after sorting */ if (deepfirst) { SUMA_Show_PickList(sv->SelAdo, 1, "After sorting deepest first:\n", NULL); } else { SUMA_Show_PickList(sv->SelAdo, 1, "After sorting deepest last:\n", NULL); } } /* Now apply the top pick */ SUMA_LH("On to apply the PR"); el = dlist_head(sv->SelAdo); ss = (SUMA_SEL_ADO_DATUM *)el->data; SUMA_Apply_PR(sv, &(ss->PR)); SUMA_LH("Returning"); SUMA_RETURN(YUP); } void SUMA_Free_SelectedDO_Datum(void *data) { SUMA_SEL_ADO_DATUM *ss = (SUMA_SEL_ADO_DATUM *)data; if (ss) { SUMA_ifree(ss->ado_idcode_str); SUMA_ifree(ss->variant); if (ss->PR) ss->PR = SUMA_free_PickResult(ss->PR); SUMA_free(ss); } } int SUMA_SV_GetShowSelectedDatum(SUMA_SurfaceViewer *sv) { static char FuncName[]={"SUMA_SV_GetShowSelectedDatum"}; if (!sv) return(0); return(sv->ShowSelectedDatum); } SUMA_Boolean SUMA_SV_SetShowSelectedDatum(SUMA_SurfaceViewer *sv, int act, int callback) { static char FuncName[]={"SUMA_SV_SetShowSelectedDatum"}; if (!sv) return(NOPE); sv->ShowSelectedDatum = act; XmToggleButtonSetState(sv->X->ViewMenu->mw[SW_ViewNodeInFocus], act, callback); return(YUP); } int SUMA_SV_GetShowSelectedFaceSet(SUMA_SurfaceViewer *sv) { static char FuncName[]={"SUMA_SV_GetShowSelectedFaceSet"}; if (!sv) return(0); return(sv->ShowSelectedFaceSet); } SUMA_Boolean SUMA_SV_SetShowSelectedFaceSet(SUMA_SurfaceViewer *sv, int act, int callback) { static char FuncName[]={"SUMA_SV_SetShowSelectedFaceSet"}; if (!sv || !sv->X) return(NOPE); sv->ShowSelectedFaceSet = act; XmToggleButtonSetState(sv->X->ViewMenu->mw[SW_ViewSelectedFaceset], act, callback); return(YUP); } SUMA_Boolean SUMA_Free_SurfaceViewer_Struct (SUMA_SurfaceViewer *SV) { static char FuncName[]={"SUMA_Free_SurfaceViewer_Struct"}; int i; SUMA_ENTRY; if (SV->WAx) SUMA_Free_Axis(SV->WAx); if (SV->C_filter) SUMA_C_free(SV->C_filter); SV->C_filter=NULL; if (SV->Ch) SUMA_Free_CrossHair (SV->Ch); if (SV->SelAdo) dlist_destroy(SV->SelAdo); SUMA_ifree(SV->SelAdo); SUMA_ifree(SV->LastSel_ado_idcode_str); if (SV->X->Title) SUMA_free(SV->X->Title); if (SV->X->LookAt_prmpt) SUMA_FreePromptDialogStruct (SV->X->LookAt_prmpt); if (SV->X->SetRot_prmpt) SUMA_FreePromptDialogStruct (SV->X->SetRot_prmpt); if (SV->X->JumpIndex_prmpt) SUMA_FreePromptDialogStruct (SV->X->JumpIndex_prmpt); if (SV->X->JumpXYZ_prmpt) SUMA_FreePromptDialogStruct (SV->X->JumpXYZ_prmpt); if (SV->X->JumpFocusNode_prmpt) SUMA_FreePromptDialogStruct (SV->X->JumpFocusNode_prmpt); if (SV->X->JumpFocusFace_prmpt) SUMA_FreePromptDialogStruct (SV->X->JumpFocusFace_prmpt); if (SV->X->HighlightBox_prmpt) SUMA_FreePromptDialogStruct (SV->X->HighlightBox_prmpt); if (SV->X->SetRenderOrder_prmpt) SUMA_FreePromptDialogStruct (SV->X->SetRenderOrder_prmpt); if (SV->X->ViewCont) SUMA_FreeViewContStruct(SV->X->ViewCont); if (SV->X) SUMA_free(SV->X); if (SV->RegistDO) SUMA_free(SV->RegistDO); if (SV->VSv) { for (i=0; i < SV->N_VSv; ++i) { if (!SUMA_Free_ViewState (&(SV->VSv[i]))) { fprintf (SUMA_STDERR, "Error %s: failed in SUMA_Free_ViewState.\n", FuncName); } } } SUMA_ifree(SV->pickrenpix4); SUMA_ifree(SV->MouseMode_ado_idcode_str); if (SV->CurGroupName) SUMA_free(SV->CurGroupName); SV->CurGroupName= NULL; if (SV->GVS) SUMA_free(SV->GVS); if (SV->State) SV->State = NULL; /* never free that one */ if (SV->ColList) { for (i=0; i < SV->N_ColList; ++i) { if (!SUMA_EmptyColorList (SV, NULL)) SUMA_S_Err("Failed in SUMA_EmptyColorList."); } /* done dumping structure contents, now free pointer */ SUMA_ifree(SV->ColList); SV->N_ColList = 0; } if (SV->BS) { SUMA_EmptyDestroyList(SV->BS); } SUMA_ifree(SV->FOV); SUMA_ifree(SV->FOV_last_PickMode); SUMA_ifree(SV->auto_FOV_val); SUMA_RETURN(YUP); } SUMA_Boolean SUMA_Free_SurfaceViewer_Struct_Vect (SUMA_SurfaceViewer *SVv, int N) { static char FuncName[]={"SUMA_Free_SurfaceViewer_Struct_Vect"}; int i; SUMA_Boolean Ret= YUP; SUMA_ENTRY; for (i=0; i < N; ++i) { if (&SVv[i] != NULL) { Ret = Ret * SUMA_Free_SurfaceViewer_Struct (&SVv[i]); } } if (SVv) SUMA_free(SVv); SUMA_RETURN(Ret); } /*! \brief SUMA_ADO_FillColorList_Params(SUMA_ALL_DO *ADO, int *N_points, char *idcode); */ SUMA_Boolean SUMA_ADO_FillColorList_Params(SUMA_ALL_DO *ADO, int *N_points, char **idcode) { static char FuncName[]={"SUMA_ADO_FillColorList_Params"}; SUMA_Boolean LocalHead = NOPE; SUMA_ENTRY; if (!ADO || !N_points || !idcode) { SUMA_RETURN(NOPE); } *N_points = -1; *idcode = NULL; switch (ADO->do_type) { case SO_type: { SUMA_SurfaceObject *SO=(SUMA_SurfaceObject *)ADO; *N_points = SO->N_Node; *idcode = SO->idcode_str; SUMA_LHv("Filling a color list for surface %s (%s).\n", SO->Label, SO->idcode_str); } break; case GDSET_type: { SUMA_DSET *dset=(SUMA_DSET *)ADO; if (!SUMA_isGraphDset(dset)) { SUMA_S_Err("Dataset should be graph type"); SUMA_RETURN(NOPE); } *N_points = 1+SUMA_GDSET_Max_Edge_Index(dset); *idcode = SDSET_ID(dset); SUMA_LHv("Filling a color list for dset %s (%s).\n", SDSET_LABEL(dset), SDSET_ID(dset)); } break; case GRAPH_LINK_type: { SUMA_DSET *dset=SUMA_find_GLDO_Dset((SUMA_GraphLinkDO *)ADO); if (dset) SUMA_RETURN(SUMA_ADO_FillColorList_Params( (SUMA_ALL_DO *)dset, N_points, idcode)); } break; case CDOM_type: case MASK_type: case TRACT_type: case VO_type: { *N_points = SUMA_ADO_N_Datum(ADO); *idcode = ADO_ID(ADO); SUMA_LHv("Filling a color list for tract %s (%s) %d points.\n", ADO_LABEL(ADO), ADO_ID(ADO), *N_points); break; } default: SUMA_S_Errv("Not ready for type %d (%s)\n", ADO->do_type, SUMA_ObjectTypeCode2ObjectTypeName(ADO->do_type)); SUMA_RETURN(NOPE); } SUMA_RETURN(YUP); } SUMA_Boolean SUMA_BlankColorListStruct(SUMA_COLORLIST_STRUCT *cl) { static char FuncName[]={"SUMA_BlankColorListStruct"}; int i = 0; SUMA_ENTRY; if (!cl || !cl->glar_ColorList_private) { SUMA_S_Err("NULL input"); SUMA_RETURN(NOPE); } /* fill up with blanks, may be unecessary ... */ i=0; while (i < cl->N_glar_ColorList) { cl->glar_ColorList_private[i] = SUMA_GRAY_NODE_COLOR; ++i; cl->glar_ColorList_private[i] = SUMA_GRAY_NODE_COLOR; ++i; cl->glar_ColorList_private[i] = SUMA_GRAY_NODE_COLOR; ++i; cl->glar_ColorList_private[i] = SUMA_NODE_ALPHA; ++i; } SUMA_RETURN(YUP); } /* Accessor function to pointer cl->glar_ColorList_private */ GLfloat * SUMA_GetColorListPtr (SUMA_COLORLIST_STRUCT *cl) { static char FuncName[]={"SUMA_GetColorListPtr"}; if (!cl) return(NULL); if (cl->glar_ColorList_private) { return(cl->glar_ColorList_private); } else if (cl->idcode_str) { SUMA_ALL_DO *ado = SUMA_whichADOg(cl->idcode_str); int N_points; char *idcode=NULL; SUMA_Boolean LocalHead = NOPE; SUMA_LH("Found but must allocate for colorlist"); if (!SUMA_ADO_FillColorList_Params(ado, &N_points, &idcode)) { SUMA_S_Err("Failed to initialize params"); return(NULL); } /* That previous call to get N_points may be a waste. The number of values is already in cl->N_glar_ColorList so just check for redundancy now */ if (4*N_points != cl->N_glar_ColorList) { SUMA_S_Err("This should not happen: %d clashes with %d", N_points, cl->N_glar_ColorList); return(NULL); } if (!(cl->glar_ColorList_private = (GLfloat *) SUMA_calloc (N_points*4, sizeof(GLfloat)))) { SUMA_S_Crit("Failed to allocate %d floats for glar_ColorList_private", N_points*4); return(NULL); } if (!SUMA_BlankColorListStruct(cl)) { SUMA_S_Err("Failed to fill with gray?!?"); return(NULL); } /* Will need mixing in all likelihood, go ahead, why not */ cl->Remix = YUP; ++cl->RemixID; return(cl->glar_ColorList_private); } return(NULL); } /*! \brief ans = SUMA_FillColorList (sv, so); Creates a colorlist structure for a certain displayable. \param sv (SUMA_SurfaceViewer *) pointer to surface viewer \param so (SUMA_SurfaceObject *) pointer to surface object Remix flag is set to YUP since this function is called when DOs are being registered with a viewer and a remix is highly likely. \return ans YUP/NOPE */ SUMA_Boolean SUMA_FillColorList (SUMA_SurfaceViewer *sv, SUMA_ALL_DO *ADO) { static char FuncName[]={"SUMA_FillColorList"}; int i, N_points; char *idcode=NULL; SUMA_COLORLIST_STRUCT *clinh=NULL; SUMA_Boolean LocalHead = NOPE; SUMA_ENTRY; if (!sv) { SUMA_RETURN(NOPE); } switch(ADO->do_type) { case GRAPH_LINK_type: { SUMA_DSET *dset=SUMA_find_GLDO_Dset((SUMA_GraphLinkDO *)ADO); if (dset) SUMA_RETURN(SUMA_FillColorList(sv,(SUMA_ALL_DO *)dset)); SUMA_S_Err("GLDO without dset!"); SUMA_RETURN(NOPE); } break; } if (!SUMA_ADO_FillColorList_Params(ADO, &N_points, &idcode) ) { SUMA_S_Err("Failed to initialize variables"); SUMA_RETURN(NOPE); } if (!idcode) { fprintf (SUMA_STDERR,"Error %s: idcode is NULL.\n", FuncName); SUMA_RETURN (NOPE); } /* make sure idcode is not in the list already */ for (i=0; iN_ColList; ++i) { if (strcmp (idcode, sv->ColList[i]->idcode_str) == 0) { if (ADO->do_type != GDSET_type && ADO->do_type != CDOM_type && ADO->do_type != TRACT_type && ADO->do_type != VO_type && ADO->do_type != MASK_type ) { SUMA_S_Err("idcode is already in sv->ColList, \n" "This is an error for a SurfaceObject, though I doubt\n" "it is of serious consequence. Type is question is %s\n" "for DO type %s\n", ADO_LABEL(ADO), ADO_TNAME(ADO)); SUMA_RETURN (NOPE); } else { SUMA_LH("List for %s found", SUMA_DO_dbg_info(idcode)); /* No harm done, no need to get upset. */ SUMA_RETURN(YUP); } } } SUMA_LH("Need to create or link to existing list." "will insert into position: %d", sv->N_ColList); if (sv->N_ColList >= SUMA_MAX_DISPLAYABLE_OBJECTS) { SUMA_SL_Crit("sv->N_ColList >= SUMA_MAX_DISPLAYABLE_OBJECTS"); SUMA_RETURN (NOPE); } /* create the ColList struct */ if (sv->ColList[sv->N_ColList]) { SUMA_S_Err("sv->ColorList[%d] is not NULL. Cannot reallocate.\n" "not expecting this scenario at this stage", sv->N_ColList); SUMA_RETURN (NOPE); } /* Should we inherit? */ clinh = NULL; switch (ADO->do_type) { case SO_type: /* Don't inherit, surfaces can have different colorings in different viewers */ break; case VO_type: /* Those beasts cost a lot of memory, share with sv[0] */ if (sv != SUMAg_SVv+0) { clinh = SUMA_GetColorListStruct (SUMAg_SVv, idcode); } break; default: /* For all others, make new copy, decide later whether should inherit */ break; } if (clinh) { SUMA_LH("Inheriting"); sv->ColList[sv->N_ColList] = (SUMA_COLORLIST_STRUCT*)SUMA_LinkToPointer((void *)clinh); } else { SUMA_LH("Recreating, N_points = %d", N_points); sv->ColList[sv->N_ColList] = (SUMA_COLORLIST_STRUCT *)SUMA_calloc(1, sizeof(SUMA_COLORLIST_STRUCT)); sv->ColList[sv->N_ColList]->N_links = 0; memset(sv->ColList[sv->N_ColList]->per_sv_extra, PSV_NOTHING, SUMA_MAX_SURF_VIEWERS*sizeof(int)); sv->ColList[sv->N_ColList]->owner_id[0] = '\0'; sv->ColList[sv->N_ColList]->LinkedPtrType = SUMA_LINKED_COLORLIST_TYPE; sv->ColList[sv->N_ColList]->do_type = not_DO_type; sv->ColList[sv->N_ColList]->N_glar_ColorList = N_points*4; sv->ColList[sv->N_ColList]->idcode_str = (char *)SUMA_malloc((strlen(idcode)+1) * sizeof(char)); if (!sv->ColList[sv->N_ColList]->idcode_str) { SUMA_S_Err("Failed to allocate for idcode_str."); SUMA_RETURN (NOPE); } sv->ColList[sv->N_ColList]->idcode_str = strcpy (sv->ColList[sv->N_ColList]->idcode_str, idcode); if (sv->Open) { /* ZSS: Feb 2014 */ /* Only allocate if Viewer is open, otherwise don't allocate for float vector, let SUMA_GetColorList() take care of that */ if (!SUMA_GetColorListPtr(sv->ColList[sv->N_ColList])) { SUMA_S_Err("Failed to allocate for glar_ColorList."); SUMA_RETURN (NOPE); } } else { sv->ColList[sv->N_ColList]->glar_ColorList_private = NULL; } } ++sv->N_ColList; SUMA_LH("N_ColList now %d, latest glar %p", sv->N_ColList, sv->ColList[sv->N_ColList-1]->glar_ColorList_private); SUMA_RETURN (YUP); } /*! glar_ColorList = SUMA_GetColorList (sv, DO_idstr); returns the pointer to the colorlist of the DO (or SO) with ID string DO_idstr \param sv (SUMA_SurfaceViewer *) pointer to surface viewer in question \param DO_idstr (char *) ID string of DO (usually a Surface Object) \return glar_ColorList (GLfloat *) a pointer to the array containing node colors \sa SUMA_GetColorListStruct, SUMA_GetColorListPtr */ GLfloat * SUMA_GetColorList (SUMA_SurfaceViewer *sv, char *DO_idstr) { static char FuncName[]={"SUMA_GetColorList"}; int i; SUMA_Boolean Found = NOPE; SUMA_Boolean LocalHead = NOPE; SUMA_ENTRY_LH; if (!DO_idstr) { SUMA_S_Err("DO_idstr is NULL, this should not be."); SUMA_RETURN (NULL); } SUMA_LH("Looking for colorlist of %s", SUMA_DO_dbg_info(DO_idstr)); /* find the culprit */ Found = NOPE; i = 0; while (!Found && i < sv->N_ColList) { if (strcmp (DO_idstr, sv->ColList[i]->idcode_str) == 0) { Found = YUP; SUMA_RETURN (SUMA_GetColorListPtr(sv->ColList[i])); } ++i; } #if 0 /* This should be deleted, but is left for Rick's joy at the moment. In the first pass, a CIFTI DO got its own colorplane and the colorization was done for the entire multi domain data set then to colorize a sub-domain the color list pointer was offset by the total number of data in sub-domains preceding it. That scheme however proved problematic when going back and forth between sub-domains and the whole dataset because datum indices in a multi-domain dataset are not unique. So this approach was scrapped in favor of dealing only with elementary datasets and domains */ if (!Found) { /* Perhaps this DO is part of a CIFTI conglomorate */ SUMA_CIFTI_DO *CO=NULL; int ksub; if ((CO = SUMA_find_CIFTI_subdom_container(DO_idstr, &ksub, NULL, 0))) { /* search again to find the pointer to the colorlist of the entire CIFTI domain */ Found = NOPE; i = 0; while (!Found && i < sv->N_ColList) { if (strcmp (ADO_ID((SUMA_ALL_DO *)CO), sv->ColList[i]->idcode_str) == 0) { Found = YUP; } else ++i; } if (Found) { GLfloat *glc=NULL; SUMA_LH( "Found CIFTI container's colorlist for id %s, offset of 4*%d", DO_idstr, SUMA_CIFTI_SubDomFullOffset(CO, ksub)); /* Now return the pointer offset to the sub-domain */ glc = SUMA_GetColorListPtr(sv->ColList[i])+ 4*SUMA_CIFTI_SubDomFullOffset(CO, ksub); SUMA_RETURN(glc); } else { SUMA_LH("Found cobwebs"); } } } #endif if (!Found) { SUMA_S_Err("DO_idstr %s was not found.\n", DO_idstr); SUMA_RETURN (NULL); } /* should not get to this point */ SUMA_S_Err("Logic error. Should not get here."); SUMA_RETURN (NULL); } /* Return the color list structure for a particular object */ SUMA_COLORLIST_STRUCT * SUMA_GetColorListStruct (SUMA_SurfaceViewer *sv, char *DO_idstr) { static char FuncName[]={"SUMA_GetColorListStruct"}; int i; SUMA_COLORLIST_STRUCT *cls = NULL; SUMA_Boolean Found = NOPE; SUMA_Boolean LocalHead = NOPE; SUMA_ENTRY_LH; if (!DO_idstr) { SUMA_S_Err("DO_idstr is NULL, this should not be."); SUMA_RETURN (NULL); } SUMA_LH("Looking for colorlist struct of %s", SUMA_DO_dbg_info(DO_idstr)); /* find the culprit */ Found = NOPE; i = 0; while (!Found && i < sv->N_ColList) { if (strcmp (DO_idstr, sv->ColList[i]->idcode_str) == 0) { Found = YUP; if (!SUMA_GetColorListPtr(sv->ColList[i])) { SUMA_RETURN(NULL); } SUMA_RETURN (sv->ColList[i]); } ++i; } if (!Found) { SUMA_S_Err("DO_idstr %s was not found.\n", DO_idstr); SUMA_RETURN (NULL); } /* should not get to this point */ SUMA_S_Err("Logic error. Should not get here."); SUMA_RETURN (NULL); } /*! frees the Node Neighbor structure formed in SUMA_Build_FirstNeighb */ SUMA_Boolean SUMA_Free_ColorList (SUMA_COLORLIST_STRUCT *cl) { static char FuncName[]={"SUMA_Free_ColorList"}; SUMA_Boolean LocalHead = NOPE; SUMA_ENTRY; if (!cl) SUMA_RETURN(YUP); SUMA_LH("Entered to free colorlist of %s, (glar = %p)", SUMA_DO_dbg_info(cl->idcode_str), cl->glar_ColorList_private); if (LocalHead) SUMA_DUMP_TRACE("At Free ColorList"); if (cl->N_links) { SUMA_LH("Just a link release"); cl = (SUMA_COLORLIST_STRUCT *)SUMA_UnlinkFromPointer((void *)cl); SUMA_RETURN (YUP); } /* no more links, go for it */ SUMA_LH("No more links freeing time"); SUMA_ifree(cl->idcode_str); SUMA_ifree(cl->glar_ColorList_private); SUMA_ifree(cl); SUMA_RETURN (YUP); } /*! \brief Empty a colorlist structure ans = SUMA_EmptyColorList (sv, DO_idstr) \param sv (SUMA_SurfaceViewer *) pointer to surface viewer in question \param DO_idstr (char *) ID string of DO (usually a Surface Object) If you want to delete all color lists, set this pointer to NULL \return ans (SUMA_Boolean) YUP/NOPE */ SUMA_Boolean SUMA_EmptyColorList (SUMA_SurfaceViewer *sv, char *DO_idstr) { static char FuncName[]={"SUMA_EmptyColorList"}; int i; SUMA_Boolean Found = NOPE; SUMA_Boolean LocalHead = NOPE; SUMA_ENTRY; if (!sv->ColList) { SUMA_S_Err("sv->ColList is NULL, this should not be."); SUMA_RETURN (NOPE); } SUMA_LH("Emptying collist for sv %p, id %s", sv, DO_idstr); if (!DO_idstr) { /* empty them all */ for (i=0; i < sv->N_ColList; ++i) { if (!SUMA_Free_ColorList(sv->ColList[i])) { SUMA_S_Err("Failed to free colorlist"); SUMA_RETURN(NOPE); } sv->ColList[i]=NULL; sv->N_ColList = 0; } } else { /* just empty one */ SUMA_LH("Emptying col list for %s", DO_idstr); Found = NOPE; i = 0; while (!Found && i < sv->N_ColList) { if (strcmp (DO_idstr, sv->ColList[i]->idcode_str) == 0) { Found = YUP; if (!SUMA_Free_ColorList(sv->ColList[i])) { SUMA_S_Err("Failed to free colorlist"); SUMA_RETURN(NOPE); } /* copy the last in the list here */ if (i < sv->N_ColList) { sv->ColList[i] = sv->ColList[sv->N_ColList-1]; /* mark the last element as empty */ sv->ColList[sv->N_ColList-1] = NULL; /* decrement the number of full elements in ColList */ --sv->N_ColList; } } ++i; } if (!Found) { SUMA_S_Errv("item %s (%s) was not found, this should not be.\n", DO_idstr, SUMA_DO_dbg_info(DO_idstr)); SUMA_DUMP_TRACE("Whence the unfound"); SUMA_RETURN (NOPE); } } SUMA_RETURN (YUP); } /*! ans = SUMA_SetShownLocalRemixFlag (SUMA_SurfaceViewer *sv) Set Remix flags for all surfaces in sv->RegistDO regardless of their relationship. This is useful when you change the settings for background color modulation and the like. \param sv (SUMA_SurfaceViewer *) pointer to surface viewer \return ans (SUMA_Boolean) YUP/NOPE \sa SUMA_SetRemixFlag \sa SUMA_SetLocalRemixFlag */ SUMA_Boolean SUMA_SetShownLocalRemixFlag (SUMA_SurfaceViewer *sv) { static char FuncName[]={"SUMA_SetShownLocalRemixFlag"}; int k; SUMA_ENTRY; for (k=0; k < sv->N_ColList; ++k) { sv->ColList[k]->Remix = YUP; } SUMA_RETURN (YUP); } /*! ans = SUMA_SetLocalRemixFlag (char *idcode_str, SUMA_SurfaceViewer *sv); Search RegistDO for sv and if a Surface in RegistDO is related to DO_idcode_str then its remix flag is set to yes. \param idcode_str (char *) IDcode of the surface that had its colorplanes modified \param sv (SUMA_SurfaceViewer *) pointer to surface viewer \return ans (SUMA_Boolean) YUP/NOPE \sa SUMA_SetRemixFlag \sa SUMA_SetShownLocalRemixFlag Will I ever use that one, not common to have related surfaces in one view ...? */ SUMA_Boolean SUMA_SetLocalRemixFlag (char *DO_idcode_str, SUMA_SurfaceViewer *sv) { static char FuncName[]={"SUMA_SetLocalRemixFlag"}; SUMA_SurfaceObject *SO1 = NULL, *SO2 = NULL; int k, kk; void *pp = NULL; SUMA_DSET *dset=NULL; char *p2=NULL; SUMA_ALL_DO *ado2=NULL; SUMA_DO_Types tp; SUMA_Boolean Found = NOPE; SUMA_ENTRY; if (!DO_idcode_str || !sv) { SUMA_S_Err("NULL sv or DO_idcode_str. BAD"); SUMA_RETURN (NOPE); } if (!(pp = SUMA_find_any_object(DO_idcode_str, &tp))) { SUMA_S_Errv("Bad id %s\n", DO_idcode_str); SUMA_RETURN (NOPE); } switch (tp) { case SO_type: SO1 = (SUMA_SurfaceObject *)pp; /* search for relatives in RegistDO */ for (k=0; k < sv->N_DO; ++k) { ado2 = (SUMA_ALL_DO *)SUMAg_DOv[sv->RegistDO[k].dov_ind].OP; if (SUMA_isRelated((SUMA_ALL_DO *)SO1, ado2, 1)){ /* kinship of da first order */ SO2 = (SUMA_SurfaceObject *)SUMAg_DOv[sv->RegistDO[k].dov_ind].OP; /* related, set flag for remixing SO2 */ kk = 0; Found = NOPE; while (!Found && kk < sv->N_ColList) { if (strcmp(SO2->idcode_str,sv->ColList[kk]->idcode_str) == 0) { Found = YUP; sv->ColList[kk]->Remix = YUP; } ++kk; } if (!Found) { SUMA_S_Err("Failed to find surface in ColList structs. BAD."); SUMA_RETURN (NOPE); } } } break; case CDOM_type: SUMA_S_Err("Is this needed (perhaps once we have isotopic COs ? " "If so then do it"); SUMA_RETURN (NOPE); break; case GDSET_type: dset = (SUMA_DSET *)pp; for (k=0; k < sv->N_DO; ++k) { ado2 = (SUMA_ALL_DO *)SUMAg_DOv[sv->RegistDO[k].dov_ind].OP; if (SUMA_isRelated((SUMA_ALL_DO *)dset, ado2, 1)){ kk = 0; Found = NOPE; while (!Found && kk < sv->N_ColList) { if ((p2 = SUMA_ADO_idcode(ado2)) && !strcmp(p2, sv->ColList[kk]->idcode_str)) { Found = YUP; sv->ColList[kk]->Remix = YUP; } ++kk; } if (!Found) { SUMA_S_Err("Failed to find DSET in ColList structs. BAD."); SUMA_RETURN (NOPE); } } } break; default: SUMA_S_Errv("Type %d is not welcome here\n", tp); SUMA_RETURN(NOPE); } SUMA_RETURN (YUP); } /*! ans = SUMA_SetRemixFlag (char *idcode_str, SUMA_SurfaceViewer *SVv, int N_SVv); Search RegistDO for each Surface Viewer and if a Surface in RegistDO is related to DO_idcode_str then its remix flag is set to yes. \param idcode_str (char *) IDcode of the surface that had its colorplanes modified \param SVv (SUMA_SurfaceViewer *) vector of existing surface viewers (typically, that is SUMAg_SVv) \param N_SVv (int) number of surface viewers (typically that is N_SUMAg_SVv) \return ans (SUMA_Boolean) YUP/NOPE \sa SUMA_SetLocalRemixFlag \sa SUMA_SetShownLocalRemix DO NOT SET THE REMIXFLAG unless you have modified the colorplanes of a certain surface. This function will set a remix flags to all related surfaces that are being displayed in all viewers. You want to do this when one (or all) of the colorplanes is changed. Alternately, if you make changes that only affect the surface as is it shown in the viewer (change background modulation for example), you want to do the remixing for the concerned surface or surfaces only in that viewer and not in all viewers open. Perhaps I should write a function to set the remix flags for surfaces within the viewer only. Something like SUMA_SetLocalRemixFlag or SUMA_SetShownLocalRemix. */ SUMA_Boolean SUMA_SetRemixFlag (char *DO_idcode_str, SUMA_SurfaceViewer *SVv, int N_SVv) { static char FuncName[]={"SUMA_SetRemixFlag"}; SUMA_SurfaceViewer *sv; int i, k, kk; void *pp=NULL; SUMA_DO_Types tp; SUMA_ALL_DO*ado2; char *p2=NULL; SUMA_Boolean Found = NOPE; SUMA_Boolean LocalHead = NOPE; SUMA_ENTRY; if (!DO_idcode_str || !SVv) { fprintf (SUMA_STDERR, "Error %s: NULL SVv or DO_idcode_str. BAD\n", FuncName); SUMA_RETURN (NOPE); } SUMA_LH("Look for any object"); SUMA_LHv(" id %s\n", DO_idcode_str); if (!(pp = SUMA_find_any_object(DO_idcode_str, &tp))) { SUMA_S_Errv("Bad id %s\n", DO_idcode_str); SUMA_RETURN (NOPE); } SUMA_LH("Switching..."); switch (tp) { case SO_type: { SUMA_SurfaceObject *SO1 = NULL, *SO2 = NULL; SO1 = (SUMA_SurfaceObject *)pp; /* search all viewers */ for (i=0; i < N_SVv; ++i) { if (LocalHead) fprintf (SUMA_STDERR,"%s: Searching viewer %d.\n", FuncName, i); sv = &(SVv[i]); /* search for relatives in RegistDO */ for (k=0; k < sv->N_DO; ++k) { if (SUMA_isSO(SUMAg_DOv[sv->RegistDO[k].dov_ind])) { SO2 = (SUMA_SurfaceObject *) SUMAg_DOv[sv->RegistDO[k].dov_ind].OP; if (SUMA_isRelated_SO(SO1, SO2, 1)) { /* only 1st order kinship allowed */ /* related, set flag for remixing SO2 */ kk = 0; Found = NOPE; while (!Found && kk < sv->N_ColList) { if (strcmp (SO2->idcode_str, sv->ColList[kk]->idcode_str) == 0) { Found = YUP; SUMA_LHv("Setting remix for %d\n", kk); sv->ColList[kk]->Remix = YUP; } ++kk; } if (!Found) { SUMA_S_Err("Failed to find surface in ColList structs."); SUMA_RETURN (NOPE); } } } } } break; } case VO_type: case CDOM_type: case MASK_type: case TRACT_type: { SUMA_ALL_DO *ADO = (SUMA_ALL_DO*)pp; /* search all viewers */ for (i=0; i < N_SVv; ++i) { if (LocalHead) fprintf (SUMA_STDERR,"%s: Searching viewer %d.\n", FuncName, i); sv = &(SVv[i]); /* search for relatives in RegistDO */ for (k=0; k < sv->N_DO; ++k) { if (iDO_isTDO(sv->RegistDO[k].dov_ind) || iDO_isVO(sv->RegistDO[k].dov_ind)) { ado2 = (SUMA_ALL_DO *)SUMAg_DOv[sv->RegistDO[k].dov_ind].OP; if (SUMA_isRelated(ADO, ado2, 1)) { SUMA_LHv("%s RELATED to %s\n", SUMA_ADO_Label(ADO), SUMA_ADO_Label(ado2)); kk = 0; Found = NOPE; while (!Found && kk < sv->N_ColList) { p2=SUMA_ADO_idcode(ado2); if ( p2 && !strcmp(p2, sv->ColList[kk]->idcode_str)) { Found = YUP; sv->ColList[kk]->Remix = YUP; } ++kk; } if (!Found) { SUMA_S_Err("No relative in ColList structs. BAD."); SUMA_RETURN (NOPE); } } else { SUMA_LHv("%s Not related to %s\n", SUMA_ADO_Label(ADO), SUMA_ADO_Label(ado2)); } } } } break; } case GDSET_type: { SUMA_DSET *dset = (SUMA_DSET *)pp; if (!dset) { SUMA_S_Errv("Failed to find dset for dset %p, %s\n", pp, SUMA_ADO_Label((SUMA_ALL_DO *)pp)); SUMA_RETURN(NOPE); } /* search all viewers */ for (i=0; i < N_SVv; ++i) { if (LocalHead) fprintf (SUMA_STDERR,"%s: Searching viewer %d.\n", FuncName, i); sv = &(SVv[i]); /* search for relatives in RegistDO */ for (k=0; k < sv->N_DO; ++k) { if (iDO_isGLDO(sv->RegistDO[k].dov_ind)) { ado2 = (SUMA_ALL_DO *)SUMAg_DOv[sv->RegistDO[k].dov_ind].OP; if (SUMA_isRelated((SUMA_ALL_DO *)dset, ado2, 1)) { SUMA_LHv("%s RELATED to %s\n", SUMA_ADO_Label((SUMA_ALL_DO *)dset), SUMA_ADO_Label(ado2)); kk = 0; Found = NOPE; while (!Found && kk < sv->N_ColList) { switch (ado2->do_type) { case GRAPH_LINK_type: /* The idcode in ColList is that of the dataset for GLDO */ p2 = SDSET_ID(SUMA_find_GLDO_Dset( (SUMA_GraphLinkDO *)ado2)); break; default: p2=SUMA_ADO_idcode(ado2); break; } if ( p2 && !strcmp(p2, sv->ColList[kk]->idcode_str)) { Found = YUP; sv->ColList[kk]->Remix = YUP; } ++kk; } if (!Found) { SUMA_S_Err("No relative in ColList structs. BAD."); SUMA_RETURN (NOPE); } } else { SUMA_LHv("%s Not related to %s\n", SUMA_ADO_Label((SUMA_ALL_DO *)dset), SUMA_ADO_Label(ado2)); } } } } break; } case GRAPH_LINK_type: { SUMA_GraphLinkDO *gldo = (SUMA_GraphLinkDO *)pp; SUMA_DSET *dset = SUMA_find_GLDO_Dset(gldo); SUMA_S_Warn("So do I need to call function with dset for you?"); SUMA_RETURN (YUP); break; } default: SUMA_S_Errv("Type %d is not welcome here\n", tp); SUMA_RETURN(NOPE); } SUMA_RETURN (YUP); } /*! \brief sets remix flags for all color lists in viewers specified in SVv Use this function whenever global color changes occur \sa SUMA_SetRemixFlag for detailed help */ SUMA_Boolean SUMA_SetAllRemixFlag (SUMA_SurfaceViewer *SVv, int N_SVv) { static char FuncName[]={"SUMA_SetAllRemixFlag"}; SUMA_SurfaceViewer *sv; int i, kk; SUMA_Boolean LocalHead = NOPE; SUMA_ENTRY; if (!SVv) { fprintf (SUMA_STDERR,"Error %s: NULL SVv . BAD\n", FuncName); SUMA_RETURN (NOPE); } /* search all viewers */ for (i=0; i < N_SVv; ++i) { if (LocalHead) fprintf (SUMA_STDERR,"%s: Searching viewer %d.\n", FuncName, i); sv = &(SVv[i]); for (kk = 0; kk < sv->N_ColList; ++kk) sv->ColList[kk]->Remix = YUP; } SUMA_RETURN (YUP); } SUMA_Boolean SUMA_SetGLHome(SUMA_SurfaceViewer *sv) { static char FuncName[]={"SUMA_SetGLHome"}; SUMA_ENTRY; if (!sv) SUMA_RETURN(NOPE); SUMA_SET_AS_NEEDED_2D_VIEW_ANGLE(sv); sv->GVS[sv->StdView].translateVec[0]=0; sv->GVS[sv->StdView].translateVec[1]=0; glMatrixMode(GL_PROJECTION); sv->GVS[sv->StdView].ViewFrom[0] = sv->GVS[sv->StdView].ViewFromOrig[0]; sv->GVS[sv->StdView].ViewFrom[1] = sv->GVS[sv->StdView].ViewFromOrig[1]; sv->GVS[sv->StdView].ViewFrom[2] = sv->GVS[sv->StdView].ViewFromOrig[2]; sv->GVS[sv->StdView].ViewCenter[0] = sv->GVS[sv->StdView].ViewCenterOrig[0]; sv->GVS[sv->StdView].ViewCenter[1] = sv->GVS[sv->StdView].ViewCenterOrig[1]; sv->GVS[sv->StdView].ViewCenter[2] = sv->GVS[sv->StdView].ViewCenterOrig[2]; glMatrixMode(GL_MODELVIEW); glLoadIdentity(); gluLookAt ( sv->GVS[sv->StdView].ViewFrom[0], sv->GVS[sv->StdView].ViewFrom[1], sv->GVS[sv->StdView].ViewFrom[2], sv->GVS[sv->StdView].ViewCenter[0], sv->GVS[sv->StdView].ViewCenter[1], sv->GVS[sv->StdView].ViewCenter[2], sv->GVS[sv->StdView].ViewCamUp[0], sv->GVS[sv->StdView].ViewCamUp[1], sv->GVS[sv->StdView].ViewCamUp[2]); SUMA_RETURN(YUP); } DListElmt *SUMA_Fetch_VisX_Element(char *label, DList *dl) { static char FuncName[]={"SUMA_Fetch_VisX_Element"}; DListElmt *el=NULL, *ref=NULL; SUMA_VIS_XFORM_DATUM *uu=NULL; SUMA_ENTRY; if (!label || !dl) { SUMA_S_Err("NULL label or NULL list"); SUMA_RETURN(ref); } if (!dlist_size(dl)) SUMA_RETURN(ref); el = NULL; do { if (!el) el = dlist_head(dl); else el = dlist_next(el); uu = (SUMA_VIS_XFORM_DATUM *)el->data; if (uu && !strcmp(uu->label, label)) { ref = el; } } while(!ref && el != dlist_tail(dl)); SUMA_RETURN(ref); } SUMA_VIS_XFORM_DATUM *SUMA_Fetch_VisX_Datum (char *label, DList *dl, SUMA_VISX_ADD_POSITIONS add, char *ref_pos_label) { static char FuncName[]={"SUMA_Fetch_VisX_Datum"}; SUMA_VIS_XFORM_DATUM *xx=NULL, *uu=NULL; DListElmt *el=NULL, *ref=NULL; SUMA_ENTRY; if (!label || !dl) { SUMA_S_Err("NULL label or NULL list"); SUMA_RETURN(xx); } if (!dlist_size(dl)) { if (add != ADD_NOT) { xx = SUMA_NewVisXdatum(label); dlist_ins_next(dl, dlist_tail(dl), xx); } else { SUMA_RETURN(xx); } } else { if ((el=SUMA_Fetch_VisX_Element(label, dl))) { if ((xx = (SUMA_VIS_XFORM_DATUM *)el->data)) { SUMA_RETURN(xx); } } /* should we add, and if so, where ? */ if (add != ADD_NOT) { if (ref_pos_label) { ref = SUMA_Fetch_VisX_Element(ref_pos_label, dl); } xx = SUMA_NewVisXdatum(label); if (ref) { switch (add) { case ADD_AFTER: dlist_ins_next(dl, ref, xx); break; case ADD_BEFORE: dlist_ins_prev(dl, ref, xx); break; default: SUMA_S_Errv("Bad add=%d\n", add); break; } } else { switch (add) { case ADD_AFTER: dlist_ins_next(dl, dlist_tail(dl), xx); break; case ADD_BEFORE: dlist_ins_prev(dl, dlist_head(dl), xx); break; default: SUMA_S_Errv("Bad add=%d\n", add); break; } } } SUMA_RETURN(xx); } /* should not get here...*/ SUMA_RETURN(xx); } SUMA_Boolean SUMA_Apply_VisX_Chain(float *xyz, int N, DList *dl, int inv) { static char FuncName[]={"SUMA_Apply_VisX_Chain"}; SUMA_VIS_XFORM_DATUM *xx=NULL; DListElmt *el=NULL; int ii, iii, N3; SUMA_Boolean LocalHead = NOPE; SUMA_ENTRY; if (!dl || !xyz) { SUMA_S_Errv("No list (%p) or coords (%p)\n", dl, xyz); SUMA_RETURN(NOPE); } if (!dlist_size(dl)) { SUMA_S_Err("Empty list"); SUMA_RETURN(NOPE); } N3 = 3*N; if (!inv) { do { if (!el) el = dlist_head(dl); else el = dlist_next(el); xx = (SUMA_VIS_XFORM_DATUM *)el->data; switch (xx->XformType) { case ID: break; case SHIFT: SUMA_LHv("Applying shift %s\n", xx->label); for (ii=0, iii=0; iiXform[0][3]; ++iii; xyz[iii] += xx->Xform[1][3]; ++iii; xyz[iii] += xx->Xform[2][3]; ++iii; } break; case AFFINE: SUMA_LHv("Applying affine %s\n", xx->label); SUMA_Apply_Coord_xform( xyz,N,3, xx->Xform, 0, NULL); break; case DISP: SUMA_LHv("Applying disp (%p) %s\n", xx->dxyz, xx->label); if (!xx->dxyz) SUMA_RETURN(NOPE); for (iii=0; iiidxyz[iii]; } break; default: SUMA_RETURN(NOPE); break; } } while (el != dlist_tail(dl)) ; } else { do { if (!el) el = dlist_tail(dl); else el = dlist_prev(el); xx = (SUMA_VIS_XFORM_DATUM *)el->data; switch (xx->XformType) { case ID: break; case SHIFT: SUMA_LHv("Removing shift %s\n", xx->label); for (ii=0, iii=0; iiXform[0][3]; ++iii; xyz[iii] -= xx->Xform[1][3]; ++iii; xyz[iii] -= xx->Xform[2][3]; ++iii; } break; case AFFINE: SUMA_LHv("Removing affine %s\n", xx->label); SUMA_Apply_Coord_xform( xyz,N,3, xx->Xform, 1, NULL); break; case DISP: SUMA_LHv("Removing displacement (%p) %s\n", xx->dxyz, xx->label); if (!xx->dxyz) SUMA_RETURN(NOPE); for (iii=0; iiidxyz[iii]; } break; default: SUMA_RETURN(NOPE); break; } } while (el != dlist_head(dl)) ; } SUMA_RETURN(YUP); } /*! Update view points of all viewers with this ADO registered */ SUMA_Boolean SUMA_UpdateViewPoint_RegisteredADO(SUMA_ALL_DO *ado, byte keepzoom) { static char FuncName[]={"SUMA_UpdateViewPoint_RegisteredADO"}; int ii; SUMA_SurfaceViewer *sv=NULL; SUMA_ENTRY; if (!ado) SUMA_RETURN(NOPE); for (ii=0; iiN_DO) { do_id = SV->RegistDO[i].dov_ind; switch (dov[do_id].ObjectType) { case SO_type: so_op = (SUMA_SurfaceObject *)dov[do_id].OP; if (SV->UsePatchDims) { SUMA_LH("Using patch center"); SUMA_COPY_VEC(so_op->patchCenter, UsedCenter, 3, float, float); } else { SUMA_LH("Using center of mass or sphere's center."); if (!SUMA_IS_GEOM_SYMM(so_op->isSphere) || so_op->VisX.Applied) { SUMA_COPY_VEC(so_op->Center, UsedCenter, 3, float, float); } else { SUMA_COPY_VEC( so_op->SphereCenter, UsedCenter, 3, float, float); } } if (so_op->VisX.Applied) { if (!SUMA_Apply_VisX_Chain( UsedCenter,1,so_op->VisX.Xchain, 0)) { SUMA_S_Warn("Sir! Have you no decency left?"); } } if (so_op->ViewCenterWeight) { NewCenter[0] += so_op->ViewCenterWeight*UsedCenter[0]; NewCenter[1] += so_op->ViewCenterWeight*UsedCenter[1]; NewCenter[2] += so_op->ViewCenterWeight*UsedCenter[2]; TotWeight += so_op->ViewCenterWeight; } break; case GRAPH_LINK_type: if(!(dset=SUMA_find_GLDO_Dset((SUMA_GraphLinkDO *)(dov[do_id].OP)))){ SUMA_S_Err("Gildaaaaaaaaaaaaaaaaaa"); SUMA_RETURN(NOPE); } if (!SUMA_IS_REAL_VARIANT(iDO_variant(do_id))) break; xyzr = SUMA_GDSET_XYZ_Center(dset, iDO_variant(do_id), NULL); NewCenter[0] += GDSET_N_SEG_POINTS(dset)*xyzr[0]; NewCenter[1] += GDSET_N_SEG_POINTS(dset)*xyzr[1]; NewCenter[2] += GDSET_N_SEG_POINTS(dset)*xyzr[2]; TotWeight += GDSET_N_SEG_POINTS(dset); break; case TRACT_type: { SUMA_TractDO *TDO=(SUMA_TractDO *)dov[do_id].OP; int N_tracts; if (TDO_HAS_GRID(TDO)) { xyzr = SUMA_TDO_Grid_Center(TDO, NULL); } else { xyzr = SUMA_TDO_Points_Center(TDO, NULL); } N_tracts = SUMA_TDO_N_tracts(TDO); NewCenter[0] += N_tracts*xyzr[0]; NewCenter[1] += N_tracts*xyzr[1]; NewCenter[2] += N_tracts*xyzr[2]; TotWeight += N_tracts; break; } case MASK_type: { int N_pts; SUMA_MaskDO *MDO=(SUMA_MaskDO *)dov[do_id].OP; xyzr = SUMA_MDO_Center(MDO, NULL); N_pts = SUMA_ADO_N_Datum((SUMA_ALL_DO*)MDO); NewCenter[0] += N_pts*xyzr[0]; NewCenter[1] += N_pts*xyzr[1]; NewCenter[2] += N_pts*xyzr[2]; TotWeight += N_pts; break; } case VO_type: { SUMA_VolumeObject *VO=(SUMA_VolumeObject *)dov[do_id].OP; xyzr = SUMA_VO_Grid_Center(VO, NULL); NewCenter[0] += VO_NVOX(VO)*xyzr[0]; NewCenter[1] += VO_NVOX(VO)*xyzr[1]; NewCenter[2] += VO_NVOX(VO)*xyzr[2]; TotWeight += VO_NVOX(VO); break; } default: SUMA_LHv("Ignoring object %s, variant %s\n", iDO_label(do_id), iDO_variant(do_id)); break; } ++i; } odelta = SV->GVS[SV->StdView].ViewFrom[2] -SV->GVS[SV->StdView].ViewCenter[2]; oviewd = SV->GVS[SV->StdView].ViewDistance; if (TotWeight) { SV->GVS[SV->StdView].ViewCenter[0] = NewCenter[0]/(float)TotWeight; SV->GVS[SV->StdView].ViewCenter[1] = NewCenter[1]/(float)TotWeight; SV->GVS[SV->StdView].ViewCenter[2] = NewCenter[2]/(float)TotWeight; SV->GVS[SV->StdView].ViewFrom[0] = SV->GVS[SV->StdView].ViewCenter[0]; SV->GVS[SV->StdView].ViewFrom[1] = SV->GVS[SV->StdView].ViewCenter[1]; if (!KeepZoom) { SV->GVS[SV->StdView].ViewFrom[2] = SV->GVS[SV->StdView].ViewCenter[2]+ SUMA_DEFAULT_VIEW_FROM/SV->GVS[SV->StdView].DimSclFac; SV->GVS[SV->StdView].ViewDistance = SUMA_DEFAULT_VIEW_FROM/SV->GVS[SV->StdView].DimSclFac; } else { SV->GVS[SV->StdView].ViewFrom[2] = SV->GVS[SV->StdView].ViewCenter[2]+odelta; SV->GVS[SV->StdView].ViewDistance = oviewd; } } else {/* default back to o.o, o.o, o.o */ SV->GVS[SV->StdView].ViewCenter[0] = SV->GVS[SV->StdView].ViewCenter[1] = SV->GVS[SV->StdView].ViewCenter[2] = 0.0; SV->GVS[SV->StdView].ViewFrom[0] = SV->GVS[SV->StdView].ViewFrom[1] = 0.0; if (!KeepZoom) { SV->GVS[SV->StdView].ViewFrom[2] = SUMA_DEFAULT_VIEW_FROM/SV->GVS[SV->StdView].DimSclFac; SV->GVS[SV->StdView].ViewDistance = SUMA_DEFAULT_VIEW_FROM/SV->GVS[SV->StdView].DimSclFac; } } /* Store that info in case subjects change things */ SV->GVS[SV->StdView].ViewCenterOrig[0] = SV->GVS[SV->StdView].ViewCenter[0]; SV->GVS[SV->StdView].ViewCenterOrig[1] = SV->GVS[SV->StdView].ViewCenter[1]; SV->GVS[SV->StdView].ViewCenterOrig[2] = SV->GVS[SV->StdView].ViewCenter[2]; SV->GVS[SV->StdView].ViewFromOrig[0] = SV->GVS[SV->StdView].ViewFrom[0]; SV->GVS[SV->StdView].ViewFromOrig[1] = SV->GVS[SV->StdView].ViewFrom[1]; SV->GVS[SV->StdView].ViewFromOrig[2] = SV->GVS[SV->StdView].ViewFrom[2]; SUMA_RETURN (YUP); } /*! Updates the Rotation Center of SV based on the contents of RegistDO */ SUMA_Boolean SUMA_UpdateRotaCenter ( SUMA_SurfaceViewer *SV, SUMA_DO *dov, int N_dov) { static char FuncName[]={"SUMA_UpdateRotaCenter"}; int i, do_id, TotWeight; float NewCenter[3], UsedCenter[3]; float *xyzr; SUMA_SurfaceObject *so_op; SUMA_DSET *dset=NULL; SUMA_Boolean LocalHead = NOPE; SUMA_ENTRY; NewCenter[0] = 0.0; NewCenter[1] = 0.0; NewCenter[2] = 0.0; TotWeight = 0; i = 0; while (i < SV->N_DO) { do_id = SV->RegistDO[i].dov_ind; switch (dov[do_id].ObjectType) { case SO_type: so_op = (SUMA_SurfaceObject *)dov[do_id].OP; if (SV->UsePatchDims) { SUMA_COPY_VEC(so_op->patchCenter, UsedCenter, 3, float, float); } else { if (SUMA_IS_GEOM_SYMM(so_op->isSphere) && !so_op->VisX.Applied) { SUMA_COPY_VEC(so_op->SphereCenter, UsedCenter, 3, float, float); } else { SUMA_COPY_VEC(so_op->Center, UsedCenter, 3, float, float); } } SUMA_LHv("Used Center (%s) Pre Xform: [%f %f %f]\n", so_op->Label, UsedCenter[0], UsedCenter[1], UsedCenter[2]); if (so_op->VisX.Applied) { if (!SUMA_Apply_VisX_Chain( UsedCenter,1,so_op->VisX.Xchain, 0)) { SUMA_S_Warn("Oh please don't break my heart."); } } SUMA_LHv("Used Center (%s) Post Xform: [%f %f %f]\n", so_op->Label, UsedCenter[0], UsedCenter[1], UsedCenter[2]); if (so_op->RotationWeight) { NewCenter[0] += so_op->RotationWeight*UsedCenter[0]; NewCenter[1] += so_op->RotationWeight*UsedCenter[1]; NewCenter[2] += so_op->RotationWeight*UsedCenter[2]; TotWeight += so_op->RotationWeight; } break; case GRAPH_LINK_type: if(!(dset=SUMA_find_GLDO_Dset((SUMA_GraphLinkDO *)(dov[do_id].OP)))){ SUMA_S_Err("Gildaaaaaaaaaaaaaaaaaa"); SUMA_RETURN(NOPE); } if (!SUMA_IS_REAL_VARIANT(iDO_variant(do_id))) break; xyzr = SUMA_GDSET_XYZ_Center(dset, iDO_variant(do_id), NULL); NewCenter[0] += GDSET_N_SEG_POINTS(dset)*xyzr[0]; NewCenter[1] += GDSET_N_SEG_POINTS(dset)*xyzr[1]; NewCenter[2] += GDSET_N_SEG_POINTS(dset)*xyzr[2]; TotWeight += GDSET_N_SEG_POINTS(dset); break; case MASK_type: { int N_pts; SUMA_MaskDO *MDO=(SUMA_MaskDO *)dov[do_id].OP; xyzr = SUMA_MDO_Center(MDO, NULL); N_pts = SUMA_ADO_N_Datum((SUMA_ALL_DO *)MDO); NewCenter[0] += N_pts*xyzr[0]; NewCenter[1] += N_pts*xyzr[1]; NewCenter[2] += N_pts*xyzr[2]; TotWeight += N_pts; break; } case TRACT_type: { SUMA_TractDO *TDO=(SUMA_TractDO *)dov[do_id].OP; int N_tracts; if (TDO_HAS_GRID(TDO)) { xyzr = SUMA_TDO_Grid_Center(TDO, NULL); } else { xyzr = SUMA_TDO_Points_Center(TDO, NULL); } N_tracts = SUMA_TDO_N_tracts(TDO); NewCenter[0] += N_tracts*xyzr[0]; NewCenter[1] += N_tracts*xyzr[1]; NewCenter[2] += N_tracts*xyzr[2]; TotWeight += N_tracts; break; } case VO_type: { SUMA_VolumeObject *VO=(SUMA_VolumeObject *)dov[do_id].OP; xyzr = SUMA_VO_Grid_Center(VO, NULL); NewCenter[0] += VO_NVOX(VO)*xyzr[0]; NewCenter[1] += VO_NVOX(VO)*xyzr[1]; NewCenter[2] += VO_NVOX(VO)*xyzr[2]; TotWeight += VO_NVOX(VO); break; } case CDOM_type: { int kkk; xyzr = SUMA_ADO_Center((SUMA_ALL_DO *)dov[do_id].OP, NULL); kkk = SUMA_ADO_N_Datum((SUMA_ALL_DO *)dov[do_id].OP); NewCenter[0] += kkk*xyzr[0]; NewCenter[1] += kkk*xyzr[1]; NewCenter[2] += kkk*xyzr[2]; TotWeight += kkk; break; } default: if (SUMA_is_iDO_Selectable(do_id)) { static int nwarn=0; if (!nwarn) { SUMA_S_Warnv("Just ignored DO %s, I hope that's cool\n", iDO_label(do_id)); } ++nwarn; } SUMA_LHv("Ignoring object %s, variant %s\n", iDO_label(do_id), iDO_variant(do_id)); break; } ++i; } if (TotWeight) { SV->GVS[SV->StdView].RotaCenter[0] = NewCenter[0]/(float)TotWeight; SV->GVS[SV->StdView].RotaCenter[1] = NewCenter[1]/(float)TotWeight; SV->GVS[SV->StdView].RotaCenter[2] = NewCenter[2]/(float)TotWeight; } else {/* default back to o.o, o.o, o.o */ SV->GVS[SV->StdView].RotaCenter[0] = SV->GVS[SV->StdView].RotaCenter[1] = SV->GVS[SV->StdView].RotaCenter[2] = 0.0; } SUMA_RETURN (YUP); } /*! output the state variable contents of the Surface Viewer */ void SUMA_Show_SurfaceViewer_Struct (SUMA_SurfaceViewer *SV, FILE *Out, int detail) { static char FuncName[]={"SUMA_Show_SurfaceViewer_Struct"}; char *s = NULL; SUMA_ENTRY; if (Out == NULL) Out = stdout; s = SUMA_SurfaceViewer_StructInfo (SV, detail); if (s) { fprintf(Out, "%s", s); SUMA_free(s); s = NULL; }else { SUMA_SL_Err("Failed in SUMA_SurfaceViewer_StructInfo"); } SUMA_RETURNe; } SUMA_Boolean SUMA_isViewerStateAnatomical(SUMA_SurfaceViewer *sv) { int i; if (!sv) return(NOPE); for (i=0; i < sv->N_VSv; ++i) { if (!strcmp(sv->State, sv->VSv[i].Name)) return(sv->VSv[i].AnatCorrect); } return(NOPE); } char *SUMA_SurfaceViewer_StructInfo (SUMA_SurfaceViewer *SV, int detail) { static char FuncName[]={"SUMA_SurfaceViewer_StructInfo"}; SUMA_STRING *SS = NULL; char *s=NULL; int i; SUMA_Boolean LocalHead = NOPE; SUMA_ENTRY; SS = SUMA_StringAppend (NULL, NULL); if (!SV) { SS = SUMA_StringAppend (SS,"NULL SV.\n"); SS = SUMA_StringAppend (SS, NULL); /* copy s pointer and free SS */ s = SS->s; SUMA_free(SS); SUMA_RETURN(s); } SS = SUMA_StringAppend(SS, "SV contents:\n"); SS = SUMA_StringAppend_va(SS, " verbose = %d\n", SV->verbose); if (SV->ShowLeft) SS = SUMA_StringAppend_va(SS," Show Left = YES\n"); else SS = SUMA_StringAppend_va(SS," Show Left = NO\n"); if (SV->ShowRight) SS = SUMA_StringAppend_va(SS," Show Right = YES\n"); else SS = SUMA_StringAppend_va(SS," Show Right = NO\n"); if (SV->ortho) SS = SUMA_StringAppend_va(SS," Projection: Orthographic\n"); else SS = SUMA_StringAppend_va(SS," Projection: Perspective\n"); SS = SUMA_StringAppend_va(SS," Display Labels At Crosshair: %s\n", SV->ShowLabelAtXhair ? "ON": "OFF"); SS = SUMA_StringAppend_va(SS," Aspect = %f\n", SV->Aspect); SS = SUMA_StringAppend_va( SS," Freeze Zoom across states = %d\n", SV->FreezeZoomXstates); SS = SUMA_StringAppend_va(SS, " Dim. Scale Factor = %f\n", SV->GVS[SV->StdView].DimSclFac); SS = SUMA_StringAppend_va(SS, " ViewDistance = %f\n", SV->GVS[SV->StdView].ViewDistance); SS = SUMA_StringAppend_va(SS, " ViewFrom = [%f %f %f]\n", SV->GVS[SV->StdView].ViewFrom[0], SV->GVS[SV->StdView].ViewFrom[1], SV->GVS[SV->StdView].ViewFrom[2]); SS = SUMA_StringAppend_va(SS," ViewFromOrig = [%f %f %f]\n", SV->GVS[SV->StdView].ViewFromOrig[0], SV->GVS[SV->StdView].ViewFromOrig[1], SV->GVS[SV->StdView].ViewFromOrig[2]); SS = SUMA_StringAppend_va(SS," ViewCenter = [%f %f %f]\n", SV->GVS[SV->StdView].ViewCenter[0], SV->GVS[SV->StdView].ViewCenter[1], SV->GVS[SV->StdView].ViewCenter[2]); SS = SUMA_StringAppend_va(SS," ViewCenterOrig = [%f %f %f]\n", SV->GVS[SV->StdView].ViewCenterOrig[0], SV->GVS[SV->StdView].ViewCenterOrig[1], SV->GVS[SV->StdView].ViewCenterOrig[2]); SS = SUMA_StringAppend_va(SS," ViewCamUp = [%f %f %f]\n", SV->GVS[SV->StdView].ViewCamUp[0], SV->GVS[SV->StdView].ViewCamUp[1], SV->GVS[SV->StdView].ViewCamUp[2]); SS = SUMA_StringAppend_va(SS," RotaCenter = [%f %f %f]\n", SV->GVS[SV->StdView].RotaCenter[0], SV->GVS[SV->StdView].RotaCenter[1], SV->GVS[SV->StdView].RotaCenter[2]); SS = SUMA_StringAppend_va(SS, " Convolution Filter = %d\n [ ", SV->C_mode); if (SV->C_filter) { int ii, jj; for(jj = 0; jj < SV->C_filter->rows; jj++) { for(ii = 0; ii < SV->C_filter->cols; ii++) { SS = SUMA_StringAppend_va(SS, "%.3f ", SV->C_filter->array[ii + jj * SV->C_filter->cols]); } if (jj < SV->C_filter->rows-1) SS = SUMA_StringAppend(SS, "\n "); else SS = SUMA_StringAppend(SS, " ]\n" ); } } else { SS = SUMA_StringAppend_va(SS, " NULL! ]\n"); } SS = SUMA_StringAppend_va(SS, " light0_position = [%f %f %f %f] (lit for %d)\n", SV->light0_position[0], SV->light0_position[1], SV->light0_position[2], SV->light0_position[3], SV->lit_for); SS = SUMA_StringAppend_va(SS," light1_position = [%f %f %f %f]\n", SV->light1_position[0], SV->light1_position[1], SV->light1_position[2], SV->light1_position[3]); SS = SUMA_StringAppend_va(SS," ZoomCompensate = %f\n", SV->ZoomCompensate); SS = SUMA_StringAppend_va(SS," WindWidth/WIDTH/Offset = %d/%d/%d\n", SV->wWindWidth, SV->X->aWIDTH, SV->DrawAreaWidthOffset); SS = SUMA_StringAppend_va(SS," WindHeight/HEIGHT/Offset = %d/%d/%d\n", SV->wWindHeight, SV->X->aHEIGHT, SV->DrawAreaHeightOffset); SS = SUMA_StringAppend_va(SS," ShowWorldAxis = %d\n", SV->ShowWorldAxis); if (SV->WAx) { SS = SUMA_StringAppend_va(SS, " WorldAxis: Center = [%f %f %f] \n" " BR = [%f %f %f ,\n" " %f %f %f]\n", SV->WAx->Center[0], SV->WAx->Center[1], SV->WAx->Center[2], SV->WAx->BR[0][0], SV->WAx->BR[1][0], SV->WAx->BR[2][0], SV->WAx->BR[0][1], SV->WAx->BR[1][1], SV->WAx->BR[2][1]); } else { SS = SUMA_StringAppend_va(SS," WorldAxis: NULL\n"); } SS = SUMA_StringAppend_va(SS," currentQuat = [%f %f %f %f]\n", SV->GVS[SV->StdView].currentQuat[0], SV->GVS[SV->StdView].currentQuat[1], SV->GVS[SV->StdView].currentQuat[2], SV->GVS[SV->StdView].currentQuat[3]); SS = SUMA_StringAppend_va(SS," deltaQuat = [%f %f %f %f]\n", SV->GVS[SV->StdView].deltaQuat[0], SV->GVS[SV->StdView].deltaQuat[1], SV->GVS[SV->StdView].deltaQuat[2], SV->GVS[SV->StdView].deltaQuat[3]); SS = SUMA_StringAppend_va(SS," ApplyMomentum = %d\n", SV->GVS[SV->StdView].ApplyMomentum); SS = SUMA_StringAppend_va(SS," MinIdleDelta = %d\n", SV->GVS[SV->StdView].MinIdleDelta); SS = SUMA_StringAppend_va(SS," zoomDelta = %f, zoomBegin = %f\n", SV->GVS[SV->StdView].zoomDelta, SV->GVS[SV->StdView].zoomBegin); SS = SUMA_StringAppend_va(SS," ArrowRotationAngle=%f rad (%f deg)\n", SV->ArrowRotationAngle, SV->ArrowRotationAngle * 180.0 / SUMA_PI); SS = SUMA_StringAppend_va(SS," KeyZoomGain=%f \n", SV->KeyZoomGain); SS = SUMA_StringAppend_va(SS," FOV_original=%f\n", SV->FOV_original); SS = SUMA_StringAppend_va(SS," spinDeltaX/Y = %.4f/%.4f\n", SV->GVS[SV->StdView].spinDeltaX, SV->GVS[SV->StdView].spinDeltaY); SS = SUMA_StringAppend_va(SS," spinBeginX/Y = %.4f/%.4f\n", SV->GVS[SV->StdView].spinBeginX, SV->GVS[SV->StdView].spinBeginY); SS = SUMA_StringAppend_va(SS," TranslateGain = %f\n", SV->GVS[SV->StdView].TranslateGain); SS = SUMA_StringAppend_va(SS," ArrowtranslateDeltaX/Y = %f/%f\n", SV->GVS[SV->StdView].ArrowtranslateDeltaX, SV->GVS[SV->StdView].ArrowtranslateDeltaY); SS = SUMA_StringAppend_va(SS," translateBeginX/Y = %.4f/%.4f\n", SV->GVS[SV->StdView].translateBeginX, SV->GVS[SV->StdView].translateBeginY); SS = SUMA_StringAppend_va(SS," translateDeltaX/Y = %f/%f\n", SV->GVS[SV->StdView].translateDeltaX, SV->GVS[SV->StdView].translateDeltaY); SS = SUMA_StringAppend_va(SS," translateVec = [%f %f 0.0]\n", SV->GVS[SV->StdView].translateVec[0], SV->GVS[SV->StdView].translateVec[1]); SS = SUMA_StringAppend_va(SS, " LHpry = [%f %f %f], LHpry0 = [%f %f %f], LHlol = %d\n", SV->GVS[SV->StdView].vLHpry[0], SV->GVS[SV->StdView].vLHpry[1], SV->GVS[SV->StdView].vLHpry[2], SV->GVS[SV->StdView].vLHpry0[0], SV->GVS[SV->StdView].vLHpry0[1], SV->GVS[SV->StdView].vLHpry0[2], SV->GVS[SV->StdView].LHlol); SS = SUMA_StringAppend_va(SS," Show Mesh Axis %d\n", SV->ShowMeshAxis); SS = SUMA_StringAppend_va(SS," Show Eye Axis %d\n", SV->ShowEyeAxis); SS = SUMA_StringAppend_va(SS," Show Cross Hair %d\n", SV->ShowCrossHair); SS = SUMA_StringAppend_va(SS," PolyMode %d\n", SV->PolyMode); SS = SUMA_StringAppend_va(SS," DO_DrawMask %d\n", SV->DO_DrawMask); SS = SUMA_StringAppend_va(SS," Blend_Mode %d\n", SV->Blend_Mode); if (SV->N_otseq) { SS = SUMA_StringAppend(SS,"Object rendering sequence:\n "); for (i=0; iN_otseq; ++i) { SS = SUMA_StringAppend_va(SS,"%s%s", SUMA_ObjectTypeCode2ObjectTypeName(SV->otseq[i]), (iN_otseq-1)?", ":""); } SS = SUMA_StringAppend(SS,"\n"); } else { SS = SUMA_StringAppend(SS,"No object ordering sequence"); } SS = SUMA_StringAppend_va(SS," Group Name %s, indexed %d\n", SV->CurGroupName, SV->iCurGroup); SS = SUMA_StringAppend_va(SS, " Current State %s, indexed %d, Anatomical %s\n", SV->State, SV->iState, SUMA_isViewerStateAnatomical(SV)?"YES":"NO"); SS = SUMA_StringAppend_va(SS," N_DO = %d\n", SV->N_DO); SS = SUMA_StringAppend(SS, " RegistDO = "); for (i=0; i< SV->N_DO; ++i) { SS = SUMA_StringAppend_va(SS,"[id %d] %s \n%s", SV->RegistDO[i].dov_ind, iDO_label(SV->RegistDO[i].dov_ind), (i<(SV->N_DO-1)) ? " ":""); } SS = SUMA_StringAppend_va(SS," N_ColList = %d\n", SV->N_ColList); SS = SUMA_StringAppend(SS, " ColList = "); for (i = 0; i < SV->N_ColList; ++i) { SS = SUMA_StringAppend_va(SS, "[%d] for DO %s, id %s, Remix %d, RemixID %d, %d colors, N_links = %d\n%s", i, iDO_label(SUMA_whichDO(SV->ColList[i]->idcode_str, SUMAg_DOv, SUMAg_N_DOv)), SV->ColList[i]->idcode_str, SV->ColList[i]->Remix, SV->ColList[i]->RemixID, SV->ColList[i]->N_glar_ColorList, SV->ColList[i]->N_links, (i<(SV->N_ColList-1)) ? " ":""); } if (SV->X == NULL) SS = SUMA_StringAppend_va(SS," X struct is NULL!\n"); else { SS = SUMA_StringAppend_va(SS," X struct defined.\n"); } SS = SUMA_StringAppend_va(SS," DO in focus %d\n", SV->Focus_DO_ID); /* show some state stuff */ SS = SUMA_StringAppend_va(SS, "\nView States (%d total):\n", SV->N_VSv); for (i=0; i < SV->N_VSv; ++i) { SS = SUMA_StringAppend_va(SS, "\nView State %d/%d (FOV = %f) (autoFOVval = %f):\n", i, SV->N_VSv-1, SV->FOV[i], SV->auto_FOV_val[i]); s = SUMA_ViewStateInfo (&(SV->VSv[i]), 0); if (!s) { SS = SUMA_StringAppend(SS, "*** Error in SUMA_Show_ViewState ***\n"); } else { SS = SUMA_StringAppend(SS, s); SUMA_free(s); s = NULL; } } SS = SUMA_StringAppend_va(SS, "\nStandard viewing mode: %d\n", SV->StdView ); SS = SUMA_StringAppend_va(SS, "\nBackground Modulation Factor= %f\n", SV->Back_Modfact); SS = SUMA_StringAppend_va(SS, "\nLast non mappable visited %d\n", SV->LastNonMapStateID); s = SUMA_EnablingState_Info(&(SV->SER)); SS = SUMA_StringAppend_va(SS,"Enabling state in sv->SER\n%s",s); SUMA_free(s); s = NULL; SS = SUMA_StringAppend(SS,"\n"); /* trim SS */ SS = SUMA_StringAppend (SS, NULL); /* copy s pointer and free SS */ s = SS->s; SUMA_free(SS); SUMA_RETURN(s); } /*! Show the ViewState structure */ SUMA_Boolean SUMA_Show_ViewState(SUMA_ViewState *VS, FILE *Out, int detail) { static char FuncName[]={"SUMA_Show_ViewState"}; char *s = NULL; SUMA_ENTRY; if (Out == NULL) Out = stdout; s = SUMA_ViewStateInfo(VS, detail); if (!s) { SUMA_SL_Err("Failed in SUMA_ViewStateInfo"); SUMA_RETURN(NOPE); } else { fprintf(Out, "%s", s); SUMA_free(s); s = NULL; } SUMA_RETURN(YUP); } /* A function that must be called each time order of objects in SUMAg_DOv is disturbed */ SUMA_Boolean SUMA_ViewState_MembsRefresh(SUMA_ViewState *VS) { static char FuncName[]={"SUMA_ViewState_MembsRefresh"}; int ii=0, found = -1; SUMA_Boolean LocalHead = NOPE; SUMA_ENTRY; if (!VS) SUMA_RETURN(NOPE); ii = 0; while (ii < VS->N_MembDO) { if ( VS->MembDO && (found = SUMA_whichDOg(VS->MembDO[ii].idcode_str)) >= 0) { /* A good thing, refresh index mapping */ VS->MembDO[ii].dov_ind = found; } else { SUMA_LH("A bad entry in MembDO at index %d/%d, cleaning", ii, VS->N_MembDO); if (ii != VS->N_MembDO-1) { strcpy(VS->MembDO[ii].idcode_str, VS->MembDO[VS->N_MembDO-1].idcode_str); } VS->MembDO[ii].dov_ind = VS->MembDO[VS->N_MembDO-1].dov_ind; VS->N_MembDO = VS->N_MembDO-1; VS->MembDO = (SUMA_DO_LOCATOR *)SUMA_realloc(VS->MembDO, VS->N_MembDO*sizeof(SUMA_DO_LOCATOR)); } ++ii; } SUMA_RETURN(YUP); } SUMA_Boolean SUMA_AllViewState_MembsRefresh(void) { static char FuncName[]={"SUMA_AllViewState_MembsRefresh"}; int i,j; SUMA_SurfaceViewer *sv; SUMA_Boolean state = YUP; SUMA_ENTRY; state = YUP; for (i=0; iN_VSv; ++j) { if (!SUMA_ViewState_MembsRefresh(sv->VSv+j)) state = NOPE; } } SUMA_RETURN(state); } /*! Search for certain types of DOs members of a particular state*/ int *SUMA_ViewState_Membs(SUMA_ViewState *VS, SUMA_DO_Types *ttv, int *uN_Membs) { static char FuncName[]={"SUMA_ViewState_Membs"}; int ii, jj, N_ttv; SUMA_DO_Types tt; int *Membs = NULL, N_Membs = 0; SUMA_Boolean LocalHead = NOPE; SUMA_ENTRY; if (uN_Membs) *uN_Membs = 0; SUMA_LHv("VS %p, Name %s, MembDOs %p, N_MembDOs %d\n", VS, (VS && VS->Name) ? VS->Name:"NULL", (VS && VS->MembDO) ? VS->MembDO:NULL, (VS && VS->N_MembDO) ? VS->N_MembDO:0); if (!VS || !VS->MembDO) SUMA_RETURN(Membs); jj = 0; N_ttv=0; while (ttv[jj] != NOT_SET_type) { ++N_ttv; ++jj; } if (!N_ttv) SUMA_RETURN(Membs); jj = 0; while (ttv[jj] != NOT_SET_type) { tt = ttv[jj]; for (ii=0; iiN_MembDO; ++ii) { SUMA_LHv("Checking %s type %s \n", iDO_label(VS->MembDO[ii].dov_ind), SUMA_ObjectTypeCode2ObjectTypeName(tt)); if (iDO_type(VS->MembDO[ii].dov_ind) == tt) { switch (tt) { case SO_type: if (!Membs) Membs = (int *) SUMA_malloc(N_ttv*(VS->N_MembDO+1)*sizeof(int)); Membs[N_Membs++] = VS->MembDO[ii].dov_ind; Membs[N_Membs]=-1;/* a plug, if uN_Membs is NULL*/ break; case GRAPH_LINK_type: if (!Membs) Membs = (int *) SUMA_malloc(N_ttv*(VS->N_MembDO+1)*sizeof(int)); Membs[N_Membs++] = VS->MembDO[ii].dov_ind; Membs[N_Membs]=-1;/* a plug, if uN_Membs is NULL*/ break; case CDOM_type: case TRACT_type: case MASK_type: case VO_type: if (!Membs) Membs = (int *) SUMA_malloc(N_ttv*(VS->N_MembDO+1)*sizeof(int)); Membs[N_Membs++] = VS->MembDO[ii].dov_ind; Membs[N_Membs]=-1;/* a plug, if uN_Membs is NULL*/ break; default: SUMA_S_Err("Not ready for type %d (%s)", tt, SUMA_otc2otn(tt)); break; } } } ++jj;} if (uN_Membs) *uN_Membs = N_Membs; SUMA_RETURN(Membs); } /*! Show the ViewState structure */ char *SUMA_ViewStateInfo(SUMA_ViewState *VS, int detail) { static char FuncName[]={"SUMA_ViewStateInfo"}; int i, ifound; SUMA_STRING *SS = NULL; char *s=NULL; SUMA_ENTRY; SS = SUMA_StringAppend (NULL, NULL); if (!VS) { SS = SUMA_StringAppend (SS,"NULL VS.\n"); SS = SUMA_StringAppend (SS, NULL); /* copy s pointer and free SS */ s = SS->s; SUMA_free(SS); SUMA_RETURN(s); } if (VS->Name) SS = SUMA_StringAppend_va(SS, " Name: %s, Anatomically Correct: %s\n", VS->Name, VS->AnatCorrect?"YES":"NO"); else SS = SUMA_StringAppend_va(SS, " Name: NULL\n"); if (VS->Group) SS = SUMA_StringAppend_va(SS, " Group: %s\n", VS->Group); else SS = SUMA_StringAppend_va(SS, " Group: NULL\n"); if (VS->N_MembDO) { SS = SUMA_StringAppend_va(SS, " %2d MembDOs in VS->MembDO %p:\n ", VS->N_MembDO, VS->MembDO); for (i=0; i < VS->N_MembDO; ++i) { ifound = SUMA_whichDOg(VS->MembDO[i].idcode_str); SS = SUMA_StringAppend_va(SS, "id %s DOv[%d] %s (%s) -- id check: %s\n ", VS->MembDO[i].idcode_str, VS->MembDO[i].dov_ind, iDO_label(VS->MembDO[i].dov_ind), iDO_typename(VS->MembDO[i].dov_ind), (ifound == VS->MembDO[i].dov_ind)?"OK":"NO - Must run Refresh"); } } else { SS = SUMA_StringAppend_va(SS, " No MembDOs in VS->MembDO %p\n", VS->MembDO); } if (VS->Hist) { } else { SS = SUMA_StringAppend_va(SS, " Hist is NULL\n"); } SS = SUMA_StringAppend (SS, NULL); /* copy s pointer and free SS */ s = SS->s; SUMA_free(SS); SUMA_RETURN (s); } /*! Create & free ViewState_Hist structure */ SUMA_ViewState_Hist *SUMA_Alloc_ViewState_Hist (void) { static char FuncName[]={"SUMA_Alloc_ViewState_Hist"}; SUMA_ViewState_Hist *vsh; SUMA_ENTRY; vsh = (SUMA_ViewState_Hist *)SUMA_calloc(1,sizeof(SUMA_ViewState_Hist)); if (vsh == NULL) { fprintf(SUMA_STDERR,"Error %s: Could not allocate for vsh.\n", FuncName); SUMA_RETURN (NULL); } SUMA_RETURN (vsh); } SUMA_Boolean SUMA_Free_ViewState_Hist (SUMA_ViewState_Hist *vsh) { static char FuncName[]={"SUMA_Free_ViewState_Hist"}; SUMA_ENTRY; if (vsh == NULL) SUMA_RETURN (YUP); if (vsh) SUMA_free(vsh); SUMA_RETURN (YUP); } /*! Add a new SUMA_ViewState structure This is meant to replace SUMA_Alloc_ViewState - Both csv->VSv and csv->N_VSv are updated here */ SUMA_Boolean SUMA_New_ViewState (SUMA_SurfaceViewer *cs) { static char FuncName[]={"SUMA_New_ViewState"}; int i; SUMA_ENTRY; if (!cs->VSv) { /* a new baby */ cs->N_VSv = 1; cs->VSv = (SUMA_ViewState *)SUMA_calloc(1,sizeof(SUMA_ViewState)); } else { /* realloc */ ++cs->N_VSv; cs->VSv = (SUMA_ViewState *)SUMA_realloc(cs->VSv, cs->N_VSv*sizeof(SUMA_ViewState) ); } /* check on allocation */ if (!cs->VSv) { SUMA_SL_Err("Failed to allocate"); SUMA_RETURN(YUP); } /* initialization of last element */ cs->VSv[cs->N_VSv-1].Name = NULL; cs->VSv[cs->N_VSv-1].AnatCorrect = NOPE; cs->VSv[cs->N_VSv-1].Group = NULL; cs->VSv[cs->N_VSv-1].MembDO = NULL; cs->VSv[cs->N_VSv-1].N_MembDO = 0; cs->VSv[cs->N_VSv-1].Hist = SUMA_Alloc_ViewState_Hist (); if (cs->VSv[cs->N_VSv-1].Hist == NULL) { SUMA_S_Err("Could not allocate for cs->VSv->Hist."); SUMA_free(cs->VSv); SUMA_RETURN (NOPE); } /* allocate for FOV */ if (!cs->FOV) { cs->FOV = (float *)SUMA_calloc(cs->N_VSv, sizeof(float)); if (!cs->FOV_last_PickMode) { cs->FOV_last_PickMode = (float *)SUMA_calloc(cs->N_VSv, sizeof(float)); } if (!cs->auto_FOV_val) { cs->auto_FOV_val = (float *)SUMA_calloc(cs->N_VSv, sizeof(float)); } for (i=0; i < cs->N_VSv; ++i) { /* This will get reset in SUMA_SetupSVforDOs */ cs->FOV[i] = cs->FOV_original; cs->auto_FOV_val[i] = -1.0; } } else { cs->FOV = (float *)SUMA_realloc(cs->FOV, cs->N_VSv * sizeof(float)); cs->FOV[cs->N_VSv-1] = cs->FOV[0]; /* used to be = cs->FOV_original, but it is best to set to 0th view, gives user ability to set display before auto-movie making via talk-suma */ cs->FOV_last_PickMode = (float *)SUMA_realloc(cs->FOV_last_PickMode, cs->N_VSv * sizeof(float)); cs->auto_FOV_val = (float *)SUMA_realloc(cs->auto_FOV_val, cs->N_VSv * sizeof(float)); cs->auto_FOV_val[cs->N_VSv-1] = -1.0; } /* Done */ SUMA_RETURN(YUP); } /*! Create & free SUMA_ViewState structure */ SUMA_ViewState *SUMA_Alloc_ViewState (int N) { SUMA_ViewState *vs; int i; static char FuncName[]={"SUMA_Alloc_ViewState"}; SUMA_ENTRY; SUMA_SL_Err("Should not be using this anymore.\n" "Start using SUMA_New_ViewState.\n" " ZSS Jan 12 04 \n"); SUMA_RETURN(NULL); vs = (SUMA_ViewState *)SUMA_calloc(N,sizeof(SUMA_ViewState)); if (vs == NULL) { fprintf(SUMA_STDERR,"Error %s: Could not allocate for vs.\n", FuncName); SUMA_RETURN (NULL); } for (i=0; i< N; ++i) { vs[i].Name = NULL; vs[i].Group = NULL; vs[i].MembDO = NULL; vs[i].N_MembDO = 0; vs[i].Hist = SUMA_Alloc_ViewState_Hist (); if (vs[i].Hist == NULL) { SUMA_S_Err("Could not allocate for vs->Hist."); SUMA_free(vs); SUMA_RETURN (NULL); } } SUMA_RETURN (vs); } SUMA_Boolean SUMA_Free_ViewState (SUMA_ViewState *vs) { static char FuncName[]={"SUMA_Free_ViewState"}; SUMA_ENTRY; if (vs == NULL) SUMA_RETURN (YUP); if (vs->Name) SUMA_free(vs->Name); if (vs->Group) SUMA_free(vs->Group); if (vs->MembDO) SUMA_free(vs->MembDO); if (vs->Hist) SUMA_Free_ViewState_Hist (vs->Hist); if (vs) SUMA_free(vs); SUMA_RETURN (YUP); } /*! locate the index i (into SVv[i]) of sv -1 if not found */ int SUMA_WhichSV (SUMA_SurfaceViewer *sv, SUMA_SurfaceViewer *SVv, int N_SVv) { static char FuncName[]={"SUMA_WhichSV"}; int i = 0; SUMA_ENTRY; if (!SVv || !sv) { fprintf (SUMA_STDERR, "Error %s: NULL SVv or sv.\n", FuncName); SUMA_RETURN (-1); } for (i=0; i= 0) return(65+isv); return('\0'); } /* return 1st viewer that is open and has a particular surface visible AND in focus */ SUMA_SurfaceViewer *SUMA_OneViewerWithSOinFocus( SUMA_SurfaceObject *curSO) { static char FuncName[]={"SUMA_OneViewerWithSOinFocus"}; int i=0; SUMA_SurfaceViewer *sv=NULL; SUMA_ENTRY; /* look for 1st viewer that is showing this surface and has this surface in focus*/ for (i=0; iTOPLEVEL) { /* is this viewer showing curSO ? */ if (SUMA_isVisibleDO(&(SUMAg_SVv[i]), SUMAg_DOv, (SUMA_ALL_DO *)curSO)) { if ((SUMAg_DOv[SUMAg_SVv[i].Focus_DO_ID].OP) == curSO) { sv = &(SUMAg_SVv[i]); SUMA_RETURN(sv); } } } } SUMA_RETURN(sv); } /* return 1st viewer that is open and has a particular object visible AND in focus */ SUMA_SurfaceViewer *SUMA_OneViewerWithADOinFocus(SUMA_ALL_DO *ADO) { static char FuncName[]={"SUMA_OneViewerWithADOinFocus"}; int i=0; SUMA_SurfaceViewer *sv=NULL; SUMA_ENTRY; /* look for 1st viewer that has this ADO in focus*/ for (i=0; iTOPLEVEL) { /* is this viewer showing ADO ? */ if (SUMA_isVisibleDO(&(SUMAg_SVv[i]), SUMAg_DOv, ADO)) { if ((SUMAg_DOv[SUMAg_SVv[i].Focus_DO_ID].OP) == ADO) { sv = &(SUMAg_SVv[i]); SUMA_RETURN(sv); } } } } SUMA_RETURN(sv); } SUMA_SurfaceViewer *SUMA_OneViewerWithADOVisible(SUMA_ALL_DO *ADO) { static char FuncName[]={"SUMA_OneViewerWithADOVisible"}; int i=0; SUMA_SurfaceViewer *sv=NULL; SUMA_ENTRY; /* look for 1st viewer that is showing this surface and has this surface in focus*/ for (i=0; iTOPLEVEL) { /* is this viewer showing ADO ? */ if (SUMA_isVisibleDO(&(SUMAg_SVv[i]), SUMAg_DOv, ADO)) { sv = &(SUMAg_SVv[i]); SUMA_RETURN(sv); } } } SUMA_RETURN(sv); } SUMA_SurfaceViewer *SUMA_OneViewerWithSOVisible( SUMA_SurfaceObject *curSO) { static char FuncName[]={"SUMA_OneViewerWithSOVisible"}; int i=0; SUMA_SurfaceViewer *sv=NULL; SUMA_ENTRY; /* look for 1st viewer that is showing this surface and has this surface in focus*/ for (i=0; iTOPLEVEL) { /* is this viewer showing curSO ? */ if (SUMA_isVisibleDO(&(SUMAg_SVv[i]), SUMAg_DOv, (SUMA_ALL_DO *)curSO)) { sv = &(SUMAg_SVv[i]); SUMA_RETURN(sv); } } } SUMA_RETURN(sv); } SUMA_SurfaceViewer *SUMA_OneViewerWithADORegistered(SUMA_ALL_DO *ADO) { static char FuncName[]={"SUMA_OneViewerWithADORegistered"}; int i=0; SUMA_SurfaceViewer *sv=NULL; SUMA_ENTRY; /* look for 1st viewer that is showing this surface and has this surface in focus*/ for (i=0; iTOPLEVEL) { if (SUMA_ADO_isRegistered(SUMAg_SVv+i, ADO)) { sv = &(SUMAg_SVv[i]); SUMA_RETURN(sv); } } } SUMA_RETURN(sv); } SUMA_SurfaceViewer *SUMA_OneViewerWithSORegistered( SUMA_SurfaceObject *curSO) { static char FuncName[]={"SUMA_OneViewerWithSORegistered"}; int i=0; SUMA_SurfaceViewer *sv=NULL; SUMA_ENTRY; /* look for 1st viewer that is showing this surface and has this surface in focus*/ for (i=0; iTOPLEVEL) { /* is this viewer showing curSO ? */ if (SUMA_isRegisteredSO(&(SUMAg_SVv[i]), SUMAg_DOv, curSO)) { sv = &(SUMAg_SVv[i]); SUMA_RETURN(sv); } } } SUMA_RETURN(sv); } SUMA_SurfaceViewer *SUMA_BestViewerForADO(SUMA_ALL_DO *ado) { static char FuncName[]={"SUMA_BestViewerForADO"}; int i=0; SUMA_SurfaceViewer *sv=NULL; SUMA_ENTRY; if (!ado) { SUMA_RETURN(&(SUMAg_SVv[0])); } switch (ado->do_type) { case SO_type: { SUMA_SurfaceObject *curSO = (SUMA_SurfaceObject *)ado; /* best bet, visible, and in focus */ if ((sv=SUMA_OneViewerWithSOinFocus(curSO))) { SUMA_RETURN(sv); } /* just visible */ if ((sv=SUMA_OneViewerWithSOVisible(curSO))) { SUMA_RETURN(sv); } /* registered */ if ((sv=SUMA_OneViewerWithSORegistered(curSO))) { SUMA_RETURN(sv); } /* crap! */ sv = &(SUMAg_SVv[0]); break; } case GRAPH_LINK_type: case GDSET_type: case CDOM_type: case VO_type: case MASK_type: case TRACT_type: if ((sv=SUMA_OneViewerWithADOinFocus(ado))) { SUMA_RETURN(sv); } /* just visible */ if ((sv=SUMA_OneViewerWithADOVisible(ado))) { SUMA_RETURN(sv); } /* registered */ if ((sv=SUMA_OneViewerWithADORegistered(ado))) { SUMA_RETURN(sv); } sv = &(SUMAg_SVv[0]); break; default: SUMA_S_Errv("Not ready for %s\n", SUMA_ObjectTypeCode2ObjectTypeName(ado->do_type)); sv = &(SUMAg_SVv[0]); break; } SUMA_RETURN(sv); } /*! locate the index i (into csv->VSv[i]) of state -1 if not found */ int SUMA_WhichState (char *state, SUMA_SurfaceViewer *csv, char *ForceGroup) { static char FuncName[]={"SUMA_WhichState"}; int i = 0; SUMA_Boolean LocalHead = NOPE; SUMA_ENTRY; if (!ForceGroup || !strcmp(ForceGroup,"ANY")) { SUMA_LH("Searching for state: %s, %d states\n" "Viewer %p [%c], %d viewers realized out of %d total", state, csv->N_VSv, csv, SUMA_WhichSVc(csv, SUMAg_SVv, SUMA_MAX_SURF_VIEWERS), SUMAg_N_SVv, SUMA_MAX_SURF_VIEWERS); while (i < csv->N_VSv) { if (LocalHead) fprintf(SUMA_STDERR," %d? %s ...\n", i, csv->VSv[i].Name); if (!csv->VSv[i].Name || !state) { SUMA_LH("Null Name or State \n"); SUMA_RETURN (-1); } if (strcmp(csv->VSv[i].Name, state) == 0) { if (LocalHead) fprintf(SUMA_STDERR,"%s: FOUND, i=%d!\n", FuncName, i); SUMA_RETURN (i); } if (strcmp(state,"ANY_ANATOMICAL") == 0 && csv->VSv[i].AnatCorrect) { if (LocalHead) fprintf(SUMA_STDERR,"%s: FOUND, i=%d for ANY_ANATOMICAL!\n", FuncName, i); SUMA_RETURN (i); } ++i; } } else { if (LocalHead) fprintf(SUMA_STDERR,"%s: Searching for: %s, %s...\n", FuncName, state, ForceGroup); while (i < csv->N_VSv) { if (LocalHead) fprintf(SUMA_STDERR," %d? %s, %s ...\n", i, csv->VSv[i].Name, csv->VSv[i].Group); if (!csv->VSv[i].Name || !state || !csv->CurGroupName) { SUMA_LH("Null Name or State or CurGroupName.\n"); SUMA_RETURN (-1); } if (strcmp(csv->VSv[i].Name, state) == 0 && (strcmp(csv->VSv[i].Group, ForceGroup) == 0 || strcmp(csv->VSv[i].Group, "ANY") == 0 ) ) { if (LocalHead) fprintf(SUMA_STDERR,"%s: FOUND, i=%d!\n", FuncName, i); SUMA_RETURN (i); } ++i; } } SUMA_RETURN (-1); } /*! Return the viewer state for a particular displayable object dov_id (int): Identifier of object by index into SUMAg_DOv cSV (SUMA_SurfaceViewer *): Viewer in question. I null, all viewers get the treatment addifmissing (int): If 1 then add state if not found Returns the index of the state into cSV->VSv, if cSV was null, the index would be into SUMAg_SVv[0] -1 if none were found and adding was not allowed. -2 in error */ int SUMA_Which_iDO_State(int dov_id, SUMA_SurfaceViewer *cSV, int addifmissing) { static char FuncName[]={"SUMA_Which_iDO_State"}; int is = -2, do_all=0, iic, isd; char *sid=NULL; SUMA_Boolean LocalHead = NOPE; SUMA_ENTRY; if (SUMAg_N_SVv <= 0) SUMA_RETURN(is); if (!cSV) do_all=1; iic=SUMAg_N_SVv-1; do { if (do_all) cSV = &(SUMAg_SVv[iic]); is = SUMA_WhichState (SUMA_iDO_state(dov_id), cSV, SUMA_iDO_group(dov_id)); SUMA_LHv("is %d, addifmissing %d", is, addifmissing); if (is < 0 && addifmissing) { /* add state, it is a new one */ SUMA_LHv("For DO %s, type %s\n State:%s to be added, group %s\n", iDO_label(dov_id), iDO_typename(dov_id), iDO_state(dov_id), iDO_group(dov_id)); SUMA_New_ViewState (cSV); is = cSV->N_VSv-1; cSV->VSv[is].Name = SUMA_copy_string(SUMA_iDO_state(dov_id)); cSV->VSv[is].AnatCorrect = SUMA_is_iDO_AnatCorrect(dov_id); cSV->VSv[is].Group = iDO_group(dov_id); if (!cSV->VSv[is].Name || !cSV->VSv[is].Group) { SUMA_S_Err("Failed to allocate for cSV->VSv[is]." "Name or .Group."); SUMA_RETURN (NOPE); } cSV->VSv[is].N_MembDO = 1; cSV->VSv[is].MembDO = (SUMA_DO_LOCATOR *)SUMA_calloc(cSV->VSv[is].N_MembDO, sizeof(SUMA_DO_LOCATOR)); cSV->VSv[is].MembDO[cSV->VSv[is].N_MembDO-1].dov_ind = dov_id; sid = iDO_idcode(dov_id); strcpy(cSV->VSv[is].MembDO[cSV->VSv[is].N_MembDO-1].idcode_str, sid); if (LocalHead) { char *s = SUMA_ViewStateInfo (&(cSV->VSv[is]), 0); SUMA_LHv("State Info of cSV->VSv[%d] now:\n%s\n",is, s); SUMA_ifree(s); } } if (!cSV->State && cSV->N_VSv) { if (is < 0) isd = 0; else isd = is; /* happens if loading DOs without surfaces already loaded Should always have something decent if you can*/ SUMA_LHv( "No state yet for cSV %p, defaulting to cSV->VSv[%d].Name -->%s<--\n", cSV, isd, cSV->VSv[isd].Name); cSV->State = cSV->VSv[isd].Name; cSV->iState = isd; } --iic; } while (do_all && iic >= 0); SUMA_RETURN(is); } /*! register the different view states and surfaces belonging to different view states in the surface viewer's structure Essentially, it creates the vector VSv that is a part of the surface viewer structure */ SUMA_Boolean SUMA_RegisterSpecSO (SUMA_SurfSpecFile *Spec, SUMA_SurfaceViewer *csv, SUMA_DO* dov, int N_dov, int viewopt) { static char FuncName[]={"SUMA_RegisterSpecSO"}; int is, i, old_N_VSv = 0; char *sid=NULL; SUMA_SurfaceObject * SO; SUMA_Boolean LocalHead = NOPE; SUMA_ENTRY; if (!viewopt) viewopt = UPDATE_ALL_VIEWING_PARAMS_MASK; if (LocalHead && SUMA_WhichSV(csv, SUMAg_SVv, SUMA_MAX_SURF_VIEWERS) != 0) { fprintf(SUMA_STDERR,"%s: Muted for viewer[%c]\n", FuncName, SUMA_SV_CHAR(csv) ); /* turn off the LocalHead, too much output*/ LocalHead = NOPE; } /* allocate for space depending on the number of states present */ if (LocalHead) fprintf(SUMA_STDERR,"%s: Entering, Spec->Group[0] = %s ...\n", FuncName, Spec->Group[0]); if (Spec->N_Groups != 1) { SUMA_SL_Err("A spec file is to have 1 and only 1 group in it"); SUMA_RETURN(NOPE); } #if 0 /* the old way */ if (!csv->VSv) { /* first pass */ csv->VSv = SUMA_Alloc_ViewState (Spec->N_States); if (csv->VSv == NULL) { fprintf(SUMA_STDERR, "Error %s: Failed to allocate for VSv.\n", FuncName); SUMA_RETURN (NOPE); } csv->N_VSv = 0; } #endif /* register the various states from each SO in DOv */ SUMA_LHv("Cycling through DOvs, looking for surfaces/DOs of group %s\n", Spec->Group[0]); old_N_VSv = csv->N_VSv; for (i=0; i < N_dov; ++i) { switch (dov[i].ObjectType) { case SO_type: if (SUMA_isSO_G(dov[i], Spec->Group[0])) { SO = (SUMA_SurfaceObject *)(dov[i].OP); is = SUMA_WhichState (SO->State, csv, SO->Group); if (is < 0) { /* add state if it is a new one */ /* make room */ if (LocalHead) { fprintf(SUMA_STDERR, "%s: For %s\nState:%s,Group:%s to be added\n", FuncName, SO->Label, SO->State, SO->Group); } SUMA_New_ViewState (csv); csv->VSv[csv->N_VSv-1].Name = SUMA_copy_string(SO->State); csv->VSv[csv->N_VSv-1].AnatCorrect = SO->AnatCorrect; csv->VSv[csv->N_VSv-1].Group = SUMA_copy_string(SO->Group); if (!csv->VSv[csv->N_VSv-1].Name || !csv->VSv[csv->N_VSv-1].Group) { SUMA_S_Err("Failed to allocate for csv->VSv[csv->N_VSv-1]." "Name or .Group."); SUMA_RETURN (NOPE); } csv->VSv[csv->N_VSv-1].N_MembDO = 1; } else { /* old one, count it */ if (LocalHead) { fprintf(SUMA_STDERR, "%s: For %s\n State:%s,Group:%s found\n", FuncName, SO->Label, SO->State, SO->Group); } csv->VSv[is].N_MembDO += 1; } } break; case TRACT_type: case MASK_type: case VO_type: case CDOM_type: case GRAPH_LINK_type: is = SUMA_WhichState (SUMA_iDO_state(i), csv,SUMA_iDO_group(i)); if (is < 0) { if (LocalHead) { fprintf(SUMA_STDERR, "%s: For %s\nState:%s,Group:%s to be added\n", FuncName, iDO_label(i), SUMA_iDO_state(i), SUMA_iDO_group(i)); } SUMA_New_ViewState (csv); csv->VSv[csv->N_VSv-1].Name = SUMA_copy_string(SUMA_iDO_state(i)); csv->VSv[csv->N_VSv-1].AnatCorrect = SUMA_isDO_AnatCorrect(&(dov[i])); csv->VSv[csv->N_VSv-1].Group = SUMA_copy_string(SUMA_iDO_group(i)); if (!csv->VSv[csv->N_VSv-1].Name || !csv->VSv[csv->N_VSv-1].Group) { SUMA_S_Err("Failed to allocate for csv->VSv[csv->N_VSv-1]." "Name or .Group."); SUMA_RETURN (NOPE); } csv->VSv[csv->N_VSv-1].N_MembDO = 1; } else { if (LocalHead) { fprintf(SUMA_STDERR, "%s: For %s\n State:%s,Group:%s found\n", FuncName, iDO_label(i), SUMA_iDO_state(i), SUMA_iDO_group(i)); } csv->VSv[is].N_MembDO += 1; } break; default: SUMA_LHv("Just ignored DO %s...\n", DO_label(dov[i].OP)); break; } } SUMA_LH("Allocating..."); /* allocate space for MembDOs counters will be reset for later use */ for (i=0; i < csv->N_VSv; ++i) { if (!csv->VSv[i].MembDO) { csv->VSv[i].MembDO = (SUMA_DO_LOCATOR *) SUMA_calloc(csv->VSv[i].N_MembDO, sizeof(SUMA_DO_LOCATOR)); } else { csv->VSv[i].MembDO = (SUMA_DO_LOCATOR *) SUMA_realloc(csv->VSv[i].MembDO, csv->VSv[i].N_MembDO * sizeof(SUMA_DO_LOCATOR)); } if (csv->VSv[i].MembDO == NULL) { SUMA_S_Err("Failed to allocate for csv->VSv[i].MembDO.\n"); SUMA_RETURN (NOPE); } csv->VSv[i].N_MembDO = 0; } /*fprintf(SUMA_STDERR,"%s: placement ...\n", FuncName);*/ /* place each SO where it belongs, don't worry about the group they're in */ for (i=0; i < N_dov; ++i) { switch (dov[i].ObjectType) { case SO_type: SO = (SUMA_SurfaceObject *)(dov[i].OP); /* find out which state it goes in */ if (!SO->State || !SO->Group) { SUMA_S_Errv("Sent me SO (%s) with null State (%s) " "or null Group (%s)!\n", SO->Label, SO->State, SO->Group); if (LocalHead) SUMA_Print_Surface_Object (SO, NULL); SUMA_RETURN (NOPE); } is = SUMA_WhichState (SO->State, csv, SO->Group); if (is < 0) { SUMA_S_Errv("This should not be.\n" "Failed to find %s %s in csv\n", SO->State, SO->Group); SUMA_RETURN (NOPE); } if (LocalHead) { fprintf (SUMA_STDERR,"%s: Trying to house %s in: State[%d]\n", \ FuncName, SO->Label, is); } /* store its id as a valid member of the state */ csv->VSv[is].MembDO[csv->VSv[is].N_MembDO].dov_ind = i; strcpy(csv->VSv[is].MembDO[csv->VSv[is].N_MembDO].idcode_str, SO->idcode_str); csv->VSv[is].N_MembDO += 1; /* count it, again */ break; case TRACT_type: case MASK_type: case VO_type: case GRAPH_LINK_type: case CDOM_type: is = SUMA_WhichState (SUMA_iDO_state(i), csv,SUMA_iDO_group(i)); if (is < 0) { SUMA_S_Errv("This should not be.\n" "Failed to find %s %s in csv\n", SUMA_iDO_state(i), SUMA_iDO_group(i)); SUMA_RETURN (NOPE); } if (LocalHead) { fprintf (SUMA_STDERR,"%s: Trying to house %s in: State[%d]\n", \ FuncName, SO->Label, is); } /* store its id as a valid member of the state */ csv->VSv[is].MembDO[csv->VSv[is].N_MembDO].dov_ind = i; sid = iDO_idcode(i); strcpy(csv->VSv[is].MembDO[csv->VSv[is].N_MembDO].idcode_str, sid); csv->VSv[is].N_MembDO += 1; /* count it, again */ break; default: if (SUMA_is_iDO_Selectable(i)) { static int nwarn=0; if (!nwarn) { SUMA_S_Warnv("Just ignored DO %s, I hope that's cool\n", iDO_label(i)); } ++nwarn; } SUMA_LHv("Just ignored DO %s...\n", DO_label(dov[i].OP)); break; } } /*fprintf(SUMA_STDERR,"%s: Leaving ...\n", FuncName);*/ SUMA_RETURN (YUP); } /*! allocate and intialize SUMA_CommonFields No fancy allocation, No fancy macros. \sa SUMA_Free_CommonFields */ SUMA_CommonFields * SUMA_Create_CommonFields () { static char FuncName[]={"SUMA_Create_CommonFields"}; SUMA_CommonFields *cf; int i, n, pb=-1; char *eee=NULL; float dsmw=5*60; SUMA_Boolean LocalHead = NOPE; /* This is the function that creates the debugging flags, do not use them here */ cf = NULL; /* allocate */ /* DO NOT USE SUMA_malloc here, too early for that */ cf = (SUMA_CommonFields *)calloc(1,sizeof(SUMA_CommonFields)); if (cf == NULL) { fprintf(SUMA_STDERR,"Error %s: Failed to allocate.\n", FuncName); return (cf); } cf->Dev = NOPE; cf->Fake_Cmap = NOPE; cf->InOut_Notify = NOPE; cf->Echo_KeyPress = NOPE; cf->InOut_Level = 0; cf->MemTrace = NOPE; cf->dcom = NULL; cf->N_dcom = 0; /* verify pointer size. I use INT_MAX and LONG_MAX to guess whether or not we have 64 bit pointers. I need to guess with #if to do proper type casting and avoid compiler warnings */ cf->PointerSize = sizeof(void *); if (INT_MAX < LONG_MAX && cf->PointerSize < 8) { SUMA_S_Warn("Using INT_MAX and LONG_MAX fails to" "guess pointer size."); } #ifdef USE_SUMA_MALLOC SUMA_SL_Err("NO LONGER SUPPORTED"); return(NULL); cf->Mem = SUMA_Create_MemTrace(); #else cf->Mem = NULL; #endif /* communications initialization happens in SUMA_init_ports_assignments() it cannot be done here because some parameters are not available yet. Setup here is to be sure there is no garbage lying around. */ for (i=0; ins_v[i] = NULL; cf->ns_to[i] = SUMA_WRITECHECKWAITMAX; cf->ns_flags_v[i] = 0; cf->Connected_v[i] = NOPE; cf->TrackingId_v[i] = 0; cf->NimlStream_v[i][0] = '\0'; cf->HostName_v[i][0] = '\0'; cf->TalkMode[i] = NI_BINARY_MODE; cf->TCP_port[i] = 0; } cf->Listening = NOPE; cf->niml_work_on = NOPE; /* viewer locking */ for (i=0; iLocked[i] = SUMA_I_Lock; if (SUMA_isEnv("SUMA_LockViewers","YES")) { cf->ViewLocked[i] = YUP; } else { cf->ViewLocked[i] = NOPE; } } eee = getenv("SUMA_SwapButtons_1_3"); if (eee) { if (strcasecmp (eee, "YES") == 0) cf->SwapButtons_1_3 = YUP; else cf->SwapButtons_1_3 = NOPE; } else { cf->SwapButtons_1_3 = NOPE; } cf->X = (SUMA_X_AllView *)calloc(1,sizeof(SUMA_X_AllView)); if (!cf->X) { fprintf(SUMA_STDERR,"Error %s: Failed to allocate.\n", FuncName); return (NULL); } cf->X->SumaCont = SUMA_CreateSumaContStruct(); cf->X->DrawROI = SUMA_CreateDrawROIStruct(); cf->X->AllMaskCont = NULL; cf->X->MaskStateID = 0; cf->X->DPY_controller1 = NULL; cf->X->Cr = (SUMA_GLCONTEXT_RECORD *)calloc(1, sizeof(SUMA_GLCONTEXT_RECORD)); cf->X->Cr->last_context_DPY = NULL; cf->X->Cr->last_context_WDW = -1; cf->X->Cr->setting_function[0] = '\0'; cf->X->Cr->widget_label[0] = '\0'; eee = getenv("SUMA_ColorPattern"); if (eee) { if (strcmp (eee, "AFNI") == 0) { cf->X->X_Resources = SXR_Afni; if (LocalHead) fprintf(SUMA_STDERR,"%s: Afni resources\n", FuncName); } else if (strcmp (eee, "EURO") == 0) { cf->X->X_Resources = SXR_Euro; if (LocalHead) fprintf(SUMA_STDERR,"%s: Euro resources\n", FuncName); } else if (strcmp (eee, "BONAIRE") == 0) { cf->X->X_Resources = SXR_Bonaire; if (LocalHead) fprintf(SUMA_STDERR,"%s: Bonaire resources\n", FuncName); } else if (strcmp (eee, "DEFAULT") == 0) { cf->X->X_Resources = SXR_default; if (LocalHead) fprintf(SUMA_STDERR,"%s: default resources\n", FuncName); } else if (strcmp (eee, "PRINT") == 0) { cf->X->X_Resources = SXR_Print; if (LocalHead) fprintf(SUMA_STDERR,"%s: Print resources\n", FuncName); } else { cf->X->X_Resources = SXR_Euro; fprintf(SUMA_STDERR, "%s:\nUnrecognized option %s for SUMA_ColorPattern.\n" "Using default = EURO\n", FuncName, eee); } } else { cf->X->X_Resources = SXR_Euro; if (LocalHead) fprintf( SUMA_STDERR, "%s: Undefined environment. Using default\n", FuncName); } cf->X->Help_TextShell = NULL; cf->X->Help_Cmap_TextShell = NULL; cf->X->Help_Plot_TextShell = NULL; cf->X->Whereami_TextShell = NULL; cf->X->Log_TextShell = NULL; cf->X->FileSelectDlg = NULL; cf->X->N_ForeSmooth_prmpt = NULL; cf->X->N_FinalSmooth_prmpt = NULL; cf->X->Clip_prmpt = NULL; cf->X->ClipObj_prmpt = NULL; cf->X->TableTextFontList = NULL; cf->X->CommonSurfContTLW = NULL; cf->X->ButtonDown = 0; cf->X->roffx = -1; /* Not set */ cf->X->roffy = -1; /* Not set */ { char *eee = getenv("SUMA_WindowOffset"); float fv3[3]; if (eee) { if (!strcmp(eee,"Auto")) { cf->X->roffx = cf->X->roffy = -1; } else { if (SUMA_StringToNum (eee, (void *)fv3, 2,1) != 2) { SUMA_S_Err("Syntax error in environment\n" "variable SUMA_WindowOffset of (%s). " "Setting offset to 0", eee); cf->X->roffx = cf->X->roffy = 0; } else { cf->X->roffx = (int)fv3[0]; cf->X->roffy = (int)fv3[1]; } } } else { /* leave it for auto setup */ cf->X->roffx = cf->X->roffy = -1; } } cf->X->SC_Notebook = NULL; cf->X->SameSurfContOpen = 0; { /* Changed default to YES March 05 2014 */ char *eee = getenv("SUMA_SameSurfCont"); if (eee) { if (strcmp(eee,"NO") == 0) cf->X->UseSameSurfCont = NOPE; else if (strcmp(eee,"YES") == 0) cf->X->UseSameSurfCont = YUP; else { fprintf (SUMA_STDERR, "Warning %s:\n" "Bad value for environment variable UseSameSurfCont\n" "Assuming default of YES", FuncName); cf->X->UseSameSurfCont = YUP; } } else cf->X->UseSameSurfCont = YUP; } { char *eee = getenv("SUMA_HomeAfterPrying"); if (eee) { if (strcmp(eee,"NO") == 0) cf->Home_After_Prying = NOPE; else if (strcmp(eee,"YES") == 0) cf->Home_After_Prying = YUP; else { fprintf (SUMA_STDERR, "Warning %s:\n" "Bad value for environment variable SUMA_HomeAfterPrying\n" "Assuming default of YES", FuncName); cf->Home_After_Prying = YUP; } } else cf->Home_After_Prying = YUP; } { char *eee = getenv("SUMA_NumForeSmoothing"); if (eee) { int rotval = (int)strtod(eee, NULL); if (rotval >= 0) cf->X->NumForeSmoothing = rotval; else { fprintf (SUMA_STDERR, "Warning %s:\n" "Bad value for environment variable SUMA_NumForeSmoothing\n" "Assuming default of 0", FuncName); cf->X->NumForeSmoothing = 0; } } else cf->X->NumForeSmoothing = 0; } { char *eee = getenv("SUMA_NumFinalSmoothing"); if (eee) { int rotval = (int)strtod(eee, NULL); if (rotval >= 0) cf->X->NumFinalSmoothing = rotval; else { fprintf (SUMA_STDERR, "Warning %s:\n" "Bad value for environment variable SUMA_NumFinalSmoothing\n" "Assuming default of 0", FuncName); cf->X->NumFinalSmoothing = 0; } } else cf->X->NumFinalSmoothing = 0; } { char *eee = getenv("SUMA_ThresholdScalePower"); if (eee) { cf->SUMA_ThrScalePowerBias = (int)strtod(eee, NULL); if (cf->SUMA_ThrScalePowerBias < 2) { fprintf (SUMA_STDERR, "Warning %s:\n" "Bad value for environment variable\n" "SUMA_ThresholdScalePower.\n" "Assuming default of 2", FuncName); cf->SUMA_ThrScalePowerBias = 2; } } else cf->SUMA_ThrScalePowerBias = 2; } { char *eee = getenv("SUMA_SnapshotOverSampling"); if (eee) { cf->SUMA_SnapshotOverSampling = (int)strtod(eee, NULL); if ( cf->SUMA_SnapshotOverSampling < 1 || cf->SUMA_SnapshotOverSampling>10) { fprintf (SUMA_STDERR, "Warning %s:\n" "Bad value for environment variable\n" "SUMA_SnapshotOverSampling.\n" "Assuming default of 1", FuncName); cf->SUMA_SnapshotOverSampling = 1; } } else cf->SUMA_SnapshotOverSampling = 1; } { char *eee = getenv("SUMA_WarnBeforeClose"); if (eee) { if (strcmp(eee,"NO") == 0) cf->X->WarnClose = NOPE; else if (strcmp(eee,"YES") == 0) cf->X->WarnClose = YUP; else { fprintf (SUMA_STDERR, "Warning %s:\n" "Bad value for environment variable SUMA_WarnBeforeClose\n" "Assuming default of YES", FuncName); cf->X->WarnClose = YUP; } } else cf->X->WarnClose = YUP; } cf->X->SwitchCmapLst = NULL; cf->MessageList = SUMA_CreateMessageList (); #ifdef USE_SUMA_MALLOC SUMA_SL_Err("NO LONGER SUPPORTED"); return(NULL); /*SUMA_ShowMemTrace (cf->Mem, NULL);*/ #endif cf->ROI_mode = NOPE; cf->ROI_contmode = YUP; cf->Pen_mode = NOPE; cf->nimlROI_Datum_type = NI_rowtype_define("SUMA_NIML_ROI_DATUM", "int,int,int,int[#3]"); if (cf->nimlROI_Datum_type < 0) { fprintf(SUMA_STDERR,"Error %s: Failed defining niml code.", FuncName); return(NULL); } if (LocalHead) fprintf(SUMA_STDERR, "%s: roi_type code = %d\n", FuncName, cf->nimlROI_Datum_type) ; cf->ROI_CM = NULL; cf->ROI_FillMode = SUMA_ROI_FILL_TO_THISROI; cf->ROI2afni = NOPE; eee = getenv("SUMA_ColorMixingMode"); if (eee) { if (strcmp (eee, "ORIG") == 0) { cf->ColMixMode = SUMA_ORIG_MIX_MODE; } else if (strcmp (eee, "MOD1") == 0) { cf->ColMixMode = SUMA_4AML; } else { cf->ColMixMode = SUMA_ORIG_MIX_MODE; fprintf(SUMA_STDERR, "%s:\nUnrecognized option %s for SUMA_ColorMixingMode.\n" "Using default = ORIG\n", FuncName, eee); } } else { cf->ColMixMode = SUMA_ORIG_MIX_MODE; } cf->GroupList = NULL; cf->N_Group = -1; SUMA_RegisterGroup(cf,"ANY"); /* register a default group that would match any group*/ cf->scm = NULL; cf->DsetList = (DList *)SUMA_calloc(1,sizeof(DList)); dlist_init (cf->DsetList, SUMA_FreeDset); { char *eee = getenv("SUMA_AllowDsetReplacement"); if (eee) { if (strcmp(eee,"NO") == 0) cf->Allow_Dset_Replace = NOPE; else if (strcmp(eee,"YES") == 0) cf->Allow_Dset_Replace = YUP; else { fprintf (SUMA_STDERR, "Warning %s:\n" "Bad value for environment variable SUMA_AllowDsetReplacement\n" "Assuming default of YES", FuncName); cf->Allow_Dset_Replace = YUP; } } else cf->Allow_Dset_Replace = YUP; } cf->IgnoreVolreg = NOPE; cf->isGraphical = NOPE; cf->N_ClipPlanes = 0; for (i=0; iClipPlanes[4*i] = cf->ClipPlanes[4*i+1] = cf->ClipPlanes[4*i+2] = cf->ClipPlanes[4*i+3]= 0.0; cf->ClipPlaneType[i] = SUMA_NO_CLIP_PLANE_TYPE; cf->ClipPlanesLabels[i][0]='\0'; } for (i=0; iTimer[i].name[0] = '\0'; cf->Timer[i].lastcall = -1.0; } cf->N_Timer = 0; { char *eee = getenv("SUMA_NoDuplicatesInRecorder"); if (eee) { if (strcmp(eee,"NO") == 0) cf->NoDuplicatesInRecorder = 0; else if (strcmp(eee,"YES") == 0) cf->NoDuplicatesInRecorder = 1; else { fprintf (SUMA_STDERR, "Warning %s:\n" "Bad value for environment variable:\n" " SUMA_NoDuplicatesInRecorder\n" "Assuming default of YES", FuncName); cf->NoDuplicatesInRecorder = 1; } } else cf->NoDuplicatesInRecorder = 1; } /* if (SUMAg_CF->NoDuplicatesInRecorder) SNAP_NoDuplicates(); else SNAP_OkDuplicates(); */ cf->cwd = SUMA_getcwd(); { char *eee = getenv("SUMA_ColorMapRotationFraction"); if (eee) { cf->CmapRotaFrac = atof(eee); if (cf->CmapRotaFrac < 0.01 || cf->CmapRotaFrac > 0.99) { SUMA_S_Warn( "Values for environment variable SUMA_ColorMapRotationFraction\n" "are outside valid range of [0.01 .. 0.99]. \n" "Setting value to default of 0.05."); cf->CmapRotaFrac = 0.05; } } else { cf->CmapRotaFrac = 0.05; } } { char *eee = getenv("SUMA_Transparency_Step"); if (eee) { cf->TransModeStep = atof(eee); if (cf->TransModeStep < 1 || cf->TransModeStep > 8) { SUMA_S_Warn( "Values for environment variable SUMA_Transparency_Step\n" "are outside valid range of [1 .. 8]. \n" "Setting value to default of 4."); cf->TransModeStep = 4; } } else { cf->TransModeStep = 4; } } cf->xforms = (DList *)SUMA_calloc(1,sizeof(DList)); dlist_init (cf->xforms, SUMA_FreeXform); cf->callbacks = (DList *)SUMA_calloc(1,sizeof(DList)); dlist_init (cf->callbacks, SUMA_FreeCallback); cf->HoldClickCallbacks = 0; cf->PointerLastInViewer = -1; cf->giset = NULL; cf->ITset = NULL; cf->autorecord = SUMA_SetAutoRecord(getenv("SUMA_AutoRecordPrefix")); cf->SaveList = NULL; cf->YokeIntToNode = 0; cf->lev = NULL; return (cf); } SUMA_PARSED_NAME *SUMA_SetAutoRecord(char *pref) { static char FuncName[]={"SUMA_SetAutoRecord"}; SUMA_PARSED_NAME *pn=NULL; SUMA_ENTRY; if (!pref) SUMA_RETURN(SUMA_ParseFname("./autorecord", NULL)); if (!(pn = SUMA_ParseFname(pref, NULL))) { SUMA_S_Errv("Failed to parse %s\n", pref); SUMA_RETURN(SUMA_ParseFname("./autorecord", NULL)); } SUMA_RETURN(pn); } /*! \brief creates the structure for storing the radio buttons used to control viewer locking Do not use CommonFields structure here. */ SUMA_rb_group *SUMA_CreateLock_rbg (int N_rb_group, int N_but) { static char FuncName[]={"SUMA_CreateLock_rbg"}; SUMA_rb_group *Lock_rb; Lock_rb = (SUMA_rb_group *) calloc(1,sizeof(SUMA_rb_group)); if (!Lock_rb) { fprintf (SUMA_STDERR,"Error %s: Failed to allocate.\n", FuncName); return(NULL); } Lock_rb->N_rb_group = N_rb_group; Lock_rb->N_but = N_but; Lock_rb->tb = (Widget *) calloc(N_rb_group*N_but, sizeof(Widget)); Lock_rb->rb = (Widget *) calloc(N_rb_group, sizeof(Widget)); Lock_rb->atb = (Widget *) calloc(N_but, sizeof(Widget)); Lock_rb->arb = NULL; if (!Lock_rb->tb || !Lock_rb->rb || !Lock_rb->atb) { fprintf (SUMA_STDERR,"Error %s: Failed to allocate.\n", FuncName); return(NULL); } return(Lock_rb); } /*! free SUMA_rb_group * Do not use CommonFields structure here. */ void * SUMA_FreeLock_rbg (SUMA_rb_group *Lock_rb) { static char FuncName[]={"SUMA_FreeLock_rb"}; if (Lock_rb->rb) free(Lock_rb->rb); if (Lock_rb->tb) free(Lock_rb->tb); if (Lock_rb->atb) free (Lock_rb->atb); if (Lock_rb) free(Lock_rb); return (NULL); } /*! \brief DrawROI = SUMA_CreateDrawROIStruct(); allocates and initializes structure of type \return SUMA_X_DrawROI * */ SUMA_X_DrawROI *SUMA_CreateDrawROIStruct (void) { static char FuncName[]={"SUMA_CreateDrawROIStruct"}; SUMA_X_DrawROI *DrawROI = NULL; /* do not use commonfields related stuff here for obvious reasons */ DrawROI = (SUMA_X_DrawROI *)calloc (1, sizeof(SUMA_X_DrawROI)); DrawROI->AppShell = NULL; DrawROI->ROIval = (SUMA_ARROW_TEXT_FIELD *)calloc(1, sizeof(SUMA_ARROW_TEXT_FIELD)); DrawROI->ROIlbl = (SUMA_ARROW_TEXT_FIELD *)calloc(1, sizeof(SUMA_ARROW_TEXT_FIELD)); DrawROI->curDrawnROI = NULL; /* DO NOT FREE THIS POINTER */ DrawROI->SwitchROIlst = NULL; DrawROI->Delete_first = YUP; DrawROI->SaveMode = SW_DrawROI_SaveModeNIML; DrawROI->SaveWhat = SW_DrawROI_SaveWhatRelated; DrawROI->WhatDist = SW_DrawROI_WhatDistNothing; return (DrawROI); } /*! \brief SumaCont = SUMA_CreateSumaContStruct(); allocates and initializes structure of type SUMA_X_SumaCont \return SUMA_X_SumaCont * */ SUMA_X_SumaCont *SUMA_CreateSumaContStruct (void) { static char FuncName[]={"SUMA_CreateSumaContStruct"}; SUMA_X_SumaCont *SumaCont = NULL; /* do not use commonfields related stuff here for obvious reasons */ SumaCont = (SUMA_X_SumaCont *)calloc(1,sizeof(SUMA_X_SumaCont)); SumaCont->AppShell = NULL; SumaCont->quit_pb = NULL; SumaCont->quit_first = YUP; SumaCont->Lock_rbg = SUMA_CreateLock_rbg (SUMA_MAX_SURF_VIEWERS, 3); if (!SumaCont->Lock_rbg) { fprintf (SUMA_STDERR, "Error %s: Failed in SUMA_CreateLock_rb.\n", FuncName); return (NULL); } SumaCont->LockView_tbg = (Widget *)calloc (SUMA_MAX_SURF_VIEWERS, sizeof(Widget)); SumaCont->LockAllView_tb = NULL; SumaCont->SumaInfo_TextShell = NULL; return (SumaCont); } /*! \brief frees structure SUMA_X_SumaCont, returns null */ void *SUMA_FreeSumaContStruct (SUMA_X_SumaCont *SumaCont) { static char FuncName[]={"SUMA_FreeSumaContStruct"}; /* do not use commonfields related stuff here for obvious reasons */ if (SumaCont->Lock_rbg) SUMA_FreeLock_rbg (SumaCont->Lock_rbg); if (SumaCont->LockView_tbg) free (SumaCont->LockView_tbg); if (SumaCont->SumaInfo_TextShell) { SUMA_SL_Warn("SumaCont->SumaInfo_TextShell is not being freed") }; if (SumaCont) free(SumaCont); return (NULL); } /*! \brief frees structure SUMA_X_DrawROI, returns null */ void *SUMA_FreeDrawROIStruct (SUMA_X_DrawROI *DrawROI) { static char FuncName[]={"SUMA_FreeDrawROIStruct"}; /* do not use commonfields related stuff here for obvious reasons, Well, you can, it is no big deal, memory tracing variables are wiped out at the end*/ if (DrawROI->ROIval) free (DrawROI->ROIval); if (DrawROI->ROIlbl) free (DrawROI->ROIlbl); if (DrawROI->SwitchROIlst) SUMA_FreeScrolledList (DrawROI->SwitchROIlst); if (DrawROI) free(DrawROI); return (NULL); } /*! \brief ViewCont = SUMA_CreateViewContStruct(); allocates and initializes structure of type SUMA_X_ViewCont \return SUMA_X_ViewCont * */ SUMA_X_ViewCont *SUMA_CreateViewContStruct (void) { static char FuncName[]={"SUMA_CreateViewContStruct"}; SUMA_X_ViewCont *ViewCont = NULL; /* do not use commonfields related stuff here for obvious reasons */ ViewCont = (SUMA_X_ViewCont *)calloc(1,sizeof(SUMA_X_ViewCont)); ViewCont->TopLevelShell = NULL; ViewCont->ViewerInfo_TextShell = NULL; ViewCont->Info_lb = NULL; ViewCont->ViewerInfo_pb = NULL; ViewCont->Mainform = NULL; ViewCont->SwitchGrouplst = NULL; ViewCont->SwitchStatelst = NULL; ViewCont->ViewerInfo_pb = NULL; return (ViewCont); } /*! \brief frees structure SUMA_X_ViewCont, returns null */ void *SUMA_FreeViewContStruct (SUMA_X_ViewCont *ViewCont) { static char FuncName[]={"SUMA_FreeViewContStruct"}; /* do not use commonfields related stuff here for obvious reasons */ if (ViewCont->TopLevelShell) { SUMA_SL_Warn("ViewCont->TopLevelShell is not being freed"); } if (ViewCont->SwitchGrouplst) ViewCont->SwitchGrouplst = SUMA_FreeScrolledList(ViewCont->SwitchGrouplst); if (ViewCont->SwitchStatelst) ViewCont->SwitchStatelst = SUMA_FreeScrolledList(ViewCont->SwitchStatelst); if (ViewCont) free(ViewCont); return (NULL); } /*! \brief SurfCont = SUMA_CreateSurfContStruct(); allocates and initializes structure of type SUMA_X_SurfCont \return SUMA_X_SurfCont * This function is now creating controllers for other types of displayable objects too. */ SUMA_X_SurfCont *SUMA_CreateSurfContStruct (char *idcode_str, SUMA_DO_Types tp) { static char FuncName[]={"SUMA_CreateSurfContStruct"}; SUMA_X_SurfCont *SurfCont = NULL; char wname[64]={"UNNAMED"}, *s=NULL; /* do not use commonfields related stuff here for obvious reasons */ SurfCont = (SUMA_X_SurfCont *)malloc(sizeof(SUMA_X_SurfCont)); memset(SurfCont, 0, sizeof(SUMA_X_SurfCont)); /* take care of linking fields */ if (idcode_str) sprintf(SurfCont->owner_id, "%s", idcode_str); else SurfCont->owner_id[0] = '\0'; SurfCont->N_links = 0; SurfCont->Open = 0; SurfCont->LinkedPtrType = SUMA_LINKED_SURFCONT_TYPE; SurfCont->do_type = tp; SurfCont->DsetMap_fr = NULL; SurfCont->ColPlane_fr = NULL; SurfCont->DispFrame = NULL; SurfCont->SurfFrame = NULL; SurfCont->Xhair_fr = NULL; SurfCont->TLS = NULL; SurfCont->Mainform = NULL; SurfCont->Page = NULL; SurfCont->SurfInfo_pb = NULL; SurfCont->SurfInfo_label = NULL; SurfCont->SurfInfo_TextShell = NULL; SurfCont->SurfContPage_label = NULL; SurfCont->SurfContPage = (SUMA_ARROW_TEXT_FIELD *)calloc(1, sizeof(SUMA_ARROW_TEXT_FIELD)); SurfCont->NodeRadGainAF = (SUMA_ARROW_TEXT_FIELD *)calloc(1, sizeof(SUMA_ARROW_TEXT_FIELD)); SurfCont->EdgeThickGainAF = (SUMA_ARROW_TEXT_FIELD *)calloc(1, sizeof(SUMA_ARROW_TEXT_FIELD)); SurfCont->ColPlaneOrder = (SUMA_ARROW_TEXT_FIELD *)calloc(1, sizeof(SUMA_ARROW_TEXT_FIELD)); SurfCont->ColPlaneOpacity = (SUMA_ARROW_TEXT_FIELD *)calloc(1, sizeof(SUMA_ARROW_TEXT_FIELD)); SurfCont->ColPlaneDimFact = (SUMA_ARROW_TEXT_FIELD *)calloc(1, sizeof(SUMA_ARROW_TEXT_FIELD)); SurfCont->ColPlaneAlphaThresh = (SUMA_ARROW_TEXT_FIELD *)calloc(1, sizeof(SUMA_ARROW_TEXT_FIELD)); SurfCont->TractMaskGray = (SUMA_ARROW_TEXT_FIELD *)calloc(1, sizeof(SUMA_ARROW_TEXT_FIELD)); s = SUMA_do_type_2_contwname(SurfCont->do_type); snprintf(wname,63,"%s->XhairTable", s); SurfCont->XhairTable = SUMA_AllocTableField(wname); snprintf(wname,63,"%s->MaskTable", s); SurfCont->MaskTable = SUMA_AllocTableField(wname); snprintf(wname,63,"%s->MaskEvalTable", s); SurfCont->MaskEvalTable = SUMA_AllocTableField(wname); SurfCont->MaskEval_tb = NULL; snprintf(wname,63,"%s->MaskLenTable", s); SurfCont->MaskLenTable = SUMA_AllocTableField(wname); SurfCont->MaskLen_tb = NULL; SurfCont->UseMaskEval = 0; SurfCont->DeleteMask_pb = NULL; SurfCont->DeleteMask_first = YUP; SurfCont->DeleteMask_row = -1; snprintf(wname,63,"%s->SetRangeTable", s); SurfCont->SetRangeTable = SUMA_AllocTableField(wname); snprintf(wname,63,"%s->SetThrScaleTable", s); SurfCont->SetThrScaleTable = SUMA_AllocTableField(wname); snprintf(wname,63,"%s->RangeTable", s); SurfCont->RangeTable = SUMA_AllocTableField(wname); snprintf(wname,63,"%s->NodeTable", s); SurfCont->NodeTable = SUMA_AllocTableField(wname); snprintf(wname,63,"%s->FaceTable", s); SurfCont->FaceTable = SUMA_AllocTableField(wname); snprintf(wname,63,"%s->DataTable", s); SurfCont->DataTable = SUMA_AllocTableField(wname); snprintf(wname,63,"%s->LabelTable", s); SurfCont->LabelTable = SUMA_AllocTableField(wname); /* SurfCont->ColPlaneShow_tb = NULL; Obsolete */ SurfCont->ColPlaneShowOneFore_tb = NULL; SurfCont->SymIrange_tb = NULL; SurfCont->AbsThresh_tb = NULL; SurfCont->ShowZero_tb = NULL; SurfCont->SwitchDsetlst = NULL; snprintf(wname,63,"%s->ColPlaneLabelTable", s); SurfCont->ColPlaneLabelTable = SUMA_AllocTableField(wname); snprintf(wname,63,"%s->SetClustTable", s); SurfCont->SetClustTable = SUMA_AllocTableField(wname); snprintf(wname,63,"%s->Ax_slc", s); SurfCont->Ax_slc = SUMA_AllocSliceField(wname); snprintf(wname,63,"%s->Sa_slc", s); SurfCont->Sa_slc = SUMA_AllocSliceField(wname); snprintf(wname,63,"%s->Co_slc", s); SurfCont->Co_slc = SUMA_AllocSliceField(wname); snprintf(wname,63,"%s->VR", s); SurfCont->VR_fld = SUMA_AllocVRField(wname); SurfCont->curColPlane = NULL; { char *eee = getenv("SUMA_ShowOneOnly"); if (eee) { SUMA_TO_LOWER(eee); if (strcmp (eee, "yes") == 0) SurfCont->ShowCurForeOnly = YUP; else SurfCont->ShowCurForeOnly = NOPE; } else { SurfCont->ShowCurForeOnly = YUP; } } { char *eee = getenv("SUMA_GraphHidden"); if (eee) { SUMA_TO_LOWER(eee); if (strcmp (eee, "yes") == 0) SurfCont->GraphHidden = YUP; else SurfCont->GraphHidden = NOPE; } else { SurfCont->GraphHidden = YUP; } } SurfCont->prv_curDOp = (void **)calloc(1, sizeof(void*)); SurfCont->prv_variant = NULL; SurfCont->PosRef = NULL; SurfCont->cmp_ren = (SUMA_CMAP_RENDER_AREA *)SUMA_calloc(1, sizeof(SUMA_CMAP_RENDER_AREA)); SurfCont->cmp_ren->CrappyDrawable = 0; SurfCont->cmp_ren->cmap_wid = NULL; SurfCont->cmp_ren->FOV = SUMA_CMAP_FOV_INITIAL; SurfCont->cmp_ren->cmap_context = NULL; SurfCont->cmp_ren->translateVec[0] = SurfCont->cmp_ren->translateVec[0] = SurfCont->cmp_ren->translateVec[1] = 0.0; SurfCont->thr_sc = NULL; SurfCont->brt_sc = NULL; SurfCont->thr_lb = NULL; SurfCont->thrstat_lb = NULL; SurfCont->cmaptit_lb = NULL; SurfCont->cmapswtch_pb = NULL; SurfCont->CmapLoad_pb = NULL; SurfCont->SwitchIntMenu = NULL; SurfCont->SwitchBrtMenu = NULL; SurfCont->SwitchThrMenu = NULL; #if 0 /* Now in SwitchIntMenu */ SurfCont->SwitchIntLst = NULL; SurfCont->SwitchThrLst = NULL; SurfCont->SwitchBrtLst = NULL; SurfCont->SwitchIntArrow = NULL; SurfCont->SwitchBrtArrow = NULL; SurfCont->SwitchThrArrow = NULL; #endif SurfCont->SwitchCmapMenu = NULL; SurfCont->Fake_pbar = NULL; SurfCont->rc_CmapCont = NULL; SurfCont->CoordBiasMenu = SUMA_Alloc_Menu_Widget(SW_N_CoordBias); SurfCont->LinkModeMenu = SUMA_Alloc_Menu_Widget(SW_N_LinkMode); SurfCont->CmapModeMenu = SUMA_Alloc_Menu_Widget(SW_N_CmapMode); SurfCont->opts_rc = NULL; SurfCont->opts_form = NULL; SurfCont->rcvo = NULL; SurfCont->rcsw = NULL; SurfCont->rcsw_v1 = NULL; SurfCont->rcsw_v2 = NULL; SurfCont->rcswr = NULL; SurfCont->rccm = NULL; SurfCont->rccm_swcmap = NULL; SurfCont->IntRange_lb = NULL; SurfCont->Int_tb = NULL; SurfCont->Thr_tb = NULL; SurfCont->Brt_tb = NULL; SurfCont->IntRangeLocked = 0; SurfCont->BrtRangeLocked = 0; SurfCont->rcclust = NULL; SurfCont->tract_length_mask[0] = 0.0; /* SUMA_floatEnv("TEMP_MIN_LENGTH", 0.0); */ SurfCont->tract_length_mask[1] = -1.0; /* SUMA_floatEnv("TEMP_MAX_LENGTH", -1.0); */ if (SurfCont->tract_length_mask[1]>= SurfCont->tract_length_mask[0]) SurfCont->UseMaskLen = 0; /* leave it off, let it be activated in GUI */ else SurfCont->UseMaskLen = 0; return (SurfCont); } SUMA_X_SurfCont *SUMA_GlobalMaskContStruct(char *idcode) { static char FuncName[]={"SUMA_GlobalMaskContStruct"}; if (!SUMAg_CF->X->AllMaskCont) { if (!(SUMAg_CF->X->AllMaskCont = SUMA_CreateSurfContStruct(idcode, MASK_type))){ fprintf(SUMA_STDERR, "Error %s: Failed to create global mask controller struct", FuncName); return(NULL); } } /* Return a link to that struct */ return((SUMA_X_SurfCont*) SUMA_LinkToPointer((void *)SUMAg_CF->X->AllMaskCont)); } /*! \brief frees structure SUMA_X_SurfCont, returns null */ void *SUMA_FreeSurfContStruct (SUMA_X_SurfCont *SurfCont) { static char FuncName[]={"SUMA_FreeSurfContStruct"}; /* do not use commonfields related stuff here for obvious reasons */ if (!SurfCont) return(NULL); if (SurfCont->N_links) { SurfCont = (SUMA_X_SurfCont*)SUMA_UnlinkFromPointer((void *)SurfCont); return (NULL); } /* no more links, go for it */ if (SurfCont->TractMaskGray) free (SurfCont->TractMaskGray); if (SurfCont->SurfContPage) free (SurfCont->SurfContPage); if (SurfCont->NodeRadGainAF) free (SurfCont->NodeRadGainAF); if (SurfCont->EdgeThickGainAF) free (SurfCont->EdgeThickGainAF); if (SurfCont->ColPlaneOrder) free (SurfCont->ColPlaneOrder); if (SurfCont->ColPlaneOpacity) free (SurfCont->ColPlaneOpacity); if (SurfCont->ColPlaneDimFact) free (SurfCont->ColPlaneDimFact); if (SurfCont->ColPlaneAlphaThresh) free (SurfCont->ColPlaneAlphaThresh); if (SurfCont->SetRangeTable) SUMA_FreeTableField (SurfCont->SetRangeTable); if (SurfCont->RangeTable) SUMA_FreeTableField (SurfCont->RangeTable); if (SurfCont->MaskTable) SUMA_FreeTableField (SurfCont->MaskTable); if (SurfCont->MaskEvalTable) SUMA_FreeTableField (SurfCont->MaskEvalTable); if (SurfCont->MaskLenTable) SUMA_FreeTableField (SurfCont->MaskLenTable); if (SurfCont->XhairTable) SUMA_FreeTableField (SurfCont->XhairTable); if (SurfCont->NodeTable) SUMA_FreeTableField (SurfCont->NodeTable); if (SurfCont->FaceTable) SUMA_FreeTableField (SurfCont->FaceTable); if (SurfCont->DataTable) SUMA_FreeTableField (SurfCont->DataTable); if (SurfCont->LabelTable) SUMA_FreeTableField (SurfCont->LabelTable); if (SurfCont->Ax_slc) SUMA_FreeSliceField (SurfCont->Ax_slc); if (SurfCont->Sa_slc) SUMA_FreeSliceField (SurfCont->Sa_slc); if (SurfCont->Co_slc) SUMA_FreeSliceField (SurfCont->Co_slc); if (SurfCont->VR_fld) SUMA_FreeVRField (SurfCont->VR_fld); if (SurfCont->ColPlaneLabelTable) SUMA_FreeTableField (SurfCont->ColPlaneLabelTable); if (SurfCont->SetClustTable) SUMA_FreeTableField (SurfCont->SetClustTable); if (SurfCont->SwitchDsetlst) SUMA_FreeScrolledList (SurfCont->SwitchDsetlst); if (SurfCont->SurfInfo_TextShell) { SUMA_SL_Warn("SurfCont->SurfInfo_TextShell is not being freed") }; SurfCont->SwitchIntMenu = SUMA_Free_Menu_Widget(SurfCont->SwitchIntMenu); SurfCont->SwitchThrMenu = SUMA_Free_Menu_Widget(SurfCont->SwitchThrMenu); SurfCont->SwitchBrtMenu = SUMA_Free_Menu_Widget(SurfCont->SwitchBrtMenu); SurfCont->SwitchCmapMenu = SUMA_Free_Menu_Widget(SurfCont->SwitchCmapMenu); SurfCont->DsetNodeRadMenu = SUMA_Free_Menu_Widget(SurfCont->DsetNodeRadMenu); SurfCont->DsetEdgeThickMenu = SUMA_Free_Menu_Widget(SurfCont->DsetEdgeThickMenu); SurfCont->DsetEdgeStipMenu = SUMA_Free_Menu_Widget(SurfCont->DsetEdgeStipMenu); SurfCont->TractStyleMenu = SUMA_Free_Menu_Widget(SurfCont->TractStyleMenu); SurfCont->DsetAlphaValMenu = SUMA_Free_Menu_Widget(SurfCont->DsetAlphaValMenu); SurfCont->TractMaskMenu = SUMA_Free_Menu_Widget(SurfCont->TractMaskMenu); if (SurfCont->prv_curDOp) free(SurfCont->prv_curDOp); if (SurfCont->prv_variant) free(SurfCont->prv_variant); if (SurfCont->cmp_ren) free(SurfCont->cmp_ren); if (SurfCont) free(SurfCont); return (NULL); } SUMA_MENU_WIDGET *SUMA_Free_Menu_Widget(SUMA_MENU_WIDGET *smw) { static char FuncName[]={"SUMA_Free_Menu_Widget"}; if (!smw) return(NULL); if (smw->mw) { XtDestroyWidget(smw->mw[0]); SUMA_free(smw->mw); } if (smw->lw) { SUMA_cb_CloseSwitchLst (NULL, (XtPointer)smw->lw, NULL); smw->lw = SUMA_FreeScrolledList(smw->lw); } if (smw) SUMA_free(smw); return(NULL); } SUMA_MENU_WIDGET *SUMA_Alloc_Menu_Widget(int nw) { SUMA_MENU_WIDGET *smw=NULL; smw = (SUMA_MENU_WIDGET *)SUMA_calloc(1, sizeof(SUMA_MENU_WIDGET)); if (nw) { smw->mw = (Widget *)SUMA_calloc(nw, sizeof(Widget)); smw->N_mw = nw; } return(smw); } /*! free SUMA_CommonFields NOTE: the SUMA_CommonFields * itself is not freed. You'll have to free it manually with free function; \sa SUMA_Create_CommonFields */ SUMA_Boolean SUMA_Free_CommonFields (SUMA_CommonFields *cf) { static char FuncName[]={"SUMA_Free_CommonFields"}; int i; /* do not use commonfields related stuff here for obvious reasons */ if (cf->cwd) SUMA_free(cf->cwd); cf->cwd = NULL; if (cf->GroupList) { for (i=0; i< cf->N_Group; ++i) if (cf->GroupList[i]) SUMA_free(cf->GroupList[i]); SUMA_free(cf->GroupList); cf->GroupList = NULL; } #if 0 /* not anymore!, that is now a pointer copy */ if (cf->ROI_CM) SUMA_Free_ColorMap(cf->ROI_CM); /* free the colormap */ #endif cf->ROI_CM = NULL; if (cf->X->FileSelectDlg) SUMA_FreeFileSelectionDialogStruct(cf->X->FileSelectDlg); cf->X->FileSelectDlg = NULL; if (cf->X->AllMaskCont) SUMA_FreeSurfContStruct(cf->X->AllMaskCont); SUMA_ifree(cf->X->Cr); for (i=0; iN_dcom; ++i) { SUMA_ifree(cf->dcom[i]); } SUMA_ifree(cf->dcom); if (cf->X->SumaCont) SUMA_FreeSumaContStruct (cf->X->SumaCont); SUMA_free_DocumentedWidgets(); cf->X->SumaCont = NULL; if (cf->X->DrawROI) SUMA_FreeDrawROIStruct (cf->X->DrawROI); cf->X->DrawROI = NULL; if (cf->X->N_ForeSmooth_prmpt) SUMA_FreePromptDialogStruct (cf->X->N_ForeSmooth_prmpt); cf->X->N_ForeSmooth_prmpt = NULL; if (cf->X->N_FinalSmooth_prmpt) SUMA_FreePromptDialogStruct (cf->X->N_FinalSmooth_prmpt); cf->X->N_FinalSmooth_prmpt = NULL; if (cf->X->Clip_prmpt) SUMA_FreePromptDialogStruct (cf->X->Clip_prmpt); cf->X->Clip_prmpt = NULL; if (cf->X->ClipObj_prmpt) SUMA_FreePromptDialogStruct (cf->X->ClipObj_prmpt); cf->X->ClipObj_prmpt = NULL; if (cf->X->SwitchCmapLst) SUMA_FreeScrolledList (cf->X->SwitchCmapLst); if (cf->X) free(cf->X); cf->X = NULL; if (cf->MessageList) SUMA_EmptyDestroyList(cf->MessageList); cf->MessageList = NULL; if (cf->scm) cf->scm = SUMA_DestroyAfniColors (cf->scm); cf->scm = NULL; if (cf->DsetList) { dlist_destroy(cf->DsetList); SUMA_free(cf->DsetList); cf->DsetList = NULL; } if (cf->xforms) { dlist_destroy(cf->xforms); SUMA_free(cf->xforms); } if (cf->callbacks) { dlist_destroy(cf->callbacks); SUMA_free(cf->callbacks); } #ifdef USE_SUMA_MALLOC SUMA_SL_Err("NO LONGER SUPPORTED"); return(NOPE); if (cf->Mem) SUMA_Free_MemTrace (cf->Mem); cf->Mem = NULL;/* always free this right before the end */ #endif if (cf->giset) { if (cf->giset->dset) { SUMA_S_Warn("dset is not being freed"); } DESTROY_GICOR_setup(cf->giset); cf->giset=NULL; } if (cf->ITset) { Free_Insta_Tract_Setup(cf->ITset); SUMA_ifree(cf->ITset); } if (cf->autorecord) { cf->autorecord = SUMA_Free_Parsed_Name(cf->autorecord); } if (cf->SaveList) { dlist_destroy(cf->SaveList); SUMA_free(cf->SaveList); } if (cf->lev) { SUMA_free(cf->lev); cf->lev = NULL; } /* if (cf) free(cf); */ /* don't free this stupid pointer since it is used when main returns with SUMA_ RETURN (typo on purpose to avoid upsetting AnalyzeTrace. It is not quite a leak since the OS will clean it up after exit Thu Apr 8 2004*/ return (YUP); } void SUMA_Show_Clip_Planes (SUMA_CommonFields *cf, FILE *out) { static char FuncName[]={"SUMA_Show_Clip_Planes"}; char *s=NULL; SUMA_ENTRY; s = SUMA_Show_Clip_Planes_Info (cf); /* //Don't output number of planes or their equations if (!out) fprintf(SUMA_STDERR,"%s", s); else fprintf(out,"%s", s); */ SUMA_free(s); SUMA_RETURNe; } const char * SUMA_Clip_Type_to_Clip_Name (SUMA_CLIP_PLANE_TYPES tp) { switch (tp) { case SUMA_NO_CLIP_PLANE_TYPE: return("No_type"); case SUMA_SCREEN_CLIP: return("Screen_Clip"); case SUMA_ALL_OBJECT_CLIP: return("All_Objects_Clip"); default: return("Bad value"); } } char * SUMA_Show_Clip_Planes_Info (SUMA_CommonFields *cf) { static char FuncName[]={"SUMA_Show_Clip_Planes_Info"}; int i; char *s=NULL; SUMA_STRING *SS=NULL; SUMA_ENTRY; SS = SUMA_StringAppend_va(NULL, NULL); if (cf == NULL) { SS = SUMA_StringAppend_va(SS," NULL cf structure.\n"); SS = SUMA_StringAppend_va(SS, NULL); s = SS->s; SUMA_free(SS); SS= NULL; SUMA_RETURN(s); } SS = SUMA_StringAppend_va(SS," Number of Clip Planes: %d\n", cf->N_ClipPlanes); for (i=0; iN_ClipPlanes; ++i) { SS = SUMA_StringAppend_va(SS," %d: Clip plane >>%s<< of type %s. Eq: %.4fX + %.4fY + %.4fZ + %.4f = 0\n", i, cf->ClipPlanesLabels[i], SUMA_Clip_Type_to_Clip_Name(cf->ClipPlaneType[i]), (float)cf->ClipPlanes[4*i], (float)cf->ClipPlanes[4*i+1], (float)cf->ClipPlanes[4*i+2], (float)cf->ClipPlanes[4*i+3]); } SS = SUMA_StringAppend_va(SS, NULL); s = SS->s; SUMA_free(SS); SS= NULL; SUMA_RETURN(s); } void SUMA_Show_CommonFields (SUMA_CommonFields *cf, FILE *out) { static char FuncName[]={"SUMA_Show_CommonFields"}; char *s=NULL; SUMA_ENTRY; s = SUMA_CommonFieldsInfo (cf,1); if (!out) fprintf(SUMA_STDERR,"%s", s); else fprintf(out,"%s", s); SUMA_free(s); SUMA_RETURNe; } char * SUMA_CommonFieldsInfo (SUMA_CommonFields *cf, int detail) { static char FuncName[]={"SUMA_CommonFieldsInfo"}; int i; char *s=NULL; SUMA_DSET *dset=NULL; DListElmt *el=NULL; SUMA_STRING *SS=NULL; SUMA_ENTRY; SS = SUMA_StringAppend_va(NULL, NULL); if (cf == NULL) { SS = SUMA_StringAppend_va(SS," NULL cf structure.\n"); SS = SUMA_StringAppend_va(SS, NULL); s = SS->s; SUMA_free(SS); SS= NULL; SUMA_RETURN(s); } SS = SUMA_StringAppend_va(SS," CWD: %s\n", cf->cwd); for (i=0; i < SUMA_MAX_STREAMS; ++i) { SS = SUMA_StringAppend_va(SS," HostName: %s\n", cf->HostName_v[i]); SS = SUMA_StringAppend_va(SS," NimlStream: %s\n", cf->NimlStream_v[i]); } if (cf->X) { SS = SUMA_StringAppend_va(SS," X struct\n", " NumForeSmoothing %d, NumFinalSmmothing %d\n" " WarnClose %d, UseSameSurfCont %d, \n" " SameSurfContOpen %d ButtonDown %d, WinOffsets %d %d\n" , cf->X->NumForeSmoothing, cf->X->NumFinalSmoothing, cf->X->WarnClose, cf->X->UseSameSurfCont, cf->X->SameSurfContOpen, cf->X->ButtonDown, cf->X->roffx, cf->X->roffy); } else { SS = SUMA_StringAppend(SS," NULL X struct\n"); } SS = SUMA_StringAppend_va(SS," Available Groups: %d\n", cf->N_Group); for (i=0; iN_Group; ++i) { SS = SUMA_StringAppend_va(SS," Group[%d]: %s\n", i, cf->GroupList[i]); } #ifdef USE_SUMA_MALLOC SUMA_SL_Err("NO LONGER SUPPORTED"); SUMA_RETURN(NULL); #else SS = SUMA_StringAppend_va(SS," DBG_trace = %d\n", DBG_trace); SS = SUMA_StringAppend_va(SS," InOut_Notify = %d\n", cf->InOut_Notify); SS = SUMA_StringAppend_va(SS," Echo_KeyPress = %d\n", cf->Echo_KeyPress); SS = SUMA_StringAppend_va(SS," MemTrace = %d\n", cf->MemTrace); #endif SS = SUMA_StringAppend_va(SS," PointerSize = %d\n", cf->PointerSize); /* add the displayable objects Info */ s = SUMA_DOv_Info(SUMAg_DOv, SUMAg_N_DOv, 0); SS = SUMA_StringAppend_va(SS, "%s\n", s); SUMA_free(s); s = NULL; if (cf->DsetList) { SS = SUMA_StringAppend_va( SS, "DsetList (Allow Replacement = %d):\n", cf->Allow_Dset_Replace); el = NULL; do { if (!el) el = dlist_head(cf->DsetList); else el = dlist_next(el); if (el) { dset = (SUMA_DSET *)el->data; if (!dset) { SUMA_SLP_Err("Unexpected NULL dset element in list!\n" "Please report this occurrence to saadz@mail.nih.gov."); } else { s = SUMA_DsetInfo (dset,0); SS = SUMA_StringAppend_va(SS, "\n%s\n", s); SUMA_free(s); s = NULL; } } else { SUMA_S_Err("Unexpected nullness"); } } while ( (el != dlist_tail(cf->DsetList))); } else { SS = SUMA_StringAppend_va(SS, "NULL DsetList\n"); } SS = SUMA_StringAppend_va( SS, "SUMA's list of environment variables:\n"); s = SUMA_env_list_help(0, TXT); SS = SUMA_StringAppend( SS, s); SUMA_free(s); s = NULL; SS = SUMA_StringAppend( SS, "\n"); /* add the colormap info */ if (cf->scm) { SS = SUMA_StringAppend(SS, " Colormaps:\n"); s = SUMA_ColorMapVec_Info (cf->scm->CMv, cf->scm->N_maps, detail); SS = SUMA_StringAppend_va( SS, "%s\n",s); SUMA_free(s); s = NULL; } else { SS = SUMA_StringAppend_va(SS, " No Colormaps.\n"); } /* add xforms and callbacks */ s = SUMA_Xforms_Info(cf->xforms, detail); SS = SUMA_StringAppend_va( SS, "%s\n",s); SUMA_free(s); s = NULL; s = SUMA_Callbacks_Info(cf->callbacks, detail); SS = SUMA_StringAppend_va( SS, "%s\n",s); SUMA_free(s); s = NULL; SS = SUMA_StringAppend_va(SS, "Pointer last seen in viewer: %d\n", cf->PointerLastInViewer); s = SUMA_GISET_Info(cf->giset, 0); SS = SUMA_StringAppend_va(SS, "%s\n",s); SUMA_free(s); s = NULL; SS = SUMA_StringAppend_va(SS, "autorecord: %s\n", cf->autorecord ? cf->autorecord->FullName:"NULL"); SS = SUMA_StringAppend_va(SS, "SaveList: %d elements\n", cf->SaveList ? dlist_size(cf->SaveList):0); SS = SUMA_StringAppend_va(SS, "Documented Widgets:\n%s", SUMA_get_DocumentedWidgets() ? SUMA_get_DocumentedWidgets():"NULL"); /* clean up */ SS = SUMA_StringAppend_va(SS, NULL); s = SS->s; SUMA_free(SS); SS= NULL; SUMA_RETURN(s); } /*! This function determines the most suitable standard view of a surface viewer This is based on the surface objects being displayed and their embedding dimension. The highest Embedding dimension of the lot determines what view to use ans = SUMA_BestStandardView (SUMA_SurfaceViewer *sv, SUMA_DO *dov, int N_dov) \param sv (SUMA_SurfaceViewer *) Surface viewer structure \param dov (SUMA_DO *) vector of displayable objects \param N_dov (int) number of displayable objects \ret ans (SUMA_STANDARD_VIEWS) recommended view */ SUMA_STANDARD_VIEWS SUMA_BestStandardView ( SUMA_SurfaceViewer *sv, SUMA_DO *dov, int N_dov) { static char FuncName[] = {"SUMA_BestStandardView"}; SUMA_STANDARD_VIEWS ans; int N_MembSOs, *MembSOs=NULL, i, maxdim = -1, is, balls, vslice=0; SUMA_SurfaceObject *SO = NULL; char *variant=NULL; SUMA_DO_Types ttv[10]={SO_type, GRAPH_LINK_type, TRACT_type, VO_type, CDOM_type, MASK_type, NOT_SET_type}; SUMA_SO_SIDE side=SUMA_NO_SIDE; SUMA_Boolean LocalHead = NOPE; SUMA_ENTRY; is = sv->iState; if (is < 0) { fprintf(SUMA_STDERR, "Error %s: sv->iState undefined.\n", FuncName); SUMA_RETURN (SUMA_N_STANDARD_VIEWS); } side = SUMA_LEFT; balls = 0; vslice = 0; MembSOs = SUMA_ViewState_Membs(&(sv->VSv[is]), ttv, &N_MembSOs); SUMA_LHv("Working on %d members\n", N_MembSOs); for (i=0; iEmbedDim > maxdim) maxdim = SO->EmbedDim; if (SO->Side != SUMA_LEFT) side = SUMA_RIGHT; if (SUMA_IS_GEOM_SYMM(SO->isSphere)) balls = 1; break; case GRAPH_LINK_type: maxdim = 3; variant = SUMA_ADO_variant((SUMA_ALL_DO *)dov[MembSOs[i]].OP); if (!strcmp(variant,"GMATRIX")) { vslice = 1; /* a slice viewed from above */ } break; case VO_type: case CDOM_type: case MASK_type: case TRACT_type: maxdim = 3; break; default: SUMA_S_Err("Should not be here"); break; } } SUMA_ifree(MembSOs); switch (maxdim) { case 2: if (side == SUMA_LEFT) { /* left flat maps*/ SUMA_RETURN (SUMA_2D_Z0L); } else { /* default */ SUMA_RETURN (SUMA_2D_Z0); } case 3: if (!balls && !vslice) SUMA_RETURN(SUMA_3D); else SUMA_RETURN(SUMA_3D_Z0); default: fprintf(SUMA_STDERR, "Error %s: No provision for a maximum embedding dimension of %d.\n", FuncName, maxdim); SUMA_RETURN(SUMA_N_STANDARD_VIEWS); } } /*! \brief Apply the group of a surface to the surface viewer */ SUMA_Boolean SUMA_AdoptSurfGroup(SUMA_SurfaceViewer *csv, SUMA_SurfaceObject *SO) { static char FuncName[]={"SUMA_AdoptSurfGroup"}; SUMA_ENTRY; csv->iCurGroup = SUMA_WhichGroup(SUMAg_CF, SO->Group); if (csv->iCurGroup < 0) { SUMA_SL_Err("Bad, unexpected error.\nGroup was not found"); SUMA_RETURN(NOPE); } if (csv->CurGroupName) SUMA_free(csv->CurGroupName); csv->CurGroupName = SUMA_copy_string(SO->Group); SUMA_RETURN(YUP); } /*! \brief Apply a group to the surface viewer */ SUMA_Boolean SUMA_AdoptGroup(SUMA_SurfaceViewer *csv, char *group) { static char FuncName[]={"SUMA_AdoptGroup"}; SUMA_ENTRY; csv->iCurGroup = SUMA_WhichGroup(SUMAg_CF, group); if (csv->iCurGroup < 0) { SUMA_SL_Err("Bad, unexpected error.\nGroup was not found"); SUMA_RETURN(NOPE); } if (csv->CurGroupName) SUMA_free(csv->CurGroupName); csv->CurGroupName = SUMA_copy_string(SUMAg_CF->GroupList[csv->iCurGroup]); SUMA_RETURN(YUP); } SUMA_Boolean SUMA_SetViewerLightsForSO(SUMA_SurfaceViewer *cSV, SUMA_SurfaceObject *SO) { static char FuncName[]={"SUMA_SetViewerLightsForSO"}; SUMA_Boolean LocalHead = NOPE; SUMA_ENTRY; if (!cSV || !SO) SUMA_RETURN(NOPE); if (cSV->lit_for == 0) { /* olde way */ /* if surface is SureFit , flip lights */ if (SO->normdir == 0 && ( SO->FileType == SUMA_SUREFIT || SO->FileType == SUMA_OPENDX_MESH || SO->FileType == SUMA_OBJ_MESH || SO->FileType == SUMA_BRAIN_VOYAGER)) { SUMA_LH("Flippo for safety"); cSV->light0_position[0] *= -1; cSV->light0_position[1] *= -1; cSV->light0_position[2] *= -1; glLightfv(GL_LIGHT0, GL_POSITION, cSV->light0_position); } else if (SO->normdir == -1) { SUMA_LH("Flippo for safety"); cSV->light0_position[0] *= -1; cSV->light0_position[1] *= -1; cSV->light0_position[2] *= -1; glLightfv(GL_LIGHT0, GL_POSITION, cSV->light0_position); } } else { SUMA_LHv("Auto Flippo for safety, %d, %d\n", cSV->lit_for , SO->normdir); if (cSV->lit_for * SO->normdir < 0) { cSV->light0_position[0] *= -1; cSV->light0_position[1] *= -1; cSV->light0_position[2] *= -1; cSV->lit_for *= -1; glLightfv(GL_LIGHT0, GL_POSITION, cSV->light0_position); } } SUMA_RETURN(YUP); } /*! ans = SUMA_SetupSVforDOs (Spec, DOv, N_DOv, cSV, vo); This functions registers all surfaces in a spec file with a surface viewer. The following steps are performed: SUMA_RegisterSpecSO (register info on all surfaces loaded) SUMA_RegisterDO (only Surface Objects) SUMA_RegisterDO (all non SO objects) SUMA_BestStandardView (decide on best standard view) SUMA_UpdateRotaCenter (based on surfaces in first view) if vo & UPDATE_ROT_MASK SUMA_UpdateViewPoint (based on surfaces in first view) if vo & UPDATE_VIEW_POINT_MASK SUMA_EyeAxisStandard (based on surfaces in first view) if vo & UPDATE_EYE_AXIS_STD_MASK Set the Current SO pointer to the first surface object if surface is SureFit, flip lights \param Spec (SUMA_SurfSpecFile *) \param DOv (SUMA_DO *) Pointer to vector of displayable objects \param N_DOv (int) Number of displayable objects in DOv \param cSV (SUMA_SurfaceViewer *) Surface viewer structure \ret ans (SUMA_Boolean) YUP/NOPE */ SUMA_Boolean SUMA_SetupSVforDOs (SUMA_SurfSpecFile *Spec, SUMA_DO *DOv, int N_DOv, SUMA_SurfaceViewer *cSV, int viewopt) { static char FuncName[] = {"SUMA_SetupSVforDOs"}; int kar, ws, i; SUMA_SurfaceObject *SO; SUMA_Axis *EyeAxis; SUMA_DO_Types ttv[10]={SO_type, GRAPH_LINK_type, TRACT_type, MASK_type, VO_type, CDOM_type, NOT_SET_type}; int EyeAxis_ID, *MembDOs=NULL, N_MembDOs=0; int haveSpec=0; SUMA_Boolean LocalHead = NOPE; SUMA_ENTRY; if (!viewopt) { viewopt = UPDATE_ALL_VIEWING_PARAMS_MASK; } if (Spec && Spec->N_Groups > 0) haveSpec=1; /* register all surface specs */ /* find out what group the viewer will have and assign the current group to be that of the new surfaces */ if (LocalHead) { fprintf (SUMA_STDERR, "%s: Registering DOs with viewer [%d]%p, %d\n", FuncName, SUMA_WhichSV(cSV, SUMAg_SVv, SUMA_MAX_SURF_VIEWERS), cSV, SUMAg_N_SVv); if (haveSpec) { SUMA_LHv("HaveSpec, %d surfs, %d DOs, Group %s\n", Spec->N_Surfs, Spec->N_DO, Spec->Group[0]); } } cSV->iCurGroup = SUMA_WhichGroup(SUMAg_CF, haveSpec ? Spec->Group[0]:NULL); /* only one group per spec */ if (cSV->iCurGroup < 0) { SUMA_SL_Err("Group not found.\n"); SUMA_RETURN(NOPE); } else { cSV->CurGroupName = SUMA_copy_string(SUMAg_CF->GroupList[cSV->iCurGroup]); } if (haveSpec && !SUMA_RegisterSpecSO(Spec, cSV, DOv, N_DOv, viewopt)) { SUMA_S_Err("Failed in SUMA_RegisterSpecSO."); SUMA_RETURN(NOPE); } /* register all SOs of the first state if no state is current or register all surfaces if they are of the current state and group (it is possible that no new surfaces are registered, but overhead is tiny) The logic fails when you load two different groups with the one surface in each having the same state as in the other. {G1, SO->State='a'} {G2, SO->State='a'} in that case both surfaces will show up at the same time when first loaded. You'll need to switch groups before you see just the one surface from that group. For now, I don't care to set this up properly since no one uses multi-group business. */ if (!SUMA_IS_GOOD_STATE(cSV->State)) { SUMA_LHv("Looking for something better than %s\n", CNS(cSV->State)); ws = SUMA_FirstGoodState(cSV); } else { ws = SUMA_WhichState (cSV->State, cSV, cSV->CurGroupName) ; } if ( ws < 0) { /* This happens for example when running @DO.examples */ SUMA_LH("No good state at this stage? Defaulting to ws = 0;"); ws = 0; } /* Make sure we have something */ do { SUMA_LHv("Seeking members of state %s\n", CNS(cSV->VSv[ws].Name) ); MembDOs = SUMA_ViewState_Membs(&(cSV->VSv[ws]), ttv, &N_MembDOs); if (!N_MembDOs) ++ws; } while (!N_MembDOs && ws < cSV->N_VSv); if (!N_MembDOs) { SUMA_S_Err("Nothing to see (check ttv for necessary types)"); SUMA_RETURN(NOPE); } SUMA_LHv("Registering All DOs of the %dth state %s...\n" " N_MembDOs = %d\n", ws, cSV->VSv[ws].Name, N_MembDOs); cSV->State = cSV->VSv[ws].Name; cSV->iState = ws; for (kar=0; kar < N_MembDOs; ++ kar) { SUMA_LHv(" About to register DOv[%d] ...\n", MembDOs[kar]); switch (DOv[MembDOs[kar]].ObjectType) { case SO_type: SO = (SUMA_SurfaceObject *)DOv[MembDOs[kar]].OP; SUMA_LHv("SO->Group %s, cSV->CurGroupName %s\n", SO->Group, cSV->CurGroupName); if (!SUMA_RegisterDO(MembDOs[kar], cSV)) { SUMA_error_message (FuncName,"Failed to register DO", 1); SUMA_RETURN(NOPE); } break; case GRAPH_LINK_type: SUMA_LH( "Registration was done after loading in SUMA_LoadDsetOntoSO_eng\n" "No further action needed here"); break; case TRACT_type: SUMA_LH( "Registration was done after loading in SUMA_LoadSegDO\n" "No further action needed here"); break; case MASK_type: case CDOM_type: case VO_type: SUMA_LH( "Registration was done after loading in object\n" "No further action needed here"); break; default: SUMA_S_Err("Ney, no good"); break; } } /* Set the index Current SO pointer to the first surface object read of the first state, tiz NOT (Fri Jan 31 15:18:49 EST 2003) a surface of course*/ cSV->Focus_DO_ID = MembDOs[0]; SUMA_ifree(MembDOs); if (LocalHead) fprintf(SUMA_STDERR,"%s: Done.\n", FuncName); /* register all non SO objects */ if (LocalHead) fprintf(SUMA_STDERR,"%s: Registering All Non SO ...", FuncName); for (kar=0; kar < N_DOv; ++kar) { if (!SUMA_isSO(DOv[kar])) { /*fprintf(SUMA_STDERR," About to register DOv[%d] ...\n", kar);*/ if (!SUMA_RegisterDO(kar, cSV)) { SUMA_error_message (FuncName,"Failed to register DO", 1); SUMA_RETURN(NOPE); } } } if (LocalHead) fprintf(SUMA_STDERR,"%s: Done.\n", FuncName); /* decide what the best state is */ if (viewopt & UPDATE_STANDARD_VIEW_MASK) { cSV->StdView = SUMA_BestStandardView (cSV, DOv, N_DOv); if (LocalHead) fprintf(SUMA_STDERR, "%s: Standard View Now %d\n", FuncName, cSV->StdView); if (cSV->StdView == SUMA_N_STANDARD_VIEWS) { fprintf(SUMA_STDERR, "Error %s: Could not determine the best standard view. " "Choosing default SUMA_3D\n", FuncName); cSV->StdView = SUMA_3D; } } if (viewopt & UPDATE_ROT_MASK) { /* Set the Rotation Center */ if (LocalHead) fprintf(SUMA_STDERR,"%s: Setting the Rotation Center \n", FuncName); if (!SUMA_UpdateRotaCenter(cSV, DOv, N_DOv)) { fprintf (SUMA_STDERR, "Error %s: Failed to update center of rotation\n", FuncName); SUMA_RETURN(NOPE); } } if (viewopt & UPDATE_VIEW_POINT_MASK) { /* set the viewing points */ SUMA_LH("setting the viewing points"); if (!SUMA_UpdateViewPoint(cSV, DOv, N_DOv, 0)) { SUMA_S_Err("Failed to update view point"); SUMA_RETURN(NOPE); } } if (viewopt & UPDATE_EYE_AXIS_STD_MASK) { /* Change the defaults of the eye axis to fit standard EyeAxis */ SUMA_LH("Changing defaults of the eye axis to fit standard EyeAxis"); EyeAxis_ID = SUMA_GetEyeAxis (cSV, DOv); if (EyeAxis_ID < 0) { fprintf (SUMA_STDERR,"Error %s: Failed to get Eye Axis.\n", FuncName); SUMA_RETURN(NOPE); } SUMA_EyeAxisStandard ((SUMA_Axis *)DOv[EyeAxis_ID].OP, cSV); } /*set the GroupName info of the viewer correctly */ if ((SO = SUMA_SV_Focus_SO(cSV))) { if (!SUMA_AdoptSurfGroup(cSV,SO)) { SUMA_SL_Err("Failed to adopt surface's group"); SUMA_RETURN(NOPE); } if (!SUMA_SetViewerLightsForSO(cSV, SO)) { SUMA_S_Warn("Failed to set viewer lights.\n" "Use 'F' key to flip lights in SUMA if necessary."); } } /* do the axis setup */ SUMA_WorldAxisStandard (cSV->WAx, cSV); /* do the FOV thingy */ if (cSV->FOV[cSV->iState] == cSV->FOV_original || cSV->FOV[cSV->iState] < 0) { cSV->FOV[cSV->iState] = SUMA_sv_auto_fov(cSV); } SUMA_RETURN(YUP); } /*! \brief updates the cursor in all viewers */ void SUMA_UpdateAllViewerCursor() { static char FuncName[]={"SUMA_UpdateAllViewerCursor"}; int i; SUMA_ENTRY; for (i=0; iX) SUMA_RETURNe; if (!sv->X->GLXAREA) SUMA_RETURNe; if (SUMAg_CF->ROI_mode) { if (SUMAg_CF->Pen_mode) MCW_set_widget_cursor( sv->X->GLXAREA , -XC_pencil ) ; else MCW_set_widget_cursor( sv->X->GLXAREA , -XC_target ) ; } else { if (0) { MCW_set_widget_cursor( sv->X->GLXAREA , -XC_dotbox); } else { MCW_set_widget_cursor( sv->X->GLXAREA , -XC_top_left_arrow ) ; } } SUMA_RETURNe; } /*! \brief updates the title string of a viewer window */ void SUMA_UpdateViewerTitle_old(SUMA_SurfaceViewer *sv) { static char FuncName[]={"SUMA_UpdateViewerTitle_old"}; int isv, i, N_SOlist, nalloc; char slabel[30], sside[30], srec[10], cl='\0', cr='\0', smoment[30]; SUMA_SurfaceObject *SO = NULL; int SOlist[SUMA_MAX_DISPLAYABLE_OBJECTS]; SUMA_Boolean LeftSide, RightSide, RightShown, LeftShown; SUMA_Boolean LocalHead = NOPE; SUMA_ENTRY; if (!sv->X) SUMA_RETURNe; if (!sv->X->TOPLEVEL) SUMA_RETURNe; isv = SUMA_WhichSV (sv, SUMAg_SVv, SUMAg_N_SVv); if (sv->X->Title) SUMA_free(sv->X->Title); sv->X->Title = NULL; if (isv >= 0) sprintf(slabel,"[%c] SUMA", 65+isv); else sprintf(slabel,"[DOH] SUMA"); N_SOlist = SUMA_RegisteredSOs(sv, SUMAg_DOv, SOlist); i = 0; nalloc = 0; LeftSide = NOPE; LeftShown = NOPE; RightSide = NOPE; RightShown = NOPE; while (i < N_SOlist) { SO = (SUMA_SurfaceObject *)(SUMAg_DOv[SOlist[i]].OP); if (SO->Label) { nalloc += (strlen(SO->Label)+5); } if (SO->Side == SUMA_LEFT) { SUMA_LH("Left found"); LeftSide = YUP; if (sv->ShowLeft) LeftShown = YUP; } else if (SO->Side == SUMA_RIGHT) { SUMA_LH("Right found"); RightSide = YUP; if (sv->ShowRight) RightShown = YUP; } ++i; } if (LeftSide && LeftShown) cl = 'L'; else if (LeftSide && !LeftShown) cl = 'h'; else cl = 'x'; if (RightSide && RightShown) cr = 'R'; else if (RightSide && !RightShown) cr = 'h'; else cr = 'x'; sprintf(sside, ":%c%c:", cl, cr); if (sv->Record==1) sprintf(srec,":Rec"); else if (sv->Record==2) sprintf(srec,":REC"); else srec[0] = '\0'; if (sv->GVS[sv->StdView].ApplyMomentum) sprintf(smoment,":M"); else smoment[0] = '\0'; if (LocalHead) fprintf (SUMA_STDERR, "%s: Found %d surface models.\n", FuncName, N_SOlist); i = 0; if (N_SOlist >= 0) { SUMA_LH("title surfaces found"); sv->X->Title = (char *)SUMA_calloc(nalloc + strlen(slabel)+ 130, sizeof(char)); sv->X->Title[0] = '\0'; while (i < N_SOlist) { SO = (SUMA_SurfaceObject *)(SUMAg_DOv[SOlist[i]].OP); SUMA_LHv("sv->Focus_DO_ID = %d, SOlist[%d] = %d\n", sv->Focus_DO_ID, i, SOlist[i]); if (!i) { if (sv->Focus_DO_ID == SOlist[i]) { sprintf (sv->X->Title,"%s%s%s%s [%s]", slabel, srec, smoment, sside, SO->Label); } else { sprintf (sv->X->Title,"%s%s%s%s %s", slabel, srec, smoment, sside, SO->Label); } } else { sv->X->Title = strcat (sv->X->Title, " & "); if (sv->Focus_DO_ID == SOlist[i]) { sv->X->Title = strcat (sv->X->Title, " ["); sv->X->Title = strcat (sv->X->Title, SO->Label); sv->X->Title = strcat (sv->X->Title, "] "); } else { sv->X->Title = strcat (sv->X->Title, SO->Label); } } ++i; } } else { SUMA_LH("No title could be made up"); sv->X->Title = (char *)SUMA_calloc(strlen(slabel)+3, sizeof(char)); sprintf (sv->X->Title,"%s:-", slabel); } XtVaSetValues(sv->X->TOPLEVEL, XmNtitle, sv->X->Title, NULL); SUMA_RETURNe; } /*! \brief updates the title string of a viewer window */ void SUMA_UpdateViewerTitle(SUMA_SurfaceViewer *sv) { static char FuncName[]={"SUMA_UpdateViewerTitle"}; int isv, i, N_SOlist; char cl='\0', cr='\0', *s=NULL; SUMA_SurfaceObject *SO = NULL; int SOlist[SUMA_MAX_DISPLAYABLE_OBJECTS]; SUMA_STRING *SS = NULL; SUMA_Boolean LeftSide, RightSide, RightShown, LeftShown; SUMA_Boolean LocalHead = NOPE; SUMA_ENTRY; if (!sv->X) SUMA_RETURNe; if (!sv->X->TOPLEVEL) SUMA_RETURNe; SUMA_LH("Finding SV"); isv = SUMA_WhichSV (sv, SUMAg_SVv, SUMAg_N_SVv); if (sv->X->Title) SUMA_free(sv->X->Title); sv->X->Title = NULL; SS = SUMA_StringAppend_va(NULL, NULL); SUMA_LH("Number"); if (isv >= 0) { if ((i = get_user_np_bloc())>-1) { SS = SUMA_StringAppend_va(SS, "[%c%d] SUMA", 65+isv, i); } else { SS = SUMA_StringAppend_va(SS, "[%c] SUMA", 65+isv); } } else SS = SUMA_StringAppend_va(SS,"[DOH] SUMA"); SUMA_LH("Rec"); if (sv->Record == 1) SS = SUMA_StringAppend_va(SS,":Rec"); else if (sv->Record == 2) SS = SUMA_StringAppend_va(SS,":REC"); SUMA_LH("Momentum"); if (sv->GVS[sv->StdView].ApplyMomentum) SS = SUMA_StringAppend_va(SS,":M"); SUMA_LH("ClippingPlane"); // if (sv->GVS[sv->StdView].ClippingPlane) SS = SUMA_StringAppend_va(SS,":C"); if (sv->GVS[sv->StdView].ClippingPlane){ char strTmp[32]; sprintf(strTmp, ":C:Inc=%.4f", sv->clippingPlaneIncrement); SS = SUMA_StringAppend_va(SS,strTmp); } SUMA_LH("Surf List"); N_SOlist = SUMA_RegisteredSOs(sv, SUMAg_DOv, SOlist); i = 0; LeftSide = NOPE; LeftShown = NOPE; RightSide = NOPE; RightShown = NOPE; while (i < N_SOlist) { SUMA_LH(" + +"); SO = (SUMA_SurfaceObject *)(SUMAg_DOv[SOlist[i]].OP); if (SO->Side == SUMA_LEFT) { SUMA_LH("Left found"); LeftSide = YUP; if (sv->ShowLeft) LeftShown = YUP; } else if (SO->Side == SUMA_RIGHT) { SUMA_LH("Right found"); RightSide = YUP; if (sv->ShowRight) RightShown = YUP; } ++i; } if (LeftSide && LeftShown) cl = 'L'; else if (LeftSide && !LeftShown) cl = 'h'; else cl = 'x'; if (RightSide && RightShown) cr = 'R'; else if (RightSide && !RightShown) cr = 'h'; else cr = 'x'; SUMA_LH("Sides"); SS = SUMA_StringAppend_va(SS, ":%c%c:", cl, cr); SUMA_LHv("Found %d surface models.\n", N_SOlist); /* add the group's name */ if (LocalHead) { if (sv->CurGroupName) { SUMA_S_Notev("Calling with sv->CurGroupName = %p\n", sv->CurGroupName); } else { SUMA_S_Note("Calling with NULL sv->CurGroupName\n"); } } if (sv->CurGroupName) SS = SUMA_StringAppend_va(SS," %s:", sv->CurGroupName); else SS = SUMA_StringAppend_va(SS," xx:"); i = 0; if (N_SOlist >= 0) { SUMA_LH("title surfaces found"); while (i < N_SOlist) { SO = (SUMA_SurfaceObject *)(SUMAg_DOv[SOlist[i]].OP); SUMA_LHv("sv->Focus_DO_ID = %d, SOlist[%d] = %d\n", sv->Focus_DO_ID, i, SOlist[i]); if (!i) { if (sv->Focus_DO_ID == SOlist[i]) { SS = SUMA_StringAppend_va(SS," [%s]", SO->Label); } else { SS = SUMA_StringAppend_va(SS," %s", SO->Label); } } else { SS = SUMA_StringAppend_va(SS," & "); if (sv->Focus_DO_ID == SOlist[i]) { SS = SUMA_StringAppend_va(SS, " ["); SS = SUMA_StringAppend_va(SS, "%s", SO->Label); SS = SUMA_StringAppend_va(SS, "] "); } else { SS = SUMA_StringAppend_va(SS, "%s", SO->Label); } } ++i; } } else { SUMA_LH("No title could be made up"); SS = SUMA_StringAppend_va(SS,":-"); } /* compact SS */ SS = SUMA_StringAppend_va(SS, NULL); sv->X->Title = SS->s; SUMA_free(SS); SS= NULL; XtVaSetValues(sv->X->TOPLEVEL, XmNtitle, sv->X->Title, NULL); SUMA_RETURNe; } /*! \brief finds the index into the grouplist of a certain group */ int SUMA_WhichGroup (SUMA_CommonFields *cf, char *nm) { static char FuncName[]={"SUMA_WhichGroup"}; int i = -1; SUMA_ENTRY; if (!cf) { SUMA_SL_Err("Null nm"); SUMA_RETURN(i); } if (!nm) nm = "ANY"; if (cf->N_Group <=0) { SUMA_RETURN(i); } for (i=0; iN_Group; ++i) { if (!strcmp(cf->GroupList[i], nm)) SUMA_RETURN(i); } SUMA_RETURN(-1); } SUMA_Boolean SUMA_RegisterSpecGroup (SUMA_CommonFields *cf, SUMA_SurfSpecFile *spec) { static char FuncName[]={"SUMA_RegisterSpecGroup"}; SUMA_ENTRY; if (!spec || spec->N_Groups != 1) { SUMA_SL_Err("!Spec || Spec->N_Groups != 1. This is unacceptable.\n"); SUMA_RETURN(NOPE); } SUMA_RETURN(SUMA_RegisterGroup(cf, spec->Group[0])); } /*! \brief Register a new group with SUMA */ SUMA_Boolean SUMA_RegisterGroup (SUMA_CommonFields *cf, char *gname) { static char FuncName[]={"SUMA_RegisterGroup"}; int n=0; SUMA_Boolean LocalHead = NOPE; SUMA_ENTRY; if (!gname || !cf) { SUMA_S_Err("NULL gname or !cf") SUMA_RETURN(NOPE); } if (!cf->GroupList){ cf->GroupList = (char **) SUMA_malloc(sizeof(char*)*SUMA_MAX_N_GROUPS); for (n=0; nGroupList[n]=NULL; cf->N_Group = 0; } /* does the group exist already ? */ if (SUMA_WhichGroup (cf, gname) < 0) { /* new group */ SUMA_LH("Adding group"); if (cf->N_Group >= SUMA_MAX_N_GROUPS) { SUMA_SL_Err("Exceeding maximum number of groups allowed.\n"); SUMA_RETURN(NOPE); } cf->GroupList[cf->N_Group] = SUMA_copy_string(gname); ++cf->N_Group; } else{ /* an old group */ SUMA_LH("Group exists already"); } SUMA_RETURN(YUP); } /*! \brief Returns a list of the Groups available to a viewer. \param sv (SUMA_SurfaceViewer *) pointer to surface viewer \return clist (SUMA_ASSEMBLE_LIST_STRUCT *) pointer to structure containing results \sa SUMA_FreeAssembleListStruct \sa SUMA_CreateAssembleListStruct */ SUMA_ASSEMBLE_LIST_STRUCT * SUMA_AssembleGroupList (SUMA_SurfaceViewer *sv) { static char FuncName[]={"SUMA_AssembleGroupList"}; SUMA_ASSEMBLE_LIST_STRUCT *clist_str = NULL; int i=-1, N_clist=-1; char *store=NULL; char **clist=NULL; void **oplist=NULL; DList *list=NULL, *listop = NULL; DListElmt *Elm = NULL, *Elmop = NULL; SUMA_Boolean Found = NOPE; SUMA_Boolean LocalHead = NOPE; SUMA_ENTRY; list = (DList *)SUMA_malloc(sizeof(DList)); listop = (DList *)SUMA_malloc(sizeof(DList)); clist = NULL; N_clist = -1; dlist_init(list, NULL); dlist_init(listop, NULL); for (i=0; i< SUMAg_CF->N_Group; ++i) { store = SUMA_copy_string(SUMAg_CF->GroupList[i]); if (!list->size) { dlist_ins_next(list, dlist_tail(list), (void*)store); dlist_ins_next(listop, dlist_tail(listop), NULL); } else { /* must sort first */ Elm = NULL; Elmop = NULL; do { Found = NOPE; if (!Elm) { Elm = dlist_head(list); Elmop = dlist_head(listop); } else { Elm = dlist_next(Elm); Elmop = dlist_next(Elmop); } if (strcmp(store, (char*)Elm->data) <= 0) { dlist_ins_prev(list, Elm, (void *)store); dlist_ins_prev(listop, Elmop, NULL); Found = YUP; } else if (Elm == dlist_tail(list)) { /* reached the end, append */ dlist_ins_next(list, Elm, (void *)store); dlist_ins_next(listop, Elmop, NULL); Found = YUP; } } while (!Found); } } if (!list->size) { /* Nothing found */ N_clist = 0; }else { Elm = NULL; Elmop = NULL; clist = (char **)SUMA_calloc(list->size, sizeof(char *)); oplist = (void **)SUMA_calloc(list->size, sizeof(void*)); for (i=0; i< list->size; ++i) { if (!Elm) { Elm = dlist_head(list); Elmop = dlist_head(listop); } else { Elm = dlist_next(Elm); Elmop = dlist_next(Elmop); } clist[i] = (char*)Elm->data; oplist[i] = Elmop->data; } N_clist = list->size; /* destroy list */ dlist_destroy(list);SUMA_free(list); dlist_destroy(listop);SUMA_free(listop); } clist_str = SUMA_CreateAssembleListStruct(); clist_str->clist = clist; clist_str->oplist = oplist; clist_str->N_clist = N_clist; /* return */ SUMA_RETURN (clist_str); } /*! \brief Switch viewer between two groups */ SUMA_Boolean SUMA_SwitchGroups (SUMA_SurfaceViewer *sv, char *group) { static char FuncName[]={"SUMA_SwitchGroups"}; int ig, i, nxtstateID; SUMA_SurfaceObject *SO = NULL; DList *list = NULL; SUMA_Boolean LocalHead = NOPE; SUMA_ENTRY; if (!group) { SUMA_SL_Err("NULL group"); SUMA_RETURN(NOPE); } if (!strcmp(group, sv->CurGroupName)) { SUMA_LH("Same group, nothing to do."); SUMA_RETURN(YUP); } if (SUMAg_CF->N_Group == 1) { SUMA_LH("One group, nothing to do."); SUMA_RETURN(YUP); } /* which group are we going to ? */ ig = SUMA_WhichGroup (SUMAg_CF, group); if (ig < 0) { SUMA_SL_Err("No such group"); SUMA_RETURN(NOPE); } /* It does not seem necessary to close surface controllers or ROI controllers*/ /* find me a surface in that new group */ SO = NULL; i = 0; while (!SUMA_isSO_G(SUMAg_DOv[i], group) && i < SUMAg_N_DOv) { ++i; } if (i < SUMAg_N_DOv) { /* found a surface */ SO = (SUMA_SurfaceObject *)SUMAg_DOv[i].OP; } else { SUMA_SL_Err("No candidate surface"); SUMA_RETURN(NOPE); } /* what is the state ID of that surface ? */ nxtstateID = SUMA_WhichState(SO->State, sv, SO->Group); if (nxtstateID < 0) { SUMA_SL_Err("Bad! State not found."); SUMA_RETURN(NOPE); } if (!SUMA_SwitchState (SUMAg_DOv, SUMAg_N_DOv, sv, nxtstateID, group)) { SUMA_SL_Err("Failed to switch states"); SUMA_RETURN(NOPE); } /* home call */ #if 0 /* now redisplay (won't work alone with multiple viewers, GL state problems) */ if (!list) list = SUMA_CreateList(); /* SUMA_REGISTER_HEAD_COMMAND_NO_DATA(list, SE_Home, SES_Suma, sv); */ SUMA_REGISTER_HEAD_COMMAND_NO_DATA(list, SE_Redisplay, SES_Suma, sv); if (!SUMA_Engine (&list)) { fprintf(stderr, "Error %s: SUMA_Engine call failed.\n", FuncName); } #elif 0 /* redisplay all others (won't work alone with multiple viewers, GL state problems) */ if (!list) list = SUMA_CreateList (); SUMA_REGISTER_TAIL_COMMAND_NO_DATA(list, SE_RedisplayNow_AllOtherVisible, SES_SumaWidget, sv); SUMA_Engine (&list); /* redisplay . DO NOT REDISPLAY WITH SE_Redisplay_AllVisible or you will have GL state synchronization problems */ sv->ResetGLStateVariables = YUP; SUMA_handleRedisplay((XtPointer)sv->X->GLXAREA); #elif 1 /* got to do this, in addition to SE_Redisplay_AllVisible to get the views to look good. I don't know why that is yet */ for (i=0; i < SUMAg_N_SVv; ++i) { SUMA_SurfaceViewer *svtmp= &(SUMAg_SVv[i]); if (!svtmp->isShaded && svtmp->X->TOPLEVEL) { if (!list) list = SUMA_CreateList(); SUMA_REGISTER_HEAD_COMMAND_NO_DATA(list, SE_Home, SES_Suma, svtmp); SUMA_REGISTER_HEAD_COMMAND_NO_DATA(list, SE_RedisplayNow, SES_Suma, svtmp); if (!SUMA_Engine (&list)) { SUMA_S_Err("SUMA_Engine call failed."); } } } if (!list) list = SUMA_CreateList(); SUMA_REGISTER_HEAD_COMMAND_NO_DATA(list, SE_Redisplay_AllVisible, SES_Suma, sv); if (!SUMA_Engine (&list)) { SUMA_S_Err("SUMA_Engine call failed."); } #endif /* update titles */ SUMA_UpdateViewerTitle(sv); SUMA_RETURN(YUP); } /* Return a viewer's surface in focus, if there is a surface and it is in focus */ SUMA_SurfaceObject *SUMA_SV_Focus_SO(SUMA_SurfaceViewer *sv) { static char FuncName[]={"SUMA_SV_Focus_SO"}; if (!sv || sv->Focus_DO_ID < 0) return(NULL); if (SUMAg_DOv[sv->Focus_DO_ID].ObjectType == SO_type) { return((SUMA_SurfaceObject *)(SUMAg_DOv[sv->Focus_DO_ID].OP)); } return(NULL); } SUMA_SurfaceObject *SUMA_SV_Focus_any_SO(SUMA_SurfaceViewer *sv, int *dov_id) { static char FuncName[]={"SUMA_SV_Focus_any_SO"}; SUMA_SurfaceObject *SO=NULL; int ii; if (dov_id) *dov_id = -1; if (sv && sv->Focus_DO_ID >= 0 && SUMAg_DOv[sv->Focus_DO_ID].ObjectType == SO_type) { if (dov_id) *dov_id = sv->Focus_DO_ID; return((SUMA_SurfaceObject *)(SUMAg_DOv[sv->Focus_DO_ID].OP)); } /* give me anything */ SO = SUMA_findanySOp_inDOv(SUMAg_DOv, SUMAg_N_DOv, dov_id); return(SO); } SUMA_ALL_DO *SUMA_SV_Focus_ADO(SUMA_SurfaceViewer *sv) { static char FuncName[]={"SUMA_SV_Focus_ADO"}; if (!sv || sv->Focus_DO_ID < 0) return(NULL); return((SUMA_ALL_DO *)(SUMAg_DOv[sv->Focus_DO_ID].OP)); return(NULL); } SUMA_ALL_DO *SUMA_findanyFocusable_ADO(int *dov_id) { static char FuncName[]={"SUMA_findanyFocusable_ADO"}; SUMA_ALL_DO *ado=NULL; int ii; void *pp; if (dov_id) *dov_id = -1; if ((ado = (SUMA_ALL_DO *)SUMA_findanySOp_inDOv( SUMAg_DOv, SUMAg_N_DOv, dov_id))) return(ado); if ((ado = (SUMA_ALL_DO *)SUMA_FindDset_eng("NA", SUMAg_CF->DsetList, NULL, "isGraphDset"))) return(ado); if ((ado = (SUMA_ALL_DO *)SUMA_findanyVOp_inDOv( SUMAg_DOv, SUMAg_N_DOv, dov_id))) return(ado); if ((ado = (SUMA_ALL_DO *)SUMA_findanyTDOp_inDOv( SUMAg_DOv, SUMAg_N_DOv, dov_id))) return(ado); return(NULL); } SUMA_ALL_DO *SUMA_SV_Focus_any_ADO(SUMA_SurfaceViewer *sv, int *dov_id) { static char FuncName[]={"SUMA_SV_Focus_any_ADO"}; SUMA_ALL_DO *ado=NULL; int ii; if (dov_id) *dov_id = -1; if (sv && sv->Focus_DO_ID >= 0) { if (dov_id) *dov_id = sv->Focus_DO_ID; return((SUMA_ALL_DO *)(SUMAg_DOv[sv->Focus_DO_ID].OP)); } /* give me anything */ ado = SUMA_findanyFocusable_ADO(dov_id); return(ado); } SUMA_ALL_DO *SUMA_SV_any_ADO_WithSurfContWidget(SUMA_SurfaceViewer *sv, int *dov_id, SUMA_DO_Types thisdotp) { static char FuncName[]={"SUMA_SV_any_ADO_WithSurfContWidget"}; SUMA_ALL_DO *ado=NULL; SUMA_X_SurfCont *SurfCont=NULL; int ii; if (dov_id) *dov_id = -1; if (sv && sv->Focus_DO_ID >= 0 && (SurfCont = SUMA_ADO_Cont( (SUMA_ALL_DO *)(SUMAg_DOv[sv->Focus_DO_ID].OP))) && SurfCont->TLS) { if (dov_id) *dov_id = sv->Focus_DO_ID; return((SUMA_ALL_DO *)(SUMAg_DOv[sv->Focus_DO_ID].OP)); } /* give me anything */ ado = SUMA_findany_ADO_WithSurfContWidget(dov_id, thisdotp); return(ado); } SUMA_ALL_DO *SUMA_findany_ADO_WithSurfContWidget(int *dov_id, SUMA_DO_Types thisdotp) { static char FuncName[]={"SUMA_findany_ADO_WithSurfContWidget"}; SUMA_ALL_DO *ado=NULL; SUMA_DO_Types ttv[N_DO_TYPES] = { SO_type, GRAPH_LINK_type, VO_type, TRACT_type, CDOM_type, NOT_SET_type }; int ii, tt; void *pp; SUMA_X_SurfCont *SurfCont=NULL; if (dov_id) *dov_id = -1; if (thisdotp != NOT_SET_type) { ttv[0] = thisdotp; ttv[1] = NOT_SET_type; } tt = 0; while (ttv[tt] != NOT_SET_type) { for (ii=0; iiTLS) { if (dov_id) *dov_id = ii; return(ado); } } } ++tt; } return(NULL); } SUMA_ALL_DO *SUMA_SurfCont_GetcurDOp(SUMA_X_SurfCont *SurfCont) { static char FuncName[]={"SUMA_SurfCont_GetcurDOp"}; SUMA_Boolean LocalHead = NOPE; if (SurfCont && SurfCont->prv_curDOp) { SUMA_ALL_DO *ado = (SUMA_ALL_DO *)*(SurfCont->prv_curDOp); if (!ado) return(NULL); /* This will happen if controller not yet open */ SUMA_LHv("ADO Type name %s, variant %s\n", ADO_TNAME(ado), SurfCont->prv_variant); switch (ado->do_type) { case GRAPH_LINK_type: SUMA_S_Err("This should not be, See SUMA_SurfCont_SetcurDOp"); return(NULL); break; case GDSET_type: return((SUMA_ALL_DO *)SUMA_find_Dset_GLDO((SUMA_DSET *)ado, SurfCont->prv_variant,NULL)); break; default: return(ado); break; } } return(NULL); } SUMA_Boolean SUMA_SurfCont_SetcurDOp(SUMA_X_SurfCont *SurfCont, SUMA_ALL_DO *ado) { static char FuncName[]={"SUMA_SurfCont_SetcurDOp"}; SUMA_Boolean LocalHead = NOPE; if (!(SurfCont && SurfCont->prv_curDOp && ado)) return(NOPE); switch(ado->do_type) { case SO_type: *(SurfCont->prv_curDOp) = (void *)ado; return(YUP); break; case GRAPH_LINK_type: /* Store the GDSET, a pointer of some permanence, NOT gldo which has no right to property*/ *(SurfCont->prv_curDOp) = (void *)SUMA_find_GLDO_Dset((SUMA_GraphLinkDO*)ado); SurfCont->prv_variant = SUMA_replace_string(SurfCont->prv_variant, SUMA_ADO_variant(ado)); SUMA_LHv("Now have %p, %s\n", *(SurfCont->prv_curDOp), SurfCont->prv_variant); return(YUP); break; case GDSET_type: SUMA_S_Err( "You should not set the current DOp to a DO that has\n" "ambiguous rendering\n"); return(NOPE); break; SUMA_S_Err("Have to figure out this machinery"); return(NOPE); break; case CDOM_type: case TRACT_type: *(SurfCont->prv_curDOp) = (void *)ado; return(YUP); break; case MASK_type: case VO_type: *(SurfCont->prv_curDOp) = (void *)ado; return(YUP); break; default: SUMA_S_Warnv("Not ready for the likes of %s\n", SUMA_ObjectTypeCode2ObjectTypeName(ado->do_type)); return(NOPE); break; } return(NOPE); } SUMA_Boolean SUMA_SetMouseMode(SUMA_SurfaceViewer *sv, SUMA_MOUSE_MODES mmode, void *val) { static char FuncName[]={"SUMA_SetMouseMode"}; int ival; SUMA_ALL_DO *ado; SUMA_Boolean LocalHead = NOPE; SUMA_ENTRY; if (!sv) SUMA_RETURN(NOPE); SUMA_LH("Should this function handle cross hair modification?"); switch (mmode) { case SUMA_ROI_MMODE: ival = (int)(long int)val; /* Double cast to quiet compiler warning */ if (ival) { if (SUMAg_CF->ROI_mode==YUP) { /* nothing to do */ SUMA_RETURN(YUP); } else { /* Need to turn on ROI_mode, and kill other modes */ SUMAg_CF->ROI_mode = YUP; // [PT: Feb 22, 2021] fix '=' sv->MouseMode = SUMA_DEF_MMODE; SUMA_RETURN(YUP); } } else { if (SUMAg_CF->ROI_mode==NOPE) { /* nothing to do */ SUMA_RETURN(YUP); } else { /* Need to turn off ROI_mode */ SUMAg_CF->ROI_mode = NOPE; // [PT: Feb 22, 2021] fix '=' SUMA_RETURN(YUP); } } break; case SUMA_MASK_MANIP_MMODE: if (val) { if (SUMAg_CF->ROI_mode == YUP) SUMAg_CF->ROI_mode = NOPE; sv->MouseMode = SUMA_MASK_MANIP_MMODE; if ((ado=SUMA_whichADOg(sv->MouseMode_ado_idcode_str)) && ado->do_type == MASK_type) { SUMA_MDO_New_parent((SUMA_MaskDO*)ado, val, -1); } SUMA_ifree(sv->MouseMode_ado_idcode_str); sv->MouseMode_ado_idcode_str = SUMA_copy_string((char*)val); } else if (sv->MouseMode == SUMA_MASK_MANIP_MMODE) { sv->MouseMode = SUMA_DEF_MMODE; SUMA_ifree(sv->MouseMode_ado_idcode_str); if ((ado=SUMA_whichADOg(sv->MouseMode_ado_idcode_str)) && ado->do_type == MASK_type) { SUMA_MDO_New_parent((SUMA_MaskDO*)ado, NULL, -1); } } SUMA_RETURN(YUP); break; case SUMA_DEF_MMODE: sv->MouseMode = SUMA_DEF_MMODE; SUMA_RETURN(YUP); break; default: SUMA_S_Err("Not ready for %d", mmode); break; } SUMA_RETURN(NOPE); } char *SUMA_ADO_ContName(SUMA_ALL_DO *ado) { if (!ado) return("NULL"); switch(ado->do_type) { case MASK_type: return ("Mask"); case GRAPH_LINK_type: return ("Graph"); case TRACT_type: return ("Network"); case VO_type: return ("Volume"); case SO_type: return ("Surface"); default: return ("Not set"); } return("Should not get here"); }