-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathFractControllerContrast.j
198 lines (164 loc) · 8.62 KB
/
FractControllerContrast.j
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
/*
This file is part of FrACT10, a vision test battery.
Copyright © 2021 Michael Bach, bach@uni-freiburg.de, <https://michaelbach.de>
FractControllerContrast.j
Created by Bach on 2020-09-02
*/
@import "Dithering.j"
@implementation FractControllerContrast: FractController {
float stimStrengthInDeviceunitsUnquantised;
}
- (CPString) composeExportString {return [self contrastComposeExportString];}
- (void) modifyThresholderStimulus {
if (iTrial == 1) // make the first more visible
stimStrengthInThresholderUnits = Math.min(stimStrengthInThresholderUnits + 0.3, 1);
if ([Settings contrastEasyTrials]) // don't forget bonus
[self modifyThresholderStimulusWithBonus];
}
- (void) calculateForeBackColors {
let gray1 = [MiscLight lowerLuminanceFromContrastLogCSWeber: stimStrengthInDeviceunits];
gray1 = [MiscLight devicegrayFromLuminance: gray1];
let gray2 = [MiscLight upperLuminanceFromContrastLogCSWeber: stimStrengthInDeviceunits];
gray2 = [MiscLight devicegrayFromLuminance: gray2];
if (![Settings contrastDarkOnLight]) {
[gray1, gray2] = [gray2, gray1]; // "modern" swapping of variables
}
gColorFore = [MiscLight colorFromGreyBitStealed: gray1];//console.info(gColorFore);
gColorBack = [MiscLight colorFromGreyBitStealed: gray2];//console.info(gColorBack);
colorForeUndithered = gColorFore; colorBackUndithered = gColorBack;
stimStrengthInDeviceunitsUnquantised = stimStrengthInDeviceunits;
if ([Settings contrastDithering]) {
gColorFore = [CPColor colorWithPatternImage: [Dithering image3x3withGray: gray1]];
gColorBack = [CPColor colorWithPatternImage: [Dithering image3x3withGray: gray2]];
}
}
// Strategy: draw fixmark, delay (onTimeoutFixMark) and draw optotype, erasing fixmark.
// With eccentricity: Need to draw again right after optotype so it seamlessly stays put
// No marked eccentricity: don't draw it again
- (void) drawFixMark3 { // check marked eccentricity is set, otherwise don't draw it
const eccRadiusInPix = Math.sqrt(xEccInPix * xEccInPix + yEccInPix * yEccInPix);
if ((strokeSizeInPix * 4) > eccRadiusInPix) return; // we don't want overlap between fixmark and optotype
[self drawFixMark2];
}
- (void) drawFixMark2 {
CGContextSaveGState(cgc);
CGContextSetStrokeColor(cgc, [CPColor colorWithRed: 0 green: 0 blue: 1 alpha: 0.7]);
CGContextSetLineWidth(cgc, 0.5);
CGContextTranslateCTM(cgc, +xEccInPix, +yEccInPix); // counter eccentricity
[optotypes strokeStarAtX: 0 y: 0 size: strokeSizeInPix * 3];
CGContextRestoreGState(cgc);
}
- (void) drawFixMark {
let t = [Settings contrastTimeoutFixmark] / 1000; // ms → seconds
if ([Settings contrastShowFixMark] && (currentTestID != kTestContrastDitherUnittest)) {
[self drawFixMark2];
timerFixMark = [CPTimer scheduledTimerWithTimeInterval: t target:self selector:@selector(onTimeoutFixMark:) userInfo:nil repeats:NO];
} else {
t = 0.02;
}
timerFixMark = [CPTimer scheduledTimerWithTimeInterval: t target:self selector:@selector(onTimeoutFixMark:) userInfo:nil repeats:NO];
}
- (void) onTimeoutFixMark: (CPTimer) timer { //console.info("FractController>onTimeoutFixCross");
state = kStateDrawFore2;
[[[self window] contentView] setNeedsDisplay: YES];
}
- (void) runStart { //console.info("FractControllerContrast>runStart");
[super runStart];
[self setCurrentTestResultUnit: "logCSWeber"];
}
// this manages stuff after the optotypes have been drawn
- (void) drawStimulusInRect: (CGRect) dirtyRect { //console.info("FractControllerContrast>drawStimulusInRect");
if ([Settings contrastDithering]) {
stimStrengthInDeviceunits = stimStrengthInDeviceunitsUnquantised;
}
[trialHistoryController setValue: stimStrengthInDeviceunits];
[super drawStimulusInRect: dirtyRect];
}
- (void) runEnd { //console.info("FractControllerContrast>runEnd");
if (iTrial < nTrials) { //premature end
[self setResultString: @"Aborted"];
} else {
stimStrengthInDeviceunits = Math.min(stimStrengthInDeviceunits, gMaxResultLogCSWeber);
[self setResultString: [self contrastComposeResultString]];
}
[super runEnd];
}
///////////////////////// CONTRAST UTILs
/*
basic flow:
Thresholder → thresholderunits
thresholderunits → deviceunits (logCSWeber)
deviceunits → upper/lower luminance, considering gamma → discrete values for [CPColor colorWithWhite] (0…255)
present stimulus
stimStrengthInDeviceunits → thresholderunits (currently ignoring gamma because "locally everything is linear")
thresholderunits + correctness → Thresholder
// contrast: 0.1 … 100, thresholder: 0 … 1
// deviceUnits are in logCSWeber for all contrast tests; for gratings that is converted to Michelson%
// logCSW: 2 … 0, thresholder: 0 … 1 */
- (float) getCurrentContrastMichelsonPercent {
return [MiscLight contrastMichelsonPercentFromColor1: colorForeUndithered color2: colorBackUndithered];
}
/*- (float) getCurrentContrastWeberPercent {
return [MiscLight contrastWeberPercentFromMichelsonPercent: [self getCurrentContrastMichelsonPercent]];
}*/
// Problem here: 0% weber contrast corresponds to infinite logCSWeber.
// But since the latter is clamped at gMaxAllowedLogCSWeber, after rounding this will still read 0%. Solved.
- (float) getCurrentContrastLogCSWeber {
const michelsonPercent = [self getCurrentContrastMichelsonPercent];
const weberPercent = [MiscLight contrastWeberPercentFromMichelsonPercent: michelsonPercent];
return [MiscLight contrastLogCSWeberFromWeberPercent: weberPercent];
}
- (float) stimDeviceunitsFromThresholderunits: (float) thresholderunit { //console.info("FractControllerContrast>stimDeviceunitsFromThresholderunits");
const logCSWMaximal = [Settings contrastMaxLogCSWeber];
const deviceVal = logCSWMaximal - logCSWMaximal * thresholderunit; // logCSWMinimal = 0 anyway
return deviceVal;
}
- (float) stimThresholderunitsFromDeviceunits: (float) d { //console.info("FractControllerContrast>stimThresholderunitsFromDeviceunits");
//console.info("d: ", d, ", retVal: ", retVal)
const logCSWMaximal = [Settings contrastMaxLogCSWeber];
const retVal = (logCSWMaximal - d) / logCSWMaximal
return retVal;
}
- (void) unittestContrastDeviceThresholdConversion {
for (let i = 0; i <= 1.0; i += 0.1) {
const d = [self contrastStimDeviceunitsFromThresholderunits: i];
console.info("threshh: ", i, ", devUnits: ", d, ", threshh: ", [self stimThresholderunitsFromDeviceunits: d]);
}
}
- (CPString) contrastComposeTrialInfoString {
let s = "trial: " + iTrial + "/" + nTrials;
s += ", contrast: " + [Misc stringFromNumber: [MiscLight contrastWeberPercentFromLogCSWeber: stimStrengthInDeviceunits] decimals: 2 localised: YES] + "%";
s += ", logCSW: " + [Misc stringFromNumber: stimStrengthInDeviceunits decimals: 2 localised: YES];
s += ", alternative: " + [alternativesGenerator currentAlternative];
return s;
}
- (CPString) contrastComposeResultString { //console.info("contrastComposeResultString");
// console.info("rangeLimitStatus: ", rangeLimitStatus);
rangeLimitStatus = kRangeLimitOk;
if (stimStrengthInDeviceunits >= gMaxResultLogCSWeber) { // todo: do this while testing
rangeLimitStatus = kRangeLimitValueAtCeiling;
}
let s = "Contrast threshold: " + crlf;
s += [self rangeStatusIndicatorStringInverted: YES];
s += [Misc stringFromNumber: stimStrengthInDeviceunits decimals: 2 localised: YES];
s += " logCS(Weber) ≙ ";
s += [self rangeStatusIndicatorStringInverted: NO];
s += [Misc stringFromNumber: [MiscLight contrastWeberPercentFromLogCSWeber: stimStrengthInDeviceunits] decimals: 2 localised: YES];
s += "%";
return s;
}
- (CPString) contrastComposeExportString { //console.info("FractController>contrastComposeExportString");
if ([[self parentController] runAborted]) return "";
let s = [self generalComposeExportString];
const nDigits = 3;
s += tab + "value" + tab + [Misc stringFromNumber: stimStrengthInDeviceunits decimals: nDigits localised: YES];
s += tab + "unit1" + tab + currentTestResultUnit
s += tab + "distanceInCm" + tab + [Misc stringFromNumber: [Settings distanceInCM] decimals: 2 localised: YES];
s += tab + "diameter" + tab + [Misc stringFromNumber: [Settings contrastOptotypeDiameter] decimals: 2 localised: YES];
s += tab + "unit2" + tab + "arcmin";
s += tab + "nTrials" + tab + [Misc stringFromNumber: nTrials decimals: 0 localised: YES];
s += tab + "rangeLimitStatus" + tab + rangeLimitStatus;
s += tab + "crowding" + tab + 0; // does not apply, but let's not NaN this
return [self generalComposeExportStringFinalize: s];
}
@end