00001
00002
00003
00004
00005
00006
00007
00008
00009
00010
00011
00012
00013
00014
00015
00016
00017
00018
00019
00020
00021
00022
00023
00024
00025
00026
00027
00028
00029
00030
00031
00032
00033
00034
00035
00036
00037
00038
00039
00040
00041
00042
00043
00044
00045
00046
00047
00048
00049
00050
00051
00052
00053
00054
00055
00056
00057
00058
00059
00060
00061
00062
00063
00064
00065
00066
00067
00068
00069
00070
00071
00072
00073
00074
00075
00076
00077
00078
00079
00080
00081
00082
00083
00084
00085
00086
00087
00088
00089
00090
00091
00092
00093
00094
00095
00096
00097 #include <gtk/gtk.h>
00098 #include "config.h"
00099 #include "gxsm/plugin.h"
00100
00101
00102 static void plane_max_prop_init( void );
00103 static void plane_max_prop_about( void );
00104 static void plane_max_prop_configure( void );
00105 static void plane_max_prop_cleanup( void );
00106
00107
00108 #define GXSM_ONE_SRC_PLUGIN__DEF
00109
00110
00111
00112 #ifdef GXSM_ONE_SRC_PLUGIN__DEF
00113
00114 static gboolean plane_max_prop_run( Scan *Src, Scan *Dest );
00115 #else
00116
00117 static gboolean plane_max_prop_run( Scan *Src1, Scan *Src2, Scan *Dest );
00118 #endif
00119
00120
00121 GxsmPlugin plane_max_prop_pi = {
00122 NULL,
00123 NULL,
00124 0,
00125 NULL,
00126
00127
00128
00129
00130 "plane_max_prop-"
00131 #ifdef GXSM_ONE_SRC_PLUGIN__DEF
00132 "M1S"
00133 #else
00134 "M2S"
00135 #endif
00136 "-BG",
00137
00138
00139
00140
00141
00142 NULL,
00143
00144 "Calculates a max propability plane and subtracts it.",
00145
00146 "Percy Zahl",
00147
00148 N_("_Math/_Background/"),
00149
00150 N_("Plane max prop"),
00151
00152 N_("Calculates a max propability plane and subtracts it."),
00153
00154 "Plane max propability",
00155 NULL,
00156 NULL,
00157
00158
00159 plane_max_prop_init,
00160
00161
00162 NULL,
00163
00164
00165 plane_max_prop_about,
00166
00167
00168 plane_max_prop_configure,
00169
00170
00171 NULL,
00172
00173
00174 plane_max_prop_cleanup
00175 };
00176
00177
00178
00179
00180 #ifdef GXSM_ONE_SRC_PLUGIN__DEF
00181 GxsmMathOneSrcPlugin plane_max_prop_m1s_pi
00182 #else
00183 GxsmMathTwoSrcPlugin plane_max_prop_m2s_pi
00184 #endif
00185 = {
00186
00187 plane_max_prop_run
00188 };
00189
00190
00191 static const char *about_text = N_("Gxsm plane_max_prop Plugin\n\n"
00192 "Calculates a max propability plane and subtracts it.");
00193
00194
00195
00196 GxsmPlugin *get_gxsm_plugin_info ( void ){
00197 plane_max_prop_pi.description = g_strdup_printf(N_("Gxsm MathOneArg plane_max_prop plugin %s"), VERSION);
00198 return &plane_max_prop_pi;
00199 }
00200
00201
00202
00203
00204 #ifdef GXSM_ONE_SRC_PLUGIN__DEF
00205 GxsmMathOneSrcPlugin *get_gxsm_math_one_src_plugin_info( void ) {
00206 return &plane_max_prop_m1s_pi;
00207 }
00208 #else
00209 GxsmMathTwoSrcPlugin *get_gxsm_math_two_src_plugin_info( void ) {
00210 return &plane_max_prop_m2s_pi;
00211 }
00212 #endif
00213
00214
00215
00216 static void plane_max_prop_init(void)
00217 {
00218 PI_DEBUG (DBG_L2, "plane_max_prop Plugin Init");
00219 }
00220
00221
00222 static void plane_max_prop_about(void)
00223 {
00224 const gchar *authors[] = { plane_max_prop_pi.authors, NULL};
00225 gtk_widget_show(gnome_about_new ( plane_max_prop_pi.name,
00226 VERSION,
00227 N_("(C) 2000 the Free Software Foundation"),
00228 about_text,
00229 authors,
00230 NULL, NULL, NULL
00231 ));
00232 }
00233
00234
00235 static void plane_max_prop_configure(void)
00236 {
00237 if(plane_max_prop_pi.app)
00238 plane_max_prop_pi.app->message("plane_max_prop Plugin Configuration");
00239 }
00240
00241
00242 static void plane_max_prop_cleanup(void)
00243 {
00244 PI_DEBUG (DBG_L2, "plane_max_prop Plugin Cleanup");
00245 }
00246
00247
00248
00249
00250
00251
00252 #define F1 6
00253 #define F2 12
00254 #define F3 5
00255 #define DFAC 2
00256 #define WEIGHTFAC 3
00257 #define EPS 1
00258
00259
00260
00261
00262 #define fbound(a,b,c) \
00263 { \
00264 double uttemp; \
00265 \
00266 if ((a) < (uttemp = (b)) || (a) > (uttemp = (c))) \
00267 (a) = uttemp; \
00268 }
00269
00270 #define DATA double
00271 #define LONGDATA double
00272 #define INT int
00273
00274 struct mode_out {double low, high, best; };
00275
00276 int random_number(int);
00277
00278 #define NRAND 97
00279
00280 #ifndef RAND_MAX
00281 # define RAND_MAX 32767
00282 #endif
00283
00284 int random_number(int x)
00285 {
00286 static gboolean initted = FALSE;
00287 static int v[NRAND];
00288 static unsigned vv;
00289
00290 if (x<0) x=0;
00291
00292 if(!initted) {
00293 int i;
00294 initted = TRUE;
00295 for(i=0;i<NRAND;i++)
00296 (void)rand();
00297 for(i=0;i<NRAND;i++)
00298 v[i] = rand();
00299 vv = rand();
00300 }
00301 {
00302 const unsigned y = vv%NRAND;
00303 vv = v[ y ];
00304 v[y] = rand();
00305 }
00306
00307 {
00308 const int out = ((long)vv*x)/((long)RAND_MAX+1);
00309
00310 return ((out>0)?out:0);
00311 }
00312 }
00313
00314
00315 gboolean mode(DATA *d, int n, int stop, struct mode_out *rv)
00316 {
00317 DATA qmin, qmax;
00318 DATA split1, split2, lastqmax, lastqmin;
00319 INT low, med, high, nmax, i, delta;
00320 double tmp;
00321
00322
00323 for(i=0,qmin=qmax=d[0];i<n;i++)
00324 if(d[i] > qmax) qmax = d[i];
00325 else if(d[i] < qmin) qmin = d[i];
00326 qmax ++;
00327 if((LONGDATA)qmax-qmin < 3*stop)
00328 qmax = qmin + 3*stop;
00329
00330 while(1) {
00331 lastqmax = qmax;
00332 lastqmin = qmin;
00333 split1 = qmin + (qmax-qmin+EPS)/3;
00334 split2 = qmax - (qmax-qmin+EPS)/3;
00335
00336 for(low=med=high=0,i=0;i<n;i++) {
00337 if(d[i]>=qmin && d[i]<split1)
00338 low++;
00339 else if(d[i]>=split1 && d[i]<split2)
00340 med++;
00341 else if(d[i]>=split2 && d[i]<qmax)
00342 high++;
00343 }
00344
00345
00346
00347 low = (INT)(((DATA)low * (qmax-qmin))/(3*(split1-qmin)));
00348 med = (INT)(((DATA)med * (qmax-qmin))/(3*(split2-split1)));
00349 high = (INT)(((DATA)high * (qmax-qmin))/(3*(qmax-split2)));
00350
00351
00352 nmax = low>med ? (low>high ? low : high) : (med>high ? med : high);
00353 delta = (INT)(DFAC * sqrt((double)nmax) + 0.5);
00354
00355 if(low+delta<nmax && med+delta<nmax) {
00356 qmin = split2 - ((LONGDATA)qmax-qmin+EPS)/F1;
00357 qmax = qmax + ((LONGDATA)qmax-qmin+EPS)/F3;
00358 }
00359 else if(med+delta<nmax && high+delta<nmax) {
00360 qmin = qmin - ((LONGDATA)qmax-qmin+EPS)/F3;
00361 qmax = split1 + ((LONGDATA)qmax-qmin+EPS)/F1;
00362 }
00363 else if(low+delta<nmax && high+delta<nmax) {
00364 qmin = split1 - ((LONGDATA)qmax-qmin+EPS)/F1;
00365 qmax = split2 + ((LONGDATA)qmax-qmin+EPS)/F1;
00366 }
00367 else if(low+delta<nmax) {
00368 qmin = split1 - ((LONGDATA)qmax-qmin+EPS)/F2;
00369 qmax = qmax + ((LONGDATA)qmax-qmin+EPS)/F2;
00370 }
00371 else if(high+delta<nmax) {
00372 qmin = qmin - ((LONGDATA)qmax-qmin+EPS)/F2;
00373 qmax = split2 + ((LONGDATA)qmax-qmin+EPS)/F2;
00374 }
00375 else break;
00376
00377 if((LONGDATA)qmax-qmin<3*stop && (LONGDATA)lastqmax-lastqmin>=4*stop) {
00378 if(lastqmax-qmax>qmin-lastqmin)
00379 qmax = qmin+3*stop;
00380 else if(qmin-lastqmin>lastqmax-qmax)
00381 qmin = qmax-3*stop;
00382 else if((LONGDATA)lastqmax-lastqmin>=qmax-qmin+3*stop)
00383 {qmax += stop; qmin -= stop;}
00384 else if(qmax-qmin >= 2*stop)
00385 qmax += stop;
00386 else {
00387 qmax = lastqmax; qmin = lastqmin;
00388 break;
00389 }
00390 }
00391 else if((LONGDATA)qmax-qmin < 3*stop) {
00392 qmax = lastqmax; qmin = lastqmin;
00393 break;
00394 }
00395 }
00396
00397 tmp = WEIGHTFAC * delta/(double)DFAC;
00398 split1 = qmin + (qmax-qmin+EPS)/3;
00399 split2 = qmax - (qmax-qmin+EPS)/3;
00400
00401 rv->best = (((LONGDATA)qmin+split1-EPS)*exp((low-nmax)/tmp) +
00402 ((LONGDATA)split1+split2-EPS)* exp((med-nmax)/tmp) +
00403 ((LONGDATA)split2+qmax-EPS)*exp((high-nmax)/tmp)) /
00404 (2.0*(exp((low-nmax)/tmp) +
00405 exp((med-nmax)/tmp) +
00406 exp((high-nmax)/tmp)));
00407 rv->low = ((LONGDATA)qmin+split1-EPS)/2.0;
00408 rv->high = ((LONGDATA)split2+qmax-EPS)/2.0;
00409
00410 return (TRUE);
00411 }
00412
00413
00414 #define SAMPLE 10000
00415
00416 gboolean deltamode(Scan *Src, int m, int n, int dm, int dn,
00417 struct mode_out *answer)
00418 {
00419 int l, sample;
00420 DATA *ds;
00421
00422 sample = (long)m*n<SAMPLE ? m*n : SAMPLE;
00423 ds = (DATA *)malloc(sizeof(*ds) * sample);
00424
00425 for(l=0;l<sample;l++) {
00426 int i, j;
00427 i = random_number(m-abs(dm))+(dm>0 ? 0 : -dm);
00428 j = random_number(n-abs(dn))+(dn>0 ? 0 : -dn);
00429 ds[l] = Src->mem2d->GetDataPkt (j+dn, i+dm) - Src->mem2d->GetDataPkt (j, i);
00430 }
00431 if ( (l > sample) || (l<1)) {
00432 return (FALSE);
00433 }
00434 mode(ds, l, 1, answer);
00435 free((void *)ds);
00436 return (TRUE);
00437 }
00438
00439
00440
00441 #define START 8
00442
00443
00444
00445
00446 static gboolean plane_max_prop_run(Scan *Src, Scan *Dest)
00447 {
00448 struct mode_out x1, xy1, y1, yx1;
00449 int i, j, delta, n, m;
00450 double xs, ys, xt, yt, xmi, xma, ymi, yma, xtu, ytu, xu, yu;
00451 double tosub;
00452
00453 m = Dest->mem2d->GetNy ();
00454 n = Dest->mem2d->GetNx ();
00455 xmi = xma = ymi = yma = xs = ys = 0.0;
00456 for(delta=START; delta<m/2 && delta<n/2; delta*=2) {
00457 if (!deltamode(Src, m, n, delta, 0, &x1))
00458 return(MATH_UNDEFINED);
00459 if (!deltamode(Src, m, n, 0, delta, &y1))
00460 return(MATH_UNDEFINED);
00461 if (!deltamode(Src, m, n, delta, delta, &xy1))
00462 return(MATH_UNDEFINED);
00463 if (!deltamode(Src, m, n, delta, -delta, &yx1))
00464 return(MATH_UNDEFINED);
00465
00466 xt = (xy1.best + yx1.best)/(2.0*delta);
00467 yt = (xy1.best - yx1.best)/(2.0*delta);
00468 xtu = (xy1.high+yx1.high-xy1.low-yx1.low)/2.0;
00469 ytu = (xy1.high-xy1.low-yx1.high+yx1.low)/2.0;
00470 xu = (x1.high-x1.low+xtu)/(2.0*delta);
00471 yu = (y1.high-y1.low+ytu)/(2.0*delta);
00472 x1.best /= (double)delta;
00473 y1.best /= (double)delta;
00474
00475 if(delta > START) {
00476 if(((xt>xma || xt<xmi) && (x1.best>xma || x1.best<xmi)) ||
00477 ((yt>yma || yt<ymi) && (y1.best>yma || y1.best<ymi)))
00478 break;
00479 fbound(xt, xmi, xma);
00480 fbound(yt, ymi, yma);
00481 fbound(x1.best, xmi, xma);
00482 fbound(y1.best, ymi, yma);
00483 xs = (x1.best + xt)/2.0;
00484 ys = (y1.best + yt)/2.0;
00485
00486 xmi = MAX(xmi, xs-0.60*xu);
00487 xma = MIN(xma, xs+0.60*xu);
00488 ymi = MAX(ymi, ys-0.60*yu);
00489 yma = MIN(yma, ys+0.60*yu);
00490 }
00491 else {
00492 xs = (x1.best + xt)/2.0;
00493 ys = (y1.best + yt)/2.0;
00494 xmi = xs - 0.60*xu;
00495 xma = xs + 0.60*xu;
00496 ymi = ys - 0.60*yu;
00497 yma = ys + 0.60*yu;
00498 }
00499 }
00500
00501 for(i=0; i<m; i++)
00502 for(j=0; j<n; j++)
00503 Dest->mem2d->PutDataPkt (Src->mem2d->GetDataPkt (j, i) - i*ys - j*xs, j, i);
00504
00505
00506
00507
00508 double max, min, off;
00509 Dest->mem2d->HiLo(&max, &min);
00510
00511 off=(max+min)/2.;
00512 for(i=0;i<m;i++)
00513 for(j=0;j<n;j++)
00514 Dest->mem2d->data->Zadd(-off, j, i);
00515
00516 return MATH_OK;
00517 }
00518