forked from jakehilborn/displayplacer
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathdisplayplacer.c
379 lines (313 loc) · 15.1 KB
/
displayplacer.c
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
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
// displayplacer.c
// Created by Jake Hilborn on 5/16/15.
#include <IOKit/graphics/IOGraphicsLib.h>
#include <ApplicationServices/ApplicationServices.h>
#include <unistd.h>
#include <math.h>
#include "header.h"
int main(int argc, char * argv[]) {
if(argc == 1 || strcmp(argv[1], "--help") == 0) {
printHelp();
return 0;
}
if(strcmp(argv[1], "--version") == 0) {
printVersion();
return 0;
}
if (strcmp(argv[1], "list") == 0) {
listScreens();
return 0;
}
ScreenConfig* screenConfigs = malloc((argc - 1) * sizeof(ScreenConfig));
for (int i = 0; i < argc - 1; i++) {
screenConfigs[i].modeNum = -1; //set modeNum -1 in case user wants to set and use mode 0
char* propGroup = argv[i + 1];
char* propSetSavePtr = NULL;
char* propSetToken = strtok_r(propGroup, " \t", &propSetSavePtr);
while (propSetToken) {
char* propSavePtr = NULL;
char* propToken = strtok_r(propSetToken, ":", &propSavePtr);
switch (propToken[0]) {
case 'i': //id
propToken = strtok_r(NULL, ":", &propSavePtr);
char* idToken = strtok_r(propToken, "+", &propToken);
screenConfigs[i].id = atoi(idToken);
int j = 0;
while ((idToken = strtok_r(propToken, "+", &propToken))) {
screenConfigs[i].mirrors[j] = atoi(idToken);
j++;
if (j > 127) {
fprintf(stderr, "Current code only supports 128 screens mirroring. Please execute `displayplacer --version` for info on contacting the developer to change this.\n");
}
}
screenConfigs[i].mirrorCount = j;
break;
case 'r': //res
propToken = strtok_r(NULL, ":", &propSavePtr);
char* resSavePtr = NULL;
char* resToken = strtok_r(propToken, "x", &resSavePtr);
screenConfigs[i].width = atoi(resToken);
resToken = strtok_r(NULL, "x", &resSavePtr);
screenConfigs[i].height = atoi(resToken);
resToken = strtok_r(NULL, "x", &resSavePtr);
if (resToken) {
screenConfigs[i].hz = atoi(resToken);
} else {
screenConfigs[i].hz = 0;
}
break;
case 's': //scaling
propToken = strtok_r(NULL, ":", &propSavePtr);
if (strcmp(propToken, "on") == 0) {
screenConfigs[i].scaled = true;
} else {
screenConfigs[i].scaled = false;
}
break;
case 'o': //origin
propToken = strtok_r(NULL, ":", &propSavePtr);
char* originSavePtr = NULL;
char* originToken = strtok_r(propToken, ",", &originSavePtr);
screenConfigs[i].x = atoi(originToken + 1); //skip the '(' character
originToken = strtok_r(NULL, ",", &originSavePtr);
screenConfigs[i].y = atoi(originToken);
break;
case 'm': //mode
propToken = strtok_r(NULL, ":", &propSavePtr);
screenConfigs[i].modeNum = atoi(propToken);
break;
case 'd': //rotation degree
propToken = strtok_r(NULL, ":", &propSavePtr);
screenConfigs[i].degree = atoi(propToken);
break;
default:
fprintf(stderr, "Argument parsing error\n");
exit(1);
}
propSetToken = strtok_r(NULL, " \t", &propSetSavePtr);
}
}
CGDisplayCount screenCount;
CGGetOnlineDisplayList(INT_MAX, NULL, &screenCount); //get number of online screens and store in screenCount
CGDirectDisplayID screenList[screenCount];
CGGetOnlineDisplayList(INT_MAX, screenList, &screenCount); //store display list in array of size screenCount
CGDisplayConfigRef configRef;
CGBeginDisplayConfiguration(&configRef);
bool isSuccess = true; //returns non-zero exit code on any errors but allows for completing remaining program execution
for (int i = 0; i < argc - 1; i++) {
if (!validateScreenOnline(screenList, screenCount, screenConfigs[i].id)) {
isSuccess = false;
continue;
}
if (CGDisplayRotation(screenConfigs[i].id) != screenConfigs[i].degree) {
isSuccess = rotateScreen(screenConfigs[i].id, screenConfigs[i].degree) && isSuccess;
}
for (int j = 0; j < screenConfigs[i].mirrorCount; j++) {
if (!validateScreenOnline(screenList, screenCount, screenConfigs[i].mirrors[j])) {
isSuccess = false;
continue;
}
if (CGDisplayRotation(screenConfigs[i].mirrors[j]) != screenConfigs[i].degree) {
isSuccess = rotateScreen(screenConfigs[i].mirrors[j], screenConfigs[i].degree) && isSuccess;
}
isSuccess = configureMirror(configRef, screenConfigs[i].id, screenConfigs[i].mirrors[j]) && isSuccess;
}
isSuccess = configureResolution(configRef, screenConfigs[i].id, screenConfigs[i].width, screenConfigs[i].height, screenConfigs[i].scaled, screenConfigs[i].hz, screenConfigs[i].modeNum) && isSuccess;
isSuccess = configureOrigin(configRef, screenConfigs[i].id, screenConfigs[i].x, screenConfigs[i].y) && isSuccess;
}
int retVal = CGCompleteDisplayConfiguration(configRef, kCGConfigurePermanently);
if (retVal != 0) {
fprintf(stderr, "Error finalizing display configurations\n");
isSuccess = false;
}
free(screenConfigs);
if (isSuccess) {
return 0;
} else {
return 1;
}
}
void printHelp() {
printf(
"Usage:\n"
" Show current screen info and possible resolutions: displayplacer list\n"
"\n"
" Screen config: displayplacer 'id:<screenId> res:<width>x<height>x<hz> scaling:<on/off> origin:(<x>,<y>) degree:<0/90/180/270>'\n"
"\n"
" Screen config using mode: displayplacer 'id:<screenId> mode:<modeNum> origin:(<x>,<y>) degree:<0/90/180/270>'\n"
"\n"
" Set layout with a mirrored screen: displayplacer 'id:<mainScreenId>+<mirrorScreenId>+<mirrorScreenId> res:<width>x<height>x<hz> scaling:<on/off> origin:(<x>,<y>) degree:<0/90/180/270>'\n"
"\n"
" Example w/ all features: displayplacer 'id:69731906+862792382 res:1440x900 scaling:on origin:(0,0) degree:0' 'id:374164677 res:768x1360x60 scaling:off origin:(1440,0) degree:90' 'id:173529877 mode:3 origin:(-1440,0) degree:270'\n"
"\n"
"Instructions:\n"
" 1. Manually set rotations 1st*, resolutions 2nd, and arrangement 3rd. For extra resolutions and rotations read 'Notes' below.\n"
" - Open System Preferences -> Displays\n"
" - Choose desired screen rotations (use displayplacer for rotating internal MacBook screen).\n"
" - Choose desired resolutions (use displayplacer for extra resolutions).\n"
" - Drag the white bar to your desired primary screen.\n"
" - Arrange screens as desired and/or enable mirroring.\n"
" - To enable partial mirroring hold the option key and drag a display on top of another.\n"
" 2. Use `displayplacer list` to get the info about your current layout so you can create profiles for scripting/hotkeys.\n"
"\n"
"Notes:\n"
" - *`displayplacer list` and system prefs only show resolutions for the screen's current rotation.\n"
" - ScreenIDs change when cables are plugged into different ports. To ensure screenIDs match your saved profiles, always plug cables into the same ports.\n"
" - Use an extra resolution shown in `displayplacer list` by executing `displayplacer 'id:<screenId> mode:<modeNum>'`\n"
" - Rotate your internal MacBook screen by executing `displayplacer 'id:<screenId> degree:<0/90/180/270>'`\n"
" - The screen set to origin (0,0) will be set as the primary screen (white bar in system prefs).\n"
" - The first screenId in a mirroring set will be the 'Optimize for' screen in the system prefs. You can only choose resolutions for the 'Optimize for' screen. If there is a mirroring resolution you need but cannot find, try making a different screenId to first of the set.\n"
);
}
void printVersion() {
printf(
"displayplacer v1.1.0-dev\n"
"\n"
"Developer: Jake Hilborn\n"
"GitHub: https://github.com/jakehilborn/displayplacer\n"
"LinkedIn: https://www.linkedin.com/in/jakehilborn\n"
"Email: jakehilborn@gmail\n"
);
}
void listScreens() {
CGDisplayCount screenCount;
CGGetOnlineDisplayList(INT_MAX, NULL, &screenCount); //get number of online screens and store in screenCount
CGDirectDisplayID screenList[screenCount];
CGGetOnlineDisplayList(INT_MAX, screenList, &screenCount);
for (int i = 0; i < screenCount; i++) {
UInt32 curScreen = screenList[i];
printf("Screen ID: %i\n", curScreen);
if (CGDisplayIsBuiltin(curScreen)) {
printf("Type: MacBook built in screen\n");
} else {
CGSize size = CGDisplayScreenSize(curScreen);
int diagonal = round(sqrt((size.width * size.width) + (size.height * size.height)) / 25.4); //25.4mm in an inch
printf("Type: %i inch external screen\n", diagonal);
}
printf("Resolution: %lux%lu\n", CGDisplayPixelsWide(curScreen), CGDisplayPixelsHigh(curScreen));
printf("Rotation: %i", (int) CGDisplayRotation(curScreen));
if (CGDisplayIsBuiltin(curScreen)) {
printf(" - rotate internal screen example: `displayplacer 'id:%i degree:90'`", curScreen);
}
printf("\n");
printf("Origin: (%i,%i)", (int) CGDisplayBounds(curScreen).origin.x, (int) CGDisplayBounds(curScreen).origin.y);
if (CGDisplayIsMain(curScreen)) {
printf(" - main display");
}
printf("\n");
int modeCount;
modes_D4* modes;
CopyAllDisplayModes(curScreen, &modes, &modeCount);
for(int i = 0; i < modeCount; i++) {
modes_D4 mode = modes[i];
if(mode.derived.density == 2.0) { //scaling on
if(mode.derived.freq) { //if screen supports different framerates
printf("mode %i: Res=%dx%dx%i, scaled\n", i, mode.derived.width, mode.derived.height, mode.derived.freq);
} else {
printf("mode %i: Res=%dx%d, scaled\n", i, mode.derived.width, mode.derived.height);
}
}
else { //scaling off
if(mode.derived.freq) { //if screen supports different framerates
printf("mode %i: Res=%dx%dx%i\n", i, mode.derived.width, mode.derived.height, mode.derived.freq);
} else {
printf("mode %i: Res=%dx%d\n", i, mode.derived.width, mode.derived.height);
}
}
}
printf("\n");
}
}
bool validateScreenOnline(CGDirectDisplayID onlineDisplayList[], int screenCount, CGDirectDisplayID screenId) {
for (int i = 0; i < screenCount; i++) {
if (onlineDisplayList[i] == screenId) {
return true;
}
}
fprintf(stderr, "Unable to find screen %i - skipping changes for that screen\n", screenId);
return false;
}
bool rotateScreen(CGDirectDisplayID screenId, int degree) {
io_service_t service = CGDisplayIOServicePort(screenId);
IOOptionBits options;
switch(degree) {
default:
options = (0x00000400 | (kIOScaleRotate0) << 16);
break;
case 90:
options = (0x00000400 | (kIOScaleRotate90) << 16);
break;
case 180:
options = (0x00000400 | (kIOScaleRotate180) << 16);
break;
case 270:
options = (0x00000400 | (kIOScaleRotate270) << 16);
break;
}
int retVal = IOServiceRequestProbe(service, options);
if (retVal != 0) {
fprintf(stderr, "Error rotating screen %i\n", screenId);
return false;
}
return true;
}
bool configureMirror(CGDisplayConfigRef configRef, CGDirectDisplayID primaryScreenId, CGDirectDisplayID mirrorScreenId) {
int retVal = CGConfigureDisplayMirrorOfDisplay(configRef, mirrorScreenId, primaryScreenId);
if (retVal != 0) {
fprintf(stderr, "Error making the secondary screen %i mirror the primary screen %i\n", mirrorScreenId, primaryScreenId);
return false;
}
return true;
}
bool configureResolution(CGDisplayConfigRef configRef, CGDirectDisplayID screenId, int width, int height, bool scaled, int hz, int modeNum) {
int modeCount;
modes_D4* modes;
if (modeNum != -1) { //user specified modeNum instead of height/width/hz
CGSConfigureDisplayMode(configRef, screenId, modeNum);
return true;
}
CopyAllDisplayModes(screenId, &modes, &modeCount);
//loop through all modes looking for one that matches user input resolution
for (int i = 0; i < modeCount; i++) {
modes_D4 mode = modes[i];
if (mode.derived.width != width) {
continue;
}
if (mode.derived.height != height) {
continue;
}
if (scaled && mode.derived.density != 2.0) {
continue;
}
if (hz && mode.derived.freq != hz) {
continue;
}
//matching resolution found
CGSConfigureDisplayMode(configRef, screenId, i);
return true;
}
//no matching resolution found
if(scaled) {
if(hz) { //if screen supports different framerates
fprintf(stderr, "Screen ID %i: could not find Res=%ix%ix%i, scaled\n", screenId, width, height, hz);
} else {
fprintf(stderr, "Screen ID %i: could not find Res=%ix%i, scaled\n", screenId, width, height);
}
}
else { //scaling off
if(hz) { //if screen supports different framerates
fprintf(stderr, "Screen ID %i: could not find Res=%ix%ix%i\n", screenId, width, height, hz);
} else {
fprintf(stderr, "Screen ID %i: could not find Res=%ix%i\n", screenId, width, height);
}
}
return false;
}
bool configureOrigin(CGDisplayConfigRef configRef, CGDirectDisplayID screenId, int x, int y) {
int retVal = CGConfigureDisplayOrigin(configRef, screenId, x, y);
if (retVal != 0) {
fprintf(stderr, "Error moving screen %i to %ix%i\n", screenId, x, y);
return false;
}
return true;
}