command.c 37 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486
  1. /* command.c, Ait, BSD 3-Clause, Kevin Bloom, 2023-2024,
  2. Derived from: Atto January 2017
  3. Derived from: AnthonyEditor January 93
  4. */
  5. #include "header.h"
  6. #include "termbox.h"
  7. #include "util.h"
  8. void quit() { done = 1; }
  9. void up()
  10. {
  11. curbp->b_point = lncolumn(curbp, upup(curbp, curwp, curbp->b_point),curbp->b_pcol - curwp->w_left);
  12. }
  13. void down()
  14. {
  15. curbp->b_point = lncolumn(curbp, dndn(curbp, curwp, curbp->b_point),curbp->b_pcol - curwp->w_left);
  16. }
  17. void lnbegin()
  18. {
  19. char_t *p;
  20. if(curbp->b_point == 0)
  21. return;
  22. p = ptr(curbp, curbp->b_point);
  23. while(*(p = ptr(curbp, curbp->b_point-1)) != '\n' && p > curbp->b_buf)
  24. --curbp->b_point;
  25. if(curbp->b_point != 0 && p == curbp->b_buf)
  26. --curbp->b_point;
  27. curbp->b_pcol = 0 + curwp->w_left;
  28. }
  29. void version() { msg(VERSION); }
  30. void top()
  31. {
  32. shift_pmark(TRUE, NOMARK);
  33. curbp->b_point = 0;
  34. curbp->b_pcol = 0 + curwp->w_left;
  35. }
  36. void bottom()
  37. {
  38. shift_pmark(TRUE, NOMARK);
  39. curbp->b_point = pos(curbp, curbp->b_ebuf);
  40. if (curbp->b_epage < pos(curbp, curbp->b_ebuf))
  41. curbp->b_reframe = 1;
  42. curbp->b_pcol = 0 + curwp->w_left;
  43. }
  44. void block() { curbp->b_mark = curbp->b_point; }
  45. void copy() { copy_cut(FALSE, TRUE, FALSE); }
  46. void cut() { copy_cut(TRUE, TRUE, FALSE); }
  47. void resize_terminal()
  48. {
  49. LINES = tb_height();
  50. COLS = tb_width();
  51. MSGLINE = LINES-1;
  52. one_window(curwp);
  53. }
  54. void print_to_msgline(const char *msg)
  55. {
  56. printf_tb(0, MSGLINE, TB_DEFAULT, TB_DEFAULT, msg);
  57. tb_set_cursor(strlen(msg), MSGLINE);
  58. }
  59. void quit_ask()
  60. {
  61. if (modified_buffers() > 0) {
  62. const char *msg = "Modified buffers exist; really exit (y/N) ?";
  63. print_to_msgline(msg);
  64. clrtoeol(msg, MSGLINE);
  65. if (!yesno(FALSE)) {
  66. clrtoeol("", MSGLINE);
  67. return;
  68. }
  69. }
  70. quit();
  71. }
  72. void redraw()
  73. {
  74. window_t *wp;
  75. for (wp=wheadp; wp != NULL; wp = wp->w_next)
  76. wp->w_update = TRUE;
  77. update_display();
  78. }
  79. void left()
  80. {
  81. int n = prev_utf8_char_size();
  82. if(curbp->b_point == 0)
  83. return;
  84. while (0 < curbp->b_point && n-- > 0)
  85. --curbp->b_point;
  86. }
  87. void right()
  88. {
  89. if(curbp->b_point == pos(curbp, curbp->b_ebuf))
  90. return;
  91. int n = utf8_size(*ptr(curbp,curbp->b_point));
  92. while ((curbp->b_point < pos(curbp, curbp->b_ebuf)) && n-- > 0)
  93. ++curbp->b_point;
  94. }
  95. /* work out number of bytes based on first byte */
  96. int utf8_size(char_t c)
  97. {
  98. return tb_utf8_char_length(c);
  99. }
  100. int prev_utf8_char_size()
  101. {
  102. int n;
  103. for (n=2;n<5;n++)
  104. if (-1 < curbp->b_point - n && (utf8_size(*(ptr(curbp, curbp->b_point - n))) == n))
  105. return n;
  106. return 1;
  107. }
  108. void lnend()
  109. {
  110. char_t *p;
  111. int cols = 0;
  112. lnbegin(); // reset the line so we get the right number for `cols`
  113. while(*(p = ptr(curbp, curbp->b_point)) != '\n' && curbp->b_ebuf > p) {
  114. ++curbp->b_point;
  115. }
  116. /* loop until we get to the correct column */
  117. while(cols > curwp->w_cols) {
  118. cols -= curwp->w_cols;
  119. }
  120. curbp->b_pcol = cols + curwp->w_left; // set it for column-memory
  121. }
  122. void wleft()
  123. {
  124. char_t *p;
  125. if ((!isspace(*(p = ptr(curbp, curbp->b_point))) || !is_symbol(*p)) && curbp->b_buf < p)
  126. --curbp->b_point;
  127. while ((isspace(*(p = ptr(curbp, curbp->b_point))) || is_symbol(*p)) && curbp->b_buf < p)
  128. --curbp->b_point;
  129. while (!isspace(*(p = ptr(curbp, curbp->b_point))) && !is_symbol(*p) && curbp->b_buf < p)
  130. --curbp->b_point;
  131. if(isspace(*(p = ptr(curbp, curbp->b_point))) || is_symbol(*p))
  132. ++curbp->b_point;
  133. }
  134. void wleftdelete()
  135. {
  136. currentcommand = KBD_DELETE_CHAR;
  137. iblock();
  138. wleft();
  139. copy_cut(TRUE, TRUE, FALSE);
  140. }
  141. void pgdown()
  142. {
  143. curbp->b_page = curbp->b_point = upup(curbp, curwp, curbp->b_epage);
  144. while (0 < curwp->w_top - curbp->b_row) {
  145. down();
  146. curbp->b_row--;
  147. }
  148. curbp->b_epage = pos(curbp, curbp->b_ebuf);
  149. curbp->b_pcol = 0 + curwp->w_left;
  150. }
  151. void pgup()
  152. {
  153. int i = curwp->w_rows;
  154. while (0 < --i) {
  155. curbp->b_page = upup(curbp, curwp, curbp->b_page);
  156. up();
  157. }
  158. curbp->b_pcol = 0 + curwp->w_left;
  159. }
  160. void wright()
  161. {
  162. char_t *p;
  163. if ((!isspace(*(p = ptr(curbp, curbp->b_point))) || !is_symbol(*p)) && p < curbp->b_ebuf)
  164. ++curbp->b_point;
  165. while ((isspace(*(p = ptr(curbp, curbp->b_point))) || is_symbol(*p)) && p < curbp->b_ebuf)
  166. ++curbp->b_point;
  167. while (!isspace(*(p = ptr(curbp, curbp->b_point))) && !is_symbol(*p) && p < curbp->b_ebuf)
  168. ++curbp->b_point;
  169. }
  170. void wrightdelete()
  171. {
  172. currentcommand = KBD_DELETE_CHAR;
  173. iblock();
  174. wright();
  175. copy_cut(TRUE, TRUE, FALSE);
  176. }
  177. void insert()
  178. {
  179. assert(curbp->b_gap <= curbp->b_egap);
  180. if (curbp->b_gap == curbp->b_egap && !growgap(curbp, CHUNK))
  181. return;
  182. curbp->b_point = movegap(curbp, curbp->b_point);
  183. /* overwrite if mid line, not EOL or EOF, CR will insert as normal */
  184. if ((curbp->b_flags & B_OVERWRITE) && *input != '\r' && *(ptr(curbp, curbp->b_point)) != '\n' && curbp->b_point < pos(curbp,curbp->b_ebuf) ) {
  185. *(ptr(curbp, curbp->b_point)) = *input;
  186. if (curbp->b_point < pos(curbp, curbp->b_ebuf))
  187. ++curbp->b_point;
  188. } else {
  189. *curbp->b_gap++ = *input == '\r' ? '\n' : *input;
  190. curbp->b_point = pos(curbp, curbp->b_egap);
  191. // force reframe if scrolled off bottom of screen and at EOF
  192. if (curbp->b_point == pos(curbp, curbp->b_ebuf) && curbp->b_point >= curbp->b_epage &&
  193. curwp->w_rows == (curwp->w_row - curwp->w_top))
  194. curbp->b_reframe = 1;
  195. }
  196. curbp->b_flags |= B_MODIFIED;
  197. undoset_flag = TRUE;
  198. currentcommand = KBD_INSERT;
  199. }
  200. void insert_str()
  201. {
  202. int len = strlen((const char *)input);
  203. assert(curbp->b_gap <= curbp->b_egap);
  204. undoset(1, FALSE);
  205. if (curbp->b_gap == curbp->b_egap && !growgap(curbp, CHUNK))
  206. return;
  207. curbp->b_point = movegap(curbp, curbp->b_point);
  208. /* overwrite if mid line, not EOL or EOF, CR will insert as normal */
  209. if ((curbp->b_flags & B_OVERWRITE) && input[0] != '\r' && *(ptr(curbp, curbp->b_point)) != '\n' && curbp->b_point < pos(curbp,curbp->b_ebuf) ) {
  210. *(ptr(curbp, curbp->b_point)) = *input;
  211. if (curbp->b_point < pos(curbp, curbp->b_ebuf))
  212. ++curbp->b_point;
  213. } else {
  214. for(int i = 0; i < len; i++) {
  215. *curbp->b_gap++ = input[i] == '\r' ? '\n' : input[i];
  216. // if(input[i] == '\n' || input[i] == '\r')
  217. // curbp->b_line++;
  218. }
  219. curbp->b_point = pos(curbp, curbp->b_egap);
  220. // force reframe if scrolled off bottom of screen and at EOF
  221. if (curbp->b_point == pos(curbp, curbp->b_ebuf) && curbp->b_point >= curbp->b_epage &&
  222. curwp->w_rows == (curwp->w_row - curwp->w_top))
  223. curbp->b_reframe = 1;
  224. }
  225. curbp->b_flags |= B_MODIFIED;
  226. undoset_flag = TRUE;
  227. }
  228. void insert_unicode()
  229. {
  230. int len = strlen((const char *)unicode_buf);
  231. assert(curbp->b_gap <= curbp->b_egap);
  232. undoset(INSERT, lastcommand == KBD_INSERT);
  233. if (curbp->b_gap == curbp->b_egap && !growgap(curbp, CHUNK))
  234. return;
  235. curbp->b_point = movegap(curbp, curbp->b_point);
  236. /* overwrite if mid line, not EOL or EOF, CR will insert as normal */
  237. for(int i = 0; i < len; i++) {
  238. *curbp->b_gap++ = unicode_buf[i];
  239. }
  240. curbp->b_point = pos(curbp, curbp->b_egap);
  241. // force reframe if scrolled off bottom of screen and at EOF
  242. if (curbp->b_point == pos(curbp, curbp->b_ebuf) && curbp->b_point >= curbp->b_epage &&
  243. curwp->w_rows == (curwp->w_row - curwp->w_top))
  244. curbp->b_reframe = 1;
  245. curbp->b_flags |= B_MODIFIED;
  246. undoset_flag = TRUE;
  247. unicode_buf[0] = '\0';
  248. currentcommand = KBD_INSERT;
  249. }
  250. void backsp()
  251. {
  252. undoset(3, lastcommand == KBD_DELETE_CHAR);
  253. if(curbp->b_point != 0 && *ptr(curbp, curbp->b_point - 1) == '\n')
  254. curbp->b_line--;
  255. curbp->b_point = movegap(curbp, curbp->b_point);
  256. if (curbp->b_buf < curbp->b_gap) {
  257. curbp->b_gap -= prev_utf8_char_size();
  258. curbp->b_flags |= B_MODIFIED;
  259. }
  260. curbp->b_point = pos(curbp, curbp->b_egap);
  261. currentcommand = KBD_DELETE_CHAR;
  262. }
  263. void delete()
  264. {
  265. undoset(2, lastcommand == KBD_DELETE_CHAR);
  266. curbp->b_point = movegap(curbp, curbp->b_point);
  267. if (curbp->b_egap < curbp->b_ebuf) {
  268. curbp->b_egap += utf8_size(*curbp->b_egap);
  269. curbp->b_point = pos(curbp, curbp->b_egap);
  270. curbp->b_flags |= B_MODIFIED;
  271. }
  272. currentcommand = KBD_DELETE_CHAR;
  273. }
  274. void gotoline()
  275. {
  276. int line;
  277. point_t p;
  278. if (getinput("Goto line: ", temp, STRBUF_S, F_CLEAR, FALSE)) {
  279. line = atoi(temp);
  280. p = line_to_point(line);
  281. if (p != -1) {
  282. shift_pmark(TRUE, NOMARK);
  283. curbp->b_point = p;
  284. curbp->b_pcol = 0 + curwp->w_left;
  285. if (curbp->b_epage < pos(curbp, curbp->b_ebuf)) curbp->b_reframe = 1;
  286. curwp->w_update = TRUE;
  287. msg("Line %d", line);
  288. } else {
  289. msg("Line %d, not found", line);
  290. }
  291. }
  292. clrtoeol("", MSGLINE);
  293. }
  294. void gotocolumn()
  295. {
  296. int col, remainder = 0;
  297. point_t opoint = curbp->b_point, end = pos(curbp, curbp->b_ebuf);
  298. if (getinput("Goto column: ", temp, STRBUF_S, F_CLEAR, FALSE)) {
  299. col = atoi(temp);
  300. remainder = col - curbp->b_col - 1;
  301. if(remainder > 0) {
  302. shift_pmark(TRUE, NOMARK);
  303. if(*ptr(curbp, curbp->b_point) == '\n') {
  304. msg("Column %d, not found.", col);
  305. return;
  306. }
  307. for(; remainder > 0; remainder--) {
  308. right();
  309. printf("%d,",remainder);
  310. if((*ptr(curbp, curbp->b_point) == '\n' ||
  311. curbp->b_point == end) &&
  312. remainder != 1) {
  313. curbp->b_point = opoint;
  314. msg("Column %d, not found.", col);
  315. return;
  316. }
  317. }
  318. } else if(remainder < 0) {
  319. shift_pmark(TRUE, NOMARK);
  320. remainder *= -1;
  321. if(curbp->b_point == 0 ||
  322. *ptr(curbp, curbp->b_point - 1) == '\n') {
  323. msg("Column %d, not found.", col);
  324. return;
  325. }
  326. for(; remainder > 0; remainder--) {
  327. left();
  328. if((*ptr(curbp, curbp->b_point - 1) == '\n' ||
  329. curbp->b_point == 0) &&
  330. remainder != 1) {
  331. curbp->b_point = opoint;
  332. msg("Column %d, not found.", col);
  333. return;
  334. }
  335. }
  336. }
  337. }
  338. clrtoeol("", MSGLINE);
  339. }
  340. void jumptorow()
  341. {
  342. int line = -1, j = 0, i = 0, current, lastln, pageln;
  343. char num[3] = { 0, 0, 0 };
  344. struct tb_event ev;
  345. char *prompt = "Jump to line reference: ";
  346. int start_col = strlen(prompt), match = FALSE;
  347. char opts[10] = {'f','j','d','k','s','l','g','h', 'a', ';'};
  348. char chars[curwp->w_rows][2];
  349. point_t point;
  350. char_t *p;
  351. char f, s;
  352. int count = 0, fp = 0, sp = 0;
  353. int w_row = curwp->w_row - curwp->w_top;
  354. get_line_stats(&current, &lastln, curbp);
  355. pageln = current - w_row;
  356. point = curbp->b_page;
  357. p = ptr(curbp, point);
  358. for(int i = 0; i < curwp->w_rows && pageln <= lastln; i++) {
  359. f = opts[fp];
  360. s = opts[sp];
  361. chars[i][0] = f;
  362. chars[i][1] = s;
  363. printf_tb(curwp->w_left, curwp->w_top+i, TB_RED, TB_CYAN, "%c%c", f,s);
  364. sp++;
  365. count++;
  366. if(count > 7) {
  367. fp++;
  368. sp = 0;
  369. count = 0;
  370. }
  371. int c = 1;
  372. while(*(p = ptr(curbp, point)) != '\n' &&
  373. curbp->b_ebuf > p && c < curwp->w_cols) {
  374. ++point;
  375. c++;
  376. }
  377. if(*p == '\n' || pageln == lastln)
  378. pageln++;
  379. ++point;
  380. p = ptr(curbp, point);
  381. }
  382. display_prompt_and_response(prompt, num);
  383. tb_present();
  384. while(j < 2) {
  385. display_prompt_and_response(prompt, num);
  386. tb_present();
  387. if(tb_poll_event(&ev) != TB_OK) break;
  388. if(ev.key == TB_KEY_CTRL_G) {
  389. clrtoeol("", MSGLINE);
  390. return;
  391. }
  392. if(j < 2) {
  393. num[j] = ev.ch;
  394. tb_set_cursor(start_col, MSGLINE);
  395. addstr(num);
  396. point = curbp->b_page;
  397. p = ptr(curbp, point);
  398. if(j == 0) {
  399. pageln = current - w_row;
  400. for(int i = 0; i < curwp->w_rows && pageln <= lastln; i++) {
  401. if(chars[i][0] == ev.ch) {
  402. match = TRUE;
  403. if(*p != '\n')
  404. p = ptr(curbp, point+1);
  405. if(*p == '\0')
  406. *p = ' ';
  407. printf_tb(curwp->w_left, curwp->w_top+i, TB_RED, TB_CYAN, "%c", chars[i][1]);
  408. printf_tb(curwp->w_left+1, curwp->w_top+i, TB_DEFAULT, TB_DEFAULT, "%c", *p == '\n' ? ' ' : *p);
  409. } else {
  410. printf_tb(curwp->w_left, curwp->w_top+i, TB_DEFAULT, TB_DEFAULT, "%c", *p == '\n' ? ' ' : *p);
  411. if(*p != '\n')
  412. p = ptr(curbp, point+1);
  413. if(*p == '\0')
  414. *p = ' ';
  415. printf_tb(curwp->w_left+1, curwp->w_top+i, TB_DEFAULT, TB_DEFAULT, "%c", *p == '\n' ? ' ' : *p);
  416. }
  417. int c = 1;
  418. while(*(p = ptr(curbp, point)) != '\n' &&
  419. curbp->b_ebuf > p && c < curwp->w_cols) {
  420. ++point;
  421. c++;
  422. }
  423. if(*p == '\n' || pageln == lastln)
  424. pageln++;
  425. ++point;
  426. p = ptr(curbp, point);
  427. }
  428. }
  429. j++;
  430. }
  431. if(!match) {
  432. clrtoeol("", MSGLINE);
  433. return;
  434. }
  435. }
  436. for(; i < curwp->w_rows; i++) {
  437. if(chars[i][0] == num[0] && chars[i][1] == num[1]) {
  438. line = w_row - i;
  439. break;
  440. }
  441. }
  442. if(i == curwp->w_rows) {
  443. msg("Out of bounds");
  444. return;
  445. }
  446. shift_pmark(TRUE, NOMARK);
  447. if(line > 0) {
  448. for(; line > 0; line--) {
  449. up();
  450. }
  451. } else {
  452. for(; line < 0; line++) {
  453. down();
  454. }
  455. }
  456. clrtoeol("", MSGLINE);
  457. }
  458. void jumpword()
  459. {
  460. point_t current = curbp->b_page;
  461. char num[3] = { 'f', 'f', 0 };
  462. int j = 0, match = FALSE;
  463. char starting[1];
  464. struct tb_event ev;
  465. char *prompt = "Jump to word starting with: ";
  466. int start_col = strlen(prompt);
  467. char opts[10] = {'f','j','d','k','s','l','g','h', 'a', ';'};
  468. int diff = curbp->b_epage - curbp->b_page;
  469. char chars[diff][2];
  470. point_t point = -1;
  471. int begin = TRUE, is_white = FALSE, is_symb = FALSE, charlen = 0;
  472. char_t *p, *tp;
  473. char f, s;
  474. int count = 0, fp = 0, sp = 0, x = 0, y = 0;
  475. display_prompt_and_response(prompt, starting);
  476. tb_present();
  477. if(tb_poll_event(&ev) != TB_OK) return;
  478. if(ev.key == TB_KEY_CTRL_G) {
  479. clrtoeol("", MSGLINE);
  480. return;
  481. }
  482. starting[0] = (unsigned)ev.ch;
  483. for(; current < curbp->b_epage; current++) {
  484. p = ptr(curbp, current);
  485. is_white = isspace(*p);
  486. is_symb = is_symbol(*p);
  487. if(is_white || is_symb || current == 0)
  488. begin = TRUE;
  489. if(*p == (char_t)starting[0] && begin) {
  490. f = opts[fp];
  491. s = opts[sp];
  492. chars[current-curbp->b_page][0] = f;
  493. chars[current-curbp->b_page][1] = s;
  494. charlen++;
  495. printf_tb(curwp->w_left+x, curwp->w_top+y, TB_RED, TB_CYAN, "%c%c", f,s);
  496. sp++;
  497. count++;
  498. if(count > 7) {
  499. fp++;
  500. sp = 0;
  501. count = 0;
  502. }
  503. begin = FALSE;
  504. }
  505. if(!is_white && !is_symb)
  506. begin = FALSE;
  507. x++;
  508. if(*p == '\t')
  509. x += (TAB_SIZE - 2);
  510. if(*p < 31)
  511. x++;
  512. if(*p == '\n' || x >= curwp->w_cols) {
  513. x = 0;
  514. y++;
  515. }
  516. }
  517. tb_present();
  518. if(charlen > 1) {
  519. display_prompt_and_response(prompt, num);
  520. tb_present();
  521. while(j < 2) {
  522. display_prompt_and_response(prompt, num);
  523. tb_present();
  524. if(tb_poll_event(&ev) != TB_OK) break;
  525. if(ev.key == TB_KEY_CTRL_G) {
  526. clrtoeol("", MSGLINE);
  527. return;
  528. }
  529. if(j < 2) {
  530. num[j] = ev.ch;
  531. tb_set_cursor(start_col, MSGLINE);
  532. addstr(num);
  533. x = 0;
  534. y = 0;
  535. if(j == 0) {
  536. for(current = curbp->b_page; current < curbp->b_epage; current++) {
  537. p = ptr(curbp, current);
  538. tp = ptr(curbp, current);
  539. is_white = isspace(*p);
  540. is_symb = is_symbol(*p);
  541. if(is_white || is_symb || current == 0)
  542. begin = TRUE;
  543. if(*p == (char_t)starting[0] && begin) {
  544. point_t i = current-curbp->b_page;
  545. if(chars[i][0] == ev.ch) {
  546. match = TRUE;
  547. tp = ptr(curbp, current+1);
  548. printf_tb(curwp->w_left+x, curwp->w_top+y, TB_RED, TB_CYAN, "%c", chars[i][1]);
  549. printf_tb(curwp->w_left+x+1, curwp->w_top+y, TB_DEFAULT, TB_DEFAULT, "%c", *tp == '\n' ? ' ' : *tp);
  550. } else {
  551. printf_tb(curwp->w_left+x, curwp->w_top+y, TB_DEFAULT, TB_DEFAULT, "%c", *tp == '\n' ? ' ' : *tp);
  552. tp = ptr(curbp, current+1);
  553. printf_tb(curwp->w_left+x+1, curwp->w_top+y, TB_DEFAULT, TB_DEFAULT, "%c", *tp == '\n' ? ' ' : *tp);
  554. }
  555. begin = FALSE;
  556. }
  557. if(!is_white && !is_symb)
  558. begin = FALSE;
  559. x++;
  560. if(*p == '\t')
  561. x += (TAB_SIZE - 2);
  562. if(*p < 31)
  563. x++;
  564. if(*p == '\n' || x >= curwp->w_cols) {
  565. x = 0;
  566. y++;
  567. }
  568. }
  569. }
  570. j++;
  571. if(!match) {
  572. clrtoeol("", MSGLINE);
  573. return;
  574. }
  575. }
  576. }
  577. }
  578. for(point_t cur = 0; cur < diff; cur++) {
  579. if(chars[cur][0] == num[0] && chars[cur][1] == num[1]) {
  580. point = cur + curbp->b_page;
  581. break;
  582. }
  583. }
  584. if(point == -1) {
  585. msg("Out of bounds.");
  586. } else {
  587. shift_pmark(TRUE, NOMARK);
  588. curbp->b_point = point;
  589. int cols = 0;
  590. /* Calculate the pcol value */
  591. lnbegin(); // reset the line so we get the right number for `cols`
  592. while(curbp->b_point != point) {
  593. ++curbp->b_point;
  594. cols++;
  595. }
  596. /* loop until we get to the correct column */
  597. while(cols > curwp->w_cols) {
  598. cols -= curwp->w_cols;
  599. }
  600. curbp->b_pcol = cols + curwp->w_left; // set it for column-memory
  601. clrtoeol("", MSGLINE);
  602. }
  603. /* Clear out the chars array */
  604. for(int i = 0; i < diff; i++) {
  605. chars[i][0] = 0;
  606. chars[i][1] = 0;
  607. }
  608. /* TODO: figure out why this has to be here
  609. Without this printf, the chars array doesn't appear to get
  610. cleared entirely and you end up jumping to the wrong points.
  611. */
  612. printf("%s", chars[0]);
  613. }
  614. void get_current_path(char *cur_path)
  615. {
  616. int cutoff = 0;
  617. for(int i = strlen(curbp->b_fname) - 1; i > -1; i--) {
  618. if(curbp->b_fname[i] == '/') {
  619. cutoff = i;
  620. break;
  621. }
  622. }
  623. for(int i = 0; i <= cutoff; i++)
  624. cur_path[i] = curbp->b_fname[i];
  625. cur_path[cutoff+1] = '\0';
  626. }
  627. void insertfile()
  628. {
  629. char cur_path[PATH_MAX] = "\0";
  630. if(curbp->b_path) {
  631. get_current_path(cur_path);
  632. strcpy(temp, cur_path);
  633. }
  634. else
  635. strcpy(temp, editor_dir);
  636. if (getfilename("Insert file: ", temp, PATH_MAX))
  637. (void)insert_file(temp, TRUE);
  638. }
  639. void readfile()
  640. {
  641. buffer_t *bp;
  642. char cur_path[PATH_MAX];
  643. if(curbp->b_path) {
  644. get_current_path(cur_path);
  645. strcpy(temp, cur_path);
  646. }
  647. else
  648. strcpy(temp, editor_dir);
  649. int result = getfilename("Find file: ", (char*)temp, PATH_MAX);
  650. if (result) {
  651. bp = find_buffer(temp, TRUE);
  652. disassociate_b(curwp);
  653. curbp = bp;
  654. associate_b2w(curbp, curwp);
  655. if (!growgap(curbp, CHUNK))
  656. fatal("%s: Failed to allocate required memory.\n");
  657. movegap(curbp, 0);
  658. /* load the file if not already loaded */
  659. if (bp != NULL && bp->b_fname[0] == '\0') {
  660. if (!load_file(temp)) {
  661. msg("New file %s", temp);
  662. }
  663. strncpy(curbp->b_fname, temp, PATH_MAX);
  664. curbp->b_fname[PATH_MAX] = '\0'; /* truncate if required */
  665. }
  666. }
  667. }
  668. void savebuffer()
  669. {
  670. const char *message = "No newline at the end of file, add one (Y/n) ?";
  671. if(curbp->b_flags & B_MODIFIED) {
  672. /* move the gap to point 0 so that the ebuf is updated. */
  673. (void) movegap(curbp, 0);
  674. if(*(curbp->b_ebuf - 1) != '\n') {
  675. print_to_msgline(message);
  676. clrtoeol(message, MSGLINE);
  677. if (yesno(TRUE)) {
  678. clrtoeol("", MSGLINE);
  679. *curbp->b_ebuf++ = '\n';
  680. }
  681. }
  682. if (curbp->b_fname[0] != '\0') {
  683. save(curbp->b_fname);
  684. return;
  685. } else {
  686. writefile();
  687. }
  688. } else {
  689. msg("(No changes need to be saved.)");
  690. }
  691. }
  692. void writefile()
  693. {
  694. const char *message = "Write file: ";
  695. strncpy(temp, curbp->b_fname, PATH_MAX);
  696. if (getinput((char *)message, temp, PATH_MAX, F_NONE, FALSE))
  697. if (save(temp) == TRUE)
  698. strncpy(curbp->b_fname, temp, PATH_MAX);
  699. clrtoeol(message, MSGLINE);
  700. }
  701. void killbuffer()
  702. {
  703. buffer_t *kill_bp = curbp;
  704. buffer_t *bp;
  705. int bcount = count_buffers();
  706. const char *message = "Discard changes (y/N) ?";
  707. /* do nothing if only buffer left is the scratch buffer */
  708. if (bcount == 1 && 0 == strcmp(get_buffer_name(curbp), "*scratch*"))
  709. return;
  710. if (curbp->b_flags & B_MODIFIED) {
  711. print_to_msgline(message);
  712. clrtoeol(message, MSGLINE);
  713. if (!yesno(FALSE))
  714. return;
  715. }
  716. if (bcount == 1) {
  717. /* create a scratch buffer */
  718. bp = find_buffer("*scratch*", TRUE);
  719. strncpy(bp->b_bname, "*scratch*", STRBUF_S);
  720. bp->b_path = FALSE;
  721. }
  722. next_buffer();
  723. assert(kill_bp != curbp);
  724. delete_buffer(kill_bp);
  725. for(window_t *wp = wheadp; wp != NULL; wp = wp->w_next) {
  726. if(kill_bp == wp->w_bufp) {
  727. wp->w_bufp = curbp;
  728. }
  729. }
  730. }
  731. void iblock()
  732. {
  733. block();
  734. msg("Mark set");
  735. }
  736. void unmark()
  737. {
  738. shift_pmark(TRUE, NOMARK);
  739. curbp->b_mark = NOMARK;
  740. msg("Mark removed");
  741. }
  742. void toggle_overwrite_mode() {
  743. if (curbp->b_flags & B_OVERWRITE)
  744. curbp->b_flags &= ~B_OVERWRITE;
  745. else
  746. curbp->b_flags |= B_OVERWRITE;
  747. }
  748. void killtoeol()
  749. {
  750. if (curbp->b_point == pos(curbp, curbp->b_ebuf))
  751. return; /* do nothing if at end of file */
  752. if (*(ptr(curbp, curbp->b_point)) == 0xa) {
  753. delete(); /* delete CR if at start of empty line */
  754. } else {
  755. curbp->b_mark = curbp->b_point;
  756. lnend();
  757. if (curbp->b_mark != curbp->b_point) copy_cut(TRUE, TRUE, FALSE);
  758. }
  759. }
  760. void copy_cut(int cut, int displaymsg, int internal)
  761. {
  762. char_t *p;
  763. /* if no mark or point == marker, nothing doing */
  764. if (curbp->b_mark == NOMARK || curbp->b_point == curbp->b_mark)
  765. return;
  766. if (scrap != NULL) {
  767. free(scrap);
  768. scrap = NULL;
  769. }
  770. if(cut && !internal)
  771. undoset(4, FALSE);
  772. if (curbp->b_point < curbp->b_mark) {
  773. /* point above marker: move gap under point, region = marker - point */
  774. (void) movegap(curbp, curbp->b_point);
  775. p = ptr(curbp, curbp->b_point);
  776. nscrap = curbp->b_mark - curbp->b_point;
  777. if(cut && currentcommand == KBD_DELETE_CHAR)
  778. for(point_t pt = curbp->b_mark-1; pt > curbp->b_point; pt--) {
  779. if(*ptr(curbp, pt) == '\n')
  780. curbp->b_line--;
  781. }
  782. } else {
  783. /* if point below marker: move gap under marker, region = point - marker */
  784. (void) movegap(curbp, curbp->b_mark);
  785. p = ptr(curbp, curbp->b_mark);
  786. nscrap = curbp->b_point - curbp->b_mark;
  787. if (cut && currentcommand != KBD_DELETE_CHAR)
  788. for(point_t pt = curbp->b_mark; pt < curbp->b_point; pt++) {
  789. if(*ptr(curbp, pt) == '\n')
  790. curbp->b_line--;
  791. }
  792. }
  793. if ((scrap = (char_t*) malloc(nscrap)) == NULL && displaymsg) {
  794. msg("No more memory available.");
  795. } else {
  796. (void) memcpy(scrap, p, nscrap * sizeof (char_t));
  797. if (cut) {
  798. curbp->b_egap += nscrap; /* if cut expand gap down */
  799. curbp->b_point = pos(curbp, curbp->b_egap); /* set point to after region */
  800. curbp->b_flags |= B_MODIFIED;
  801. if(displaymsg)
  802. msg("%ld bytes cut.", nscrap);
  803. currentcommand = KBD_CUT;
  804. } else {
  805. if(displaymsg)
  806. msg("%ld bytes copied.", nscrap);
  807. }
  808. curbp->b_mark = NOMARK; /* unmark */
  809. }
  810. }
  811. void paste_internal(int internal)
  812. {
  813. int new_rows = 0;
  814. int col = curwp->w_col - curwp->w_left + 1;
  815. point_t opoint = curbp->b_point;
  816. if(curbp->b_flags & B_OVERWRITE)
  817. return;
  818. if (nscrap <= 0) {
  819. msg("Scrap is empty. Nothing to paste.");
  820. } else if (nscrap < curbp->b_egap - curbp->b_gap || growgap(curbp, nscrap)) {
  821. if(!internal)
  822. undoset(5, FALSE);
  823. curbp->b_point = movegap(curbp, curbp->b_point);
  824. memcpy(curbp->b_gap, scrap, nscrap * sizeof (char_t));
  825. curbp->b_gap += nscrap;
  826. curbp->b_point = pos(curbp, curbp->b_egap);
  827. curbp->b_flags |= B_MODIFIED;
  828. /* TODO: this assumes 1 char = 1 point (not always true) */
  829. col += curbp->b_point - opoint;
  830. for(int i = 0, cc = col; scrap[i] != '\0'; i++) {
  831. cc++;
  832. if(scrap[i] == '\n' || cc >= curwp->w_cols) {
  833. new_rows++;
  834. cc = 0;
  835. }
  836. curbp->b_pcol = cc + curwp->w_left;
  837. }
  838. if ((curbp->b_row - curwp->w_top) + new_rows >= curwp->w_rows)
  839. curbp->b_reframe = 1;
  840. }
  841. }
  842. void paste()
  843. {
  844. paste_internal(FALSE);
  845. }
  846. void clipboard()
  847. {
  848. int new_rows = 0;
  849. int ntemp = strlen(gtemp);
  850. if(curbp->b_flags & B_OVERWRITE)
  851. return;
  852. if (ntemp <= 0) {
  853. msg("Temp buffer is empty. Nothing to paste.");
  854. } else if (ntemp < curbp->b_egap - curbp->b_gap || growgap(curbp, ntemp)) {
  855. undoset(CLIPBOARD, FALSE);
  856. curbp->b_point = movegap(curbp, curbp->b_point);
  857. memcpy(curbp->b_gap, gtemp, ntemp * sizeof (char_t));
  858. curbp->b_gap += ntemp;
  859. curbp->b_point = pos(curbp, curbp->b_egap);
  860. curbp->b_flags |= B_MODIFIED;
  861. for(int i = 0; gtemp[i] != '\0'; i++) {
  862. if(gtemp[i] == '\n')
  863. new_rows++;
  864. }
  865. if ((curbp->b_row - curwp->w_top) + new_rows > curwp->w_rows &&
  866. curbp->b_point >= curbp->b_epage)
  867. curbp->b_reframe = 1;
  868. free(gtemp);
  869. gtemp = malloc(sizeof(char) * TEMPBUF);
  870. }
  871. }
  872. void showpos()
  873. {
  874. int current, lastln;
  875. point_t end_p = pos(curbp, curbp->b_ebuf);
  876. get_line_stats(&current, &lastln, curbp);
  877. if (curbp->b_point == end_p) {
  878. msg("[EOB] Line = %d/%d Point = %d/%d", current, lastln,
  879. curbp->b_point, ((curbp->b_ebuf - curbp->b_buf) - (curbp->b_egap - curbp->b_gap)));
  880. } else {
  881. char c = unctrl(*(ptr(curbp, curbp->b_point)));
  882. msg("Char = %c 0x%x Line = %d/%d Point = %d/%d", c, *(ptr(curbp, curbp->b_point)),
  883. current, lastln,
  884. curbp->b_point, ((curbp->b_ebuf - curbp->b_buf) - (curbp->b_egap - curbp->b_gap)));
  885. }
  886. }
  887. /* Delete whitespace between non-whitespace */
  888. void delete_between()
  889. {
  890. char_t *p, other;
  891. struct tb_event ev;
  892. char *prompt = "Bracket to Zap Between: ";
  893. int c, is_start = FALSE;
  894. /* Delete everything between brackets. */
  895. if(universal_argument > 0) {
  896. if(character[0] == '\0') {
  897. display_prompt_and_response(prompt, character);
  898. tb_present();
  899. if(tb_poll_event(&ev) != TB_OK) return;
  900. if(!ev.mod)
  901. c = ev.ch;
  902. else
  903. c = ev.key;
  904. /* Ignore all control keys other than C-g and ESC*/
  905. if (c < 32 && c != TB_KEY_CTRL_G && c != TB_KEY_ESC)
  906. return;
  907. if(c == TB_KEY_CTRL_G || c == TB_KEY_ESC)
  908. return;
  909. else
  910. character[0] = c;
  911. display_prompt_and_response(prompt, character);
  912. tb_present();
  913. }
  914. if(!(other = is_bracket(character[0], TRUE, &is_start))) {
  915. return;
  916. }
  917. jumptochar();
  918. adjust_bline();
  919. universal_argument = 0;
  920. lastcommand = KBD_DELETE_CHAR;
  921. if(is_start) {
  922. curbp->b_point++;
  923. while (*(p = ptr(curbp, curbp->b_point)) != other && curbp->b_buf < p)
  924. delete();
  925. } else {
  926. while (*(p = ptr(curbp, curbp->b_point - 1)) != other && curbp->b_buf < p)
  927. backsp();
  928. }
  929. character[0] = '\0';
  930. return;
  931. }
  932. /* If in a word delete the word both directions.
  933. This is the same as doing a `esc f` then `esc backsp`.
  934. This does not delete the symbols, just the words.
  935. */
  936. if(!isspace(*ptr(curbp, curbp->b_point - 1)) &&
  937. !isspace(*ptr(curbp, curbp->b_point)) &&
  938. !isspace(*ptr(curbp, curbp->b_point + 1))) {
  939. wright();
  940. wleftdelete();
  941. return;
  942. }
  943. lastcommand = KBD_DELETE_CHAR;
  944. /* Otherwise just delete whitespace */
  945. while (isspace(*(p = ptr(curbp, curbp->b_point - 1))) && curbp->b_buf < p && *p != '\n')
  946. backsp();
  947. while (isspace(*(p = ptr(curbp, curbp->b_point))) && curbp->b_buf < p && *p != '\n')
  948. delete();
  949. }
  950. void insertnewlinebelow()
  951. {
  952. char_t newline[2];
  953. newline[0] = '\n';
  954. newline[1] = '\0';
  955. input = newline;
  956. undoset(1, lastcommand == KBD_INSERT);
  957. insert();
  958. up();
  959. currentcommand = KBD_INSERT;
  960. }
  961. void insertnewline()
  962. {
  963. point_t point;
  964. char_t *p, *space = NULL, *str;
  965. int spaces = 0, i;
  966. point = segstart(curbp, curwp, lnstart(curbp, curbp->b_point), curbp->b_point);
  967. while(point < pos(curbp, curbp->b_ebuf) &&
  968. isspace(*(p = ptr(curbp, point))) &&
  969. *p != '\n' &&
  970. curwp->w_col != 0) {
  971. if(spaces == 0) {
  972. space = p;
  973. }
  974. if(*p != '\n') {
  975. spaces++;
  976. point++;
  977. }
  978. }
  979. str = (char_t *) malloc(sizeof(char_t)*spaces+2);
  980. str[0] = '\n';
  981. for(i = 0; i < spaces; i++) {
  982. str[i+1] = *space;
  983. }
  984. str[i+1] = '\0';
  985. input = str;
  986. insert_str();
  987. curbp->b_pcol = spaces + curwp->w_left;
  988. currentcommand = KBD_INSERT;
  989. free(str);
  990. if((curwp->w_row - curwp->w_top) == curwp->w_rows-1) {
  991. curbp->b_reframe = TRUE;
  992. }
  993. }
  994. void inserttab()
  995. {
  996. input = (char_t *)"\t";
  997. undoset(1, FALSE);
  998. insert();
  999. }
  1000. void inserttabasspace()
  1001. {
  1002. char_t spaces[TAB_SPACE_SIZE+1];
  1003. memset(spaces, ' ', sizeof(spaces));
  1004. spaces[TAB_SPACE_SIZE] = '\0';
  1005. input = spaces;
  1006. insert_str();
  1007. }
  1008. void suspend()
  1009. {
  1010. tb_shutdown();
  1011. raise(SIGTSTP);
  1012. }
  1013. void transpose()
  1014. {
  1015. char_t *cur = ptr(curbp, curbp->b_point);
  1016. char_t *prev = ptr(curbp, curbp->b_point-1);
  1017. char_t replace[3];
  1018. if(cur == curbp->b_ebuf) {
  1019. return;
  1020. }
  1021. point_t mark = curbp->b_mark;
  1022. replace[0] = *cur;
  1023. replace[1] = *prev;
  1024. replace[2] = '\0';
  1025. curbp->b_point--;
  1026. curbp->b_mark = curbp->b_point + 2;
  1027. undoset(REPLACE, 2);
  1028. curbp->b_mark = mark;
  1029. curbp->b_point++;
  1030. memcpy(ptr(curbp, curbp->b_point-1), replace, 2 * sizeof (char_t));
  1031. curbp->b_flags |= B_MODIFIED;
  1032. }
  1033. /* Delete a word but don't display any messages. */
  1034. void deleteword(int dir) {
  1035. block();
  1036. if(dir)
  1037. wright();
  1038. else
  1039. wleft();
  1040. copy_cut(TRUE, FALSE, TRUE);
  1041. }
  1042. /* Transpose words and put scrap back to how it was. */
  1043. void transposeword()
  1044. {
  1045. char_t *current_scrap, *p;
  1046. int n_scrap = nscrap, newlines = 0;
  1047. point_t mark = curbp->b_mark, epoint, point, npoint;
  1048. /* copy the current scrap */
  1049. current_scrap = (char_t*) malloc(nscrap);
  1050. (void) memcpy(current_scrap, scrap, nscrap * sizeof (char_t));
  1051. /* Find all the key points for the undo */
  1052. wright();
  1053. epoint = curbp->b_point;
  1054. wleft();
  1055. wleft();
  1056. curbp->b_mark = epoint;
  1057. point = curbp->b_point;
  1058. /* Adjust `b_line` to match the line you'll eventually
  1059. be on. This has to happen before the undo so that the
  1060. undo's line tracker keeps it right.
  1061. */
  1062. npoint = point;
  1063. while(npoint < epoint) {
  1064. p = ptr(curbp, npoint);
  1065. if(*p == '\n')
  1066. newlines++;
  1067. npoint++;
  1068. }
  1069. curbp->b_line -= newlines;
  1070. undoset(REPLACE, curbp->b_mark - point);
  1071. /* Cut the word to the left*/
  1072. curbp->b_mark = point;
  1073. curbp->b_point = point;
  1074. wright();
  1075. copy_cut(TRUE, FALSE, TRUE);
  1076. /* paste the left word */
  1077. right();
  1078. paste_internal(TRUE);
  1079. /* cut the right word */
  1080. curbp->b_mark = curbp->b_point;
  1081. wright();
  1082. copy_cut(TRUE, FALSE, TRUE);
  1083. wleft();
  1084. /* paste the right word */
  1085. left();
  1086. paste_internal(TRUE);
  1087. /* Put it all back together */
  1088. if (scrap != NULL) {
  1089. free(scrap);
  1090. scrap = NULL;
  1091. }
  1092. nscrap = n_scrap;
  1093. scrap = (char_t*) malloc(nscrap);
  1094. (void) memcpy(scrap, current_scrap, nscrap * sizeof (char_t));
  1095. curbp->b_mark = mark;
  1096. }
  1097. void lowercaseword()
  1098. {
  1099. char_t *p, *word;
  1100. char_t c[2];
  1101. point_t sword, eword;
  1102. int olast = lastcommand;
  1103. while ((isspace(*(p = ptr(curbp, curbp->b_point))) || is_symbol(*p)) && p < curbp->b_ebuf)
  1104. ++curbp->b_point;
  1105. sword = curbp->b_point;
  1106. wright();
  1107. eword = curbp->b_point;
  1108. word = (char_t *) malloc(sizeof(char_t)*(eword - sword));
  1109. curbp->b_point = sword;
  1110. lastcommand = KBD_DELETE_CHAR;
  1111. for(int i = sword, k = 0; i < eword; i++, k++) {
  1112. word[k] = *ptr(curbp, curbp->b_point);
  1113. delete();
  1114. }
  1115. lastcommand = olast;
  1116. for(int i = sword, k = 0; i < eword; i++, k++) {
  1117. c[0] = tolower(word[k]);
  1118. c[1] = '\0';
  1119. input = c;
  1120. undoset(INSERT, i != 0);
  1121. insert();
  1122. }
  1123. free(word);
  1124. }
  1125. void capitalizeword()
  1126. {
  1127. char_t *p;
  1128. while (isspace(*(p = ptr(curbp, curbp->b_point))) && p < curbp->b_ebuf)
  1129. ++curbp->b_point;
  1130. p = ptr(curbp, curbp->b_point);
  1131. char_t c[1];
  1132. c[0] = toupper(*p);
  1133. input = c;
  1134. delete();
  1135. undoset(INSERT, FALSE);
  1136. insert();
  1137. if(isspace(*(p = ptr(curbp, curbp->b_point+1))) || is_symbol(*p))
  1138. curbp->b_point++;
  1139. else
  1140. wright();
  1141. }
  1142. void uppercaseword()
  1143. {
  1144. char_t *p, *word;
  1145. char_t c[2];
  1146. point_t sword, eword;
  1147. int olast = lastcommand;
  1148. while ((isspace(*(p = ptr(curbp, curbp->b_point))) || is_symbol(*p)) && p < curbp->b_ebuf)
  1149. ++curbp->b_point;
  1150. sword = curbp->b_point;
  1151. wright();
  1152. eword = curbp->b_point;
  1153. word = (char_t *) malloc(sizeof(char_t)*(eword - sword));
  1154. curbp->b_point = sword;
  1155. lastcommand = KBD_DELETE_CHAR;
  1156. for(int i = sword, k = 0; i < eword; i++, k++) {
  1157. word[k] = *ptr(curbp, curbp->b_point);
  1158. delete();
  1159. }
  1160. lastcommand = olast;
  1161. for(int i = sword, k = 0; i < eword; i++, k++) {
  1162. c[0] = toupper(word[k]);
  1163. c[1] = '\0';
  1164. input = c;
  1165. undoset(INSERT, i != 0);
  1166. insert();
  1167. }
  1168. free(word);
  1169. }
  1170. /* type = 0, zap
  1171. type = 1, jump
  1172. */
  1173. /* TODO: Throw error when putting non-char in.
  1174. */
  1175. void gotochar(int type, int include_char)
  1176. {
  1177. char_t *p;
  1178. point_t opoint = curbp->b_point, eol;
  1179. int c, col = 0;
  1180. struct tb_event ev;
  1181. char *prompt = type == 0 ? "Zap to Char: " : "Jump to Char: ";
  1182. if(character[0] == '\0') {
  1183. display_prompt_and_response(prompt, character);
  1184. tb_present();
  1185. if(tb_poll_event(&ev) != TB_OK) return;
  1186. if(!ev.mod)
  1187. c = ev.ch;
  1188. else
  1189. c = ev.key;
  1190. /* Ignore all control keys other than C-g and ESC*/
  1191. if (c < 32 && c != TB_KEY_CTRL_G && c != TB_KEY_ESC)
  1192. return;
  1193. if(c == TB_KEY_CTRL_G || c == TB_KEY_ESC)
  1194. return;
  1195. else
  1196. character[0] = c;
  1197. display_prompt_and_response(prompt, character);
  1198. tb_present();
  1199. }
  1200. if(type == 0) {
  1201. block();
  1202. }
  1203. if(*ptr(curbp, curbp->b_point) == character[0] || curbp->b_point == 0) {
  1204. if(negated)
  1205. left();
  1206. else
  1207. right();
  1208. }
  1209. while (*(p = ptr(curbp, curbp->b_point + (include_char ? 0 : (negated ? -1 : 1)))) != character[0] &&
  1210. p < curbp->b_ebuf && curbp->b_point > 0) {
  1211. if(negated)
  1212. left();
  1213. else
  1214. right();
  1215. }
  1216. if(type == 0 && !negated)
  1217. right();
  1218. if(type == 0)
  1219. copy_cut(TRUE, FALSE, FALSE);
  1220. tb_set_cursor(0, MSGLINE);
  1221. clrtoeol("", MSGLINE);
  1222. eol = lnstart(curbp, curbp->b_point);
  1223. for(point_t poi = curbp->b_point; poi > eol; poi -= utf8_size(*ptr(curbp,poi)))
  1224. col++;
  1225. curbp->b_pcol = col + curwp->w_left;
  1226. if(p >= ptr(curbp, curbp->b_epage)) {
  1227. curbp->b_reframe = TRUE;
  1228. }
  1229. if((!negated && p >= curbp->b_ebuf) || (negated && curbp->b_point <= 0)) {
  1230. msg("No match found.");
  1231. curbp->b_point = opoint;
  1232. }
  1233. negated = FALSE;
  1234. }
  1235. void zaptochar()
  1236. {
  1237. gotochar(0, universal_argument == 0);
  1238. universal_argument = 0;
  1239. }
  1240. void negated_zaptochar()
  1241. {
  1242. negated = TRUE;
  1243. gotochar(0, universal_argument == 0);
  1244. universal_argument = 0;
  1245. }
  1246. void jumptochar()
  1247. {
  1248. shift_pmark(TRUE, NOMARK);
  1249. gotochar(1, TRUE);
  1250. }
  1251. void negated_jumptochar()
  1252. {
  1253. shift_pmark(TRUE, NOMARK);
  1254. negated = TRUE;
  1255. gotochar(1, TRUE);
  1256. }
  1257. void poptomark()
  1258. {
  1259. if(curbp->b_mark != NOMARK)
  1260. curbp->b_point = curbp->b_mark;
  1261. else if(curbp->b_pmark[0] != NOMARK)
  1262. curbp->b_point = shift_pmark(FALSE, NOMARK);
  1263. else {
  1264. msg("No valid mark to pop to.");
  1265. return;
  1266. }
  1267. if(curbp->b_point < curbp->b_page || curbp->b_point > curbp->b_epage)
  1268. curbp->b_reframe = TRUE;
  1269. }
  1270. void universal_argument_load()
  1271. {
  1272. universal_argument++;
  1273. msg("C-u %d", universal_argument);
  1274. }
  1275. void numeric_argument_load()
  1276. {
  1277. numeric_argument = (numeric_argument * 10) + atoi((const char *)&input_char);
  1278. msg("C-u %d", numeric_argument);
  1279. }
  1280. void back_to_indentation()
  1281. {
  1282. char_t *p;
  1283. while (isspace(*(p = ptr(curbp, curbp->b_point))) && p < curbp->b_ebuf)
  1284. ++curbp->b_point;
  1285. }
  1286. void negate()
  1287. {
  1288. negated = !negated;
  1289. msg("C-u -");
  1290. }
  1291. void forward_bracket()
  1292. {
  1293. point_t p, eol;
  1294. int col = 0;
  1295. if((p = find_matching_bracket(curbp, curwp, 1, FALSE)) >= 0)
  1296. curbp->b_point = curbp->b_mark == NOMARK ? p : p + 1;
  1297. /* Make sure the column memory updates to the new column */
  1298. eol = lnstart(curbp, curbp->b_point);
  1299. for(p = curbp->b_point; p > eol; p -= utf8_size(*ptr(curbp,p)))
  1300. col++;
  1301. curbp->b_pcol = col + curwp->w_left;
  1302. }
  1303. void backward_bracket()
  1304. {
  1305. point_t p, eol;
  1306. int col = 0;
  1307. if((p = find_matching_bracket(curbp, curwp, -1, FALSE)) >= 0) {
  1308. curbp->b_point = p;
  1309. if(curbp->b_mark != NOMARK)
  1310. curbp->b_mark++;
  1311. }
  1312. /* Make sure the column memory updates to the new column */
  1313. eol = lnstart(curbp, curbp->b_point);
  1314. for(p = curbp->b_point; p > eol; p -= utf8_size(*ptr(curbp,p)))
  1315. col++;
  1316. curbp->b_pcol = col + curwp->w_left;
  1317. }
  1318. void start_kbd_macro()
  1319. {
  1320. record_input = TRUE;
  1321. for(int i = 0; i < record_buffer_index; i++) {
  1322. memset(&record_buffer[i], 0, sizeof(record_buffer[i]));
  1323. }
  1324. record_buffer_index = 0;
  1325. msg("Started keyboard macro...");
  1326. }
  1327. void end_kbd_macro()
  1328. {
  1329. record_input = FALSE;
  1330. msg("Ended keyboard macro.");
  1331. }
  1332. void run_kbd_macro()
  1333. {
  1334. if(numeric_argument > 0)
  1335. numeric_argument--;
  1336. execute_kbd_macro = TRUE;
  1337. }
  1338. void open_file_from_shell()
  1339. {
  1340. get_popen_data(1);
  1341. }
  1342. void insert_from_shell()
  1343. {
  1344. get_popen_data(0);
  1345. }
  1346. void insert_control_char()
  1347. {
  1348. struct tb_event ev;
  1349. char *prompt = "Insert Control Char: ";
  1350. display_prompt_and_response(prompt, character);
  1351. tb_present();
  1352. if(tb_poll_event(&ev) != TB_OK) return;
  1353. tb_set_cursor(0, MSGLINE);
  1354. clrtoeol("", MSGLINE);
  1355. if(ev.key > 0x1a) {
  1356. return;
  1357. }
  1358. input[0] = (char)ev.key;
  1359. input[1] = '\0';
  1360. undoset(1, lastcommand == KBD_INSERT);
  1361. insert();
  1362. currentcommand = KBD_INSERT;
  1363. ignorenotbound = TRUE;
  1364. }