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.

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