You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

159 lines
2.8 KiB

  1. #include<stdlib.h>
  2. #include<stdio.h>
  3. #include<string.h>
  4. #include<unistd.h>
  5. #include<signal.h>
  6. #include<X11/Xlib.h>
  7. #define LENGTH(X) (sizeof(X) / sizeof (X[0]))
  8. typedef struct {
  9. char* command;
  10. unsigned int interval;
  11. unsigned int signal;
  12. } Block;
  13. void sighandler(int num);
  14. void replace(char *str, char old, char new);
  15. void getcmds(int time);
  16. void getsigcmds(int signal);
  17. void setupsignals();
  18. void getstatus(char *str);
  19. void setroot();
  20. void statusloop();
  21. void statusinit();
  22. void sighandler(int signum);
  23. void termhandler(int signum);
  24. #include "blocks.h"
  25. static Display *dpy;
  26. static int screen;
  27. static Window root;
  28. static char statusbar[LENGTH(blocks)][50] = {0};
  29. static char statusstr[256];
  30. static char *statuscat;
  31. static const char *volupcmd[] = { "volup", NULL };
  32. static const char *voldowncmd[] = { "voldown", NULL };
  33. static const char *volmutecmd[] = { "volmute", NULL };
  34. static int statusContinue = 1,volmuted = 0;
  35. void replace(char *str, char old, char new)
  36. {
  37. int N = strlen(str);
  38. for(int i = 0; i < N; i++)
  39. if(str[i] == old)
  40. str[i] = new;
  41. }
  42. //opens process *cmd and stores output in *output
  43. void getcmd(char *cmd, char *output)
  44. {
  45. FILE *cmdf = popen(cmd,"r");
  46. if (!cmdf)
  47. return;
  48. int N = strlen(output);
  49. char c;
  50. int i = 0;
  51. while((c = fgetc(cmdf)) != EOF)
  52. output[i++] = c;
  53. output[i++] = '\0';
  54. pclose(cmdf);
  55. }
  56. void getcmds(int time)
  57. {
  58. const Block* current;
  59. for(int i = 0; i < LENGTH(blocks); i++)
  60. {
  61. current = blocks + i;
  62. if ((current->interval != 0 && time % current->interval == 0) || time == -1)
  63. getcmd(current->command,statusbar[i]);
  64. }
  65. }
  66. void getsigcmds(int signal)
  67. {
  68. const Block *current;
  69. for (int i = 0; i < LENGTH(blocks); i++)
  70. {
  71. current = blocks + i;
  72. if (current->signal == signal)
  73. getcmd(current->command,statusbar[i]);
  74. }
  75. }
  76. void setupsignals()
  77. {
  78. for(int i = 0; i < LENGTH(blocks); i++)
  79. {
  80. if (blocks[i].signal > 0)
  81. signal(SIGRTMIN+blocks[i].signal, sighandler);
  82. }
  83. }
  84. void getstatus(char *str)
  85. {
  86. int j = 0;
  87. for(int i = 0; i < LENGTH(blocks); j+=strlen(statusbar[i++]))
  88. {
  89. strcpy(str + j, statusbar[i]);
  90. }
  91. str[j] = '\0';
  92. }
  93. void setroot()
  94. {
  95. Display *d = XOpenDisplay(NULL);
  96. if (d) {
  97. dpy = d;
  98. }
  99. screen = DefaultScreen(dpy);
  100. root = RootWindow(dpy, screen);
  101. getstatus(statusstr);
  102. replace(statusstr,'\n',' ');//gets rid of newlines from output
  103. replace(statusstr,EOF,' ');//gets rid of EOF from output
  104. XStoreName(dpy, root, statusstr);
  105. XCloseDisplay(dpy);
  106. }
  107. void statusloop()
  108. {
  109. setupsignals();
  110. int i = 0;
  111. getcmds(-1);
  112. while(statusContinue)
  113. {
  114. getcmds(i);
  115. setroot();
  116. sleep(1.0);
  117. i++;
  118. }
  119. }
  120. void statusinit()
  121. {
  122. statusloop();
  123. }
  124. void sighandler(int signum)
  125. {
  126. getsigcmds(signum-SIGRTMIN);
  127. setroot();
  128. }
  129. void termhandler(int signum)
  130. {
  131. statusContinue = 0;
  132. exit(0);
  133. }
  134. int main()
  135. {
  136. signal(SIGTERM, termhandler);
  137. signal(SIGINT, termhandler);
  138. statusinit();
  139. }