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 #include "Component/ModelManager.H"
00040 #include "Image/ColorOps.H"
00041 #include "Image/CutPaste.H"
00042 #include "Image/Image.H"
00043 #include "Image/MathOps.H"
00044 #include "Psycho/PsychoDisplay.H"
00045 #include "Psycho/EyeTrackerConfigurator.H"
00046 #include "Psycho/EyeTracker.H"
00047 #include "Psycho/PsychoOpts.H"
00048 #include "Component/EventLog.H"
00049 #include "Component/ComponentOpts.H"
00050 #include "Raster/Raster.H"
00051 #include "Util/MathFunctions.H"
00052
00053 #include <ctype.h>
00054 #include <vector>
00055
00056
00057 #define NMASK 10
00058
00059
00060 extern "C" int main(const int argc, char** argv)
00061 {
00062 MYLOGVERB = LOG_INFO;
00063
00064
00065 ModelManager manager("Psycho Search");
00066
00067
00068 nub::soft_ref<PsychoDisplay> d(new PsychoDisplay(manager));
00069 manager.addSubComponent(d);
00070
00071 nub::soft_ref<EyeTrackerConfigurator>
00072 etc(new EyeTrackerConfigurator(manager));
00073 manager.addSubComponent(etc);
00074
00075 nub::soft_ref<EventLog> el(new EventLog(manager));
00076 manager.addSubComponent(el);
00077
00078 manager.setOptionValString(&OPT_EventLogFileName, "psychodata.psy");
00079 manager.setOptionValString(&OPT_EyeTrackerType, "ISCAN");
00080
00081
00082 if (manager.parseCommandLine(argc, argv, "<imagelist.txt>", 1, 1) == false)
00083 return(1);
00084
00085
00086 nub::soft_ref<EyeTracker> et = etc->getET();
00087 d->setEyeTracker(et);
00088 d->setEventLog(el);
00089 et->setEventLog(el);
00090
00091
00092 manager.start();
00093
00094
00095 SDL_Surface *mask[NMASK]; int mindex[NMASK]; Dims ddims = d->getDims();
00096
00097 for (int i = 0; i < NMASK; i ++) {
00098 Image<byte> r(ddims, ZEROS), g(ddims, ZEROS), b(ddims, ZEROS);
00099 inplaceSpeckleNoise(r, 1.0F, i+1, 255, true);
00100 inplaceSpeckleNoise(g, 1.0F, i+1, 255, true);
00101 inplaceSpeckleNoise(b, 1.0F, i+1, 255, true);
00102 mask[i] = d->makeBlittableSurface(makeRGB(r, g, b), false);
00103
00104
00105 mindex[i] = i;
00106 }
00107
00108
00109 FILE *f = fopen(manager.getExtraArg(0).c_str(), "r");
00110 if (f == NULL) LFATAL("Cannot read stimulus file");
00111 char line[1024];
00112 std::vector<std::string> ilist, tlist;
00113 while(fgets(line, 1024, f))
00114 {
00115
00116
00117
00118 char *line2 = line; while(*line2 != '\0' && !isspace(*line2)) line2 ++;
00119 *line2++ = '\0'; line2[strlen(line2) - 1] = '\0';
00120
00121 tlist.push_back(std::string(line));
00122 ilist.push_back(std::string(line2));
00123 }
00124 fclose(f);
00125
00126
00127 int nimg = ilist.size(); int imindex[nimg];
00128 for (int i = 0; i < nimg; i ++) imindex[i] = i;
00129 randShuffle(imindex, nimg);
00130
00131
00132 d->clearScreen();
00133 d->displayISCANcalib();
00134 d->waitForKey();
00135
00136
00137 d->displayText("<SPACE> to calibrate; other key to skip");
00138 int c = d->waitForKey();
00139 if (c == ' ') d->displayEyeTrackerCalibration(3, 3);
00140
00141
00142 d->clearScreen();
00143 d->displayText("<SPACE> to start experiment");
00144 d->waitForKey();
00145
00146
00147 for (int im = 0; im < nimg; im ++) {
00148 int imnum = imindex[im];
00149
00150
00151 d->clearScreen();
00152 LINFO("Loading '%s' / '%s'...", ilist[imnum].c_str(),tlist[imnum].c_str());
00153
00154
00155 Image< PixRGB<byte> > img = Raster::ReadRGB(tlist[imnum]);
00156 Image< PixRGB<byte> > rndimg(d->getDims(), NO_INIT);
00157 rndimg.clear(d->getGrey());
00158 int rndx = 40 + randomUpToNotIncluding(d->getDims().w() - img.getWidth() - 80);
00159 int rndy = 30 + randomUpToNotIncluding(d->getDims().h() - img.getHeight() - 60);
00160 inplacePaste(rndimg, img, Point2D<int>(rndx, rndy));
00161 SDL_Surface *surf1 = d->makeBlittableSurface(rndimg, false);
00162 char buf[256];
00163 sprintf(buf, "===== Showing imagelet: %s at (%d, %d) =====",
00164 tlist[imnum].c_str(), rndx, rndy);
00165
00166
00167 int fixx = rndx + img.getWidth() / 2;
00168 int fixy = rndy + img.getHeight() / 2;
00169
00170
00171 img = Raster::ReadRGB(ilist[imnum]);
00172 SDL_Surface *surf2 = d->makeBlittableSurface(img, false);
00173
00174
00175 randShuffle(mindex, NMASK);
00176
00177
00178 usleep(200000);
00179
00180
00181 d->displayText("Ready");
00182 d->waitForKey();
00183 d->waitNextRequestedVsync(false, true);
00184 d->pushEvent(buf);
00185
00186
00187 d->displaySurface(surf1, 0, true);
00188
00189
00190 for (int i = 0; i < 80; i ++) d->waitNextRequestedVsync();
00191
00192
00193 d->pushEvent("===== Showing mask =====");
00194 for (int j = 0; j < NMASK; j ++)
00195 d->displaySurface(mask[mindex[j]], j+1, true);
00196
00197
00198 d->clearScreen();
00199 d->displayFixation(fixx, fixy);
00200 for (int i = 0; i < 40; i ++) d->waitNextRequestedVsync();
00201
00202
00203 et->track(true);
00204
00205
00206 for (int i = 0; i < 20; i ++) d->waitNextRequestedVsync();
00207
00208
00209 d->pushEvent(std::string("===== Showing search image: ") + ilist[imnum] +
00210 std::string(" ====="));
00211 d->displaySurface(surf2, 0, true);
00212
00213
00214 d->waitForKey();
00215
00216
00217 SDL_FreeSurface(surf1); SDL_FreeSurface(surf2);
00218
00219
00220 usleep(150000);
00221 et->track(false);
00222
00223
00224 if (im > 0 && im % 20 == 0) {
00225 d->displayText("Ready for quick recalibration");
00226 d->waitForKey();
00227 d->displayEyeTrackerCalibration(3, 3);
00228 d->clearScreen();
00229 d->displayText("Ready to continue with the images");
00230 d->waitForKey();
00231 }
00232 }
00233
00234 d->clearScreen();
00235 d->displayText("Experiment complete. Thank you!");
00236 d->waitForKey();
00237
00238
00239 manager.stop();
00240
00241
00242 for (int i = 0; i < NMASK; i++) SDL_FreeSurface(mask[i]);
00243
00244
00245 return 0;
00246 }
00247
00248
00249
00250
00251
00252