render.cpp 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250
  1. #include "render.h"
  2. #include <boost/lexical_cast.hpp>
  3. #include <iomanip>
  4. std::string timestamp_format = "%T";
  5. /**
  6. * @brief length of the timestamp string.
  7. *
  8. */
  9. static int stamp_length;
  10. void stamp(std::time_t &stamp, door::Door &door) {
  11. std::string output = boost::lexical_cast<std::string>(
  12. std::put_time(std::localtime(&stamp), timestamp_format.c_str()));
  13. if (output.find('A') != std::string::npos)
  14. door << door::ANSIColor(door::COLOR::YELLOW, door::ATTR::BOLD);
  15. else
  16. door << door::ANSIColor(door::COLOR::BROWN);
  17. door << output << door::reset << " ";
  18. stamp_length = (int)output.size() + 1;
  19. // door << std::put_time(std::localtime(&stamp), timestamp_format.c_str())
  20. }
  21. void word_wrap(int left_side, door::Door &door, std::string text) {
  22. int workarea = door.width - left_side;
  23. bool first_line = true;
  24. door::ANSIColor color = door.previous;
  25. door.log() << "word_wrap " << left_side << " area " << workarea << " ["
  26. << text << "]" << std::endl;
  27. while (text.size() > 0) {
  28. if (!first_line) {
  29. // This isn't the first line, so move over to align the text.
  30. door << std::string(left_side, ' ') << color;
  31. }
  32. first_line = false;
  33. if ((int)text.size() > workarea) {
  34. // do magic
  35. int breaker = workarea;
  36. while ((breaker > workarea / 2) and (text[breaker] != ' '))
  37. --breaker;
  38. if (breaker > workarea / 2) {
  39. // Ok, we found a logical breaking point.
  40. door << text.substr(0, breaker) << door::reset << door::nl;
  41. text.erase(0, breaker + 1);
  42. } else {
  43. // We did not find a good breaking point.
  44. door << text.substr(0, workarea) << door::reset << door::nl;
  45. text.erase(0, workarea);
  46. }
  47. } else {
  48. // finish it up
  49. door << text << door::reset << door::nl;
  50. text.clear();
  51. }
  52. }
  53. }
  54. void render(message_stamp &msg_stamp, door::Door &door, ircClient &irc) {
  55. // std::vector<std::string> irc_msg = *msg;
  56. std::vector<std::string> &irc_msg = msg_stamp.buffer;
  57. door::ANSIColor info{door::COLOR::CYAN};
  58. door::ANSIColor error{door::COLOR::RED, door::ATTR::BOLD};
  59. if (irc_msg.size() == 1) {
  60. // system message
  61. stamp(msg_stamp.stamp, door);
  62. door << info << "(" << irc_msg[0] << ")" << door::reset << door::nl;
  63. return;
  64. }
  65. door::ANSIColor nick_color{door::COLOR::CYAN, door::ATTR::BOLD};
  66. door::ANSIColor channel_color{door::COLOR::WHITE, door::COLOR::BLUE};
  67. door::ANSIColor active_channel_color =
  68. door::ANSIColor{door::COLOR::YELLOW, door::COLOR::BLUE, door::ATTR::BOLD};
  69. door::ANSIColor text_color{door::COLOR::WHITE};
  70. std::string cmd = irc_msg[1];
  71. if (irc_msg[0] == "ERROR") {
  72. std::string tmp = irc_msg[1];
  73. if (tmp[0] == ':')
  74. tmp.erase(0, 1);
  75. stamp(msg_stamp.stamp, door);
  76. door << error << "* ERROR: " << tmp << door::reset << door::nl;
  77. }
  78. if (cmd == "332") {
  79. // joined channel with topic
  80. std::vector<std::string> channel_topic = split_limit(irc_msg[3], 2);
  81. channel_topic[1].erase(0, 1);
  82. std::string output =
  83. "Topic for " + channel_topic[0] + " is: " + channel_topic[1];
  84. stamp(msg_stamp.stamp, door);
  85. int left = stamp_length;
  86. door << info;
  87. word_wrap(left, door, output);
  88. // << output << door::reset << door::nl;
  89. }
  90. if (cmd == "366") {
  91. // end of names, output and clear
  92. std::string channel = split_limit(irc_msg[3], 2)[0];
  93. irc.channels_lock.lock();
  94. stamp(msg_stamp.stamp, door);
  95. door << info << "* users on " << channel << " : ";
  96. for (auto name : irc.channels[channel]) {
  97. door << name << " ";
  98. }
  99. irc.channels_lock.unlock();
  100. door << door::reset << door::nl;
  101. // names.clear();
  102. }
  103. if (cmd == "372") {
  104. // MOTD
  105. std::string temp = irc_msg[3];
  106. temp.erase(0, 1);
  107. stamp(msg_stamp.stamp, door);
  108. door << info << "* " << temp << door::reset << door::nl;
  109. }
  110. // 400 and 500 are errors? should show those.
  111. if ((cmd[0] == '4') or (cmd[0] == '5')) {
  112. std::string tmp = irc_msg[3];
  113. if (tmp[0] == ':')
  114. tmp.erase(0, 1);
  115. stamp(msg_stamp.stamp, door);
  116. door << error << "* " << tmp << door::reset << door::nl;
  117. }
  118. if (cmd == "NOTICE") {
  119. std::string tmp = irc_msg[3];
  120. tmp.erase(0, 1);
  121. stamp(msg_stamp.stamp, door);
  122. door << nick_color << parse_nick(irc_msg[0]) << " NOTICE " << tmp
  123. << door::reset << door::nl;
  124. }
  125. if (cmd == "ACTION") {
  126. if (irc_msg[2][0] == '#') {
  127. stamp(msg_stamp.stamp, door);
  128. int left = stamp_length;
  129. if (irc_msg[2] == irc.talkto())
  130. door << active_channel_color;
  131. else
  132. door << channel_color;
  133. door << irc_msg[2] << "/" << nick_color;
  134. left += irc_msg[2].size() + 1;
  135. std::string nick = parse_nick(irc_msg[0]);
  136. left += nick.size();
  137. int len = irc.max_nick_length - nick.size();
  138. if (len > 0) {
  139. door << std::string(len, ' ');
  140. left += len;
  141. }
  142. left += 3;
  143. door << "* " << nick << " ";
  144. word_wrap(left, door, irc_msg[3]);
  145. // << irc_msg[3] << door::reset << door::nl;
  146. /*
  147. door << "* " << irc_msg[2] << "/" << parse_nick(irc_msg[0]) << " "
  148. << irc_msg[3] << door::nl;
  149. */
  150. } else {
  151. stamp(msg_stamp.stamp, door);
  152. int left = stamp_length;
  153. std::string nick = parse_nick(irc_msg[0]);
  154. door << nick_color << "* " << nick << " ";
  155. left += 3 + nick.size();
  156. word_wrap(left, door, irc_msg[3]);
  157. // << irc_msg[3] << door::reset << door::nl;
  158. // door << "* " << parse_nick(irc_msg[0]) << " " << irc_msg[3] <<
  159. // door::nl;
  160. }
  161. }
  162. if (cmd == "TOPIC") {
  163. std::string tmp = irc_msg[3];
  164. tmp.erase(0, 1);
  165. stamp(msg_stamp.stamp, door);
  166. int left = stamp_length;
  167. door << info;
  168. std::string text =
  169. parse_nick(irc_msg[0]) + " set topic of " + irc_msg[2] + " to " + tmp;
  170. word_wrap(left, door, text);
  171. // door << info << parse_nick(irc_msg[0]) << " set topic of " << irc_msg[2]
  172. // << " to " << tmp << door::reset << door::nl;
  173. }
  174. if (cmd == "PRIVMSG") {
  175. if (irc_msg[2][0] == '#') {
  176. std::string tmp = irc_msg[3];
  177. tmp.erase(0, 1);
  178. stamp(msg_stamp.stamp, door);
  179. int left = stamp_length;
  180. if (irc_msg[2] == irc.talkto())
  181. door << active_channel_color;
  182. else
  183. door << channel_color;
  184. door << irc_msg[2];
  185. door << "/" << nick_color;
  186. left += irc_msg[2].size() + 1;
  187. std::string nick = parse_nick(irc_msg[0]);
  188. left += nick.size();
  189. int len = irc.max_nick_length + 2 - nick.size();
  190. if (len > 0) {
  191. door << std::string(len, ' ');
  192. left += len;
  193. }
  194. door << nick << " " << text_color;
  195. left++;
  196. word_wrap(left, door, tmp);
  197. // << tmp << door::reset << door::nl;
  198. /*
  199. door << channel_color << irc_msg[2] << "/" << nick_color
  200. << parse_nick(irc_msg[0]) << door::reset << " " << tmp << door::nl;
  201. */
  202. } else {
  203. std::string tmp = irc_msg[3];
  204. tmp.erase(0, 1);
  205. stamp(msg_stamp.stamp, door);
  206. int left = stamp_length;
  207. std::string nick = parse_nick(irc_msg[0]);
  208. door << nick_color << nick << door::reset << " ";
  209. left += nick.size() + 1;
  210. word_wrap(left, door, tmp);
  211. // << tmp << door::nl;
  212. }
  213. }
  214. if (cmd == "NICK") {
  215. std::string tmp = irc_msg[2];
  216. tmp.erase(0, 1);
  217. stamp(msg_stamp.stamp, door);
  218. door << info << "* " << parse_nick(irc_msg[0]) << " is now known as " << tmp
  219. << door::reset << door::nl;
  220. }
  221. }