forked from emacs-mirror/emacs
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathtable.el
5462 lines (5189 loc) · 225 KB
/
table.el
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
;;; table.el --- create and edit WYSIWYG text based embedded tables -*- lexical-binding: t -*-
;; Copyright (C) 2000-2020 Free Software Foundation, Inc.
;; Keywords: wp, convenience
;; Author: Takaaki Ota <[email protected]>
;; Created: Sat Jul 08 2000 13:28:45 (PST)
;; This file is part of GNU Emacs.
;; GNU Emacs is free software: you can redistribute it and/or modify
;; it under the terms of the GNU General Public License as published by
;; the Free Software Foundation, either version 3 of the License, or
;; (at your option) any later version.
;; GNU Emacs is distributed in the hope that it will be useful,
;; but WITHOUT ANY WARRANTY; without even the implied warranty of
;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
;; GNU General Public License for more details.
;; You should have received a copy of the GNU General Public License
;; along with GNU Emacs. If not, see <https://www.gnu.org/licenses/>.
;;; Commentary:
;; -------------
;; Introduction:
;; -------------
;;
;; This package provides text based table creation and editing
;; feature. With this package Emacs is capable of editing tables that
;; are embedded inside a text document, the feature similar to the
;; ones seen in modern WYSIWYG word processors. A table is a
;; rectangular text area consisting from a surrounding frame and
;; content inside the frame. The content is usually subdivided into
;; multiple rectangular cells, see the actual tables used below in
;; this document. Once a table is recognized, editing operation
;; inside a table cell is confined into that specific cell's
;; rectangular area. This means that typing and deleting characters
;; inside a cell do not affect any outside text but introduces
;; appropriate formatting only to the cell contents. If necessary for
;; accommodating added text in the cell, the cell automatically grows
;; vertically and/or horizontally. The package uses no major mode nor
;; minor mode for its implementation because the subject text is
;; localized within a buffer. Therefore the special behaviors inside
;; a table cells are implemented by using keymap text property
;; instead of buffer wide mode-map.
;;
;;
;; -----------
;; Background:
;; -----------
;;
;; Paul Georgief is one of my best friends. He became an Emacs
;; convert after I recommended him trying it several years ago. Now
;; we both are devoted disciples of Emacsism and elisp cult. One day
;; in his Emacs exploration he asked me "Tak, what is a command to
;; edit tables in Emacs?". This question started my journey of this
;; table package development. May the code be with me! In the
;; software world Emacs is probably one of the longest lifetime record
;; holders. Amazingly there have been no direct support for WYSIWYG
;; table editing tasks in Emacs. Many people must have experienced
;; manipulating existing overwrite-mode and picture-mode for this task
;; and only dreamed of having such a lisp package which supports this
;; specific task directly. Certainly, I have been one of them. The
;; most difficult part of dealing with table editing in Emacs probably
;; is how to realize localized rectangular editing effect. Emacs has
;; no rectangular narrowing mechanism. Existing rect package provides
;; basically kill, delete and yank operations of a rectangle, which
;; internally is a mere list of strings. A simple approach for
;; realizing the localized virtual rectangular operation is combining
;; rect package capability with a temporary buffer. Insertion and
;; deletion of a character to a table cell can be trapped by a
;; function that copies the cell rectangle to a temporary buffer then
;; apply the insertion/deletion to the temporary contents. Then it
;; formats the contents by filling the paragraphs in order to fit it
;; into the original rectangular area and finally copy it back to the
;; original buffer. This simplistic approach has to bear with
;; significant performance hit. As cell grows larger the copying
;; rectangle back and forth between the original buffer and the
;; temporary buffer becomes expensive and unbearably slow. It was
;; completely impractical and an obvious failure. An idea has been
;; borrowed from the original Emacs design to overcome this
;; shortcoming. When the terminal screen update was slow and
;; expensive Emacs employed a clever algorithm to reduce actual screen
;; update by removing redundant redrawing operations. Also the actual
;; redrawing was done only when there was enough idling time. This
;; technique significantly improved the previously mentioned
;; undesirable situation. Now the original buffer's rectangle is
;; copied into a cache buffer only once. Any cell editing operation
;; is done only to the cache contents. When there is enough idling
;; time the original buffer's rectangle is updated with the current
;; cache contents. This delayed operation is implemented by using
;; Emacs's timer function. To reduce the visual awkwardness
;; introduced by the delayed effect the cursor location is updated in
;; real-time as a user types while the cell contents remains the same
;; until the next idling time. A key to the success of this approach
;; is how to maintain cache coherency. As a user moves point in and
;; out of a cell the table buffer contents and the cache buffer
;; contents must be synchronized without a mistake. By observing user
;; action carefully this is possible however not easy. Once this
;; mechanism is firmly implemented the rest of table features grew in
;; relatively painless progression. Those users who are familiar with
;; Emacs internals appreciate this table package more. Because it
;; demonstrates how extensible Emacs is by showing something that
;; appears like a magic. It lets you re-discover the potential of
;; Emacs.
;;
;;
;; -------------
;; Entry Points:
;; -------------
;;
;; If this is the first time for you to try this package, go ahead and
;; load the package by M-x `load-file' RET. Specify the package file
;; name "table.el". Then switch to a new test buffer and issue the
;; command M-x `table-insert' RET. It'll ask you number of columns,
;; number of rows, cell width and cell height. Give some small
;; numbers for each of them. Play with the resulted table for a
;; while. If you have menu system find the item "Table" under "Tools"
;; and "Table" in the menu bar when the point is in a table cell.
;; Some of them are pretty intuitive and you can easily guess what
;; they do. M-x `describe-function' and get the documentation of
;; `table-insert'. The document includes a short tutorial. When you
;; are tired of guessing how it works come back to this document
;; again.
;;
;; To use the package regularly place this file in the site library
;; directory and add the next expression in your init file. Make
;; sure that directory is included in the `load-path'.
;;
;; (require 'table)
;;
;; Have the next expression also, if you want always be ready to edit
;; tables inside text files. This mechanism is analogous to
;; fontification in a sense that tables are recognized at editing time
;; without having table information saved along with the text itself.
;;
;; (add-hook 'text-mode-hook 'table-recognize)
;;
;; Following is a table of entry points and brief description of each
;; of them. The tables below are of course generated and edited by
;; using this package. Not all the commands are bound to keys. Many
;; of them must be invoked by "M-x" (`execute-extended-command')
;; command. Refer to the section "Keymap" below for the commands
;; available from keys.
;;
;; +------------------------------------------------------------------+
;; | User Visible Entry Points |
;; +-------------------------------+----------------------------------+
;; | Function | Description |
;; +-------------------------------+----------------------------------+
;; |`table-insert' |Insert a table consisting of grid |
;; | |of cells by specifying the number |
;; | |of COLUMNS, number of ROWS, cell |
;; | |WIDTH and cell HEIGHT. |
;; +-------------------------------+----------------------------------+
;; |`table-insert-row' |Insert row(s) of cells before the |
;; | |current row that matches the |
;; | |current row structure. |
;; +-------------------------------+----------------------------------+
;; |`table-insert-column' |Insert column(s) of cells before |
;; | |the current column that matches |
;; | |the current column structure. |
;; +-------------------------------+----------------------------------+
;; |`table-delete-row' |Delete row(s) of cells. The row |
;; | |must consist from cells of the |
;; | |same height. |
;; +-------------------------------+----------------------------------+
;; |`table-delete-column' |Delete column(s) of cells. The |
;; | |column must consist from cells of |
;; | |the same width. |
;; +-------------------------------+----------------------------------+
;; |`table-recognize' |Recognize all tables in the |
;; |`table-unrecognize' |current buffer and |
;; | |activate/deactivate them. |
;; +-------------------------------+----------------------------------+
;; |`table-recognize-region' |Recognize all the cells in a |
;; |`table-unrecognize-region' |region and activate/deactivate |
;; | |them. |
;; +-------------------------------+----------------------------------+
;; |`table-recognize-table' |Recognize all the cells in a |
;; |`table-unrecognize-table' |single table and |
;; | |activate/deactivate them. |
;; +-------------------------------+----------------------------------+
;; |`table-recognize-cell' |Recognize a cell. Find a cell |
;; |`table-unrecognize-cell' |which contains the current point |
;; | |and activate/deactivate that cell.|
;; +-------------------------------+----------------------------------+
;; |`table-forward-cell' |Move point to the next Nth cell in|
;; | |a table. |
;; +-------------------------------+----------------------------------+
;; |`table-backward-cell' |Move point to the previous Nth |
;; | |cell in a table. |
;; +-------------------------------+----------------------------------+
;; |`table-span-cell' |Span the current cell toward the |
;; | |specified direction and merge it |
;; | |with the adjacent cell. The |
;; | |direction is right, left, above or|
;; | |below. |
;; +-------------------------------+----------------------------------+
;; |`table-split-cell-vertically' |Split the current cell vertically |
;; | |and create a cell above and a cell|
;; | |below the point location. |
;; +-------------------------------+----------------------------------+
;; |`table-split-cell-horizontally'|Split the current cell |
;; | |horizontally and create a cell on |
;; | |the left and a cell on the right |
;; | |of the point location. |
;; +-------------------------------+----------------------------------+
;; |`table-split-cell' |Split the current cell vertically |
;; | |or horizontally. This is a |
;; | |wrapper command to the other two |
;; | |orientation specific commands. |
;; +-------------------------------+----------------------------------+
;; |`table-heighten-cell' |Heighten the current cell. |
;; +-------------------------------+----------------------------------+
;; |`table-shorten-cell' |Shorten the current cell. |
;; +-------------------------------+----------------------------------+
;; |`table-widen-cell' |Widen the current cell. |
;; +-------------------------------+----------------------------------+
;; |`table-narrow-cell' |Narrow the current cell. |
;; +-------------------------------+----------------------------------+
;; |`table-fixed-width-mode' |Toggle fixed width mode. In the |
;; | |fixed width mode, typing inside a |
;; | |cell never changes the cell width,|
;; | |while in the normal mode the cell |
;; | |width expands automatically in |
;; | |order to prevent a word being |
;; | |folded into multiple lines. Fixed|
;; | |width mode reverses video or |
;; | |underline the cell contents for |
;; | |its indication. |
;; +-------------------------------+----------------------------------+
;; |`table-query-dimension' |Compute and report the current |
;; | |cell dimension, current table |
;; | |dimension and the number of |
;; | |columns and rows in the table. |
;; +-------------------------------+----------------------------------+
;; |`table-generate-source' |Generate the source of the current|
;; | |table in the specified language |
;; | |and insert it into a specified |
;; | |buffer. |
;; +-------------------------------+----------------------------------+
;; |`table-insert-sequence' |Travel cells forward while |
;; | |inserting a specified sequence |
;; | |string into each cell. |
;; +-------------------------------+----------------------------------+
;; |`table-capture' |Convert plain text into a table by|
;; | |capturing the text in the region. |
;; +-------------------------------+----------------------------------+
;; |`table-release' |Convert a table into plain text by|
;; | |removing the frame from a table. |
;; +-------------------------------+----------------------------------+
;; |`table-justify' |Justify the contents of cell(s). |
;; +-------------------------------+----------------------------------+
;;
;;
;; *Note*
;;
;; You may find that some of commonly expected table commands are
;; missing such as copying a row/column and yanking it. Those
;; functions can be obtained through existing Emacs text editing
;; commands. Rows are easily manipulated with region commands and
;; columns can be copied and pasted through rectangle commands. After
;; all a table is still a part of text in the buffer. Only the
;; special behaviors exist inside each cell through text properties.
;;
;; `table-generate-html' which appeared in earlier releases is
;; deprecated in favor of `table-generate-source'. Now HTML is
;; treated as one of the languages used for describing the table's
;; logical structure.
;;
;;
;; -------
;; Keymap:
;; -------
;;
;; Although this package does not use a mode it does use its own
;; keymap inside a table cell by way of keymap text property. Some of
;; the standard basic editing commands bound to certain keys are
;; replaced with the table specific version of corresponding commands.
;; This replacement combination is listed in the constant alist
;; `table-command-remap-alist' declared below. This alist is
;; not meant to be user configurable but mentioned here for your
;; better understanding of using this package. In addition, table
;; cells have some table specific bindings for cell navigation and
;; cell reformation. You can find these additional bindings in the
;; constant `table-cell-bindings'. Those key bound functions are
;; considered as internal functions instead of normal commands,
;; therefore they have special prefix, *table-- instead of table-, for
;; symbols. The purpose of this is to make it easier for a user to
;; use command name completion. There is a "normal hooks" variable
;; `table-cell-map-hook' prepared for users to override the default
;; table cell bindings. Following is the table of predefined default
;; key bound commands inside a table cell. Remember these bindings
;; exist only inside a table cell. When your terminal is a tty, the
;; control modifier may not be available or applicable for those
;; special characters. In this case use "C-cC-c", which is
;; customizable via `table-command-prefix', as the prefix key
;; sequence. This should preceding the following special character
;; without the control modifier. For example, use "C-cC-c|" instead
;; of "C-|".
;;
;; +------------------------------------------------------------------+
;; | Default Bindings in a Table Cell |
;; +-------+----------------------------------------------------------+
;; | Key | Function |
;; +-------+----------------------------------------------------------+
;; | TAB |Move point forward to the beginning of the next cell. |
;; +-------+----------------------------------------------------------+
;; | "C->" |Widen the current cell. |
;; +-------+----------------------------------------------------------+
;; | "C-<" |Narrow the current cell. |
;; +-------+----------------------------------------------------------+
;; | "C-}" |Heighten the current cell. |
;; +-------+----------------------------------------------------------+
;; | "C-{" |Shorten the current cell. |
;; +-------+----------------------------------------------------------+
;; | "C--" |Split current cell vertically. (one above and one below) |
;; +-------+----------------------------------------------------------+
;; | "C-|" |Split current cell horizontally. (one left and one right) |
;; +-------+----------------------------------------------------------+
;; | "C-*" |Span current cell into adjacent one. |
;; +-------+----------------------------------------------------------+
;; | "C-+" |Insert row(s)/column(s). |
;; +-------+----------------------------------------------------------+
;; | "C-!" |Toggle between normal mode and fixed width mode. |
;; +-------+----------------------------------------------------------+
;; | "C-#" |Report cell and table dimension. |
;; +-------+----------------------------------------------------------+
;; | "C-^" |Generate the source in a language from the current table. |
;; +-------+----------------------------------------------------------+
;; | "C-:" |Justify the contents of cell(s). |
;; +-------+----------------------------------------------------------+
;;
;; *Note*
;;
;; When using `table-cell-map-hook' do not use `local-set-key'.
;;
;; (add-hook 'table-cell-map-hook
;; (function (lambda ()
;; (local-set-key [<key sequence>] '<function>))))
;;
;; Adding the above to your init file is a common way to customize a
;; mode specific keymap. However it does not work for this package.
;; This is because there is no table mode in effect. This package
;; does not use a local map therefore you must modify `table-cell-map'
;; explicitly. The correct way of achieving above task is:
;;
;; (add-hook 'table-cell-map-hook
;; (function (lambda ()
;; (define-key table-cell-map [<key sequence>] '<function>))))
;;
;; -----
;; Menu:
;; -----
;;
;; If a menu system is available a group of table specific menu items,
;; "Table" under "Tools" section of the menu bar, is globally added
;; after this package is loaded. The commands in this group are
;; limited to the ones that are related to creation and initialization
;; of tables, such as to insert a table, to insert rows and columns,
;; or recognize and unrecognize tables. Once tables are created and
;; point is placed inside of a table cell a table specific menu item
;; "Table" appears directly on the menu bar. The commands in this
;; menu give full control on table manipulation that include cell
;; navigation, insertion, splitting, spanning, shrinking, expansion
;; and unrecognizing. In addition to above two types of menu there is
;; a pop-up menu available within a table cell. The content of pop-up
;; menu is identical to the full table menu. [mouse-3] is the default
;; button, defined in `table-cell-bindings', to bring up the pop-up
;; menu. It can be reconfigured via `table-cell-map-hook'. The
;; benefit of a pop-up menu is that it combines selection of the
;; location (which cell, where in the cell) and selection of the
;; desired operation into a single clicking action.
;;
;;
;; -------------------------------
;; Definition of tables and cells:
;; -------------------------------
;;
;; There is no artificial-intelligence magic in this package. The
;; definition of a table and the cells inside the table is reasonably
;; limited in order to achieve acceptable performance in the
;; interactive operation under Emacs lisp implementation. A valid
;; table is a rectangular text area completely filled with valid
;; cells. A valid cell is a rectangle text area, which four borders
;; consist of valid border characters. Cells can not be nested one to
;; another or overlapped to each other except sharing the border
;; lines. A valid character of a cell's vertical border is either
;; table-cell-vertical-char `|' or table-cell-intersection-char `+'.
;; A valid character of a cell's horizontal border is either
;; one of table-cell-horizontal-chars (`-' or `=')
;; or table-cell-intersection-char `+'.
;; A valid character of the four corners of a cell must be
;; table-cell-intersection-char `+'. A cell must contain at least one
;; character space inside. There is no restriction about the contents
;; of a table cell, however it is advised if possible to avoid using
;; any of the border characters inside a table cell. Normally a few
;; boarder characters inside a table cell are harmless. But it is
;; possible that they accidentally align up to emulate a bogus cell
;; corner on which software relies on for cell recognition. When this
;; happens the software may be fooled by it and fail to determine
;; correct cell dimension.
;;
;; Following are the examples of valid tables.
;;
;; +--+----+---+ +-+ +--+-----+
;; | | | | | | | | |
;; +--+----+---+ +-+ | +--+--+
;; | | | | | | | |
;; +--+----+---+ +--+--+ |
;; | | |
;; +-----+--+
;;
;; The next five tables are the examples of invalid tables. (From
;; left to right, 1. nested cells 2. overlapped cells and a
;; non-rectangle cell 3. non-rectangle table 4. zero width/height
;; cells 5. zero sized cell)
;;
;; +-----+ +-----+ +--+ +-++--+ ++
;; | | | | | | | || | ++
;; | +-+ | | | | | | || |
;; | | | | +--+ | +--+--+ +-++--+
;; | +-+ | | | | | | | +-++--+
;; | | | | | | | | | || |
;; +-----+ +--+--+ +--+--+ +-++--+
;;
;; Although the program may recognizes some of these invalid tables,
;; results from the subsequent editing operations inside those cells
;; are not predictable and will most likely start destroying the table
;; structures.
;;
;; It is strongly recommended to have at least one blank line above
;; and below a table. For a table to coexist peacefully with
;; surrounding environment table needs to be separated from unrelated
;; text. This is necessary for the left table to grow or shrink
;; horizontally without breaking the right table in the following
;; example.
;;
;; +-----+-----+-----+
;; +-----+-----+ | | | |
;; | | | +-----+-----+-----+
;; +-----+-----+ | | | |
;; +-----+-----+-----+
;;
;;
;; -------------------------
;; Cell contents formatting:
;; -------------------------
;;
;; The cell contents are formatted by filling a paragraph immediately
;; after characters are inserted into or deleted from a cell. Because
;; of this, cell contents always remain fit inside a cell neatly. One
;; drawback of this is that users do not have full control over
;; spacing between words and line breaking. Only one space can be
;; entered between words and up to two spaces between sentences. For
;; a newline to be effective the new line must form a beginning of
;; paragraph, otherwise it'll automatically be merged with the
;; previous line in a same paragraph. To form a new paragraph the
;; line must start with some space characters or immediately follow a
;; blank line. Here is a typical example of how to list items within
;; a cell. Without a space at the beginning of each line the items
;; can not stand on their own.
;;
;; +---------------------------------+
;; |Each one of the following three |
;; |items starts with a space |
;; |character thus forms a paragraph |
;; |of its own. Limitations in cell |
;; |contents formatting are: |
;; | |
;; | 1. Only one space between words.|
;; | 2. Up to two spaces between |
;; |sentences. |
;; | 3. A paragraph must start with |
;; |spaces or follow a blank line. |
;; | |
;; |This paragraph stays away from |
;; |the item 3 because there is a |
;; |blank line between them. |
;; +---------------------------------+
;;
;; In the normal operation table cell width grows automatically when
;; certain word has to be folded into the next line if the width had
;; not been increased. This normal operation is useful and
;; appropriate for most of the time, however, it is sometimes useful
;; or necessary to fix the width of table and width of table cells.
;; For this purpose the package provides fixed width mode. You can
;; toggle between fixed width mode and normal mode by "C-!".
;;
;; Here is a simple example of the fixed width mode. Suppose we have
;; a table like this one.
;;
;; +-----+
;; | |
;; +-----+
;;
;; In normal mode if you type a word "antidisestablishmentarianism" it
;; grows the cell horizontally like this.
;;
;; +----------------------------+
;; |antidisestablishmentarianism|
;; +----------------------------+
;;
;; In the fixed width mode the same action produces the following
;; result. The folded locations are indicated by a continuation
;; character (`\' is the default). The continuation character is
;; treated specially so it is recommended to choose a character that
;; does not appear elsewhere in table cells. This character is
;; configurable via customization and is kept in the variable
;; `table-word-continuation-char'. The continuation character is
;; treated specially only in the fixed width mode and has no special
;; meaning in the normal mode however.
;;
;; +-----+
;; |anti\|
;; |dise\|
;; |stab\|
;; |lish\|
;; |ment\|
;; |aria\|
;; |nism |
;; +-----+
;;
;;
;; -------------------
;; Cell Justification:
;; -------------------
;;
;; By default the cell contents are filled with left justification and
;; no vertical justification. A paragraph can be justified
;; individually but only horizontally. Paragraph justification is for
;; appearance only and does not change any structural information
;; while cell justification affects table's structural information.
;; For cell justification a user can select horizontal justification
;; and vertical justification independently. Horizontal justification
;; must be one of the three 'left, 'center or 'right. Vertical
;; justification can be 'top, 'middle, 'bottom or 'none. When a cell
;; is justified, that information is recorded as a part of text
;; property therefore the information is persistent as long as the
;; cell remains within the Emacs world. Even copying tables by region
;; and rectangle manipulation commands preserve this information.
;; However, once the table text is saved as a file and the buffer is
;; killed the justification information vanishes permanently. To
;; alleviate this shortcoming without forcing users to save and
;; maintain a separate attribute file, the table code detects
;; justification of each cell when recognizing a table. This
;; detection is done by guessing the justification by looking at the
;; appearance of the cell contents. Since it is a guessing work it
;; does not guarantee the perfectness but it is designed to be
;; practically good enough. The guessing algorithm is implemented in
;; the function `table--detect-cell-alignment'. If you have better
;; algorithm or idea any suggestion is welcome.
;;
;;
;; -----
;; Todo: (in the order of priority, some are just possibility)
;; -----
;;
;; Fix incompatibilities with input methods other than quail
;; Resolve conflict with flyspell
;; Use mouse for resizing cells
;; A mechanism to link cells internally
;; Consider the use of variable width font under Emacs 21
;; Consider the use of `:box' face attribute under Emacs 21
;; Consider the use of `modification-hooks' text property instead of
;; rebinding the keymap
;;
;;
;; ---------------
;; Acknowledgment:
;; ---------------
;;
;; Table would not have been possible without the help and
;; encouragement of the following spirited contributors.
;;
;; Paul Georgief <[email protected]> has been the best tester
;; of the code as well as the constructive criticizer.
;;
;; Gerd Moellmann <[email protected]> gave me useful suggestions from Emacs
;; 21 point of view.
;;
;; Richard Stallman <[email protected]> showed the initial interest in this
;; attempt of implementing the table feature to Emacs. This greatly
;; motivated me to follow through to its completion.
;;
;; Kenichi Handa <[email protected]> kindly guided me through to
;; overcome many technical issues while I was struggling with quail
;; related internationalization problems.
;;
;; Christoph Conrad <[email protected]> suggested making symbol
;; names consistent as well as fixing several bugs.
;;
;; Paul Lew <[email protected]> suggested implementing fixed width
;; mode as well as multi column width (row height) input interface.
;;
;; Michael Smith <[email protected]> a well-informed DocBook user
;; asked for CALS table source generation and helped me following
;; through the work by offering valuable suggestions and testing out
;; the code. Jorge Godoy <[email protected]> has also suggested
;; supporting for DocBook tables.
;;
;; And many other individuals who reported bugs and suggestions.
;;; Code:
(require 'regexp-opt)
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;;
;;; Compatibility:
;;;
;; hush up the byte-compiler
(defvar quail-translating)
(defvar quail-converting)
(defvar flyspell-mode)
(defvar real-last-command)
(defvar delete-selection-mode)
;; This is evil!!
;; (eval-when-compile
;; (unless (fboundp 'set-face-property)
;; (defun set-face-property (face prop value)))
;; (unless (fboundp 'unibyte-char-to-multibyte)
;; (defun unibyte-char-to-multibyte (char)))
;; (defun table--point-in-cell-p (&optional location)))
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;;
;;; Customization:
;;;
(defgroup table nil
"Text based table manipulation utilities."
:tag "Table"
:prefix "table-"
:group 'text
:version "22.1")
(defgroup table-hooks nil
"Hooks for table manipulation utilities."
:group 'table)
(defcustom table-time-before-update 0.2
"Time in seconds before updating the cell contents after typing.
Updating the cell contents on the screen takes place only after this
specified amount of time has passed after the last modification to the
cell contents. When the contents of a table cell changes repetitively
and frequently the updating the cell contents on the screen is
deferred until at least this specified amount of quiet time passes. A
smaller number wastes more computation resource by unnecessarily
frequent screen update. A large number presents noticeable and
annoying delay before the typed result start appearing on the screen."
:tag "Time Before Cell Update"
:type 'number
:group 'table)
(defcustom table-time-before-reformat 0.2
"Time in seconds before reformatting the table.
This many seconds must pass in addition to `table-time-before-update'
before the table is updated with newly widened width or heightened
height."
:tag "Time Before Cell Reformat"
:type 'number
:group 'table)
(defcustom table-command-prefix [(control c) (control c)]
"Key sequence to be used as prefix for table command key bindings."
:type '(vector (repeat :inline t sexp))
:tag "Table Command Prefix"
:group 'table)
(defface table-cell
'((((min-colors 88) (class color)) :foreground "gray90" :background "blue1")
(((class color)) :foreground "gray90" :background "blue")
(t :weight bold))
"Face used for table cell contents."
:tag "Cell Face"
:group 'table)
(defcustom table-cell-horizontal-chars "-="
"Characters that may be used for table cell's horizontal border line."
:tag "Cell Horizontal Boundary Characters"
:type 'string
:group 'table)
(defcustom table-cell-vertical-char ?\|
"Character that forms table cell's vertical border line."
:tag "Cell Vertical Boundary Character"
:type 'character
:group 'table)
(defcustom table-cell-intersection-char ?\+
"Character that forms table cell's corner."
:tag "Cell Intersection Character"
:type 'character
:group 'table)
(defcustom table-word-continuation-char ?\\
"Character that indicates word continuation into the next line.
This character has a special meaning only in the fixed width mode,
that is when `table-fixed-width-mode' is non-nil . In the fixed width
mode this character indicates that the location is continuing into the
next line. Be careful about the choice of this character. It is
treated substantially different manner than ordinary characters. Try
select a character that is unlikely to appear in your document."
:tag "Cell Word Continuation Character"
:type 'character
:group 'table)
(defcustom table-detect-cell-alignment t
"Detect cell contents alignment automatically.
When non-nil cell alignment is automatically determined by the
appearance of the current cell contents when recognizing tables as a
whole. This applies to `table-recognize', `table-recognize-region'
and `table-recognize-table' but not to `table-recognize-cell'."
:tag "Detect Cell Alignment"
:type 'boolean
:group 'table)
(defcustom table-dest-buffer-name "table"
"Default buffer name (without a suffix) for source generation."
:tag "Source Buffer Name"
:type 'string
:group 'table)
(defcustom table-html-delegate-spacing-to-user-agent nil
"Non-nil delegates cell contents spacing entirely to user agent.
Otherwise, when nil, it preserves the original spacing and line breaks."
:tag "HTML delegate spacing"
:type 'boolean
:group 'table)
(defcustom table-html-th-rows 0
"Number of top rows to become header cells automatically in HTML generation."
:tag "HTML Header Rows"
:type 'integer
:group 'table)
(defcustom table-html-th-columns 0
"Number of left columns to become header cells automatically in HTML generation."
:tag "HTML Header Columns"
:type 'integer
:group 'table)
(defcustom table-html-table-attribute "border=\"1\""
"Table attribute that applies to the table in HTML generation."
:tag "HTML table attribute"
:type 'string
:group 'table)
(defcustom table-html-cell-attribute ""
"Cell attribute that applies to all cells in HTML generation.
Do not specify \"align\" and \"valign\" because they are determined by
the cell contents dynamically."
:tag "HTML cell attribute"
:type 'string
:group 'table)
(defcustom table-cals-thead-rows 1
"Number of top rows to become header rows in CALS table."
:tag "CALS Header Rows"
:type 'integer
:group 'table)
(defcustom table-cell-map-hook nil
"Normal hooks run when finishing construction of `table-cell-map'.
User can modify `table-cell-map' by adding custom functions here."
:tag "Cell Keymap Hooks"
:type 'hook
:group 'table-hooks)
(defcustom table-disable-incompatibility-warning nil
"Disable compatibility warning notice.
When nil user is reminded of known incompatible issues."
:tag "Disable Incompatibility Warning"
:type 'boolean
:group 'table)
(defcustom table-abort-recognition-when-input-pending t
"Abort current recognition process when input pending.
Abort current recognition process when we are not sure that no input
is available. When non-nil lengthy recognition process is aborted
simply by any key input."
:tag "Abort Recognition When Input Pending"
:type 'boolean
:group 'table)
(defcustom table-load-hook nil
"List of functions to be called after the table is first loaded."
:type 'hook
:group 'table-hooks)
(make-obsolete-variable 'table-load-hook
"use `with-eval-after-load' instead." "28.1")
(defcustom table-point-entered-cell-hook nil
"List of functions to be called after point entered a table cell."
:type 'hook
:group 'table-hooks)
(defcustom table-point-left-cell-hook nil
"List of functions to be called after point left a table cell."
:type 'hook
:group 'table-hooks)
(defvar table-yank-handler '(nil nil t nil)
"Yank handler for tables.")
(setplist 'table-disable-incompatibility-warning nil)
(defvar table-disable-menu (null (and (locate-library "easymenu")
(require 'easymenu)
(fboundp 'easy-menu-add-item)))
"When non-nil, use of menu by table package is disabled.
It must be set before loading this package `table.el' for the first
time.")
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
;;;
;;; Implementation:
;;;
;;; Internal variables and constants
;;; No need of user configuration
(defconst table-paragraph-start "[ \t\n\f]"
"Regexp for beginning of a line that starts OR separates paragraphs.")
(defconst table-cache-buffer-name " *table cell cache*"
"Cell cache buffer name.")
(defvar table-cell-info-lu-coordinate nil
"Zero based coordinate of the cached cell's left upper corner.")
(defvar table-cell-info-rb-coordinate nil
"Zero based coordinate of the cached cell's right bottom corner.")
(defvar table-cell-info-width nil
"Number of characters per cached cell width.")
(defvar table-cell-info-height nil
"Number of lines per cached cell height.")
(defvar table-cell-info-justify nil
"Justification information of the cached cell.")
(defvar table-cell-info-valign nil
"Vertical alignment information of the cached cell.")
(defvar table-cell-self-insert-command-count 0
"Counter for undo control.")
(defvar table-cell-map nil
"Keymap for table cell contents.")
(defvar table-cell-global-map-alist nil
"Alist of copy of global maps that are substituted in `table-cell-map'.")
(defvar table-global-menu-map nil
"Menu map created via `easy-menu-define'.")
(defvar table-cell-menu-map nil
"Menu map created via `easy-menu-define'.")
(defvar table-cell-buffer nil
"Buffer that contains the table cell.")
(defvar table-cell-cache-point-coordinate nil
"Cache point coordinate based from the cell origin.")
(defvar table-cell-cache-mark-coordinate nil
"Cache mark coordinate based from the cell origin.")
(defvar table-update-timer nil
"Timer id for deferred cell update.")
(defvar table-widen-timer nil
"Timer id for deferred cell update.")
(defvar table-heighten-timer nil
"Timer id for deferred cell update.")
(defvar table-inhibit-update nil
"Non-nil inhibits implicit cell and cache updates.
It inhibits `table-with-cache-buffer' to update data in both direction, cell to cache and cache to cell.")
(defvar table-inhibit-auto-fill-paragraph nil
"Non-nil inhibits auto fill paragraph when `table-with-cache-buffer' exits.
This is always set to nil at the entry to `table-with-cache-buffer' before executing body forms.")
(defvar table-mode-indicator nil
"For mode line indicator")
;; This is not a real minor-mode but placed in the minor-mode-alist
;; so that we can show the indicator on the mode line handy.
(make-variable-buffer-local 'table-mode-indicator)
(unless (assq table-mode-indicator minor-mode-alist)
(push '(table-mode-indicator (table-fixed-width-mode " Fixed-Table" " Table"))
minor-mode-alist))
(defconst table-source-languages '(html latex cals wiki mediawiki)
"Supported source languages.")
(defvar table-source-info-plist nil
"General storage for temporary information used while generating source.")
;; The following history containers not only keep the history of user
;; entries but also serve as the default value providers. When an
;; interactive command is invoked it offers a user the latest entry
;; of the history as a default selection. Therefore the values below
;; are the first default value when a command is invoked for the very
;; first time when there is no real history existing yet.
(defvar table-cell-span-direction-history '("right"))
(defvar table-cell-split-orientation-history '("horizontally"))
(defvar table-cell-split-contents-to-history '("split"))
(defvar table-insert-row-column-history '("row"))
(defvar table-justify-history '("center"))
(defvar table-columns-history '("3"))
(defvar table-rows-history '("3"))
(defvar table-cell-width-history '("5"))
(defvar table-cell-height-history '("1"))
(defvar table-source-caption-history '("Table"))
(defvar table-sequence-string-history '("0"))
(defvar table-sequence-count-history '("0"))
(defvar table-sequence-increment-history '("1"))
(defvar table-sequence-interval-history '("1"))
(defvar table-sequence-justify-history '("left"))
(defvar table-source-language-history '("html"))
(defvar table-col-delim-regexp-history '(""))
(defvar table-row-delim-regexp-history '(""))
(defvar table-capture-justify-history '("left"))
(defvar table-capture-min-cell-width-history '("5"))
(defvar table-capture-columns-history '(""))
(defvar table-target-history '("cell"))
;; Some entries in `table-cell-bindings' are duplicated in
;; `table-command-remap-alist'. There is a good reason for
;; this. Common key like return key may be taken by some other
;; function than normal `newline' function. Thus binding return key
;; directly for `*table--cell-newline' ensures that the correct enter
;; operation in a table cell. However
;; `table-command-remap-alist' has an additional role than
;; replacing commands. It is also used to construct a table command
;; list. This list is very important because it is used to check if
;; the previous command was one of them in this list or not. If the
;; previous command is found in the list the current command will not
;; refill the table cache. If the command were not listed fast
;; typing can cause unwanted cache refill.
(defconst table-cell-bindings
'(([(control ?i)] . table-forward-cell)
([(control ?I)] . table-backward-cell)
([tab] . table-forward-cell)
([(shift backtab)] . table-backward-cell) ; for HPUX console keyboard
([(shift iso-lefttab)] . table-backward-cell) ; shift-tab on a microsoft natural keyboard and redhat linux
([(shift tab)] . table-backward-cell)
([backtab] . table-backward-cell) ; for terminals (e.g., xterm)
([return] . *table--cell-newline)
([(control ?m)] . *table--cell-newline)
([(control ?j)] . *table--cell-newline-and-indent)
([mouse-3] . *table--present-cell-popup-menu)
([(control ?>)] . table-widen-cell)
([(control ?<)] . table-narrow-cell)
([(control ?})] . table-heighten-cell)
([(control ?{)] . table-shorten-cell)
([(control ?-)] . table-split-cell-vertically)
([(control ?|)] . table-split-cell-horizontally)
([(control ?*)] . table-span-cell)
([(control ?+)] . table-insert-row-column)
([(control ?!)] . table-fixed-width-mode)
([(control ?#)] . table-query-dimension)
([(control ?^)] . table-generate-source)
([(control ?:)] . table-justify)
)
"Bindings for table cell commands.")
(defvar table-command-remap-alist
'((self-insert-command . *table--cell-self-insert-command)
(completion-separator-self-insert-autofilling . *table--cell-self-insert-command)
(completion-separator-self-insert-command . *table--cell-self-insert-command)
(delete-char . *table--cell-delete-char)
(delete-forward-char . *table--cell-delete-char)
(delete-backward-char . *table--cell-delete-backward-char)
(backward-delete-char . *table--cell-delete-backward-char)
(backward-delete-char-untabify . *table--cell-delete-backward-char)
(newline . *table--cell-newline)
(newline-and-indent . *table--cell-newline-and-indent)
(open-line . *table--cell-open-line)
(quoted-insert . *table--cell-quoted-insert)
(describe-mode . *table--cell-describe-mode)
(describe-bindings . *table--cell-describe-bindings)
(dabbrev-expand . *table--cell-dabbrev-expand)
(dabbrev-completion . *table--cell-dabbrev-completion))
"List of cons cells consisting of (ORIGINAL-COMMAND . TABLE-VERSION-OF-THE-COMMAND).")
(defvar table-command-list
;; Construct the real contents of the `table-command-list'.
(mapcar #'cdr table-command-remap-alist)
"List of commands that override original commands.")
(defconst table-global-menu
'("Table"
("Insert"
["a Table..." table-insert
:active (and (not buffer-read-only) (not (table--probe-cell)))
:help "Insert a text based table at point"]
["Row" table-insert-row
:active (table--row-column-insertion-point-p)
:help "Insert row(s) of cells in table"]
["Column" table-insert-column
:active (table--row-column-insertion-point-p 'column)
:help "Insert column(s) of cells in table"])
"----"
("Recognize"
["in Buffer" table-recognize
:active t
:help "Recognize all tables in the current buffer"]
["in Region" table-recognize-region
:active (and mark-active (not (eq (mark t) (point))))
:help "Recognize all tables in the current region"]