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.

167 lines
2.9 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* icon;
  10. char* command;
  11. unsigned int interval;
  12. unsigned int signal;
  13. } Block;
  14. void sighandler(int num);
  15. void replace(char *str, char old, char new);
  16. void getcmds(int time);
  17. void getsigcmds(int signal);
  18. void setupsignals();
  19. void getstatus(char *str);
  20. void setroot();
  21. void statusloop();
  22. void statusinit();
  23. void sighandler(int signum);
  24. void termhandler(int signum);
  25. #include "blocks.h"
  26. static Display *dpy;
  27. static int screen;
  28. static Window root;
  29. static char statusbar[LENGTH(blocks)][50] = {0};
  30. static char statusstr[256];
  31. static char *statuscat;
  32. static const char *volupcmd[] = { "volup", NULL };
  33. static const char *voldowncmd[] = { "voldown", NULL };
  34. static const char *volmutecmd[] = { "volmute", NULL };
  35. static int statusContinue = 1,volmuted = 0;
  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. //int N = strlen(output);
  52. char c;
  53. int i = strlen(block->icon);
  54. while((c = fgetc(cmdf)) != EOF)
  55. output[i++] = c;
  56. if (delim != '\0' && --i)
  57. output[i++] = delim;
  58. output[i++] = '\0';
  59. pclose(cmdf);
  60. }
  61. void getcmds(int time)
  62. {
  63. const Block* current;
  64. for(int i = 0; i < LENGTH(blocks); i++)
  65. {
  66. current = blocks + i;
  67. if ((current->interval != 0 && time % current->interval == 0) || time == -1)
  68. getcmd(current,statusbar[i]);
  69. }
  70. }
  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. for(int i = 0; i < LENGTH(blocks); i++)
  84. {
  85. if (blocks[i].signal > 0)
  86. signal(SIGRTMIN+blocks[i].signal, sighandler);
  87. }
  88. }
  89. void getstatus(char *str)
  90. {
  91. int j = 0;
  92. for(int i = 0; i < LENGTH(blocks); j+=strlen(statusbar[i++]))
  93. {
  94. strcpy(str + j, statusbar[i]);
  95. }
  96. str[--j] = '\0';
  97. }
  98. void setroot()
  99. {
  100. Display *d = XOpenDisplay(NULL);
  101. if (d) {
  102. dpy = d;
  103. }
  104. screen = DefaultScreen(dpy);
  105. root = RootWindow(dpy, screen);
  106. getstatus(statusstr);
  107. XStoreName(dpy, root, statusstr);
  108. XCloseDisplay(dpy);
  109. }
  110. void statusloop()
  111. {
  112. setupsignals();
  113. int i = 0;
  114. getcmds(-1);
  115. while(statusContinue)
  116. {
  117. getcmds(i);
  118. setroot();
  119. sleep(1.0);
  120. i++;
  121. }
  122. }
  123. void statusinit()
  124. {
  125. statusloop();
  126. }
  127. void sighandler(int signum)
  128. {
  129. getsigcmds(signum-SIGRTMIN);
  130. setroot();
  131. }
  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. }
  144. signal(SIGTERM, termhandler);
  145. signal(SIGINT, termhandler);
  146. statusinit();
  147. }