input.cpp 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267
  1. #include "input.h"
  2. #include "render.h"
  3. bool allow_part = false;
  4. bool allow_join = false;
  5. int ms_input_delay = 250;
  6. std::string input;
  7. std::string prompt; // mostly for length to erase/restore properly
  8. int max_input = 100;
  9. door::ANSIColor prompt_color{door::COLOR::YELLOW, door::COLOR::BLUE,
  10. door::ATTR::BOLD};
  11. door::ANSIColor input_color{door::COLOR::WHITE}; // , door::COLOR::BLUE};
  12. void erase(door::Door &d, int count) {
  13. d << door::reset;
  14. for (int x = 0; x < count; ++x) {
  15. d << "\x08 \x08";
  16. }
  17. }
  18. void clear_input(door::Door &d) {
  19. if (prompt.empty())
  20. return;
  21. erase(d, input.size());
  22. erase(d, prompt.size() + 1);
  23. }
  24. void restore_input(door::Door &d) {
  25. if (prompt.empty())
  26. return;
  27. d << prompt_color << prompt << input_color << " " << input;
  28. }
  29. /*
  30. commands:
  31. /h /help /?
  32. /t /talk /talkto [TARGET]
  33. /msg [TO] [message]
  34. /me [message]
  35. /quit [message, maybe]
  36. /join [TARGET]
  37. /part [TARGET]
  38. future:
  39. /list ?
  40. /version ?
  41. */
  42. void parse_input(door::Door &door, ircClient &irc) {
  43. // yes, we have something
  44. std::time_t now_t;
  45. time(&now_t);
  46. if (input[0] == '/') {
  47. // command given
  48. std::vector<std::string> cmd = split_limit(input, 3);
  49. if (cmd[0] == "/motd") {
  50. irc.write("MOTD");
  51. }
  52. if (cmd[0] == "/quit") {
  53. irc.write("QUIT");
  54. }
  55. if (cmd[0] == "/talkto") {
  56. irc.talkto(cmd[1]);
  57. door << "[talkto = " << cmd[1] << "]" << door::nl;
  58. }
  59. if (cmd[0] == "/join") {
  60. if (allow_join) {
  61. std::string tmp = "JOIN " + cmd[1];
  62. irc.write(tmp);
  63. }
  64. }
  65. if (cmd[0] == "/part") {
  66. if (allow_part) {
  67. std::string tmp = "PART " + cmd[1];
  68. irc.write(tmp);
  69. }
  70. }
  71. if (cmd[0] == "/msg") {
  72. std::string tmp = "PRIVMSG " + cmd[1] + " :" + cmd[2];
  73. irc.write(tmp);
  74. stamp(now_t, door);
  75. if (cmd[1][0] == '#') {
  76. door << irc.nick << "/" << cmd[1] << " " << cmd[2] << door::nl;
  77. } else {
  78. door << cmd[1] << " " << cmd[2] << door::nl;
  79. }
  80. }
  81. if (cmd[0] == "/me") {
  82. cmd = split_limit(input, 2);
  83. std::string tmp =
  84. "PRIVMSG " + irc.talkto() + " :\x01" + "ACTION " + cmd[1] + "\x01";
  85. irc.write(tmp);
  86. stamp(now_t, door);
  87. door << "* " << irc.nick << " " << cmd[1] << door::nl;
  88. }
  89. if (cmd[0] == "/nick") {
  90. std::string tmp = "NICK " + cmd[1];
  91. irc.write(tmp);
  92. }
  93. if (cmd[0] == "/help") {
  94. door << "IRC Commands :" << door::nl;
  95. door << "/help /motd /quit /nick" << door::nl;
  96. door << "/me ACTION" << door::nl;
  97. door << "/msg TARGET Message" << door::nl;
  98. if (allow_part) {
  99. door << "/join #CHANNEL" << door::nl;
  100. door << "/part #CHANNEL" << door::nl;
  101. }
  102. }
  103. if (cmd[0] == "/info") {
  104. irc.channels_lock.lock();
  105. for (auto c : irc.channels) {
  106. door << "CH " << c.first << " ";
  107. for (auto s : c.second) {
  108. door << s << " ";
  109. }
  110. door << door::nl;
  111. }
  112. irc.channels_lock.unlock();
  113. }
  114. } else {
  115. std::string target = irc.talkto();
  116. std::string output = "PRIVMSG " + target + " :" + input;
  117. // I need to display something here to show we've said something (and
  118. // where we've said it)
  119. door::ANSIColor nick_color{door::COLOR::WHITE, door::COLOR::BLUE};
  120. stamp(now_t, door);
  121. if (target[0] == '#') {
  122. door::ANSIColor channel_color = door::ANSIColor{
  123. door::COLOR::YELLOW, door::COLOR::BLUE, door::ATTR::BOLD};
  124. door << channel_color << target << nick_color << "/" << nick_color
  125. << irc.nick << door::reset << " " << input << door::nl;
  126. } else {
  127. door << nick_color << irc.nick << " -> " << target << door::reset << " "
  128. << input << door::nl;
  129. }
  130. irc.write(output);
  131. }
  132. input.clear();
  133. }
  134. // can't do /motd it matches /me /msg
  135. const char *hot_keys[] = {"/join #", "/nick ", "/part #",
  136. "/talkto ", "/help", "/quit "};
  137. bool check_for_input(door::Door &door, ircClient &irc) {
  138. int c;
  139. // return true when we have input and is "valid" // ready
  140. if (prompt.empty()) {
  141. // ok, nothing has been displayed at this time.
  142. if (door.haskey()) {
  143. // something to do.
  144. c = door.sleep_key(1);
  145. if (c < 0) {
  146. // handle timeout/hangup/out of time
  147. return false;
  148. }
  149. if (c > 0x1000)
  150. return false;
  151. if (isprint(c)) {
  152. prompt = "[" + irc.talkto() + "]";
  153. door << prompt_color << prompt << input_color << " ";
  154. door << (char)c;
  155. input.append(1, c);
  156. }
  157. }
  158. return false;
  159. } else {
  160. // continue on with what we have displayed.
  161. c = door.sleep_ms_key(ms_input_delay);
  162. if (c != -1) {
  163. /*
  164. c = door.sleep_key(1);
  165. if (c < 0) {
  166. // handle error
  167. return false;
  168. }
  169. */
  170. if (c > 0x1000)
  171. return false;
  172. if (isprint(c)) {
  173. // string length check / scroll support?
  174. door << (char)c;
  175. input.append(1, c);
  176. // hot-keys
  177. if (input[0] == '/') {
  178. if (input.size() == 2) {
  179. char c = std::tolower(input[1]);
  180. if (!allow_part) {
  181. if (c == 'p')
  182. c = '!';
  183. }
  184. if (!allow_join) {
  185. if (c == 'j')
  186. c = '!';
  187. }
  188. for (auto hk : hot_keys) {
  189. if (c == hk[1]) {
  190. erase(door, input.size());
  191. input = hk;
  192. door << input;
  193. break;
  194. }
  195. }
  196. }
  197. }
  198. }
  199. if ((c == 0x08) or (c == 0x7f)) {
  200. // hot-keys
  201. if (input[0] == '/') {
  202. for (auto hk : hot_keys) {
  203. if (input == hk) {
  204. clear_input(door);
  205. input.clear();
  206. prompt.clear();
  207. return false;
  208. }
  209. }
  210. }
  211. if (input.size() > 1) {
  212. erase(door, 1);
  213. door << input_color;
  214. input.erase(input.length() - 1);
  215. } else {
  216. // erasing the last character
  217. erase(door, 1);
  218. input.clear();
  219. erase(door, prompt.size() + 1);
  220. prompt.clear();
  221. return false;
  222. }
  223. }
  224. if (c == 0x0d) {
  225. clear_input(door);
  226. prompt.clear();
  227. parse_input(door, irc);
  228. input.clear();
  229. return true;
  230. }
  231. }
  232. return false;
  233. }
  234. }