| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111 | ;;; test-org-table.el --- tests for org-table.el;; Copyright (c)  David Maus;; Authors: David Maus, Michael Brand;; This file is not part of GNU Emacs.;; This program 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.;; This program 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 this program.  If not, see <http://www.gnu.org/licenses/>.;;;; Comments:;; Template test file for Org tests.  Many tests are also a howto;; example collection as a user documentation, more or less all those;; using `org-test-table-target-expect'.  See also the doc string of;; `org-test-table-target-expect'.;;; Code:(require 'org-table)  ; `org-table-make-reference'(ert-deftest test-org-table/simple-formula/no-grouping/no-title-row ()  "Simple sum without grouping rows, without title row."  (org-test-table-target-expect   "|       2 ||       4 ||       8 || replace |"   "|  2 ||  4 ||  8 || 14 |"   1   ;; Calc formula   "#+TBLFM: @>$1 = vsum(@<..@>>)"   ;; Lisp formula   "#+TBLFM: @>$1 = '(+ @<..@>>); N"))(ert-deftest test-org-table/simple-formula/no-grouping/with-title-row ()  "Simple sum without grouping rows, with title row."  (org-test-table-target-expect   "|     foo ||---------||       2 ||       4 ||       8 || replace |"   "| foo ||-----||   2 ||   4 ||   8 ||  14 |"   1   ;; Calc formula   "#+TBLFM: @>$1 = vsum(@I..@>>)"   ;; Lisp formula   "#+TBLFM: @>$1 = '(+ @I..@>>); N"))(ert-deftest test-org-table/simple-formula/with-grouping/no-title-row ()  "Simple sum with grouping rows, how not to do."  ;; The first example has a problem, see the second example in this  ;; ert-deftest.  (org-test-table-target-expect   "|       2 ||       4 ||       8 ||---------|| replace |"   "|  2 ||  4 ||  8 ||----|| 14 |"   1   ;; Calc formula   "#+TBLFM: $1 = vsum(@<..@>>)"   ;; Lisp formula   "#+TBLFM: $1 = '(+ @<..@>>); N")  ;; The problem is that the first three rows with the summands are  ;; considered the header and therefore column formulas are not  ;; applied on them as shown below.  Also export behaves unexpected.  ;; See next ert-deftest how to group rows right.  (org-test-table-target-expect   "|       2 | header  ||       4 | header  ||       8 | header  ||---------+---------|| replace | replace |"   "|  2 | header ||  4 | header ||  8 | header ||----+--------|| 14 | 28     |"   2   ;; Calc formula   "#+TBLFM: @>$1 = vsum(@<..@>>) :: $2 = 2 * $1"   ;; Lisp formula   "#+TBLFM: @>$1 = '(+ @<..@>>); N :: $2 = '(* 2 $1); N"))(ert-deftest test-org-table/simple-formula/with-grouping/with-title-row ()  "Simple sum with grouping rows, how to do it right."  ;; Always add a top row with the column names separated by hline to  ;; get the desired header when you want to group rows.  (org-test-table-target-expect   "|     foo | bar     ||---------+---------||       2 | replace ||       4 | replace ||       8 | replace ||---------+---------|| replace | replace |"   "| foo | bar ||-----+-----||   2 |   4 ||   4 |   8 ||   8 |  16 ||-----+-----||  14 |  28 |"   2   ;; Calc formula   "#+TBLFM: @>$1 = vsum(@I..@>>) :: $2 = 2 * $1"   ;; Lisp formula   "#+TBLFM: @>$1 = '(+ @I..@>>); N :: $2 = '(* 2 $1); N"))(defconst references/target-normal "| 0 | 1 | replace | replace | replace | replace | replace | replace || z | 1 | replace | replace | replace | replace | replace | replace ||   | 1 | replace | replace | replace | replace | replace | replace ||   |   | replace | replace | replace | replace | replace | replace |"  "Normal numbers and non-numbers for Lisp and Calc formula.")(defconst references/target-special "|  nan | 1 | replace | replace | replace | replace | replace | replace || uinf | 1 | replace | replace | replace | replace | replace | replace || -inf | 1 | replace | replace | replace | replace | replace | replace ||  inf | 1 | replace | replace | replace | replace | replace | replace |"  "Special numbers for Calc formula.")(ert-deftest test-org-table/references/mode-string-EL ()  "Basic: Assign field reference, sum of field references, sumand len of simple range reference (no row) and complex rangereference (with row).  Mode string EL."  ;; Empty fields are kept during parsing field but lost as list  ;; elements within Lisp formula syntactically when used literally  ;; and not enclosed with " within fields, see last columns with len.  (org-test-table-target-expect   references/target-normal   ;; All the #ERROR show that for Lisp calculations N has to be used.   "| 0 | 1 | 0 |      1 |      1 |      1 | 2 | 2 || z | 1 | z | #ERROR | #ERROR | #ERROR | 2 | 2 ||   | 1 |   |      1 |      1 |      1 | 1 | 1 ||   |   |   |      0 |      0 |      0 | 0 | 0 |"   1 (concat      "#+TBLFM: $3 = '(identity \"$1\"); EL :: $4 = '(+ $1 $2); EL :: "      "$5 = '(+ $1..$2); EL :: $6 = '(+ @0$1..@0$2); EL :: "      "$7 = '(length '($1..$2)); EL :: $8 = '(length '(@0$1..@0$2)); EL"))  ;; Empty fields are kept during parsing field _and_ as list elements  ;; within Lisp formula syntactically even when used literally when  ;; enclosed with " within fields, see last columns with len.  (org-test-table-target-expect   "| \"0\" | \"1\" | repl | repl | repl | repl | repl | repl || \"z\" | \"1\" | repl | repl | repl | repl | repl | repl || \"\"  | \"1\" | repl | repl | repl | repl | repl | repl || \"\"  | \"\"  | repl | repl | repl | repl | repl | repl |"   "| \"0\" | \"1\" | \"0\" | 1 | #ERROR | #ERROR | 2 | 2 || \"z\" | \"1\" | \"z\" | 1 | #ERROR | #ERROR | 2 | 2 || \"\"  | \"1\" | \"\"  | 1 | #ERROR | #ERROR | 2 | 2 || \"\"  | \"\"  | \"\"  | 0 | #ERROR | #ERROR | 2 | 2 |"   1 (concat      "#+TBLFM: $3 = '(concat \"\\\"\" $1 \"\\\"\"); EL :: "      "$4 = '(+ (string-to-number $1) (string-to-number $2)); EL :: "      "$5 = '(+ $1..$2); EL :: $6 = '(+ @0$1..@0$2); EL :: "      "$7 = '(length '($1..$2)); EL :: $8 = '(length '(@0$1..@0$2)); EL")))(ert-deftest test-org-table/references/mode-string-E ()  "Basic: Assign field reference, sum of field references, sumand len of simple range reference (no row) and complex rangereference (with row).  Mode string E."  (let ((lisp	 (concat	  "#+TBLFM: $3 = '(identity $1); E :: $4 = '(+ $1 $2); E :: "	  "$5 = '(+ $1..$2); E :: $6 = '(+ @0$1..@0$2); E :: "	  "$7 = '(length '($1..$2)); E :: $8 = '(length '(@0$1..@0$2)); E"))	(calc	 (concat	  "#+TBLFM: $3 = $1; E :: $4 = $1 + $2; E :: "	  "$5 = vsum($1..$2); E :: $6 = vsum(@0$1..@0$2); E :: "	  "$7 = vlen($1..$2); E :: $8 = vlen(@0$1..@0$2); E")))    (org-test-table-target-expect     references/target-normal     ;; All the #ERROR show that for Lisp calculations N has to be used.     "| 0 | 1 | 0 | #ERROR | #ERROR | #ERROR | 2 | 2 || z | 1 | z | #ERROR | #ERROR | #ERROR | 2 | 2 ||   | 1 |   | #ERROR | #ERROR | #ERROR | 2 | 2 ||   |   |   | #ERROR | #ERROR | #ERROR | 2 | 2 |"     1 lisp)    (org-test-table-target-expect     references/target-normal     "| 0 | 1 |   0 |     1 |     1 |     1 | 2 | 2 || z | 1 |   z | z + 1 | z + 1 | z + 1 | 2 | 2 ||   | 1 | nan |   nan |   nan |   nan | 2 | 2 ||   |   | nan |   nan |   nan |   nan | 2 | 2 |"     1 calc)    (org-test-table-target-expect     references/target-special     "|  nan | 1 |  nan |  nan |  nan |  nan | 2 | 2 || uinf | 1 | uinf | uinf | uinf | uinf | 2 | 2 || -inf | 1 | -inf | -inf | -inf | -inf | 2 | 2 ||  inf | 1 |  inf |  inf |  inf |  inf | 2 | 2 |"     1 calc)))(ert-deftest test-org-table/references/mode-string-EN ()  "Basic: Assign field reference, sum of field references, sumand len of simple range reference (no row) and complex rangereference (with row).  Mode string EN."  (let ((lisp (concat	       "#+TBLFM: $3 = '(identity $1); EN :: $4 = '(+ $1 $2); EN :: "	       "$5 = '(+ $1..$2); EN :: $6 = '(+ @0$1..@0$2); EN :: "	       "$7 = '(length '($1..$2)); EN :: "	       "$8 = '(length '(@0$1..@0$2)); EN"))	(calc (concat	       "#+TBLFM: $3 = $1; EN :: $4 = $1 + $2; EN :: "	       "$5 = vsum($1..$2); EN :: $6 = vsum(@0$1..@0$2); EN :: "	       "$7 = vlen($1..$2); EN :: $8 = vlen(@0$1..@0$2); EN")))    (org-test-table-target-expect     references/target-normal     "| 0 | 1 | 0 | 1 | 1 | 1 | 2 | 2 || z | 1 | 0 | 1 | 1 | 1 | 2 | 2 ||   | 1 | 0 | 1 | 1 | 1 | 2 | 2 ||   |   | 0 | 0 | 0 | 0 | 2 | 2 |"     1 lisp calc)    (org-test-table-target-expect     references/target-special     "|  nan | 1 | 0 | 1 | 1 | 1 | 2 | 2 || uinf | 1 | 0 | 1 | 1 | 1 | 2 | 2 || -inf | 1 | 0 | 1 | 1 | 1 | 2 | 2 ||  inf | 1 | 0 | 1 | 1 | 1 | 2 | 2 |"     1 calc)))(ert-deftest test-org-table/references/mode-string-L ()  "Basic: Assign field reference, sum of field references, sumand len of simple range reference (no row) and complex rangereference (with row).  Mode string L."  (org-test-table-target-expect   references/target-normal   ;; All the #ERROR show that for Lisp calculations N has to be used.   "| 0 | 1 | 0 |      1 |      1 |      1 | 2 | 2 || z | 1 | z | #ERROR | #ERROR | #ERROR | 2 | 2 ||   | 1 |   |      1 |      1 |      1 | 1 | 1 ||   |   |   |      0 |      0 |      0 | 0 | 0 |"   1 (concat      "#+TBLFM: $3 = '(identity \"$1\"); L :: $4 = '(+ $1 $2); L :: "      "$5 = '(+ $1..$2); L :: $6 = '(+ @0$1..@0$2); L :: "      "$7 = '(length '($1..$2)); L :: $8 = '(length '(@0$1..@0$2)); L")))(ert-deftest test-org-table/references/mode-string-none ()  "Basic: Assign field reference, sum of field references, sumand len of simple range reference (no row) and complex rangereference (with row).  No mode string."  (let ((lisp (concat	       "#+TBLFM: $3 = '(identity $1) :: $4 = '(+ $1 $2) :: "	       "$5 = '(+ $1..$2) :: $6 = '(+ @0$1..@0$2) :: "	       "$7 = '(length '($1..$2)) :: $8 = '(length '(@0$1..@0$2))"))	(calc (concat	       "#+TBLFM: $3 = $1 :: $4 = $1 + $2 :: "	       "$5 = vsum($1..$2) :: $6 = vsum(@0$1..@0$2) :: "	       "$7 = vlen($1..$2) :: $8 = vlen(@0$1..@0$2)")))    (org-test-table-target-expect     references/target-normal     ;; All the #ERROR show that for Lisp calculations N has to be used.     "| 0 | 1 | 0 | #ERROR | #ERROR | #ERROR | 2 | 2 || z | 1 | z | #ERROR | #ERROR | #ERROR | 2 | 2 ||   | 1 |   | #ERROR | #ERROR | #ERROR | 1 | 1 ||   |   |   | #ERROR | 0      | 0      | 0 | 0 |"     1 lisp)    (org-test-table-target-expect     references/target-normal     "| 0 | 1 | 0 |     1 |     1 |     1 | 2 | 2 || z | 1 | z | z + 1 | z + 1 | z + 1 | 2 | 2 ||   | 1 | 0 |     1 |     1 |     1 | 1 | 1 ||   |   | 0 |     0 |     0 |     0 | 0 | 0 |"     1 calc)    (org-test-table-target-expect     references/target-special     "|  nan | 1 |  nan |  nan |  nan |  nan | 2 | 2 || uinf | 1 | uinf | uinf | uinf | uinf | 2 | 2 || -inf | 1 | -inf | -inf | -inf | -inf | 2 | 2 ||  inf | 1 |  inf |  inf |  inf |  inf | 2 | 2 |"     1 calc)))(ert-deftest test-org-table/references/mode-string-N ()  "Basic: Assign field reference, sum of field references, sumand len of simple range reference (no row) and complex rangereference (with row).  Mode string N."  (let ((lisp	 (concat	  "#+TBLFM: $3 = '(identity $1); N :: $4 = '(+ $1 $2); N :: "	  "$5 = '(+ $1..$2); N :: $6 = '(+ @0$1..@0$2); N :: "	  "$7 = '(length '($1..$2)); N :: $8 = '(length '(@0$1..@0$2)); N"))        (calc	 (concat	  "#+TBLFM: $3 = $1; N :: $4 = $1 + $2; N :: "	  "$5 = vsum($1..$2); N :: $6 = vsum(@0$1..@0$2); N :: "	  "$7 = vlen($1..$2); N :: $8 = vlen(@0$1..@0$2); N")))    (org-test-table-target-expect     references/target-normal     "| 0 | 1 | 0 | 1 | 1 | 1 | 2 | 2 || z | 1 | 0 | 1 | 1 | 1 | 2 | 2 ||   | 1 | 0 | 1 | 1 | 1 | 1 | 1 ||   |   | 0 | 0 | 0 | 0 | 0 | 0 |"     1 lisp calc)    (org-test-table-target-expect     references/target-special     "|  nan | 1 | 0 | 1 | 1 | 1 | 2 | 2 || uinf | 1 | 0 | 1 | 1 | 1 | 2 | 2 || -inf | 1 | 0 | 1 | 1 | 1 | 2 | 2 ||  inf | 1 | 0 | 1 | 1 | 1 | 2 | 2 |"     1 calc)))(ert-deftest test-org-table/lisp-return-value ()  "Basic: Return value of Lisp formulas."  (org-test-table-target-expect   "|                         | nil         | (list) | '() ||-------------------------+-------------+--------+-----|| type-of, no L           | replace (r) | r      | r   || type-of identity, no L  | r           | r      | r   || identity, no L          | r           | r      | r   ||-------------------------+-------------+--------+-----|| type-of \"@1\"            | r           | r      | r   || type-of (identity \"@1\") | r           | r      | r   || identity \"@1\"           | r           | r      | r   ||-------------------------+-------------+--------+-----|| type-of @1              | r           | r      | r   || type-of (identity @1)   | r           | r      | r   || identity @1             | r           | r      | r   |"   "|                         | nil    | (list) | '()    ||-------------------------+--------+--------+--------|| type-of, no L           | string | string | string || type-of identity, no L  | string | string | string || identity, no L          | nil    | (list) | '()    ||-------------------------+--------+--------+--------|| type-of \"@1\"            | string | string | string || type-of (identity \"@1\") | string | string | string || identity \"@1\"           | nil    | (list) | '()    ||-------------------------+--------+--------+--------|| type-of @1              | symbol | symbol | symbol || type-of (identity @1)   | symbol | symbol | symbol || identity @1             | nil    | nil    | nil    |"   1 (concat "#+TBLFM: @2$<<..@2$> = '(type-of @1) :: "	     "@3$<<..@3$> = '(type-of (identity @1)) :: "	     "@4$<<..@4$> = '(identity @1) :: @5$<<..@>$> = '(@0$1); L")))(ert-deftest test-org-table/compare ()  "Basic: Compare field references in Calc."  (org-test-table-target-expect   "|      | 0    | z    |      | nan  | uinf | -inf | inf  ||------+------+------+------+------+------+------+------||    0 | repl | repl | repl | repl | repl | repl | repl ||    z | repl | repl | repl | repl | repl | repl | repl ||      | repl | repl | repl | repl | repl | repl | repl ||  nan | repl | repl | repl | repl | repl | repl | repl || uinf | repl | repl | repl | repl | repl | repl | repl || -inf | repl | repl | repl | repl | repl | repl | repl ||  inf | repl | repl | repl | repl | repl | repl | repl |"   "|      | 0 | z |   | nan | uinf | -inf | inf ||------+---+---+---+-----+------+------+-----||    0 | x |   |   |     |      |      |     ||    z |   | x |   |     |      |      |     ||      |   |   | x |     |      |      |     ||  nan |   |   |   |   x |      |      |     || uinf |   |   |   |     |    x |      |     || -inf |   |   |   |     |      |    x |     ||  inf |   |   |   |     |      |      |   x |"   1   ;; Compare field reference ($1) with field reference (@1)   "#+TBLFM: @<<$<<..@>$> = if(\"$1\" == \"@1\", x, string(\"\")); E"   ;; Compare field reference ($1) with absolute term   (concat "#+TBLFM: "	   "$2 = if(\"$1\" == \"(0)\"   , x, string(\"\")); E :: "	   "$3 = if(\"$1\" == \"(z)\"   , x, string(\"\")); E :: "	   "$4 = if(\"$1\" == \"nan\"   , x, string(\"\")); E :: "	   "$5 = if(\"$1\" == \"(nan)\" , x, string(\"\")); E :: "	   "$6 = if(\"$1\" == \"(uinf)\", x, string(\"\")); E :: "	   "$7 = if(\"$1\" == \"(-inf)\", x, string(\"\")); E :: "	   "$8 = if(\"$1\" == \"(inf)\" , x, string(\"\")); E"))  ;; Check field reference converted from an empty field: Despite this  ;; field reference will not end up in a result, Calc evaluates it.  ;; Make sure that also then there is no Calc error.  (org-test-table-target-expect   "|   0 | replace ||   z | replace ||     | replace || nan | replace |"   "|   0 |     1 ||   z | z + 1 ||     |       || nan |   nan |"   1 "#+TBLFM: $2 = if(\"$1\" == \"nan\", string(\"\"), $1 + 1); E"))(ert-deftest test-org-table/empty-field ()  "Examples how to deal with empty fields."  ;; Test if one field is empty, else do a calculation  (org-test-table-target-expect   "| -1 | replace ||  0 | replace ||    | replace |"   "| -1 | 0 ||  0 | 1 ||    |   |"   1   ;; Calc formula   "#+TBLFM: $2 = if(\"$1\" == \"nan\", string(\"\"), $1 + 1); E"   ;; Lisp formula   "#+TBLFM: $2 = '(if (eq \"$1\" \"\") \"\" (1+ $1)); L")  ;; Test if several fields are empty, else do a calculation  (org-test-table-target-expect   "| 1 | 2 | replace || 4 |   | replace ||   | 8 | replace ||   |   | replace |"   "| 1 | 2 | 3 || 4 |   |   ||   | 8 |   ||   |   |   |"   1   ;; Calc formula   (concat "#+TBLFM: $3 = if(\"$1\" == \"nan\" || \"$2\" == \"nan\", "	   "string(\"\"), $1 + $2); E")   ;; Lisp formula   (concat "#+TBLFM: $3 = '(if (or (eq \"$1\" \"\") (eq \"$2\" \"\")) "	   "\"\" (+ $1 $2)); L"))  ;; $2: Use $1 + 0.5 if $1 available, else only reformat $2 if $2 available  (org-test-table-target-expect   "| 1.5 | 0 || 3.5 |   ||     | 5 ||     |   |"   "| 1.5 | 2.0 || 3.5 | 4.0 ||     | 5.0 ||     |     |"   1   ;; Calc formula   (concat "#+TBLFM: $2 = if(\"$1\" == \"nan\", "	   "if(\"$2\" == \"nan\", string(\"\"), $2 +.0), $1 + 0.5); E f-1")   ;; Lisp formula not implemented yet   )  ;; Empty fields in simple and complex range reference  (org-test-table-target-expect   "|   |   |   |   | repl | repl | repl | repl | repl | repl ||   |   | 5 | 7 | repl | repl | repl | repl | repl | repl || 1 | 3 | 5 | 7 | repl | repl | repl | repl | repl | repl |"   "|   |   |   |   |   |   |   |   | 0 | 0 ||   |   | 5 | 7 |   |   | 6 | 6 | 3 | 3 || 1 | 3 | 5 | 7 | 4 | 4 | 4 | 4 | 4 | 4 |"   1   ;; Calc formula   (concat    "#+TBLFM: "    "$5 = if(typeof(vmean($1..$4)) == 12, "    "string(\"\"), vmean($1..$4)); E :: "    "$6 = if(typeof(vmean(@0$1..@0$4)) == 12, "    "string(\"\"), vmean(@0$1..@0$4)); E :: "    "$7 = if(\"$1..$4\" == \"[]\", string(\"\"), vmean($1..$4)) :: "    "$8 = if(\"@0$1..@0$4\" == \"[]\", string(\"\"), vmean(@0$1..@0$4)) :: "    "$9 = vmean($1..$4); EN :: "    "$10 = vmean(@0$1..@0$4); EN")   ;; Lisp formula   (concat    "#+TBLFM: "    "$5 = '(let ((l '($1..$4))) (if (member \"\" l) \"\" "    "(/ (apply '+ (mapcar 'string-to-number l)) (length l)))); E :: "    "$6 = '(let ((l '(@0$1..@0$4))) (if (member \"\" l) \"\" "    "(/ (apply '+ (mapcar 'string-to-number l)) (length l)))); E :: "    "$7 = '(let ((l '($1..$4))) "    "(if l (/ (apply '+ l) (length l)) \"\")); N :: "    "$8 = '(let ((l '(@0$1..@0$4))) "    "(if l (/ (apply '+ l) (length l)) \"\")); N :: "    "$9 = '(/ (+ $1..$4) (length '($1..$4))); EN :: "    "$10 = '(/ (+ @0$1..@0$4) (length '(@0$1..@0$4))); EN")))(ert-deftest test-org-table/copy-field ()  "Experiments on how to copy one field into another field.See also `test-org-table/remote-reference-access'."  (let ((target "| 0                | replace || a b              | replace || c   d            | replace ||                  | replace || 2012-12          | replace || [2012-12-31 Mon] | replace |"))    ;; Lisp formula to copy literally    (org-test-table-target-expect     target     "| 0                | 0                || a b              | a b              || c   d            | c   d            ||                  |                  || 2012-12          | 2012-12          || [2012-12-31 Mon] | [2012-12-31 Mon] |"     1 "#+TBLFM: $2 = '(identity $1)")    ;; Calc formula to copy quite literally    (org-test-table-target-expect     target     "| 0                | 0                || a b              | a b              || c   d            | c   d            ||                  |                  || 2012-12          | 2012-12          || [2012-12-31 Mon] | [2012-12-31 Mon] |"     1 (concat "#+TBLFM: $2 = if(\"$1\" == \"nan\", "	       "string(\"\"), string(subvec(\"$1\", 2, vlen(\"$1\")))); E"))    ;; Calc formula simple    (org-test-table-target-expect     target     "| 0                | 0                || a b              | a b              || c   d            | c d              ||                  |                  || 2012-12          | 2000             || [2012-12-31 Mon] | [2012-12-31 Mon] |"     1 "#+TBLFM: $2 = if(\"$1\" == \"nan\", string(\"\"), $1); E")))(ert-deftest test-org-table/sub-total ()  "Grouped rows with sub-total.Begin range with \"@II\" to handle multiline header.  Convertinteger to float with \"+.0\" for sub-total of items c1 and c2.Sum empty fields as value zero but without ignoring them for\"vlen\" with format specifier \"EN\".  Format possibly emptyresults with the Calc formatter \"f-1\" instead of the printfformatter \"%.1f\"."  (org-test-table-target-expect   "|-------+---------+---------|| Item  |    Item | Sub-    || name  |   value | total   ||-------+---------+---------|| a1    |     4.1 | replace || a2    |     8.2 | replace || a3    |         | replace ||-------+---------+---------|| b1    |    16.0 | replace ||-------+---------+---------|| c1    |      32 | replace || c2    |      64 | replace ||-------+---------+---------|| Total | replace | replace ||-------+---------+---------|"   "|-------+-------+-------|| Item  |  Item |  Sub- || name  | value | total ||-------+-------+-------|| a1    |   4.1 |       || a2    |   8.2 |       || a3    |       |  12.3 ||-------+-------+-------|| b1    |  16.0 |  16.0 ||-------+-------+-------|| c1    |    32 |       || c2    |    64 |  96.0 ||-------+-------+-------|| Total | 124.3 |       ||-------+-------+-------|"   1 (concat "#+TBLFM: @>$2 = vsum(@II..@>>) ::"	     "$3 = if(vlen(@0..@+I) == 1, "	     "vsum(@-I$2..@+I$2) +.0, string(\"\")); EN f-1 :: "	     "@>$3 = string(\"\")")))(ert-deftest test-org-table/org-lookup-all ()  "Use `org-lookup-all' for several GROUP BY as in SQL and for ranking.See also URL `https://orgmode.org/worg/org-tutorials/org-lookups.html'."  (let ((data "#+NAME: data| Purchase | Product | Shop | Rating ||----------+---------+------+--------|| a        | p1      | s1   |      1 || b        | p1      | s2   |      4 || c        | p2      | s1   |      2 || d        | p3      | s2   |      8 |"))    ;; Product rating and ranking by average purchase from "#+NAME: data"    (org-test-table-target-expect     (concat data "| Product | Rating  | Ranking ||---------+---------+---------|| p1      | replace | replace || p2      | replace | replace || p3      | replace | replace |")     (concat data "| Product | Rating | Ranking ||---------+--------+---------|| p1      |    2.5 |       2 || p2      |    2.0 |       3 || p3      |    8.0 |       1 |")    2 (concat       "#+TBLFM: $2 = '(let ((all (org-lookup-all '$1 "       "'(remote(data, @I$2..@>$2)) '(remote(data, @I$4..@>$4))))) "       "(/ (apply '+ all) (length all) 1.0)); L :: "       "$3 = '(+ 1 (length (org-lookup-all $2 '(@I$2..@>$2) nil '<))); N"))    ;; Shop rating and ranking by average purchase from "#+NAME: data"    (org-test-table-target-expect     (concat data "| Shop | Rating  | Ranking ||------+---------+---------|| s1   | replace | replace || s2   | replace | replace |")     (concat data "| Shop | Rating | Ranking ||------+--------+---------|| s1   |    1.5 |       2 || s2   |    6.0 |       1 |")     2 (concat       "#+TBLFM: $2 = '(let ((all (org-lookup-all '$1 "       "'(remote(data, @I$3..@>$3)) '(remote(data, @I$4..@>$4))))) "       "(/ (apply '+ all) (length all) 1.0)); L :: "       "$3 = '(+ 1 (length (org-lookup-all $2 '(@I$2..@>$2) nil '<))); N"))))(ert-deftest test-org-table/org-table-make-reference/mode-string-EL ()  ;; For Lisp formula only  (should (equal "0" (org-table-make-reference "0" t nil 'literal)))  (should (equal "z" (org-table-make-reference "z" t nil 'literal)))  (should (equal "" (org-table-make-reference "" t nil 'literal)))  (should (equal "0 1" (org-table-make-reference '("0" "1") t nil 'literal)))  (should (equal "z 1" (org-table-make-reference '("z" "1") t nil 'literal)))  (should (equal " 1" (org-table-make-reference '("" "1") t nil 'literal)))  (should (equal " " (org-table-make-reference '("" "") t nil 'literal))))(ert-deftest test-org-table/org-table-make-reference/mode-string-E ()  ;; For Lisp formula  (should (equal "\"0\"" (org-table-make-reference "0" t nil t)))  (should (equal "\"z\"" (org-table-make-reference "z" t nil t)))  (should (equal"\"\"" (org-table-make-reference "" t nil t)))  (should (equal "\"0\" \"1\"" (org-table-make-reference '("0""1") t nil t)))  (should (equal "\"z\" \"1\"" (org-table-make-reference '("z""1") t nil t)))  (should (equal"\"\" \"1\"" (org-table-make-reference '("""1") t nil t)))  (should (equal"\"\" \"\""(org-table-make-reference '("""" ) t nil t)))  ;; For Calc formula  (should (equal "(0)" (org-table-make-reference "0" t nil nil)))  (should (equal "(z)" (org-table-make-reference "z" t nil nil)))  (should (equal "nan" (org-table-make-reference "" t nil nil)))  (should (equal "[0,1]" (org-table-make-reference '("0" "1") t nil nil)))  (should (equal "[z,1]" (org-table-make-reference '("z" "1") t nil nil)))  (should (equal "[nan,1]" (org-table-make-reference '("" "1") t nil nil)))  (should (equal "[nan,nan]" (org-table-make-reference '("" "") t nil nil)))  ;; For Calc formula, special numbers  (should (equal "(nan)" (org-table-make-reference "nan" t nil nil)))  (should (equal "(uinf)" (org-table-make-reference "uinf" t nil nil)))  (should (equal "(-inf)" (org-table-make-reference "-inf" t nil nil)))  (should (equal "(inf)" (org-table-make-reference "inf" t nil nil)))  (should (equal "[nan,1]" (org-table-make-reference '("nan" "1") t nil nil)))  (should (equal "[uinf,1]" (org-table-make-reference '("uinf" "1") t nil nil)))  (should (equal "[-inf,1]" (org-table-make-reference '("-inf" "1") t nil nil)))  (should (equal "[inf,1]" (org-table-make-reference '("inf" "1") t nil nil))))(ert-deftest test-org-table/org-table-make-reference/mode-string-EN ()  ;; For Lisp formula  (should (equal "0" (org-table-make-reference "0" t t t)))  (should (equal "0" (org-table-make-reference "z" t t t)))  (should (equal "0" (org-table-make-reference "" t t t)))  (should (equal "0 1" (org-table-make-reference '("0" "1") t t t)))  (should (equal "0 1" (org-table-make-reference '("z" "1") t t t)))  (should (equal "0 1" (org-table-make-reference '("" "1") t t t)))  (should (equal "0 0" (org-table-make-reference '("" "" ) t t t)))  ;; For Calc formula  (should (equal "(0)" (org-table-make-reference "0" t t nil)))  (should (equal "(0)" (org-table-make-reference "z" t t nil)))  (should (equal "(0)" (org-table-make-reference "" t t nil)))  (should (equal "[0,1]" (org-table-make-reference '("0" "1") t t nil)))  (should (equal "[0,1]" (org-table-make-reference '("z" "1") t t nil)))  (should (equal "[0,1]" (org-table-make-reference '("" "1") t t nil)))  (should (equal "[0,0]" (org-table-make-reference '("" "" ) t t nil)))  ;; For Calc formula, special numbers  (should (equal "(0)" (org-table-make-reference "nan" t t nil)))  (should (equal "(0)" (org-table-make-reference "uinf" t t nil)))  (should (equal "(0)" (org-table-make-reference "-inf" t t nil)))  (should (equal "(0)" (org-table-make-reference "inf" t t nil)))  (should (equal "[0,1]" (org-table-make-reference '( "nan" "1") t t nil)))  (should (equal "[0,1]" (org-table-make-reference '("uinf" "1") t t nil)))  (should (equal "[0,1]" (org-table-make-reference '("-inf" "1") t t nil)))  (should (equal "[0,1]" (org-table-make-reference '( "inf" "1") t t nil))))(ert-deftest test-org-table/org-table-make-reference/mode-string-L ()  ;; For Lisp formula only  (should (equal "0" (org-table-make-reference "0" nil nil 'literal)))  (should (equal "z" (org-table-make-reference "z" nil nil 'literal)))  (should (equal "" (org-table-make-reference "" nil nil 'literal)))  (should (equal "0 1" (org-table-make-reference '("0" "1") nil nil 'literal)))  (should (equal "z 1" (org-table-make-reference '("z" "1") nil nil 'literal)))  (should (equal "1" (org-table-make-reference '("" "1") nil nil 'literal)))  (should (equal "" (org-table-make-reference '("" "" ) nil nil 'literal))))(ert-deftest test-org-table/org-table-make-reference/mode-string-none ()  ;; For Lisp formula  (should (equal "\"0\"" (org-table-make-reference "0" nil nil t)))  (should (equal "\"z\"" (org-table-make-reference "z" nil nil t)))  (should (equal "" (org-table-make-reference "" nil nil t)))  (should (equal "\"0\" \"1\"" (org-table-make-reference '("0" "1") nil nil t)))  (should (equal "\"z\" \"1\"" (org-table-make-reference '("z" "1") nil nil t)))  (should (equal "\"1\"" (org-table-make-reference '("" "1") nil nil t)))  (should (equal "" (org-table-make-reference '("" "" ) nil nil t)))  ;; For Calc formula  (should (equal "(0)" (org-table-make-reference "0" nil nil nil)))  (should (equal "(z)" (org-table-make-reference "z" nil nil nil)))  (should (equal "(0)" (org-table-make-reference "" nil nil nil)))  (should (equal "[0,1]" (org-table-make-reference '("0" "1") nil nil nil)))  (should (equal "[z,1]" (org-table-make-reference '("z" "1") nil nil nil)))  (should (equal "[1]" (org-table-make-reference '("" "1") nil nil nil)))  (should (equal "[]" (org-table-make-reference '("" "" ) nil nil nil)))  ;; For Calc formula, special numbers  (should (equal "(nan)" (org-table-make-reference "nan" nil nil nil)))  (should (equal "(uinf)" (org-table-make-reference "uinf" nil nil nil)))  (should (equal "(-inf)" (org-table-make-reference "-inf" nil nil nil)))  (should (equal "(inf)" (org-table-make-reference "inf" nil nil nil)))  (should (equal "[nan,1]" (org-table-make-reference '( "nan" "1") nil nil nil)))  (should (equal "[uinf,1]" (org-table-make-reference '("uinf" "1") nil nil nil)))  (should (equal "[-inf,1]" (org-table-make-reference '("-inf" "1") nil nil nil)))  (should (equal "[inf,1]" (org-table-make-reference '( "inf" "1") nil nil nil))))(ert-deftest test-org-table/org-table-make-reference/mode-string-N ()  ;; For Lisp formula  (should (equal "0" (org-table-make-reference "0" nil t t)))  (should (equal "0" (org-table-make-reference "z" nil t t)))  (should (equal "" (org-table-make-reference "" nil t t)))  (should (equal "0 1" (org-table-make-reference '("0" "1") nil t t)))  (should (equal "0 1" (org-table-make-reference '("z" "1") nil t t)))  (should (equal "1" (org-table-make-reference '("" "1") nil t t)))  (should (equal "" (org-table-make-reference '("" "" ) nil t t)))  ;; For Calc formula  (should (equal "(0)" (org-table-make-reference "0" nil t nil)))  (should (equal "(0)" (org-table-make-reference "z" nil t nil)))  (should (equal "(0)" (org-table-make-reference "" nil t nil)))  (should (equal "[0,1]" (org-table-make-reference '("0" "1") nil t nil)))  (should (equal "[0,1]" (org-table-make-reference '("z" "1") nil t nil)))  (should (equal "[1]" (org-table-make-reference '("" "1") nil t nil)))  (should (equal "[]" (org-table-make-reference '("" "" ) nil t nil)))  ;; For Calc formula, special numbers  (should (equal "(0)" (org-table-make-reference "nan" nil t nil)))  (should (equal "(0)" (org-table-make-reference "uinf" nil t nil)))  (should (equal "(0)" (org-table-make-reference "-inf" nil t nil)))  (should (equal "(0)" (org-table-make-reference "inf" nil t nil)))  (should (equal "[0,1]" (org-table-make-reference '( "nan" "1") nil t nil)))  (should (equal "[0,1]" (org-table-make-reference '("uinf" "1") nil t nil)))  (should (equal "[0,1]" (org-table-make-reference '("-inf" "1") nil t nil)))  (should (equal "[0,1]" (org-table-make-reference '( "inf" "1") nil t nil))))(ert-deftest test-org-table/org-table-convert-refs-to-an/1 ()  "Simple reference @2$1."  (should   (string= "A2" (org-table-convert-refs-to-an "@2$1"))));; TODO: Test broken;; (ert-deftest test-org-table/org-table-convert-refs-to-an/2 ();;   "Self reference @1$1.";;   (should;;    (string= "A1 = $0" (org-table-convert-refs-to-an "@1$1 = $0"))))(ert-deftest test-org-table/org-table-convert-refs-to-an/3 ()  "Remote reference."  (should   (string= "C& = remote(FOO, @@#B&)" (org-table-convert-refs-to-an "$3 = remote(FOO, @@#$2)"))))(ert-deftest test-org-table/org-table-convert-refs-to-rc/1 ()  "Simple reference @2$1."  (should   (string= "@2$1" (org-table-convert-refs-to-rc "A2"))))(ert-deftest test-org-table/org-table-convert-refs-to-rc/2 ()  "Self reference $0."  (should   (string= "@1$1 = $0" (org-table-convert-refs-to-rc "A1 = $0"))));; TODO: Test Broken;; (ert-deftest test-org-table/org-table-convert-refs-to-rc/3 ();;   "Remote reference.";;   (should;;    (string= "$3 = remote(FOO, @@#$2)" (org-table-convert-refs-to-rc "C& = remote(FOO, @@#B&)"))))(ert-deftest test-org-table/remote-reference-access ()  "Access to remote reference.See also `test-org-table/copy-field'."  (org-test-table-target-expect   "#+NAME: table|   | x   42 |   || replace | replace |"   "#+NAME: table|   | x   42 |   || x   42 | 84 x |"   1 (concat "#+TBLFM: "	     ;; Copy text without calculation: Use Lisp formula	     "$1 = '(identity remote(table, @1$2)) :: "	     ;; Do a calculation: Use Calc (or Lisp ) formula	     "$2 = 2 * remote(table, @1$2)")))(ert-deftest test-org-table/remote-reference-indirect ()  "Access to remote reference with indirection of name or ID."  (let ((source-tables "#+NAME: 2012| amount ||--------||      1 ||      2 ||--------||      3 |#+TBLFM: @>$1 = vsum(@I..@II)#+NAME: 2013| amount ||--------||      4 ||      8 ||--------||     12 |#+TBLFM: @>$1 = vsum(@I..@II)"))    ;; Read several remote references from same column    (org-test-table-target-expect     (concat source-tables "#+NAME: summary|  year | amount  ||-------+---------||  2012 | replace ||  2013 | replace ||-------+---------|| total | replace |")     (concat source-tables "#+NAME: summary|  year | amount ||-------+--------||  2012 |      3 ||  2013 |     12 ||-------+--------|| total |     15 |")     1     ;; Calc formula     "#+TBLFM: @<<$2..@>>$2 = remote($<, @>$1) :: @>$2 = vsum(@I..@II)"     ;; Lisp formula     (concat "#+TBLFM: @<<$2..@>>$2 = '(identity remote($<, @>$1)); N :: "	     "@>$2 = '(+ @I..@II); N"))    ;; Read several remote references from same row    (org-test-table-target-expect     (concat source-tables "#+NAME: summary| year   |    2012 |    2013 | total   ||--------+---------+---------+---------|| amount | replace | replace | replace |")     (concat source-tables "#+NAME: summary| year   | 2012 | 2013 | total ||--------+------+------+-------|| amount |    3 |   12 |    15 |")     1     ;; Calc formula     "#+TBLFM: @2$<<..@2$>> = remote(@<, @>$1) :: @2$> = vsum($<<..$>>)"     ;; Lisp formula     (concat "#+TBLFM: @2$<<..@2$>> = '(identity remote(@<, @>$1)); N :: "	     "@2$> = '(+ $<<..$>>); N"))))(ert-deftest test-org-table/org-at-TBLFM-p ()  (org-test-with-temp-text-in-file      "| 1 || 2 |#+TBLFM: $2=$1*2"    (goto-char (point-min))    (forward-line 2)    (should (equal (org-at-TBLFM-p) nil))    (goto-char (point-min))    (forward-line 3)    (should (equal (org-at-TBLFM-p) t))    (goto-char (point-min))    (forward-line 4)    (should (equal (org-at-TBLFM-p) nil))))(ert-deftest test-org-table/org-table-TBLFM-begin ()  (org-test-with-temp-text-in-file      "| 1 || 2 |#+TBLFM: $2=$1*2"    (goto-char (point-min))    (should (equal (org-table-TBLFM-begin)		   nil))    (goto-char (point-min))    (forward-line 1)    (should (equal (org-table-TBLFM-begin)		   nil))    (goto-char (point-min))    (forward-line 3)    (should (= (org-table-TBLFM-begin)		   14))    (goto-char (point-min))    (forward-line 4)    (should (= (org-table-TBLFM-begin)		   14))    ))(ert-deftest test-org-table/org-table-TBLFM-begin-for-multiple-TBLFM-lines ()  "For multiple #+TBLFM lines."  (org-test-with-temp-text-in-file      "| 1 || 2 |#+TBLFM: $2=$1*1#+TBLFM: $2=$1*2"    (goto-char (point-min))    (should (equal (org-table-TBLFM-begin)		   nil))    (goto-char (point-min))    (forward-line 1)    (should (equal (org-table-TBLFM-begin)		   nil))    (goto-char (point-min))    (forward-line 3)    (should (= (org-table-TBLFM-begin)		   14))    (goto-char (point-min))    (forward-line 4)    (should (= (org-table-TBLFM-begin)		   14))    (goto-char (point-min))    (forward-line 5)    (should (= (org-table-TBLFM-begin)		   14))    ))(ert-deftest test-org-table/org-table-TBLFM-begin-for-pultiple-TBLFM-lines-blocks ()  (org-test-with-temp-text-in-file      "| 1 || 2 |#+TBLFM: $2=$1*1#+TBLFM: $2=$1*2| 6 || 7 |#+TBLFM: $2=$1*1#+TBLFM: $2=$1*2"    (goto-char (point-min))    (should (equal (org-table-TBLFM-begin)		   nil))    (goto-char (point-min))    (forward-line 1)    (should (equal (org-table-TBLFM-begin)		   nil))    (goto-char (point-min))    (forward-line 3)    (should (= (org-table-TBLFM-begin)		   14))    (goto-char (point-min))    (forward-line 4)    (should (= (org-table-TBLFM-begin)		   14))    (goto-char (point-min))    (forward-line 5)    (should (= (org-table-TBLFM-begin)		   14))    (goto-char (point-min))    (forward-line 6)    (should (= (org-table-TBLFM-begin)		   14))    (goto-char (point-min))    (forward-line 8)    (should (= (org-table-TBLFM-begin)		   61))    (goto-char (point-min))    (forward-line 9)    (should (= (org-table-TBLFM-begin)		   61))    (goto-char (point-min))    (forward-line 10)    (should (= (org-table-TBLFM-begin)		   61))))(ert-deftest test-org-table/org-table-calc-current-TBLFM ()    (org-test-with-temp-text-in-file      "| 1 |   || 2 |   |#+TBLFM: $2=$1*1#+TBLFM: $2=$1*2#+TBLFM: $2=$1*3"    (let ((got (progn (goto-char (point-min))		      (forward-line 3)		      (org-table-calc-current-TBLFM)		      (buffer-string)))	  (expect "| 1 | 1 || 2 | 2 |#+TBLFM: $2=$1*1#+TBLFM: $2=$1*2#+TBLFM: $2=$1*3"))      (should (string= got		       expect)))    (let ((got (progn (goto-char (point-min))		      (forward-line 4)		      (org-table-calc-current-TBLFM)		      (buffer-string)))	  (expect "| 1 | 2 || 2 | 4 |#+TBLFM: $2=$1*1#+TBLFM: $2=$1*2#+TBLFM: $2=$1*3"))      (should (string= got		       expect)))))(ert-deftest test-org-table/org-table-calc-current-TBLFM-when-stop-because-of-error ()  "org-table-calc-current-TBLFM should preserve the input as it was."  (org-test-with-temp-text-in-file      "| 1 | 1 || 2 | 2 |#+TBLFM: $2=$1*1#+TBLFM: $2=$1*2::$2=$1*2#+TBLFM: $2=$1*3"    (let ((expect "| 1 | 1 || 2 | 2 |#+TBLFM: $2=$1*1#+TBLFM: $2=$1*2::$2=$1*2#+TBLFM: $2=$1*3"))      (goto-char (point-min))      (forward-line 4)      (should-error (org-table-calc-current-TBLFM))      (setq got (buffer-string))      (message "%s" got)      (should (string= got		       expect)))));;; Radio Tables(ert-deftest test-org-table/to-generic ()  "Test `orgtbl-to-generic' specifications."  ;; Test :hline parameter.  (should   (equal "a\nb"	  (orgtbl-to-generic (org-table-to-lisp "| a |\n|---|\n| b |")			     '(:hline nil))))  (should   (equal "a\n~\nb"	  (orgtbl-to-generic (org-table-to-lisp "| a |\n|---|\n| b |")			     '(:hline "~"))))  ;; Test :sep parameter.  (should   (equal "a!b\nc!d"	  (orgtbl-to-generic	   (org-table-to-lisp "| a | b |\n|---+---|\n| c | d |")	   '(:sep "!"))))  ;; Test :hsep parameter.  (should   (equal "a!b\nc?d"	  (orgtbl-to-generic	   (org-table-to-lisp "| a | b |\n|---+---|\n| c | d |")	   '(:sep "?" :hsep "!"))))  ;; Test :tstart parameter.  (should   (equal "<begin>\na"	  (orgtbl-to-generic (org-table-to-lisp "| a |") '(:tstart "<begin>"))))  (should   (equal "<begin>\na"	  (orgtbl-to-generic (org-table-to-lisp "| a |")			     '(:tstart (lambda () "<begin>")))))  (should   (equal "a"	  (orgtbl-to-generic (org-table-to-lisp "| a |")			     '(:tstart "<begin>" :splice t))))  ;; Test :tend parameter.  (should   (equal "a\n<end>"	  (orgtbl-to-generic (org-table-to-lisp "| a |") '(:tend "<end>"))))  (should   (equal "a\n<end>"	  (orgtbl-to-generic (org-table-to-lisp "| a |")			     '(:tend (lambda () "<end>")))))  (should   (equal "a"	  (orgtbl-to-generic (org-table-to-lisp "| a |")			     '(:tend "<end>" :splice t))))  ;; Test :lstart parameter.  (should   (equal "> a"	  (orgtbl-to-generic	   (org-table-to-lisp "| a |") '(:lstart "> "))))  (should   (equal "> a"	  (orgtbl-to-generic (org-table-to-lisp "| a |")			     '(:lstart (lambda () "> ")))))  ;; Test :llstart parameter.  (should   (equal "> a\n>> b"	  (orgtbl-to-generic (org-table-to-lisp "| a |\n|---|\n| b |")			     '(:lstart "> " :llstart ">> "))))  ;; Test :hlstart parameter.  (should   (equal "!> a\n> b"	  (orgtbl-to-generic (org-table-to-lisp "| a |\n|---|\n| b |")			     '(:lstart "> " :hlstart "!> "))))  ;; Test :hllstart parameter.  (should   (equal "!> a\n!!> b\n> c"	  (orgtbl-to-generic (org-table-to-lisp "| a |\n| b |\n|---|\n| c |")			     '(:lstart "> " :hlstart "!> " :hllstart "!!> "))))  ;; Test :lend parameter.  (should   (equal "a <"	  (orgtbl-to-generic (org-table-to-lisp "| a |") '(:lend " <"))))  ;; Test :llend parameter.  (should   (equal "a <\nb <<"	  (orgtbl-to-generic (org-table-to-lisp "| a |\n|---|\n| b |")			     '(:lend " <" :llend " <<"))))  ;; Test :hlend parameter.  (should   (equal "a <!\nb <"	  (orgtbl-to-generic (org-table-to-lisp "| a |\n|---|\n| b |")			     '(:lend " <" :hlend " <!"))))  ;; Test :hllend parameter.  (should   (equal "a <!\nb <!!\nc <"	  (orgtbl-to-generic (org-table-to-lisp "| a |\n| b |\n|---|\n| c |")			     '(:lend " <" :hlend " <!" :hllend " <!!"))))  ;; Test :lfmt parameter.  (should   (equal "a!b"	  (orgtbl-to-generic (org-table-to-lisp "| a | b |")			     '(:lfmt "%s!%s"))))  (should   (equal "a+b"	  (orgtbl-to-generic	   (org-table-to-lisp "| a | b |")	   '(:lfmt (lambda (c) (concat (car c) "+" (cadr c)))))))  (should   (equal "a!b"	  (orgtbl-to-generic (org-table-to-lisp "| a | b |")			     '(:lfmt "%s!%s" :lstart ">" :lend "<" :sep " "))))  ;; Test :llfmt parameter.  (should   (equal "a!b"	  (orgtbl-to-generic (org-table-to-lisp "| a | b |")			     '(:llfmt "%s!%s"))))  (should   (equal "a!b\nc+d"	  (orgtbl-to-generic	   (org-table-to-lisp "| a | b |\n| c | d |")	   '(:lfmt "%s!%s" :llfmt (lambda (c) (concat (car c) "+" (cadr c)))))))  (should   (equal "a!b"	  (orgtbl-to-generic (org-table-to-lisp "| a | b |")			     '(:llfmt "%s!%s" :lstart ">" :lend "<" :sep " "))))  ;; Test :hlfmt parameter.  (should   (equal "a!b\ncd"	  (orgtbl-to-generic	   (org-table-to-lisp "| a | b |\n|---+---|\n| c | d |")	   '(:hlfmt "%s!%s"))))  (should   (equal "a+b\ncd"	  (orgtbl-to-generic	   (org-table-to-lisp "| a | b |\n|---+---|\n| c | d |")	   '(:hlfmt (lambda (c) (concat (car c) "+" (cadr c)))))))  (should   (equal "a!b\n>c d<"	  (orgtbl-to-generic	   (org-table-to-lisp "| a | b |\n|---+---|\n| c | d |")	   '(:hlfmt "%s!%s" :lstart ">" :lend "<" :sep " "))))  ;; Test :hllfmt parameter.  (should   (equal "a!b\ncd"	  (orgtbl-to-generic	   (org-table-to-lisp "| a | b |\n|---+---|\n| c | d |")	   '(:hllfmt "%s!%s"))))  (should   (equal "a+b\ncd"	  (orgtbl-to-generic	   (org-table-to-lisp "| a | b |\n|---+---|\n| c | d |")	   '(:hllfmt (lambda (c) (concat (car c) "+" (cadr c)))))))  (should   (equal "a!b\n>c d<"	  (orgtbl-to-generic	   (org-table-to-lisp "| a | b |\n|---+---|\n| c | d |")	   '(:hllfmt "%s!%s" :lstart ">" :lend "<" :sep " "))))  ;; Test :fmt parameter.  (should   (equal ">a<\n>b<"	  (orgtbl-to-generic (org-table-to-lisp "| a |\n|---|\n| b |")			     '(:fmt ">%s<"))))  (should   (equal ">a<b"	  (orgtbl-to-generic (org-table-to-lisp "| a | b |")			     '(:fmt (1 ">%s<" 2 (lambda (c) c))))))  (should   (equal "a b"	  (orgtbl-to-generic (org-table-to-lisp "| a | b |")			     '(:fmt (2 " %s")))))  (should   (equal ">a<"	  (orgtbl-to-generic (org-table-to-lisp "| a |")			     '(:fmt (lambda (c) (format ">%s<" c))))))  ;; Test :hfmt parameter.  (should   (equal ">a<\nb"	  (orgtbl-to-generic (org-table-to-lisp "| a |\n|---|\n| b |")			     '(:hfmt ">%s<"))))  (should   (equal ">a<b\ncd"	  (orgtbl-to-generic	   (org-table-to-lisp "| a | b |\n|---+---|\n| c | d |")	   '(:hfmt (1 ">%s<" 2 identity)))))  (should   (equal "a b\ncd"	  (orgtbl-to-generic	   (org-table-to-lisp "| a | b |\n|---+---|\n| c | d |")	   '(:hfmt (2 " %s")))))  (should   (equal ">a<\nb"	  (orgtbl-to-generic (org-table-to-lisp "| a |\n|---|\n| b |")			     '(:hfmt (lambda (c) (format ">%s<" c))))))  ;; Test :efmt parameter.  (should   (equal "2x10^3"	  (orgtbl-to-generic (org-table-to-lisp "| 2e3 |")			     '(:efmt "%sx10^%s"))))  (should   (equal "2x10^3"	  (orgtbl-to-generic (org-table-to-lisp "| 2e3 |")			     '(:efmt (lambda (m e) (concat m "x10^" e))))))  (should   (equal "2x10^3"	  (orgtbl-to-generic (org-table-to-lisp "| 2e3 |")			     '(:efmt (1 "%sx10^%s")))))  (should   (equal "2x10^3"	  (orgtbl-to-generic	   (org-table-to-lisp "| 2e3 |")	   '(:efmt (1 (lambda (m e) (format "%sx10^%s" m e)))))))  (should   (equal "2e3"	  (orgtbl-to-generic (org-table-to-lisp "| 2e3 |") '(:efmt nil))))  ;; Test :skip parameter.  (should   (equal "cd"	  (orgtbl-to-generic	   (org-table-to-lisp "| \ | <c> |\n| a | b |\n|---+---|\n| c | d |")	   '(:skip 2))))  ;; Test :skipcols parameter.  (should   (equal "a\nc"	  (orgtbl-to-generic	   (org-table-to-lisp "| a | b |\n| c | d |") '(:skipcols (2)))))  (should   (equal "a\nc"	  (orgtbl-to-generic	   (org-table-to-lisp	    "| / | <c> | <c> |\n| # | a | b |\n|---+---+---|\n|   | c | d |")	   '(:skipcols (2)))))  ;; Test :raw parameter.  (when (featurep 'ox-latex)    (should     (string-match-p      "/a/"      (orgtbl-to-generic (org-table-to-lisp "| /a/ | b |")			 '(:backend latex :raw t)))))  ;; Hooks are ignored.  (should   (equal    "a\nb"    (let* ((fun-list (list (lambda (backend) (search-forward "a") (insert "hook"))))	   (org-export-before-parsing-hook fun-list)	   (org-export-before-processing-hook fun-list))      (orgtbl-to-generic (org-table-to-lisp "| a |\n|---|\n| b |")			 '(:hline nil)))))  ;; User-defined export filters are ignored.  (should   (equal    "a\nb"    (let ((org-export-filter-table-cell-functions (list (lambda (c b i) "filter"))))      (orgtbl-to-generic (org-table-to-lisp "| a |\n|---|\n| b |")			 '(:hline nil)))))  ;; Macros, even if unknown, are returned as-is.  (should   (equal "{{{macro}}}"	  (orgtbl-to-generic (org-table-to-lisp "| {{{macro}}} |") nil))))(ert-deftest test-org-table/to-latex ()  "Test `orgtbl-to-latex' specifications."  (should   (equal "\\begin{tabular}{l}\na\\\\\n\\end{tabular}"	  (orgtbl-to-latex (org-table-to-lisp "| a |") nil)))  ;; Test :environment parameter.  (should   (equal "\\begin{tabularx}{l}\na\\\\\n\\end{tabularx}"	  (orgtbl-to-latex (org-table-to-lisp "| a |")			   '(:environment "tabularx"))))  ;; Test :booktabs parameter.  (should   (string-match-p    "\\toprule" (orgtbl-to-latex (org-table-to-lisp "| a |") '(:booktabs t))))  ;; Handle LaTeX snippets.  (should   (equal "\\begin{tabular}{l}\n\\(x\\)\\\\\n\\end{tabular}"	  (orgtbl-to-latex (org-table-to-lisp "| $x$ |") nil)))  ;; Test pseudo objects and :raw parameter.  (should   (string-match-p    "\\$x\\$" (orgtbl-to-latex (org-table-to-lisp "| $x$ |") '(:raw t)))))(ert-deftest test-org-table/to-html ()  "Test `orgtbl-to-html' specifications."  (should   (equal (orgtbl-to-html (org-table-to-lisp "| a |") nil)	  "<table border=\"2\" cellspacing=\"0\" cellpadding=\"6\" rules=\"groups\" frame=\"hsides\"><colgroup><col  class=\"org-left\" /></colgroup><tbody><tr><td class=\"org-left\">a</td></tr></tbody></table>"))  ;; Test :attributes parameter.  (should   (string-match-p    "<table>"    (orgtbl-to-html (org-table-to-lisp "| a |") '(:attributes nil))))  (should   (string-match-p    "<table border=\"2\">"    (orgtbl-to-html (org-table-to-lisp "| a |") '(:attributes (:border "2"))))))(ert-deftest test-org-table/to-texinfo ()  "Test `orgtbl-to-texinfo' specifications."  (should   (equal "@multitable {a}\n@item a\n@end multitable"	  (orgtbl-to-texinfo (org-table-to-lisp "| a |") nil)))  ;; Test :columns parameter.  (should   (equal "@multitable @columnfractions .4 .6\n@item a\n@tab b\n@end multitable"	  (orgtbl-to-texinfo (org-table-to-lisp "| a | b |")			     '(:columns ".4 .6"))))  (should   (equal "@multitable @columnfractions .4 .6\n@item a\n@tab b\n@end multitable"	  (orgtbl-to-texinfo (org-table-to-lisp "| a | b |")			     '(:columns "@columnfractions .4 .6"))))  (should   (equal "@multitable {xxx} {xx}\n@item a\n@tab b\n@end multitable"	  (orgtbl-to-texinfo (org-table-to-lisp "| a | b |")			     '(:columns "{xxx} {xx}")))))(ert-deftest test-org-table/to-orgtbl ()  "Test `orgtbl-to-orgtbl' specifications."  (should   (equal "| a | b |\n|---+---|\n| c | d |"	  (orgtbl-to-orgtbl	   (org-table-to-lisp "| a | b |\n|---+---|\n| c | d |") nil))))(ert-deftest test-org-table/to-unicode ()  "Test `orgtbl-to-unicode' specifications."  (should   (equal "━━━\n a \n━━━"	  (orgtbl-to-unicode (org-table-to-lisp "| a |") nil)))  ;; Test :narrow parameter.  (should   (equal "━━━━\n => \n━━━━"	  (orgtbl-to-unicode (org-table-to-lisp "| <2> |\n| xxx |")			     '(:narrow t)))))(ert-deftest test-org-table/send-region ()  "Test `orgtbl-send-table' specifications."  ;; Error when not at a table.  (should-error   (org-test-with-temp-text "Paragraph"     (orgtbl-send-table)))  ;; Error when destination is missing.  (should-error   (org-test-with-temp-text "#+ORGTBL: SEND\n<point>| a |"     (orgtbl-send-table)))  ;; Error when transformation function is not specified.  (should-error   (org-test-with-temp-text "# BEGIN RECEIVE ORGTBL table# END RECEIVE ORGTBL table#+ORGTBL: SEND table<point>| a |"     (orgtbl-send-table)))  ;; Standard test.  (should   (equal "| a |\n|---|\n| b |\n"	  (org-test-with-temp-text "# BEGIN RECEIVE ORGTBL table# END RECEIVE ORGTBL table#+ORGTBL: SEND table orgtbl-to-orgtbl :hlines nil<point>| a |\n|---|\n| b |"	    (orgtbl-send-table)	    (goto-char (point-min))	    (buffer-substring-no-properties	     (search-forward "# BEGIN RECEIVE ORGTBL table\n")	     (progn (search-forward "# END RECEIVE ORGTBL table")		    (match-beginning 0))))))  ;; Allow multiple receiver locations.  (should   (org-test-with-temp-text "# BEGIN RECEIVE ORGTBL table# END RECEIVE ORGTBL table#+ORGTBL: SEND table orgtbl-to-orgtbl :hlines nil<point>| a |# BEGIN RECEIVE ORGTBL table# END RECEIVE ORGTBL table"     (orgtbl-send-table)     (goto-char (point-min))     (search-forward "| a |" nil t 3))));;; Align(ert-deftest test-org-table/align ()  "Test `org-table-align' specifications."  ;; Regular test.  (should   (equal "| a |\n"	  (org-test-with-temp-text "|   a |"	    (org-table-align)	    (buffer-string))))  ;; Preserve alignment.  (should   (equal "  | a |\n"	  (org-test-with-temp-text "  |   a |"	    (org-table-align)	    (buffer-string))))  ;; Handle horizontal lines.  (should   (equal "| 123 |\n|-----|\n"	  (org-test-with-temp-text "| 123 |\n|-|"	    (org-table-align)	    (buffer-string))))  (should   (equal "| a | b |\n|---+---|\n"	  (org-test-with-temp-text "| a | b |\n|-+-|"	    (org-table-align)	    (buffer-string))))  ;; Handle empty fields.  (should   (equal "| a   | bc |\n| bcd |    |\n"	  (org-test-with-temp-text "| a | bc |\n| bcd |  |"	    (org-table-align)	    (buffer-string))))  (should   (equal "| abc | bc  |\n|     | bcd |\n"	  (org-test-with-temp-text "| abc | bc |\n| | bcd |"	    (org-table-align)	    (buffer-string))))  ;; Handle missing fields.  (should   (equal "| a | b |\n| c |   |\n"	  (org-test-with-temp-text "| a | b |\n| c |"	    (org-table-align)	    (buffer-string))))  (should   (equal "| a | b |\n|---+---|\n"	  (org-test-with-temp-text "| a | b |\n|---|"	    (org-table-align)	    (buffer-string))))  ;; Alignment is done to the right when the ratio of numbers in the  ;; column is superior to `org-table-number-fraction'.  (should   (equal "|   1 |\n|  12 |\n| abc |"	  (org-test-with-temp-text "| 1 |\n| 12 |\n| abc |"	    (let ((org-table-number-fraction 0.5)) (org-table-align))	    (buffer-string))))  (should   (equal "| 1   |\n| ab  |\n| abc |"	  (org-test-with-temp-text "| 1 |\n| ab |\n| abc |"	    (let ((org-table-number-fraction 0.5)) (org-table-align))	    (buffer-string))))  ;; Obey to alignment cookies.  (should   (equal "| <r> |\n|  ab |\n| abc |"	  (org-test-with-temp-text "| <r> |\n| ab |\n| abc |"	    (let ((org-table-number-fraction 0.5)) (org-table-align))	    (buffer-string))))  (should   (equal "| <l> |\n| 12  |\n| 123 |"	  (org-test-with-temp-text "| <l> |\n| 12 |\n| 123 |"	    (let ((org-table-number-fraction 0.5)) (org-table-align))	    (buffer-string))))  (should   (equal "| <c> |\n|  1  |\n| 123 |"	  (org-test-with-temp-text "| <c> |\n| 1 |\n| 123 |"	    (let ((org-table-number-fraction 0.5)) (org-table-align))	    (buffer-string)))))(ert-deftest test-org-table/align-buffer-tables ()  "Align all tables when updating buffer."  (let ((before "|  a  b  ||  c  d  |")	(after "| a  b || c  d |"))    (should (equal (org-test-with-temp-text before		     (org-table-recalculate-buffer-tables)		     (buffer-string))		   after))    (should (equal (org-test-with-temp-text before		     (org-table-iterate-buffer-tables)		     (buffer-string))		   after))));;; Sorting(ert-deftest test-org-table/sort-lines ()  "Test `org-table-sort-lines' specifications."  ;; Sort numerically.  (should   (equal "| 1 | 2 |\n| 2 | 4 |\n| 5 | 3 |\n"	  (org-test-with-temp-text "| <point>1 | 2 |\n| 5 | 3 |\n| 2 | 4 |\n"	    (org-table-sort-lines nil ?n)	    (buffer-string))))  (should   (equal "| 5 | 3 |\n| 2 | 4 |\n| 1 | 2 |\n"	  (org-test-with-temp-text "| <point>1 | 2 |\n| 5 | 3 |\n| 2 | 4 |\n"	    (org-table-sort-lines nil ?N)	    (buffer-string))))  ;; Sort alphabetically.  Enforce the C locale for consistent results.  (let ((original-string-collate-lessp (symbol-function 'string-collate-lessp)))    (cl-letf (((symbol-function 'string-collate-lessp)	       (lambda (s1 s2 &optional locale ignore-case)		 (funcall original-string-collate-lessp			  s1 s2 "C" ignore-case))))      (should       (equal "| a | x |\n| B | 4 |\n| c | 3 |\n"	      (org-test-with-temp-text "| <point>a | x |\n| c | 3 |\n| B | 4 |\n"				       (org-table-sort-lines nil ?a)				       (buffer-string))))      (should       (equal "| c | 3 |\n| B | 4 |\n| a | x |\n"	      (org-test-with-temp-text "| <point>a | x |\n| c | 3 |\n| B | 4 |\n"				       (org-table-sort-lines nil ?A)				       (buffer-string))))      ;; Sort alphabetically with case.      (should       (equal "| C |\n| a |\n| b |\n"	      (org-test-with-temp-text "| <point>a |\n| C |\n| b |\n"				       (org-table-sort-lines t ?a)				       (buffer-string))))      (should       (equal "| C |\n| b |\n| a |\n"	      (org-test-with-temp-text "| <point>a |\n| C |\n| b |\n"				       (org-table-sort-lines nil ?A)				       (buffer-string))))))  ;; Sort by time (timestamps)  (should   (equal    "| <2008-08-08 sat.> |\n| <2012-03-29 thu.> |\n| <2014-03-04 tue.> |\n"    (org-test-with-temp-text	"| <2014-03-04 tue.> |\n| <2008-08-08 sat.> |\n| <2012-03-29 thu.> |\n"      (org-table-sort-lines nil ?t)      (buffer-string))))  (should   (equal    "| <2014-03-04 tue.> |\n| <2012-03-29 thu.> |\n| <2008-08-08 sat.> |\n"    (org-test-with-temp-text	"| <2014-03-04 tue.> |\n| <2008-08-08 sat.> |\n| <2012-03-29 thu.> |\n"      (org-table-sort-lines nil ?T)      (buffer-string))))  ;; Sort by time (HH:MM values)  (should   (equal "| 1:00 |\n| 17:00 |\n| 114:00 |\n"	  (org-test-with-temp-text "| 114:00 |\n| 17:00 |\n| 1:00 |\n"	    (org-table-sort-lines nil ?t)	    (buffer-string))))  (should   (equal "| 114:00 |\n| 17:00 |\n| 1:00 |\n"	  (org-test-with-temp-text "| 114:00 |\n| 17:00 |\n| 1:00 |\n"	    (org-table-sort-lines nil ?T)	    (buffer-string))))  ;; Sort by time (durations)  (should   (equal "| 1d 3:00 |\n| 28:00 |\n"	  (org-test-with-temp-text "| 28:00 |\n| 1d 3:00 |\n"	    (org-table-sort-lines nil ?t)	    (buffer-string))))  ;; Sort with custom functions.  (should   (equal "| 22 |\n| 15 |\n| 18 |\n"	  (org-test-with-temp-text "| 15 |\n| 22 |\n| 18 |\n"	    (org-table-sort-lines nil ?f				  (lambda (s) (% (string-to-number s) 10))				  #'<)	    (buffer-string))))  (should   (equal "| 18 |\n| 15 |\n| 22 |\n"	  (org-test-with-temp-text "| 15 |\n| 22 |\n| 18 |\n"	    (org-table-sort-lines nil ?F				  (lambda (s) (% (string-to-number s) 10))				  #'<)	    (buffer-string))))  ;; Sort according to current column.  (should   (equal "| 1 | 2 |\n| 7 | 3 |\n| 5 | 4 |\n"	  (org-test-with-temp-text "| 1 | <point>2 |\n| 5 | 4 |\n| 7 | 3 |\n"	    (org-table-sort-lines nil ?n)	    (buffer-string))))  ;; Sort between horizontal separators if possible.  (should   (equal    "| 9 | 8 |\n|---+---|\n| 5 | 3 |\n| 7 | 4 |\n|---+---|\n| 1 | 2 |\n"    (org-test-with-temp-text	"| 9 | 8 |\n|---+---|\n| <point>7 | 4 |\n| 5 | 3 |\n|---+---|\n| 1 | 2 |\n"      (org-table-sort-lines nil ?n)      (buffer-string)))));;; Formulas(ert-deftest test-org-table/eval-formula ()  "Test `org-table-eval-formula' specifications."  ;; Error when not on a table field.  (should-error   (org-test-with-temp-text "Text"     (org-table-eval-formula)))  (should-error   (org-test-with-temp-text "| a |\n|---|<point>"     (org-table-eval-formula)))  (should-error   (org-test-with-temp-text "| a |\n#+TBLFM:<point>"     (org-table-eval-formula)))  ;; Handle @<, @>, $< and $>.  (should   (equal "| 1 |\n| 1 |"	  (org-test-with-temp-text "| <point>  |\n| 1 |"	    (org-table-eval-formula nil "@>" nil nil t)	    (buffer-string))))  (should   (equal "| 1 |\n| 1 |"	  (org-test-with-temp-text "| 1 |\n| <point>  |"	    (org-table-eval-formula nil "@<" nil nil t)	    (buffer-string))))  (should   (equal "| 1 | 1 |"	  (org-test-with-temp-text "| <point>  | 1 |"	    (org-table-eval-formula nil "$>" nil nil t)	    (buffer-string))))  (should   (equal "| 1 | 1 |"	  (org-test-with-temp-text "| 1 | <point>  |"	    (org-table-eval-formula nil "$<" nil nil t)	    (buffer-string)))))(ert-deftest test-org-table/field-formula-outside-table ()  "Test `org-table-formula-create-columns' variable."  ;; Refuse to create column if variable is nil.  (should-error   (org-test-with-temp-text "| 2 || 4 || 8 |<point>#+TBLFM: @1$2=5"     (let ((org-table-formula-create-columns nil))       (org-table-calc-current-TBLFM))     (buffer-string))   :type (list 'error 'user-error))  ;; If the variable is non-nil, field formulas and columns formulas  ;; can create tables.  (should   (equal    "| 2 | 5 || 4 |   || 8 |   |#+TBLFM: @1$2=5"    (org-test-with-temp-text "| 2 || 4 || 8 |<point>#+TBLFM: @1$2=5"      (let ((org-table-formula-create-columns t))	(org-table-calc-current-TBLFM))      (buffer-string))))  (should   (equal    "| 2 |   | 15 || 4 |   | 15 || 8 |   | 15 |#+TBLFM: $3=15"    (org-test-with-temp-text "| 2 || 4 || 8 |<point>#+TBLFM: $3=15"      (let ((org-table-formula-create-columns t))	(org-table-calc-current-TBLFM))      (buffer-string)))))(ert-deftest test-org-table/duration ()  "Test durations in table formulas."  ;; Durations in cells.  (should   (string-match "| 2:12 | 1:47 | 03:59:00 |"		 (org-test-with-temp-text "       | 2:12 | 1:47 | |       <point>#+TBLFM: @1$3=$1+$2;T"		   (org-table-calc-current-TBLFM)		   (buffer-string))))  (should   (string-match "| 2:12 | 1:47 | 03:59 |"		 (org-test-with-temp-text "       | 2:12 | 1:47 | |       <point>#+TBLFM: @1$3=$1+$2;U"		   (org-table-calc-current-TBLFM)		   (buffer-string))))  (should   (string-match "| 3:02:20 | -2:07:00 | 0.92 |"		 (org-test-with-temp-text "       | 3:02:20 | -2:07:00 | |       <point>#+TBLFM: @1$3=$1+$2;t"		   (org-table-calc-current-TBLFM)		   (buffer-string))))  ;; Durations set through properties.  (should   (string-match "| 16:00:00 |"		 (org-test-with-temp-text "* H  :PROPERTIES:  :time_constant: 08:00:00  :END:  |  |  <point>#+TBLFM: $1=2*$PROP_time_constant;T"		   (org-table-calc-current-TBLFM)		   (buffer-string))))  (should   (string-match "| 16.00 |"		 (org-test-with-temp-text "* H  :PROPERTIES:  :time_constant: 08:00:00  :END:  |  |  <point>#+TBLFM: $1=2*$PROP_time_constant;t"		   (org-table-calc-current-TBLFM)		   (buffer-string)))))(ert-deftest test-org-table/end-on-hline ()  "Test with a table ending on a hline."  (should   (equal    (org-test-with-temp-text	"| 1 | 2 | 3 || 4 | 5 | 6 ||   |   |   ||---+---+---|<point>#+TBLFM: @3$2..@3$>=vsum(@1..@2)"      (org-table-calc-current-TBLFM)      (buffer-string))    "| 1 | 2 | 3 || 4 | 5 | 6 ||   | 7 | 9 ||---+---+---|#+TBLFM: @3$2..@3$>=vsum(@1..@2)")))(ert-deftest test-org-table/named-field ()  "Test formula with a named field."  (should   (string-match-p    "| +| +1 +|"    (org-test-with-temp-text "|   |      || ^ | name |<point>#+TBLFM: $name=1"      (org-table-calc-current-TBLFM)      (buffer-string))))  (should   (string-match-p    "| +| +1 +|"    (org-test-with-temp-text "| _ | name ||   |      |<point>#+TBLFM: $name=1"      (org-table-calc-current-TBLFM)      (buffer-string)))))(ert-deftest test-org-table/named-column ()  "Test formula with a named field."  (should   (string-match-p    "| +| +1 +| +1 +|"    (org-test-with-temp-text "| ! | name |   ||   |    1 |   |<point>#+TBLFM: @2$3=$name"      (org-table-calc-current-TBLFM)      (buffer-string)))))(ert-deftest test-org-table/formula-priority ()  "Test field formula priority over column formula."  ;; Field formulas bind stronger than column formulas.  (should   (equal    "| 1 |  3 |\n| 2 | 99 |\n"    (org-test-with-temp-text	"| 1 |   |\n| 2 |   |\n<point>#+tblfm: $2=3*$1::@2$2=99"      (org-table-calc-current-TBLFM)      (buffer-substring-no-properties (point-min) (point)))))  ;; When field formula is removed, table formulas is applied again.  (should   (equal    "| 1 | 3 |\n| 2 | 6 |\n"    (org-test-with-temp-text	"| 1 |   |\n| 2 |   |\n#+tblfm: $2=3*$1<point>::@2$2=99"      (org-table-calc-current-TBLFM)      (delete-region (point) (line-end-position))      (org-table-calc-current-TBLFM)      (buffer-substring-no-properties (point-min) (line-beginning-position))))))(ert-deftest test-org-table/tab-indent ()  "Test named fields with tab indentation."  (should   (string-match-p    "| # | 111 |"    (org-test-with-temp-text	"	| ! |  sum |      | a |  b |   c |	|---+------+------+---+----+-----|	| # | 1011 | 1000 | 1 | 10 | 100 |	<point>#+TBLFM: $2=$a+$b+$c"      (org-table-calc-current-TBLFM)      (buffer-string)))))(ert-deftest test-org-table/first-rc ()  "Test \"$<\" and \"@<\" constructs in formulas."  (should   (string-match-p    "| 1 | 2 |"    (org-test-with-temp-text	"|   | 2 |<point>#+TBLFM: $<=1"      (org-table-calc-current-TBLFM)      (buffer-string))))  (should   (string-match-p    "| 2 |\n| 2 |"    (org-test-with-temp-text	"| 2 |\n|   |<point>#+TBLFM: @2$1=@<"      (org-table-calc-current-TBLFM)      (buffer-string)))))(ert-deftest test-org-table/last-rc ()  "Test \"$>\" and \"@>\" constructs in formulas."  (should   (string-match-p    "| 2 | 1 |"    (org-test-with-temp-text	"| 2 |   |\n<point>#+TBLFM: $>=1"      (org-table-calc-current-TBLFM)      (buffer-string))))  (should   (string-match-p    "| 2 |\n| 2 |"    (org-test-with-temp-text	"| 2 |\n|   |\n<point>#+TBLFM: @>$1=@<"      (org-table-calc-current-TBLFM)      (buffer-string)))))(ert-deftest test-org-table/time-stamps ()  "Test time-stamps handling."  ;; Standard test.  (should   (string-match-p    "| 1 |"    (org-test-with-temp-text	"| <2016-07-07 Sun> | <2016-07-08 Fri> |   |\n<point>#+TBLFM: $3=$2-$1"      (org-table-calc-current-TBLFM)      (buffer-string))))  ;; Handle locale specific time-stamps.  (should   (string-match-p    "| 1 |"    (org-test-with-temp-text	"| <2016-07-07 Do> | <2016-07-08 Fr> |   |\n<point>#+TBLFM: $3=$2-$1"      (org-table-calc-current-TBLFM)      (buffer-string)))))(ert-deftest test-org-table/orgtbl-ascii-draw ()  "Test `orgtbl-ascii-draw'."  ;; First value: Make sure that an integer input value is converted to a  ;; float before division. Further values: Show some float input value  ;; ranges corresponding to the same bar width.  (should   (equal    (org-test-with-temp-text	"|    Value | <l>     ||----------+---------||       19 | replace ||----------+---------|| -0.50001 | replace || -0.49999 | replace ||  0.49999 | replace ||  0.50001 | replace ||  1.49999 | replace || 22.50001 | replace || 23.49999 | replace || 23.50001 | replace || 24.49999 | replace || 24.50001 | replace |<point>#+TBLFM: $2 = '(orgtbl-ascii-draw $1 0 24 3 \" 12345678\")"      (org-table-calc-current-TBLFM)      (buffer-string))    "|    Value | <l>       ||----------+-----------||       19 | 883       ||----------+-----------|| -0.50001 | too small || -0.49999 |           ||  0.49999 |           ||  0.50001 | 1         ||  1.49999 | 1         || 22.50001 | 887       || 23.49999 | 887       || 23.50001 | 888       || 24.49999 | 888       || 24.50001 | too large |#+TBLFM: $2 = '(orgtbl-ascii-draw $1 0 24 3 \" 12345678\")"))  ;; Draw bars with a bullet. The bullet does not count in the parameter  ;; WIDTH of `orgtbl-ascii-draw'.  (should   (equal    (org-test-with-temp-text	"| -1 | replace ||  0 | replace ||  1 | replace ||  2 | replace ||  3 | replace ||  4 | replace |<point>#+TBLFM: $2 = '(orgtbl-ascii-draw $1 0 3 3 \"$-\")"      (org-table-calc-current-TBLFM)      (buffer-string))    "| -1 | too small ||  0 | $         ||  1 | -$        ||  2 | --$       ||  3 | ---$      ||  4 | too large |#+TBLFM: $2 = '(orgtbl-ascii-draw $1 0 3 3 \"$-\")")))(ert-deftest test-org-table/single-rowgroup ()  "Test column formula in a table with a single rowgroup."  (should   (equal    "|---+---|| 1 | 0 ||---+---|#+TBLFM: $2=$1-1"    (org-test-with-temp-text "|---+---|| 1 |   ||---+---|<point>#+TBLFM: $2=$1-1"      (org-table-calc-current-TBLFM)      (buffer-string))))  (should   (equal    "| 1 | 0 |#+TBLFM: $2=$1-1"    (org-test-with-temp-text "| 1 |   |<point>#+TBLFM: $2=$1-1"      (org-table-calc-current-TBLFM)      (buffer-string)))));;; Navigation(ert-deftest test-org-table/next-field ()  "Test `org-table-next-field' specifications."  ;; Regular test.  (should   (equal    "b"    (org-test-with-temp-text "| a<point> | b |"      (org-table-next-field)      (org-trim (org-table-get-field)))))  ;; Create new rows as needed.  (should   (equal    "| a |\n|   |\n"    (org-test-with-temp-text "| a<point> |"      (org-table-next-field)      (buffer-string))))  ;; Jump over hlines, if `org-table-tab-jumps-over-hlines' is  ;; non-nil.  (should   (equal    "b"    (org-test-with-temp-text "| a<point> |\n|---|\n| b |"      (let ((org-table-tab-jumps-over-hlines t)) (org-table-next-field))      (org-trim (org-table-get-field)))))  ;; If `org-table-tab-jumps-over-hlines' is nil, however, create  ;; a new row before the rule.  (should   (equal    "| a |\n|   |\n|---|\n| b |"    (org-test-with-temp-text "| a<point> |\n|---|\n| b |"      (let ((org-table-tab-jumps-over-hlines nil)) (org-table-next-field))      (buffer-string)))))(ert-deftest test-org-table/previous-field ()  "Test `org-table-previous-field' specifications."  ;; Regular tests.  (should   (eq ?a       (org-test-with-temp-text "| a | <point>b |"	 (org-table-previous-field)	 (char-after))))  (should   (eq ?a       (org-test-with-temp-text "| a |\n| <point>b |"	 (org-table-previous-field)	 (char-after))))  ;; Find previous field across horizontal rules.  (should   (eq ?a       (org-test-with-temp-text "| a |\n|---|\n| <point>b |"	 (org-table-previous-field)	 (char-after))))  ;; When called on a horizontal rule, find previous data field.  (should   (eq ?b       (org-test-with-temp-text "| a | b |\n|---+-<point>--|"	 (org-table-previous-field)	 (char-after))))  ;; Error when at first field.  Make sure to preserve original  ;; position.  (should-error   (org-test-with-temp-text "| <point> a|"     (org-table-previous-field)))  (should-error   (org-test-with-temp-text "|---|\n| <point>a |"     (org-table-previous-field)))  (should   (eq ?a       (org-test-with-temp-text "|---|\n| <point>a |"	 (ignore-errors (org-table-previous-field))	 (char-after)))));;; Inserting rows, inserting columns(ert-deftest test-org-table/insert-column ()  "Test `org-table-insert-column' specifications."  ;; Error when outside a table.  (should-error   (org-test-with-temp-text "Paragraph"     (org-table-insert-column)))  ;; Insert new column after current one.  (should   (equal "| a |   |\n"	  (org-test-with-temp-text "| a |"	    (org-table-insert-column)	    (buffer-string))))  (should   (equal "| a |   | b |\n"	  (org-test-with-temp-text "| <point>a | b |"	    (org-table-insert-column)	    (buffer-string))))  ;; Move point into the newly created column.  (should   (equal "  |"	  (org-test-with-temp-text "| <point>a |"	    (org-table-insert-column)	    (buffer-substring-no-properties (point) (line-end-position)))))  (should   (equal "  | b |"	  (org-test-with-temp-text "| <point>a | b |"	    (org-table-insert-column)	    (buffer-substring-no-properties (point) (line-end-position)))))  ;; Handle missing vertical bar in the last column.  (should   (equal "| a |   |\n"	  (org-test-with-temp-text "| a"	    (org-table-insert-column)	    (buffer-string))))  (should   (equal "  |"	  (org-test-with-temp-text "| <point>a"	    (org-table-insert-column)	    (buffer-substring-no-properties (point) (line-end-position)))))  ;; Handle column insertion when point is before first column.  (should   (equal " | a |   |\n"	  (org-test-with-temp-text " | a |"	    (org-table-insert-column)	    (buffer-string))))  (should   (equal " | a |   | b |\n"	  (org-test-with-temp-text " | a | b |"	    (org-table-insert-column)	    (buffer-string)))));;; Moving single cells(ert-deftest test-org-table/move-cell-down ()  "Test `org-table-move-cell-down' specifications."  ;; Error out when cell cannot be moved due to not in table, in the  ;; last row of the table, or is on a hline.  (should-error   (org-test-with-temp-text "not in\na table\n"     (org-table-move-cell-down)))  (should-error   (org-test-with-temp-text "| a |"     (org-table-move-cell-down)))  (should-error   (org-test-with-temp-text "| a |\n"     (org-table-move-cell-down)))  (should-error   (org-test-with-temp-text "| a | <point>b |\n"     (org-table-move-cell-down)))  (should-error   (org-test-with-temp-text "| a | b |\n| <point>c | d |\n"     (org-table-move-cell-down)))  (should-error   (org-test-with-temp-text "| a | b |\n| c | <point>d |\n"     (org-table-move-cell-down)))  (should-error   (org-test-with-temp-text "| <point>a |\n|---|\n"     (org-table-move-cell-down)))  (should-error   (org-test-with-temp-text "|<point>---|\n| a |\n"     (org-table-move-cell-down)))  ;; Check for correct cell movement  (should (equal (concat "| c | b |\n"			 "| a | d |\n"			 "| e | f |\n")		 (org-test-with-temp-text		     (concat "| <point>a | b |\n"			     "| c | d |\n"			     "| e | f |\n")		   (org-table-move-cell-down)		   (buffer-string))))  (should (equal (concat "| a | d |\n"			 "| c | b |\n"			 "| e | f |\n")		 (org-test-with-temp-text		     (concat "| a | <point>b |\n"			     "| c | d |\n"			     "| e | f |\n")		   (org-table-move-cell-down)		   (buffer-string))))  (should (equal (concat "| a | b |\n"			 "| e | d |\n"			 "| c | f |\n")		 (org-test-with-temp-text		     (concat "| a | b |\n"			     "| <point>c | d |\n"			     "| e | f |\n")		   (org-table-move-cell-down)		   (buffer-string))))  (should (equal (concat "| a | d |\n"			 "| c | f |\n"			 "| e | b |\n")		 (org-test-with-temp-text		     (concat "| a |<point> b |\n"			     "| c | d |\n"			     "| e | f |\n")		   (org-table-move-cell-down)		   (org-table-move-cell-down)		   (buffer-string))))  ;; Check for correct handling of hlines which should not change  ;; position on single cell moves.  (should (equal (concat "| c | b |\n"			 "|---+---|\n"			 "| a | d |\n"			 "| e | f |\n")		 (org-test-with-temp-text		     (concat "| <point>a | b |\n"			     "|---+---|\n"			     "| c | d |\n"			     "| e | f |\n")		   (org-table-move-cell-down)		   (buffer-string))))  (should (equal (concat "| a | d |\n"			 "|---+---|\n"			 "| c | f |\n"			 "| e | b |\n")		 (org-test-with-temp-text		     (concat "| a | <point>b |\n"			     "|---+---|\n"			     "| c | d |\n"			     "| e | f |\n")		   (org-table-move-cell-down)		   (org-table-move-cell-down)		   (buffer-string))))  (should (equal (concat "| a | b |\n"			 "|---+---|\n"			 "| c | f |\n"			 "| e | d |\n")		 (org-test-with-temp-text		     (concat "| a | b |\n"			     "|---+---|\n"			     "| c | <point>d |\n"			     "| e | f |\n")		   (org-table-move-cell-down)		   (buffer-string))))  ;; Move single cell even without a final newline.  (should (equal (concat "| a | d |\n"			 "|---+---|\n"			 "| c | f |\n"			 "| e | b |\n")		 (org-test-with-temp-text		     (concat "| a | <point>b |\n"			     "|---+---|\n"			     "| c | d |\n"			     "| e | f |")		   (org-table-move-cell-down)		   (org-table-move-cell-down)		   (buffer-string)))))(ert-deftest test-org-table/move-cell-up ()  "Test `org-table-move-cell-up' specifications."  ;; Error out when cell cannot be moved due to not in table, in the  ;; last row of the table, or is on a hline.  (should-error   (org-test-with-temp-text "not in\na table\n"     (org-table-move-cell-up)))  (should-error   (org-test-with-temp-text "| a |"     (org-table-move-cell-up)))  (should-error   (org-test-with-temp-text "| a |\n"     (org-table-move-cell-up)))  (should-error   (org-test-with-temp-text "| <point>a | b |\n"     (org-table-move-cell-up)))  (should-error   (org-test-with-temp-text "| a | <point>b |\n| c | d |\n"     (org-table-move-cell-up)))  (should-error   (org-test-with-temp-text "| <point>a |\n|---|\n"     (org-table-move-cell-up)))  (should-error   (org-test-with-temp-text "|<point>---|\n| a |\n"     (org-table-move-cell-up)))  ;; Check for correct cell movement.  (should (equal (concat "| c | b |\n"			 "| a | d |\n"			 "| e | f |\n")		 (org-test-with-temp-text		     (concat "| a | b |\n"			     "| <point>c | d |\n"			     "| e | f |\n")		   (org-table-move-cell-up)		   (buffer-string))))  (should (equal (concat "| a | d |\n"			 "| c | b |\n"			 "| e | f |\n")		 (org-test-with-temp-text		     (concat "| a | b |\n"			     "| c | <point>d |\n"			     "| e | f |\n")		   (org-table-move-cell-up)		   (buffer-string))))  (should (equal (concat "| a | b |\n"			 "| e | d |\n"			 "| c | f |\n")		 (org-test-with-temp-text		     (concat "| a | b |\n"			     "| c | d |\n"			     "| <point>e | f |\n")		   (org-table-move-cell-up)		   (buffer-string))))  (should (equal (concat "| a | f |\n"			 "| c | b |\n"			 "| e | d |\n")		 (org-test-with-temp-text		     (concat "| a | b |\n"			     "| c | d |\n"			     "| e |<point> f |\n")		   (org-table-move-cell-up)		   (org-table-move-cell-up)		   (buffer-string))))  ;; Check for correct handling of hlines which should not change  ;; position on single cell moves.  (should (equal (concat "| c | b |\n"			 "|---+---|\n"			 "| a | d |\n"			 "| e | f |\n")		 (org-test-with-temp-text		     (concat "| a | b |\n"			     "|---+---|\n"			     "| <point>c | d |\n"			     "| e | f |\n")		   (org-table-move-cell-up)		   (buffer-string))))  (should (equal (concat "| a | f |\n"			 "|---+---|\n"			 "| c | b |\n"			 "| e | d |\n")		 (org-test-with-temp-text		     (concat "| a | b |\n"			     "|---+---|\n"			     "| c | d |\n"			     "| e | <point>f |\n")		   (org-table-move-cell-up)		   (org-table-move-cell-up)		   (buffer-string))))  (should (equal (concat "| a | b |\n"			 "|---+---|\n"			 "| c | f |\n"			 "| e | d |\n")		 (org-test-with-temp-text		     (concat "| a | b |\n"			     "|---+---|\n"			     "| c | d |\n"			     "| e | <point>f |\n")		   (org-table-move-cell-up)		   (buffer-string))))  ;; Move single cell even without a final newline.  (should (equal (concat "| a | f |\n"			 "|---+---|\n"			 "| c | b |\n"			 "| e | d |\n")		 (org-test-with-temp-text		     (concat "| a | b |\n"			     "|---+---|\n"			     "| c | d |\n"			     "| e | <point>f |")		   (org-table-move-cell-up)		   (org-table-move-cell-up)		   (buffer-string)))))(ert-deftest test-org-table/move-cell-right ()  "Test `org-table-move-cell-right' specifications."  ;; Error out when cell cannot be moved due to not in table, in the  ;; last col of the table, or is on a hline.  (should-error   (org-test-with-temp-text "not in\na table\n"     (org-table-move-cell-right)))  (should-error   (org-test-with-temp-text "| a |"     (org-table-move-cell-right)))  (should-error   (org-test-with-temp-text "| a |\n"     (org-table-move-cell-right)))  (should-error   (org-test-with-temp-text "| <point>a |\n| b |\n"     (org-table-move-cell-right)))  (should-error   (org-test-with-temp-text "| a | <point>b |\n| c | d |\n"     (org-table-move-cell-right)))  (should-error   (org-test-with-temp-text "| <point>a |\n|---|\n"     (org-table-move-cell-right)))  (should-error   (org-test-with-temp-text "|<point>---|\n| a |\n"     (org-table-move-cell-right)))  ;; Check for correct cell movement.  (should (equal (concat "| b | a | c |\n"			 "| d | e | f |\n")		 (org-test-with-temp-text		     (concat "| <point>a | b | c |\n"			     "| d | e | f |\n")		   (org-table-move-cell-right)		   (buffer-string))))  (should (equal (concat "| b | c | a |\n"			 "| d | e | f |\n")		 (org-test-with-temp-text		     (concat "| <point>a | b | c |\n"			     "| d | e | f |\n")		   (org-table-move-cell-right)		   (org-table-move-cell-right)		   (buffer-string))))  (should (equal (concat "| a | b | c |\n"			 "| e | f | d |\n")		 (org-test-with-temp-text		     (concat "| a | b | c |\n"			     "| <point> d | e | f |\n")		   (org-table-move-cell-right)		   (org-table-move-cell-right)		   (buffer-string))))  (should (equal (concat "| a | b | c |\n"			 "| d | f | e |\n")		 (org-test-with-temp-text		     (concat "| a | b | c |\n"			     "| d | <point>e | f |\n")		   (org-table-move-cell-right)		   (buffer-string))))  (should (equal (concat "| a | b | c |\n"			 "|---+---+---|\n"			 "| e | f | d |\n")		 (org-test-with-temp-text		     (concat "| a | b | c |\n"			     "|---+---+---|\n"			     "| <point>d | e | f |\n")		   (org-table-move-cell-right)		   (org-table-move-cell-right)		   (buffer-string))))  ;; Move single cell even without a final newline.  (should (equal (concat "| a | b | c |\n"			 "|---+---+---|\n"			 "| e | d | f |\n")		 (org-test-with-temp-text		     (concat "| a | b | c |\n"			     "|---+---+---|\n"			     "| <point>d | e | f |")		   (org-table-move-cell-right)		   (buffer-string)))))(ert-deftest test-org-table/move-cell-left ()  "Test `org-table-move-cell-left' specifications."  ;; Error out when cell cannot be moved due to not in table, in the  ;; last col of the table, or is on a hline.  (should-error   (org-test-with-temp-text "not in\na table\n"     (org-table-move-cell-left)))  (should-error   (org-test-with-temp-text "| a |"     (org-table-move-cell-left)))  (should-error   (org-test-with-temp-text "| a |\n"     (org-table-move-cell-left)))  (should-error   (org-test-with-temp-text "| <point>a |\n| b |\n"     (org-table-move-cell-left)))  (should-error   (org-test-with-temp-text "| <point>a | b |\n| c | d |\n"     (org-table-move-cell-left)))  (should-error   (org-test-with-temp-text "| <point>a |\n|---|\n"     (org-table-move-cell-left)))  (should-error   (org-test-with-temp-text "|<point>---|\n| a |\n"     (org-table-move-cell-left)))  ;; Check for correct cell movement.  (should (equal (concat "| b | a | c |\n"			 "| d | e | f |\n")		 (org-test-with-temp-text		     (concat "| a | <point>b | c |\n"			     "| d | e | f |\n")		   (org-table-move-cell-left)		   (buffer-string))))  (should (equal (concat "| c | a | b |\n"			 "| d | e | f |\n")		 (org-test-with-temp-text		     (concat "| a | b | <point>c |\n"			     "| d | e | f |\n")		   (org-table-move-cell-left)		   (org-table-move-cell-left)		   (buffer-string))))  (should (equal (concat "| a | b | c |\n"			 "| f | d | e |\n")		 (org-test-with-temp-text		     (concat "| a | b | c |\n"			     "| d | e | <point>f |\n")		   (org-table-move-cell-left)		   (org-table-move-cell-left)		   (buffer-string))))  (should (equal (concat "| a | b | c |\n"			 "| d | f | e |\n")		 (org-test-with-temp-text		     (concat "| a | b | c |\n"			     "| d | e | <point>f |\n")		   (org-table-move-cell-left)		   (buffer-string))))  (should (equal (concat "| a | b | c |\n"			 "|---+---+---|\n"			 "| f | d | e |\n")		 (org-test-with-temp-text		     (concat "| a | b | c |\n"			     "|---+---+---|\n"			     "| d | e | <point>f |\n")		   (org-table-move-cell-left)		   (org-table-move-cell-left)		   (buffer-string))))  ;; Move single cell even without a final newline.  (should (equal (concat "| a | b | c |\n"			 "|---+---+---|\n"			 "| e | d | f |\n")		 (org-test-with-temp-text		     (concat "| a | b | c |\n"			     "|---+---+---|\n"			     "| d | <point>e | f |")		   (org-table-move-cell-left)		   (buffer-string)))));;; Moving rows, moving columns(ert-deftest test-org-table/move-row-down ()  "Test `org-table-move-row-down' specifications."  ;; Error out when row cannot be moved, e.g., it is the last row in  ;; the table.  (should-error   (org-test-with-temp-text "| a |"     (org-table-move-row-down)))  (should-error   (org-test-with-temp-text "| a |\n"     (org-table-move-row-down)))  (should-error   (org-test-with-temp-text "| a |\n| <point>b |"     (org-table-move-row-down)))  ;; Move data lines.  (should   (equal "| b |\n| a |\n"	  (org-test-with-temp-text "| a |\n| b |\n"	    (org-table-move-row-down)	    (buffer-string))))  (should   (equal "|---|\n| a |\n"	  (org-test-with-temp-text "| a |\n|---|\n"	    (org-table-move-row-down)	    (buffer-string))))  ;; Move hlines.  (should   (equal "| b |\n|---|\n"	  (org-test-with-temp-text "|---|\n| b |\n"	    (org-table-move-row-down)	    (buffer-string))))  (should   (equal "|---|\n|---|\n"	  (org-test-with-temp-text "|---|\n|---|\n"	    (org-table-move-row-down)	    (buffer-string))))  ;; Move rows even without a final newline.  (should   (equal "| b |\n| a |\n"	  (org-test-with-temp-text "| a |\n| b |"	    (org-table-move-row-down)	    (buffer-string)))))(ert-deftest test-org-table/move-row-up ()  "Test `org-table-move-row-up' specifications."  ;; Error out when row cannot be moved, e.g., it is the first row in  ;; the table.  (should-error   (org-test-with-temp-text "| a |"     (org-table-move-row-up)))  (should-error   (org-test-with-temp-text "| a |\n"     (org-table-move-row-up)))  ;; Move data lines.  (should   (equal "| b |\n| a |\n"	  (org-test-with-temp-text "| a |\n| <point>b |\n"	    (org-table-move-row-up)	    (buffer-string))))  (should   (equal "| b |\n|---|\n"	  (org-test-with-temp-text "|---|\n| <point>b |\n"	    (org-table-move-row-up)	    (buffer-string))))  ;; Move hlines.  (should   (equal "|---|\n| a |\n"	  (org-test-with-temp-text "| a |\n|<point>---|\n"	    (org-table-move-row-up)	    (buffer-string))))  (should   (equal "|---|\n|---|\n"	  (org-test-with-temp-text "|---|\n|<point>---|\n"	    (org-table-move-row-up)	    (buffer-string))))  ;; Move rows even without a final newline.  (should   (equal "| b |\n| a |\n"	  (org-test-with-temp-text "| a |\n| <point>b |"	    (org-table-move-row-up)	    (buffer-string)))));;; Shrunk columns(ert-deftest test-org-table/toggle-column-width ()  "Test `org-table-toggle-columns-width' specifications."  ;; Error when not at a column.  (should-error   (org-test-with-temp-text "<point>a"     (org-table-toggle-column-width)))  ;; A shrunk column is overlaid with  ;; `org-table-shrunk-column-indicator'.  (should   (equal org-table-shrunk-column-indicator	  (org-test-with-temp-text "| <point>a |"	    (org-table-toggle-column-width)	    (overlay-get (car (overlays-at (point))) 'display))))  (should   (equal org-table-shrunk-column-indicator	  (org-test-with-temp-text "| a |\n|-<point>--|"	    (org-table-toggle-column-width)	    (overlay-get (car (overlays-at (point))) 'display))))  ;; Shrink every field in the same column.  (should   (equal org-table-shrunk-column-indicator	  (org-test-with-temp-text "| a |\n|-<point>--|"	    (org-table-toggle-column-width)	    (overlay-get (car (overlays-at (1+ (line-beginning-position 0))))			 'display))))  ;; When column is already shrunk, expand it, i.e., remove overlays.  (should-not   (org-test-with-temp-text "| <point>a |"     (org-table-toggle-column-width)     (org-table-toggle-column-width)     (overlays-in (point-min) (point-max))))  (should-not   (org-test-with-temp-text "| a |\n| <point>b |"     (org-table-toggle-column-width)     (org-table-toggle-column-width)     (overlays-in (point-min) (point-max))))  ;; With a column width cookie, limit overlay to the specified number  ;; of characters.  (should   (equal "| abc"	  (org-test-with-temp-text "| <3>  |\n| <point>abcd |"	    (org-table-toggle-column-width)	    (buffer-substring (line-beginning-position)			      (overlay-start			       (car (overlays-in (line-beginning-position)						 (line-end-position))))))))  (should   (equal "| a  "	  (org-test-with-temp-text "| <3>  |\n| <point>a   |"	    (org-table-toggle-column-width)	    (buffer-substring (line-beginning-position)			      (overlay-start			       (car (overlays-in (line-beginning-position)						 (line-end-position))))))))  (should   (equal (concat "----" org-table-shrunk-column-indicator)	  (org-test-with-temp-text "| <3>  |\n|--<point>----|"	    (org-table-toggle-column-width)	    (overlay-get	     (car (overlays-in (line-beginning-position)			       (line-end-position)))	     'display))))  ;; Width only takes into account visible characters.  (should   (equal "| [[http"	  (org-test-with-temp-text "| <4> |\n| <point>[[http://orgmode.org]] |"	    (org-table-toggle-column-width)	    (buffer-substring (line-beginning-position)			      (overlay-start			       (car (overlays-in (line-beginning-position)						 (line-end-position))))))))  ;; Before the first column or after the last one, ask for columns  ;; ranges.  (should   (catch :exit     (org-test-with-temp-text "| a |"       (cl-letf (((symbol-function 'read-string)		  (lambda (&rest_) (throw :exit t))))	 (org-table-toggle-column-width)	 nil))))  (should   (catch :exit     (org-test-with-temp-text "| a |<point>"       (cl-letf (((symbol-function 'read-string)		  (lambda (&rest_) (throw :exit t))))	 (org-table-toggle-column-width)	 nil))))  ;; When optional argument ARG is a string, toggle specified columns.  (should   (equal org-table-shrunk-column-indicator	  (org-test-with-temp-text "| <point>a | b |"	    (org-table-toggle-column-width "2")	    (overlay-get (car (overlays-at (- (point-max) 2))) 'display))))  (should   (equal '("b" "c")	  (org-test-with-temp-text "| a | b | c | d |"	    (org-table-toggle-column-width "2-3")	    (sort (mapcar (lambda (o) (overlay-get o 'help-echo))			  (overlays-in (point-min) (point-max)))		  #'string-lessp))))  (should   (equal '("b" "c" "d")	  (org-test-with-temp-text "| a | b | c | d |"	    (org-table-toggle-column-width "2-")	    (sort (mapcar (lambda (o) (overlay-get o 'help-echo))			  (overlays-in (point-min) (point-max)))		  #'string-lessp))))  (should   (equal '("a" "b")	  (org-test-with-temp-text "| a | b | c | d |"	    (org-table-toggle-column-width "-2")	    (sort (mapcar (lambda (o) (overlay-get o 'help-echo))			  (overlays-in (point-min) (point-max)))		  #'string-lessp))))  (should   (equal '("a" "b" "c" "d")	  (org-test-with-temp-text "| a | b | c | d |"	    (org-table-toggle-column-width "-")	    (sort (mapcar (lambda (o) (overlay-get o 'help-echo))			  (overlays-in (point-min) (point-max)))		  #'string-lessp))))  (should   (equal '("a" "d")	  (org-test-with-temp-text "| a | b | c | d |"	    (org-table-toggle-column-width "1-3")	    (org-table-toggle-column-width "2-4")	    (sort (mapcar (lambda (o) (overlay-get o 'help-echo))			  (overlays-in (point-min) (point-max)))		  #'string-lessp))))  ;; When ARG is (16), remove any column overlay.  (should-not   (org-test-with-temp-text "| <point>a |"     (org-table-toggle-column-width)     (org-table-toggle-column-width '(16))     (overlays-in (point-min) (point-max))))  (should-not   (org-test-with-temp-text "| a | b | c | d |"     (org-table-toggle-column-width "-")     (org-table-toggle-column-width '(16))     (overlays-in (point-min) (point-max)))))(ert-deftest test-org-table/shrunk-columns ()  "Test behaviour of shrunk column."  ;; Edition automatically expands a shrunk column.  (should-not   (org-test-with-temp-text "| <point>a |"     (org-table-toggle-column-width)     (insert "a")     (overlays-in (point-min) (point-max))))  ;; Other columns are not changed.  (should   (org-test-with-temp-text "| <point>a | b |"     (org-table-toggle-column-width "-")     (insert "a")     (overlays-in (point-min) (point-max))))  ;; Moving a shrunk column doesn't alter its state.  (should   (equal "a"	  (org-test-with-temp-text "| <point>a | b |"	    (org-table-toggle-column-width)	    (org-table-move-column-right)	    (overlay-get (car (overlays-at (point))) 'help-echo))))  (should   (equal "a"	  (org-test-with-temp-text "| <point>a |\n| b |"	    (org-table-toggle-column-width)	    (org-table-move-row-down)	    (overlay-get (car (overlays-at (point))) 'help-echo))))  ;; State is preserved upon inserting a column.  (should   (equal '("a")	  (org-test-with-temp-text "| <point>a |"	    (org-table-toggle-column-width)	    (org-table-insert-column)	    (sort (mapcar (lambda (o) (overlay-get o 'help-echo))			  (overlays-in (point-min) (point-max)))		  #'string-lessp))))  ;; State is preserved upon deleting a column.  (should   (equal '("a" "c")	  (org-test-with-temp-text "| a | <point>b | c |"	    (org-table-toggle-column-width "-")	    (org-table-delete-column)	    (sort (mapcar (lambda (o) (overlay-get o 'help-echo))			  (overlays-in (point-min) (point-max)))		  #'string-lessp))))  ;; State is preserved upon deleting a row.  (should   (equal '("b1" "b2")	  (org-test-with-temp-text "| a1 | a2 |\n| b1 | b2 |"	    (org-table-toggle-column-width "-")	    (org-table-kill-row)	    (sort (mapcar (lambda (o) (overlay-get o 'help-echo))			  (overlays-in (point-min) (point-max)))		  #'string-lessp))))  (should   (equal '("a1" "a2")	  (org-test-with-temp-text "| a1 | a2 |\n| <point>b1 | b2 |"	    (org-table-toggle-column-width "-")	    (org-table-kill-row)	    (sort (mapcar (lambda (o) (overlay-get o 'help-echo))			  (overlays-in (point-min) (point-max)))		  #'string-lessp))))  ;; State is preserved upon inserting a row or hline.  (should   (equal '("" "a1" "b1")	  (org-test-with-temp-text "| a1 | a2 |\n| <point>b1 | b2 |"	    (org-table-toggle-column-width)	    (org-table-insert-row)	    (sort (mapcar (lambda (o) (overlay-get o 'help-echo))			  (overlays-in (point-min) (point-max)))		  #'string-lessp))))  (should   (equal '("a1" "b1")	  (org-test-with-temp-text "| a1 | a2 |\n| <point>b1 | b2 |"	    (org-table-toggle-column-width)	    (org-table-insert-hline)	    (sort (mapcar (lambda (o) (overlay-get o 'help-echo))			  (overlays-in (point-min) (point-max)))		  #'string-lessp))))  ;; State is preserved upon sorting a column for all the columns but  ;; the one being sorted.  (should   (equal '("a2" "b2")	  (org-test-with-temp-text "| <point>a1 | a2 |\n| <point>b1 | b2 |"	    (org-table-toggle-column-width "-")	    (org-table-sort-lines nil ?A)	    (sort (mapcar (lambda (o) (overlay-get o 'help-echo))			  (overlays-in (point-min) (point-max)))		  #'string-lessp))))  ;; State is preserved upon replacing a field non-interactively.  (should   (equal '("a")	  (org-test-with-temp-text "| <point>a |"	    (org-table-toggle-column-width)	    (org-table-get-field nil "b")	    (mapcar (lambda (o) (overlay-get o 'help-echo))		    (overlays-in (point-min) (point-max))))))  ;; Moving to next field doesn't change shrunk state.  (should   (equal "a"	  (org-test-with-temp-text "| <point>a | b |"	    (org-table-toggle-column-width)	    (org-table-next-field)	    (overlay-get (car (overlays-at (1+ (line-beginning-position))))			 'help-echo))))  (should   (equal "b"	  (org-test-with-temp-text "| a | <point>b |"	    (org-table-toggle-column-width)	    (goto-char 2)	    (org-table-next-field)	    (overlay-get (car (overlays-at (point))) 'help-echo))))  ;; Aligning table doesn't alter shrunk state.  (should   (equal "a"	  (org-test-with-temp-text "| <point>a | b   |"	    (org-table-toggle-column-width)	    (org-table-align)	    (overlay-get (car (overlays-at (1+ (line-beginning-position))))			 'help-echo))))  (should   (equal "b"	  (org-test-with-temp-text "|---+-----|\n| a | <point>b   |"	    (org-table-toggle-column-width)	    (org-table-align)	    (overlay-get (car (overlays-at (point)))			 'help-echo))))  (should   (equal    '("b")    (org-test-with-temp-text "|---+-----|\n| a | <point>b   |"      (org-table-toggle-column-width)      (org-table-align)      (mapcar (lambda (o) (overlay-get o 'help-echo))	      (overlays-in (line-beginning-position) (line-end-position)))))));;; Miscellaneous(ert-deftest test-org-table/current-column ()  "Test `org-table-current-column' specifications."  (should   (= 1 (org-test-with-temp-text "| <point>a |"	  (org-table-current-column))))  (should   (= 1 (org-test-with-temp-text "|-<point>--|"	  (org-table-current-column))))  (should   (= 2 (org-test-with-temp-text "| 1 | <point>2 |"	  (org-table-current-column))))  (should   (= 2 (org-test-with-temp-text "|---+-<point>--|"	  (org-table-current-column)))))(ert-deftest test-org-table/get-field ()  "Test `org-table-get-field' specifications."  ;; Regular test.  (should   (equal " a "	  (org-test-with-temp-text "| <point>a |" (org-table-get-field))))  ;; Get field in open last column.  (should   (equal " a "	  (org-test-with-temp-text "| <point>a " (org-table-get-field))))  ;; Get empty field.  (should   (equal ""	  (org-test-with-temp-text "|<point>|" (org-table-get-field))))  (should   (equal " "	  (org-test-with-temp-text "| <point>|" (org-table-get-field))))  ;; Outside the table, return the empty string.  (should   (equal ""	  (org-test-with-temp-text "<point>| a |" (org-table-get-field))))  (should   (equal ""	  (org-test-with-temp-text "| a |<point>" (org-table-get-field))))  ;; With optional N argument, select a particular column in current  ;; row.  (should   (equal " 3 "	  (org-test-with-temp-text "| 1 | 2 | 3 |" (org-table-get-field 3))))  (should   (equal " 4 "	  (org-test-with-temp-text "| 1 | 2 |\n<point>| 3 | 4 |"	    (org-table-get-field 2))))  ;; REPLACE optional argument is used to replace selected field.  (should   (equal "| foo |"	  (org-test-with-temp-text "| <point>1 |"	    (org-table-get-field nil " foo ")	    (buffer-string))))  (should   (equal "| 1 | 2 | foo |"	  (org-test-with-temp-text "| 1 | 2 | 3 |"	    (org-table-get-field 3 " foo ")	    (buffer-string))))  (should   (equal " 4 "	  (org-test-with-temp-text "| 1 | 2 |\n<point>| 3 | 4 |"	    (org-table-get-field 2))))  ;; An empty REPLACE string clears the field.  (should   (equal "| |"	  (org-test-with-temp-text "| <point>1 |"	    (org-table-get-field nil "")	    (buffer-string))))  ;; When using REPLACE still return old value.  (should   (equal " 1 "	  (org-test-with-temp-text "| <point>1 |"	    (org-table-get-field nil " foo ")))))(provide 'test-org-table);;; test-org-table.el ends here
 |