input.cpp 8.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361
  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 = 220;
  9. int input_scroll = 0;
  10. door::ANSIColor prompt_color{door::COLOR::YELLOW, door::COLOR::BLUE,
  11. door::ATTR::BOLD};
  12. door::ANSIColor input_color{door::COLOR::WHITE}; // , door::COLOR::BLUE};
  13. void erase(door::Door &d, int count) {
  14. d << door::reset;
  15. for (int x = 0; x < count; ++x) {
  16. d << "\x08 \x08";
  17. }
  18. }
  19. void clear_input(door::Door &d) {
  20. if (prompt.empty())
  21. return;
  22. if (input_scroll == 0)
  23. erase(d, input.size());
  24. else
  25. erase(d, input.size() - input_scroll + 3);
  26. erase(d, prompt.size() + 1);
  27. }
  28. void restore_input(door::Door &d) {
  29. if (prompt.empty())
  30. return;
  31. d << prompt_color << prompt << input_color << " ";
  32. if (input_scroll == 0)
  33. d << input;
  34. else
  35. d << "..." << input.substr(input_scroll);
  36. }
  37. /*
  38. commands:
  39. /h /help /?
  40. /t /talk /talkto [TARGET]
  41. /msg [TO] [message]
  42. /me [message]
  43. /quit [message, maybe]
  44. /join [TARGET]
  45. /part [TARGET]
  46. future:
  47. /list ?
  48. /version ?
  49. */
  50. void parse_input(door::Door &door, ircClient &irc) {
  51. // yes, we have something
  52. std::time_t now_t;
  53. time(&now_t);
  54. if (input[0] == '/') {
  55. // command given
  56. std::vector<std::string> cmd = split_limit(input, 3);
  57. if (cmd[0] == "/motd") {
  58. irc.write("MOTD");
  59. }
  60. if (cmd[0] == "/names") {
  61. std::string talk = irc.talkto();
  62. if (talk[0] == '#') {
  63. // or we could pull this from /info & talk.
  64. irc.write("NAMES " + talk);
  65. } // handle this input error
  66. }
  67. if (cmd[0] == "/quit") {
  68. irc.write("QUIT");
  69. }
  70. if (cmd[0] == "/talkto") {
  71. irc.talkto(cmd[1]);
  72. door << "[talkto = " << cmd[1] << "]" << door::nl;
  73. }
  74. if (cmd[0] == "/join") {
  75. if (allow_join) {
  76. std::string tmp = "JOIN " + cmd[1];
  77. irc.write(tmp);
  78. }
  79. }
  80. if (cmd[0] == "/part") {
  81. if (allow_part) {
  82. std::string tmp = "PART " + cmd[1];
  83. irc.write(tmp);
  84. }
  85. }
  86. // TODO: feed this and /me to render so DRY/one place to render
  87. if (cmd[0] == "/msg") {
  88. std::string tmp = "PRIVMSG " + cmd[1] + " :" + cmd[2];
  89. irc.write(tmp);
  90. // build msg for render
  91. tmp = ":" + irc.nick + "!" + " " + tmp;
  92. message_stamp msg;
  93. msg.buffer = irc_split(tmp);
  94. render(msg, door, irc);
  95. /*
  96. stamp(now_t, door);
  97. if (cmd[1][0] == '#') {
  98. door << irc.nick << "/" << cmd[1] << " " << cmd[2] << door::nl;
  99. } else {
  100. door << cmd[1] << " " << cmd[2] << door::nl;
  101. }
  102. */
  103. }
  104. if (cmd[0] == "/me") {
  105. cmd = split_limit(input, 2);
  106. std::string tmp =
  107. "PRIVMSG " + irc.talkto() + " :\x01" + "ACTION " + cmd[1] + "\x01";
  108. irc.write(tmp);
  109. // build msg for render
  110. tmp = ":" + irc.nick + "!" + " ACTION " + irc.talkto() + " " + cmd[1];
  111. message_stamp msg;
  112. msg.buffer = irc_split(tmp);
  113. render(msg, door, irc);
  114. /*
  115. stamp(now_t, door);
  116. door << "* " << irc.nick << " " << cmd[1] << door::nl;
  117. */
  118. }
  119. if (cmd[0] == "/nick") {
  120. std::string tmp = "NICK " + cmd[1];
  121. irc.write(tmp);
  122. }
  123. if (cmd[0] == "/help") {
  124. door << "IRC Commands :" << door::nl;
  125. door << "/help /motd /quit /nick" << door::nl;
  126. door << "/me ACTION" << door::nl;
  127. door << "/msg TARGET Message" << door::nl;
  128. if (allow_part) {
  129. door << "/join #CHANNEL" << door::nl;
  130. door << "/part #CHANNEL" << door::nl;
  131. }
  132. door << "[ESC] aborts input" << door::nl;
  133. }
  134. if (cmd[0] == "/info") {
  135. irc.channels_lock.lock();
  136. for (auto c : irc.channels) {
  137. door << "CH " << c.first << " ";
  138. for (auto s : c.second) {
  139. door << s << " ";
  140. }
  141. door << door::nl;
  142. }
  143. irc.channels_lock.unlock();
  144. }
  145. } else {
  146. std::string target = irc.talkto();
  147. std::string output = "PRIVMSG " + target + " :" + input;
  148. // I need to display something here to show we've said something (and
  149. // where we've said it)
  150. door::ANSIColor nick_color{door::COLOR::WHITE, door::COLOR::BLUE};
  151. irc.write(output);
  152. // build message for render
  153. message_stamp msg;
  154. output = ":" + irc.nick + "!" + " " + output;
  155. msg.buffer = irc_split(output);
  156. render(msg, door, irc);
  157. /*
  158. stamp(now_t, door);
  159. if (target[0] == '#') {
  160. door::ANSIColor channel_color = door::ANSIColor{
  161. door::COLOR::YELLOW, door::COLOR::BLUE, door::ATTR::BOLD};
  162. door << channel_color << target << nick_color << "/" << nick_color
  163. << irc.nick << door::reset << " " << input << door::nl;
  164. } else {
  165. door << nick_color << irc.nick << " -> " << target << door::reset << " "
  166. << input << door::nl;
  167. }
  168. */
  169. }
  170. input.clear();
  171. input_scroll = 0;
  172. }
  173. // can't do /motd it matches /me /msg
  174. // nick matches names.
  175. const char *hot_keys[] = {"/join #", "/part #", "/talkto ", "/help", "/quit "};
  176. bool check_for_input(door::Door &door, ircClient &irc) {
  177. int c;
  178. int width = door.width;
  179. int third = width / 3;
  180. // return true when we have input and is "valid" // ready
  181. if (prompt.empty()) {
  182. // ok, nothing has been displayed at this time.
  183. if (door.haskey()) {
  184. // something to do.
  185. c = door.sleep_key(1);
  186. if (c < 0) {
  187. // handle timeout/hangup/out of time
  188. return false;
  189. }
  190. if (c > 0x1000)
  191. return false;
  192. // How to handle "early" typing, we we're still connecting...
  193. // FAIL-WHALE (what if we part all channels?)
  194. if (irc.registered)
  195. // don't take any imput unless our talkto has been set.
  196. if (isprint(c)) {
  197. prompt = "[" + irc.talkto() + "]";
  198. door << prompt_color << prompt << input_color << " ";
  199. door << (char)c;
  200. input.append(1, c);
  201. }
  202. }
  203. return false;
  204. } else {
  205. // continue on with what we have displayed.
  206. c = door.sleep_ms_key(ms_input_delay);
  207. if (c != -1) {
  208. /*
  209. c = door.sleep_key(1);
  210. if (c < 0) {
  211. // handle error
  212. return false;
  213. }
  214. */
  215. if (c > 0x1000)
  216. return false;
  217. if (isprint(c)) {
  218. // string length check / scroll support?
  219. if ((int)input.size() == max_input) {
  220. door << (char)7;
  221. return false;
  222. }
  223. door << (char)c;
  224. input.append(1, c);
  225. int pos = input.size() - input_scroll;
  226. if (input_scroll != 0)
  227. pos += 3;
  228. // need something better then magic number here.
  229. // 3 is our extra padding here.
  230. int prompt_size = prompt.size() + 1; // prompt + space
  231. if (pos + prompt_size + 3 == width) {
  232. // Ok, scroll!
  233. clear_input(door);
  234. input_scroll = input.size() - third;
  235. restore_input(door);
  236. }
  237. // hot-keys
  238. if (input[0] == '/') {
  239. if (input.size() == 2) {
  240. char c = std::tolower(input[1]);
  241. if (!allow_part) {
  242. if (c == 'p')
  243. c = '!';
  244. }
  245. if (!allow_join) {
  246. if (c == 'j')
  247. c = '!';
  248. }
  249. for (auto hk : hot_keys) {
  250. if (c == hk[1]) {
  251. erase(door, input.size());
  252. input = hk;
  253. door << input;
  254. break;
  255. }
  256. }
  257. }
  258. }
  259. }
  260. if (c == 0x1b) {
  261. // escape key
  262. clear_input(door);
  263. input.clear();
  264. prompt.clear();
  265. input_scroll = 0;
  266. return false;
  267. }
  268. if ((c == 0x08) or (c == 0x7f)) {
  269. // hot-keys
  270. if (input[0] == '/') {
  271. for (auto hk : hot_keys) {
  272. if (input == hk) {
  273. clear_input(door);
  274. input.clear();
  275. prompt.clear();
  276. input_scroll = 0;
  277. return false;
  278. }
  279. }
  280. }
  281. if (input.size() > 1) {
  282. erase(door, 1);
  283. door << input_color;
  284. input.erase(input.length() - 1);
  285. if (input_scroll != 0) {
  286. // are we getting close?
  287. if ((int)input.size() - third < input_scroll) {
  288. // scroll the other way
  289. clear_input(door);
  290. input_scroll = (input.size() - 2 * third);
  291. if (input_scroll < 0)
  292. input_scroll = 0;
  293. restore_input(door);
  294. }
  295. }
  296. } else {
  297. // erasing the last character
  298. erase(door, 1);
  299. input.clear();
  300. erase(door, prompt.size() + 1);
  301. prompt.clear();
  302. return false;
  303. }
  304. }
  305. if (c == 0x0d) {
  306. clear_input(door);
  307. prompt.clear();
  308. parse_input(door, irc);
  309. input.clear();
  310. input_scroll = 0;
  311. return true;
  312. }
  313. }
  314. return false;
  315. }
  316. }