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.

188 lines
3.5 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. #define CMDLENGTH 50
  9. typedef struct {
  10. char* icon;
  11. char* command;
  12. unsigned int interval;
  13. unsigned int signal;
  14. } Block;
  15. void dummysighandler(int num);
  16. void sighandler(int num);
  17. void replace(char *str, char old, char new);
  18. void getcmds(int time);
  19. #ifndef __OpenBSD__
  20. void getsigcmds(int signal);
  21. void setupsignals();
  22. void sighandler(int signum);
  23. #endif
  24. int getstatus(char *str, char *last);
  25. void setroot();
  26. void statusloop();
  27. void termhandler(int signum);
  28. #include "blocks.h"
  29. static Display *dpy;
  30. static int screen;
  31. static Window root;
  32. static char statusbar[LENGTH(blocks)][CMDLENGTH] = {0};
  33. static char statusstr[2][256];
  34. static int statusContinue = 1;
  35. static void (*writestatus) () = setroot;
  36. void replace(char *str, char old, char new)
  37. {
  38. int N = strlen(str);
  39. for(int i = 0; i < N; i++)
  40. if(str[i] == old)
  41. str[i] = new;
  42. }
  43. //opens process *cmd and stores output in *output
  44. void getcmd(const Block *block, char *output)
  45. {
  46. strcpy(output, block->icon);
  47. char *cmd = block->command;
  48. FILE *cmdf = popen(cmd,"r");
  49. if (!cmdf)
  50. return;
  51. char c;
  52. int i = strlen(block->icon);
  53. fgets(output+i, CMDLENGTH-i, cmdf);
  54. i = strlen(output);
  55. if (delim != '\0' && --i)
  56. output[i++] = delim;
  57. output[i++] = '\0';
  58. pclose(cmdf);
  59. }
  60. void getcmds(int time)
  61. {
  62. const Block* current;
  63. for(int i = 0; i < LENGTH(blocks); i++)
  64. {
  65. current = blocks + i;
  66. if ((current->interval != 0 && time % current->interval == 0) || time == -1)
  67. getcmd(current,statusbar[i]);
  68. }
  69. }
  70. #ifndef __OpenBSD__
  71. void getsigcmds(int signal)
  72. {
  73. const Block *current;
  74. for (int i = 0; i < LENGTH(blocks); i++)
  75. {
  76. current = blocks + i;
  77. if (current->signal == signal)
  78. getcmd(current,statusbar[i]);
  79. }
  80. }
  81. void setupsignals()
  82. {
  83. /* initialize all real time signals with dummy handler */
  84. for(int i = SIGRTMIN; i <= SIGRTMAX; i++)
  85. signal(i, dummysighandler);
  86. for(int i = 0; i < LENGTH(blocks); i++)
  87. {
  88. if (blocks[i].signal > 0)
  89. signal(SIGRTMIN+blocks[i].signal, sighandler);
  90. }
  91. }
  92. #endif
  93. int getstatus(char *str, char *last)
  94. {
  95. strcpy(last, str);
  96. str[0] = '\0';
  97. for(int i = 0; i < LENGTH(blocks); i++)
  98. strcat(str, statusbar[i]);
  99. str[strlen(str)-1] = '\0';
  100. return strcmp(str, last);//0 if they are the same
  101. }
  102. void setroot()
  103. {
  104. if (!getstatus(statusstr[0], statusstr[1]))//Only set root if text has changed.
  105. return;
  106. Display *d = XOpenDisplay(NULL);
  107. if (d) {
  108. dpy = d;
  109. }
  110. screen = DefaultScreen(dpy);
  111. root = RootWindow(dpy, screen);
  112. XStoreName(dpy, root, statusstr[0]);
  113. XCloseDisplay(dpy);
  114. }
  115. void pstdout()
  116. {
  117. if (!getstatus(statusstr[0], statusstr[1]))//Only write out if text has changed.
  118. return;
  119. printf("%s\n",statusstr[0]);
  120. fflush(stdout);
  121. }
  122. void statusloop()
  123. {
  124. #ifndef __OpenBSD__
  125. setupsignals();
  126. #endif
  127. int i = 0;
  128. getcmds(-1);
  129. while(statusContinue)
  130. {
  131. getcmds(i);
  132. writestatus();
  133. sleep(1.0);
  134. i++;
  135. }
  136. }
  137. #ifndef __OpenBSD__
  138. /* this signal handler should do nothing */
  139. void dummysighandler(int signum)
  140. {
  141. return;
  142. }
  143. #endif
  144. #ifndef __OpenBSD__
  145. void sighandler(int signum)
  146. {
  147. getsigcmds(signum-SIGRTMIN);
  148. writestatus();
  149. }
  150. #endif
  151. void termhandler(int signum)
  152. {
  153. statusContinue = 0;
  154. exit(0);
  155. }
  156. int main(int argc, char** argv)
  157. {
  158. for(int i = 0; i < argc; i++)
  159. {
  160. if (!strcmp("-d",argv[i]))
  161. delim = argv[++i][0];
  162. else if(!strcmp("-p",argv[i]))
  163. writestatus = pstdout;
  164. }
  165. signal(SIGTERM, termhandler);
  166. signal(SIGINT, termhandler);
  167. statusloop();
  168. }