123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615361636173618361936203621362236233624362536263627362836293630363136323633363436353636363736383639364036413642364336443645364636473648364936503651365236533654365536563657365836593660366136623663366436653666366736683669367036713672367336743675367636773678367936803681368236833684368536863687368836893690369136923693369436953696369736983699370037013702370337043705370637073708370937103711371237133714371537163717371837193720372137223723372437253726372737283729373037313732373337343735373637373738373937403741374237433744374537463747374837493750375137523753375437553756375737583759376037613762376337643765376637673768376937703771377237733774377537763777377837793780378137823783378437853786378737883789379037913792379337943795379637973798379938003801380238033804380538063807380838093810381138123813381438153816381738183819382038213822382338243825382638273828382938303831383238333834383538363837383838393840384138423843384438453846384738483849385038513852385338543855385638573858385938603861386238633864386538663867386838693870387138723873387438753876387738783879388038813882388338843885388638873888388938903891389238933894389538963897389838993900390139023903390439053906390739083909391039113912391339143915391639173918391939203921392239233924392539263927 |
- ;;; org-element.el --- Parser And Applications for Org syntax
- ;; Copyright (C) 2012 Free Software Foundation, Inc.
- ;; Author: Nicolas Goaziou <n.goaziou at gmail dot com>
- ;; Keywords: outlines, hypermedia, calendar, wp
- ;; 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.
- ;; This file is not part of GNU Emacs.
- ;; You should have received a copy of the GNU General Public License
- ;; along with this program. If not, see <http://www.gnu.org/licenses/>.
- ;;; Commentary:
- ;; Org syntax can be divided into three categories: "Greater
- ;; elements", "Elements" and "Objects".
- ;; An object can be defined anywhere on a line. It may span over more
- ;; than a line but never contains a blank one. Objects belong to the
- ;; following types: `emphasis', `entity', `export-snippet',
- ;; `footnote-reference', `inline-babel-call', `inline-src-block',
- ;; `latex-fragment', `line-break', `link', `macro', `radio-target',
- ;; `statistics-cookie', `subscript', `superscript', `target',
- ;; `time-stamp' and `verbatim'.
- ;; An element always starts and ends at the beginning of a line. The
- ;; only element's type containing objects is called a `paragraph'.
- ;; Other types are: `comment', `comment-block', `example-block',
- ;; `export-block', `fixed-width', `horizontal-rule', `keyword',
- ;; `latex-environment', `babel-call', `property-drawer',
- ;; `quote-section', `src-block', `table' and `verse-block'.
- ;; Elements containing paragraphs are called greater elements.
- ;; Concerned types are: `center-block', `drawer', `dynamic-block',
- ;; `footnote-definition', `headline', `inlinetask', `item',
- ;; `plain-list', `quote-block', `section' and `special-block'.
- ;; Greater elements (excepted `headline', `item' and `section' types)
- ;; and elements (excepted `keyword', `babel-call', and
- ;; `property-drawer' types) can have a fixed set of keywords as
- ;; attributes. Those are called "affiliated keywords", to distinguish
- ;; them from others keywords, which are full-fledged elements. In
- ;; particular, the "name" affiliated keyword allows to label almost
- ;; any element in an Org buffer.
- ;; Notwithstanding affiliated keywords, each greater element, element
- ;; and object has a fixed set of properties attached to it. Among
- ;; them, three are shared by all types: `:begin' and `:end', which
- ;; refer to the beginning and ending buffer positions of the
- ;; considered element or object, and `:post-blank', which holds the
- ;; number of blank lines, or white spaces, at its end.
- ;; Some elements also have special properties whose value can hold
- ;; objects themselves (i.e. an item tag, an headline name, a table
- ;; cell). Such values are called "secondary strings".
- ;; Lisp-wise, an element or an object can be represented as a list.
- ;; It follows the pattern (TYPE PROPERTIES CONTENTS), where:
- ;; TYPE is a symbol describing the Org element or object.
- ;; PROPERTIES is the property list attached to it. See docstring of
- ;; appropriate parsing function to get an exhaustive
- ;; list.
- ;; CONTENTS is a list of elements, objects or raw strings contained
- ;; in the current element or object, when applicable.
- ;; An Org buffer is a nested list of such elements and objects, whose
- ;; type is `org-data' and properties is nil.
- ;; The first part of this file implements a parser and an interpreter
- ;; for each type of Org syntax.
- ;; The next two parts introduce three accessors and a function
- ;; retrieving the smallest element starting at point (respectively
- ;; `org-element-type', `org-element-property', `org-element-contents'
- ;; and `org-element-current-element').
- ;; The following part creates a fully recursive buffer parser. It
- ;; also provides a tool to map a function to elements or objects
- ;; matching some criteria in the parse tree. Functions of interest
- ;; are `org-element-parse-buffer', `org-element-map' and, to a lesser
- ;; extent, `org-element-parse-secondary-string'.
- ;; The penultimate part is the cradle of an interpreter for the
- ;; obtained parse tree: `org-element-interpret-data' (and its
- ;; relative, `org-element-interpret-secondary').
- ;; The library ends by furnishing a set of interactive tools for
- ;; element's navigation and manipulation, mostly based on
- ;; `org-element-at-point' function.
- ;;; Code:
- (eval-when-compile (require 'cl))
- (require 'org)
- (declare-function org-inlinetask-goto-end "org-inlinetask" ())
- ;;; Greater elements
- ;; For each greater element type, we define a parser and an
- ;; interpreter.
- ;; A parser (`item''s excepted) accepts no argument and represents the
- ;; element or object as the list described above. An interpreter
- ;; accepts two arguments: the list representation of the element or
- ;; object, and its contents. The latter may be nil, depending on the
- ;; element or object considered. It returns the appropriate Org
- ;; syntax, as a string.
- ;; Parsing functions must follow the naming convention:
- ;; org-element-TYPE-parser, where TYPE is greater element's type, as
- ;; defined in `org-element-greater-elements'.
- ;;
- ;; Similarly, interpreting functions must follow the naming
- ;; convention: org-element-TYPE-interpreter.
- ;; With the exception of `headline' and `item' types, greater elements
- ;; cannot contain other greater elements of their own type.
- ;; Beside implementing a parser and an interpreter, adding a new
- ;; greater element requires to tweak `org-element-current-element'.
- ;; Moreover, the newly defined type must be added to both
- ;; `org-element-all-elements' and `org-element-greater-elements'.
- ;;;; Center Block
- (defun org-element-center-block-parser ()
- "Parse a center block.
- Return a list whose car is `center-block' and cdr is a plist
- containing `:begin', `:end', `:hiddenp', `:contents-begin',
- `:contents-end' and `:post-blank' keywords.
- Assume point is at beginning or end of the block."
- (save-excursion
- (let* ((case-fold-search t)
- (keywords (progn
- (end-of-line)
- (re-search-backward
- (concat "^[ \t]*#\\+begin_center") nil t)
- (org-element-collect-affiliated-keywords)))
- (begin (car keywords))
- (contents-begin (progn (forward-line) (point)))
- (hidden (org-truely-invisible-p))
- (contents-end (progn (re-search-forward
- (concat "^[ \t]*#\\+end_center") nil t)
- (point-at-bol)))
- (pos-before-blank (progn (forward-line) (point)))
- (end (progn (org-skip-whitespace)
- (if (eobp) (point) (point-at-bol)))))
- `(center-block
- (:begin ,begin
- :end ,end
- :hiddenp ,hidden
- :contents-begin ,contents-begin
- :contents-end ,contents-end
- :post-blank ,(count-lines pos-before-blank end)
- ,@(cadr keywords))))))
- (defun org-element-center-block-interpreter (center-block contents)
- "Interpret CENTER-BLOCK element as Org syntax.
- CONTENTS is the contents of the element."
- (format "#+begin_center\n%s#+end_center" contents))
- ;;;; Drawer
- (defun org-element-drawer-parser ()
- "Parse a drawer.
- Return a list whose car is `drawer' and cdr is a plist containing
- `:drawer-name', `:begin', `:end', `:hiddenp', `:contents-begin',
- `:contents-end' and `:post-blank' keywords.
- Assume point is at beginning of drawer."
- (save-excursion
- (let* ((case-fold-search t)
- (name (progn (looking-at org-drawer-regexp)
- (org-match-string-no-properties 1)))
- (keywords (org-element-collect-affiliated-keywords))
- (begin (car keywords))
- (contents-begin (progn (forward-line) (point)))
- (hidden (org-truely-invisible-p))
- (contents-end (progn (re-search-forward "^[ \t]*:END:" nil t)
- (point-at-bol)))
- (pos-before-blank (progn (forward-line) (point)))
- (end (progn (org-skip-whitespace)
- (if (eobp) (point) (point-at-bol)))))
- `(drawer
- (:begin ,begin
- :end ,end
- :drawer-name ,name
- :hiddenp ,hidden
- :contents-begin ,contents-begin
- :contents-end ,contents-end
- :post-blank ,(count-lines pos-before-blank end)
- ,@(cadr keywords))))))
- (defun org-element-drawer-interpreter (drawer contents)
- "Interpret DRAWER element as Org syntax.
- CONTENTS is the contents of the element."
- (format ":%s:\n%s:END:"
- (org-element-property :drawer-name drawer)
- contents))
- ;;;; Dynamic Block
- (defun org-element-dynamic-block-parser ()
- "Parse a dynamic block.
- Return a list whose car is `dynamic-block' and cdr is a plist
- containing `:block-name', `:begin', `:end', `:hiddenp',
- `:contents-begin', `:contents-end', `:arguments' and
- `:post-blank' keywords.
- Assume point is at beginning of dynamic block."
- (save-excursion
- (let* ((case-fold-search t)
- (name (progn (looking-at org-dblock-start-re)
- (org-match-string-no-properties 1)))
- (arguments (org-match-string-no-properties 3))
- (keywords (org-element-collect-affiliated-keywords))
- (begin (car keywords))
- (contents-begin (progn (forward-line) (point)))
- (hidden (org-truely-invisible-p))
- (contents-end (progn (re-search-forward org-dblock-end-re nil t)
- (point-at-bol)))
- (pos-before-blank (progn (forward-line) (point)))
- (end (progn (org-skip-whitespace)
- (if (eobp) (point) (point-at-bol)))))
- (list 'dynamic-block
- `(:begin ,begin
- :end ,end
- :block-name ,name
- :arguments ,arguments
- :hiddenp ,hidden
- :contents-begin ,contents-begin
- :contents-end ,contents-end
- :post-blank ,(count-lines pos-before-blank end)
- ,@(cadr keywords))))))
- (defun org-element-dynamic-block-interpreter (dynamic-block contents)
- "Interpret DYNAMIC-BLOCK element as Org syntax.
- CONTENTS is the contents of the element."
- (format "#+BEGIN: %s%s\n%s#+END:"
- (org-element-property :block-name dynamic-block)
- (let ((args (org-element-property :arguments dynamic-block)))
- (and arg (concat " " args)))
- contents))
- ;;;; Footnote Definition
- (defun org-element-footnote-definition-parser ()
- "Parse a footnote definition.
- Return a list whose car is `footnote-definition' and cdr is
- a plist containing `:label', `:begin' `:end', `:contents-begin',
- `:contents-end' and `:post-blank' keywords."
- (save-excursion
- (let* ((f-def (org-footnote-at-definition-p))
- (label (car f-def))
- (keywords (progn (goto-char (nth 1 f-def))
- (org-element-collect-affiliated-keywords)))
- (begin (car keywords))
- (contents-begin (progn (looking-at (concat "\\[" label "\\]"))
- (goto-char (match-end 0))
- (org-skip-whitespace)
- (point)))
- (end (goto-char (nth 2 f-def)))
- (contents-end (progn (skip-chars-backward " \r\t\n")
- (forward-line)
- (point))))
- `(footnote-definition
- (:label ,label
- :begin ,begin
- :end ,end
- :contents-begin ,contents-begin
- :contents-end ,contents-end
- :post-blank ,(count-lines contents-end end)
- ,@(cadr keywords))))))
- (defun org-element-footnote-definition-interpreter (footnote-definition contents)
- "Interpret FOOTNOTE-DEFINITION element as Org syntax.
- CONTENTS is the contents of the footnote-definition."
- (concat (format "[%s]" (org-element-property :label footnote-definition))
- " "
- contents))
- ;;;; Headline
- (defun org-element-headline-parser ()
- "Parse an headline.
- Return a list whose car is `headline' and cdr is a plist
- containing `:raw-value', `:title', `:begin', `:end',
- `:pre-blank', `:hiddenp', `:contents-begin' and `:contents-end',
- `:level', `:priority', `:tags', `:todo-keyword',`:todo-type',
- `:scheduled', `:deadline', `:timestamp', `:clock', `:category',
- `:quotedp', `:archivedp', `:commentedp' and `:footnote-section-p'
- keywords.
- The plist also contains any property set in the property drawer,
- with its name in lowercase, the underscores replaced with hyphens
- and colons at the beginning (i.e. `:custom-id').
- Assume point is at beginning of the headline."
- (save-excursion
- (let* ((components (org-heading-components))
- (level (nth 1 components))
- (todo (nth 2 components))
- (todo-type
- (and todo (if (member todo org-done-keywords) 'done 'todo)))
- (tags (nth 5 components))
- (raw-value (nth 4 components))
- (quotedp
- (let ((case-fold-search nil))
- (string-match (format "^%s +" org-quote-string) raw-value)))
- (commentedp
- (let ((case-fold-search nil))
- (string-match (format "^%s +" org-comment-string) raw-value)))
- (archivedp
- (and tags
- (let ((case-fold-search nil))
- (string-match (format ":%s:" org-archive-tag) tags))))
- (footnote-section-p (and org-footnote-section
- (string= org-footnote-section raw-value)))
- (standard-props (let (plist)
- (mapc
- (lambda (p)
- (let ((p-name (downcase (car p))))
- (while (string-match "_" p-name)
- (setq p-name
- (replace-match "-" nil nil p-name)))
- (setq p-name (intern (concat ":" p-name)))
- (setq plist
- (plist-put plist p-name (cdr p)))))
- (org-entry-properties nil 'standard))
- plist))
- (time-props (org-entry-properties nil 'special "CLOCK"))
- (scheduled (cdr (assoc "SCHEDULED" time-props)))
- (deadline (cdr (assoc "DEADLINE" time-props)))
- (clock (cdr (assoc "CLOCK" time-props)))
- (timestamp (cdr (assoc "TIMESTAMP" time-props)))
- (begin (point))
- (pos-after-head (save-excursion (forward-line) (point)))
- (contents-begin (save-excursion (forward-line)
- (org-skip-whitespace)
- (if (eobp) (point) (point-at-bol))))
- (hidden (save-excursion (forward-line) (org-truely-invisible-p)))
- (end (progn (goto-char (org-end-of-subtree t t))))
- (contents-end (progn (skip-chars-backward " \r\t\n")
- (forward-line)
- (point)))
- title)
- ;; Clean RAW-VALUE from any quote or comment string.
- (when (or quotedp commentedp)
- (setq raw-value
- (replace-regexp-in-string
- (concat "\\(" org-quote-string "\\|" org-comment-string "\\) +")
- ""
- raw-value)))
- ;; Clean TAGS from archive tag, if any.
- (when archivedp
- (setq tags
- (and (not (string= tags (format ":%s:" org-archive-tag)))
- (replace-regexp-in-string
- (concat org-archive-tag ":") "" tags)))
- (when (string= tags ":") (setq tags nil)))
- ;; Then get TITLE.
- (setq title (org-element-parse-secondary-string
- raw-value
- (cdr (assq 'headline org-element-string-restrictions))))
- `(headline
- (:raw-value ,raw-value
- :title ,title
- :begin ,begin
- :end ,end
- :pre-blank ,(count-lines pos-after-head contents-begin)
- :hiddenp ,hidden
- :contents-begin ,contents-begin
- :contents-end ,contents-end
- :level ,level
- :priority ,(nth 3 components)
- :tags ,tags
- :todo-keyword ,todo
- :todo-type ,todo-type
- :scheduled ,scheduled
- :deadline ,deadline
- :timestamp ,timestamp
- :clock ,clock
- :post-blank ,(count-lines contents-end end)
- :footnote-section-p ,footnote-section-p
- :archivedp ,archivedp
- :commentedp ,commentedp
- :quotedp ,quotedp
- ,@standard-props)))))
- (defun org-element-headline-interpreter (headline contents)
- "Interpret HEADLINE element as Org syntax.
- CONTENTS is the contents of the element."
- (let* ((level (org-element-property :level headline))
- (todo (org-element-property :todo-keyword headline))
- (priority (org-element-property :priority headline))
- (title (org-element-property :raw-value headline))
- (tags (let ((tag-string (org-element-property :tags headline))
- (archivedp (org-element-property :archivedp headline)))
- (cond
- ((and (not tag-string) archivedp)
- (format ":%s:" org-archive-tag))
- (archivedp (concat ":" org-archive-tag tag-string))
- (t tag-string))))
- (commentedp (org-element-property :commentedp headline))
- (quotedp (org-element-property :quotedp headline))
- (pre-blank (org-element-property :pre-blank headline))
- (heading (concat (make-string level ?*)
- (and todo (concat " " todo))
- (and quotedp (concat " " org-quote-string))
- (and commentedp (concat " " org-comment-string))
- (and priority (concat " " priority))
- (cond ((and org-footnote-section
- (org-element-property
- :footnote-section-p headline))
- (concat " " org-footnote-section))
- (title (concat " " title)))))
- ;; Align tags.
- (tags-fmt (when tags
- (let ((tags-len (length tags)))
- (format "%% %ds"
- (cond
- ((zerop org-tags-column) (1+ tags-len))
- ((< org-tags-column 0)
- (max (- (+ org-tags-column (length heading)))
- (1+ tags-len)))
- (t (max (+ (- org-tags-column (length heading))
- tags-len)
- (1+ tags-len)))))))))
- (concat heading (and tags (format tags-fmt tags))
- (make-string (1+ pre-blank) 10)
- contents)))
- ;;;; Inlinetask
- (defun org-element-inlinetask-parser ()
- "Parse an inline task.
- Return a list whose car is `inlinetask' and cdr is a plist
- containing `:raw-value', `:title', `:begin', `:end', `:hiddenp',
- `:contents-begin' and `:contents-end', `:level', `:priority',
- `:raw-value', `:tags', `:todo-keyword', `:todo-type',
- `:scheduled', `:deadline', `:timestamp', `:clock' and
- `:post-blank' keywords.
- The plist also contains any property set in the property drawer,
- with its name in lowercase, the underscores replaced with hyphens
- and colons at the beginning (i.e. `:custom-id').
- Assume point is at beginning of the inline task."
- (save-excursion
- (let* ((keywords (org-element-collect-affiliated-keywords))
- (begin (car keywords))
- (components (org-heading-components))
- (todo (nth 2 components))
- (todo-type (and todo
- (if (member todo org-done-keywords) 'done 'todo)))
- (raw-value (nth 4 components))
- (standard-props (let (plist)
- (mapc
- (lambda (p)
- (let ((p-name (downcase (car p))))
- (while (string-match "_" p-name)
- (setq p-name
- (replace-match "-" nil nil p-name)))
- (setq p-name (intern (concat ":" p-name)))
- (setq plist
- (plist-put plist p-name (cdr p)))))
- (org-entry-properties nil 'standard))
- plist))
- (time-props (org-entry-properties nil 'special "CLOCK"))
- (scheduled (cdr (assoc "SCHEDULED" time-props)))
- (deadline (cdr (assoc "DEADLINE" time-props)))
- (clock (cdr (assoc "CLOCK" time-props)))
- (timestamp (cdr (assoc "TIMESTAMP" time-props)))
- (title (org-element-parse-secondary-string
- raw-value
- (cdr (assq 'inlinetask org-element-string-restrictions))))
- (contents-begin (save-excursion (forward-line) (point)))
- (hidden (org-truely-invisible-p))
- (pos-before-blank (org-inlinetask-goto-end))
- ;; In the case of a single line task, CONTENTS-BEGIN and
- ;; CONTENTS-END might overlap.
- (contents-end (max contents-begin
- (save-excursion (forward-line -1) (point))))
- (end (progn (org-skip-whitespace)
- (if (eobp) (point) (point-at-bol)))))
- `(inlinetask
- (:raw-value ,raw-value
- :title ,title
- :begin ,begin
- :end ,end
- :hiddenp ,(and (> contents-end contents-begin) hidden)
- :contents-begin ,contents-begin
- :contents-end ,contents-end
- :level ,(nth 1 components)
- :priority ,(nth 3 components)
- :tags ,(nth 5 components)
- :todo-keyword ,todo
- :todo-type ,todo-type
- :scheduled ,scheduled
- :deadline ,deadline
- :timestamp ,timestamp
- :clock ,clock
- :post-blank ,(count-lines pos-before-blank end)
- ,@standard-props
- ,@(cadr keywords))))))
- (defun org-element-inlinetask-interpreter (inlinetask contents)
- "Interpret INLINETASK element as Org syntax.
- CONTENTS is the contents of inlinetask."
- (let* ((level (org-element-property :level inlinetask))
- (todo (org-element-property :todo-keyword inlinetask))
- (priority (org-element-property :priority inlinetask))
- (title (org-element-property :raw-value inlinetask))
- (tags (org-element-property :tags inlinetask))
- (task (concat (make-string level ?*)
- (and todo (concat " " todo))
- (and priority (concat " " priority))
- (and title (concat " " title))))
- ;; Align tags.
- (tags-fmt (when tags
- (format "%% %ds"
- (cond
- ((zerop org-tags-column) 1)
- ((< 0 org-tags-column)
- (max (+ org-tags-column
- (length inlinetask)
- (length tags))
- 1))
- (t (max (- org-tags-column (length inlinetask))
- 1)))))))
- (concat inlinetask (and tags (format tags-fmt tags) "\n" contents))))
- ;;;; Item
- (defun org-element-item-parser (struct)
- "Parse an item.
- STRUCT is the structure of the plain list.
- Return a list whose car is `item' and cdr is a plist containing
- `:bullet', `:begin', `:end', `:contents-begin', `:contents-end',
- `:checkbox', `:counter', `:tag', `:raw-tag', `:structure',
- `:hiddenp' and `:post-blank' keywords.
- Assume point is at the beginning of the item."
- (save-excursion
- (beginning-of-line)
- (let* ((begin (point))
- (bullet (org-list-get-bullet (point) struct))
- (checkbox (let ((box (org-list-get-checkbox begin struct)))
- (cond ((equal "[ ]" box) 'off)
- ((equal "[X]" box) 'on)
- ((equal "[-]" box) 'trans))))
- (counter (let ((c (org-list-get-counter begin struct)))
- (cond
- ((not c) nil)
- ((string-match "[A-Za-z]" c)
- (- (string-to-char (upcase (match-string 0 c)))
- 64))
- ((string-match "[0-9]+" c)
- (string-to-number (match-string 0 c))))))
- (raw-tag (org-list-get-tag begin struct))
- (tag (and raw-tag
- (org-element-parse-secondary-string
- raw-tag
- (cdr (assq 'item org-element-string-restrictions)))))
- (end (org-list-get-item-end begin struct))
- (contents-begin (progn (looking-at org-list-full-item-re)
- (goto-char (match-end 0))
- (org-skip-whitespace)
- ;; If first line isn't empty,
- ;; contents really start at the text
- ;; after item's meta-data.
- (if (= (point-at-bol) begin) (point)
- (point-at-bol))))
- (hidden (progn (forward-line)
- (and (not (= (point) end))
- (org-truely-invisible-p))))
- (contents-end (progn (goto-char end)
- (skip-chars-backward " \r\t\n")
- (forward-line)
- (point))))
- `(item
- (:bullet ,bullet
- :begin ,begin
- :end ,end
- ;; CONTENTS-BEGIN and CONTENTS-END may be mixed
- ;; up in the case of an empty item separated
- ;; from the next by a blank line. Thus, ensure
- ;; the former is always the smallest of two.
- :contents-begin ,(min contents-begin contents-end)
- :contents-end ,(max contents-begin contents-end)
- :checkbox ,checkbox
- :counter ,counter
- :raw-tag ,raw-tag
- :tag ,tag
- :hiddenp ,hidden
- :structure ,struct
- :post-blank ,(count-lines contents-end end))))))
- (defun org-element-item-interpreter (item contents)
- "Interpret ITEM element as Org syntax.
- CONTENTS is the contents of the element."
- (let* ((bullet
- (let* ((beg (org-element-property :begin item))
- (struct (org-element-property :structure item))
- (pre (org-list-prevs-alist struct))
- (bul (org-element-property :bullet item)))
- (org-list-bullet-string
- (if (not (eq (org-list-get-list-type beg struct pre) 'ordered)) "-"
- (let ((num
- (car
- (last
- (org-list-get-item-number
- beg struct pre (org-list-parents-alist struct))))))
- (format "%d%s"
- num
- (if (eq org-plain-list-ordered-item-terminator ?\)) ")"
- ".")))))))
- (checkbox (org-element-property :checkbox item))
- (counter (org-element-property :counter item))
- (tag (org-element-property :raw-tag item))
- ;; Compute indentation.
- (ind (make-string (length bullet) 32)))
- ;; Indent contents.
- (concat
- bullet
- (and counter (format "[@%d] " counter))
- (cond
- ((eq checkbox 'on) "[X] ")
- ((eq checkbox 'off) "[ ] ")
- ((eq checkbox 'trans) "[-] "))
- (and tag (format "%s :: " tag))
- (org-trim
- (replace-regexp-in-string "\\(^\\)[ \t]*\\S-" ind contents nil nil 1)))))
- ;;;; Plain List
- (defun org-element-plain-list-parser (&optional structure)
- "Parse a plain list.
- Optional argument STRUCTURE, when non-nil, is the structure of
- the plain list being parsed.
- Return a list whose car is `plain-list' and cdr is a plist
- containing `:type', `:begin', `:end', `:contents-begin' and
- `:contents-end', `:level', `:structure' and `:post-blank'
- keywords.
- Assume point is at one of the list items."
- (save-excursion
- (let* ((struct (or structure (org-list-struct)))
- (prevs (org-list-prevs-alist struct))
- (parents (org-list-parents-alist struct))
- (type (org-list-get-list-type (point) struct prevs))
- (contents-begin (goto-char
- (org-list-get-list-begin (point) struct prevs)))
- (keywords (org-element-collect-affiliated-keywords))
- (begin (car keywords))
- (contents-end (goto-char
- (org-list-get-list-end (point) struct prevs)))
- (end (save-excursion (org-skip-whitespace)
- (if (eobp) (point) (point-at-bol))))
- (level 0))
- ;; Get list level.
- (let ((item contents-begin))
- (while (setq item
- (org-list-get-parent
- (org-list-get-list-begin item struct prevs)
- struct parents))
- (incf level)))
- ;; Blank lines below list belong to the top-level list only.
- (when (> level 0)
- (setq end (min (org-list-get-bottom-point struct)
- (progn (org-skip-whitespace)
- (if (eobp) (point) (point-at-bol))))))
- ;; Return value.
- `(plain-list
- (:type ,type
- :begin ,begin
- :end ,end
- :contents-begin ,contents-begin
- :contents-end ,contents-end
- :level ,level
- :structure ,struct
- :post-blank ,(count-lines contents-end end)
- ,@(cadr keywords))))))
- (defun org-element-plain-list-interpreter (plain-list contents)
- "Interpret PLAIN-LIST element as Org syntax.
- CONTENTS is the contents of the element."
- contents)
- ;;;; Quote Block
- (defun org-element-quote-block-parser ()
- "Parse a quote block.
- Return a list whose car is `quote-block' and cdr is a plist
- containing `:begin', `:end', `:hiddenp', `:contents-begin',
- `:contents-end' and `:post-blank' keywords.
- Assume point is at beginning or end of the block."
- (save-excursion
- (let* ((case-fold-search t)
- (keywords (progn
- (end-of-line)
- (re-search-backward
- (concat "^[ \t]*#\\+begin_quote") nil t)
- (org-element-collect-affiliated-keywords)))
- (begin (car keywords))
- (contents-begin (progn (forward-line) (point)))
- (hidden (org-truely-invisible-p))
- (contents-end (progn (re-search-forward
- (concat "^[ \t]*#\\+end_quote") nil t)
- (point-at-bol)))
- (pos-before-blank (progn (forward-line) (point)))
- (end (progn (org-skip-whitespace)
- (if (eobp) (point) (point-at-bol)))))
- `(quote-block
- (:begin ,begin
- :end ,end
- :hiddenp ,hidden
- :contents-begin ,contents-begin
- :contents-end ,contents-end
- :post-blank ,(count-lines pos-before-blank end)
- ,@(cadr keywords))))))
- (defun org-element-quote-block-interpreter (quote-block contents)
- "Interpret QUOTE-BLOCK element as Org syntax.
- CONTENTS is the contents of the element."
- (format "#+begin_quote\n%s#+end_quote" contents))
- ;;;; Section
- (defun org-element-section-parser ()
- "Parse a section.
- Return a list whose car is `section' and cdr is a plist
- containing `:begin', `:end', `:contents-begin', `contents-end'
- and `:post-blank' keywords."
- (save-excursion
- ;; Beginning of section is the beginning of the first non-blank
- ;; line after previous headline.
- (org-with-limited-levels
- (let ((begin
- (save-excursion
- (outline-previous-heading)
- (if (not (org-at-heading-p)) (point)
- (forward-line) (org-skip-whitespace) (point-at-bol))))
- (end (progn (outline-next-heading) (point)))
- (pos-before-blank (progn (skip-chars-backward " \r\t\n")
- (forward-line)
- (point))))
- `(section
- (:begin ,begin
- :end ,end
- :contents-begin ,begin
- :contents-end ,pos-before-blank
- :post-blank ,(count-lines pos-before-blank end)))))))
- (defun org-element-section-interpreter (section contents)
- "Interpret SECTION element as Org syntax.
- CONTENTS is the contents of the element."
- contents)
- ;;;; Special Block
- (defun org-element-special-block-parser ()
- "Parse a special block.
- Return a list whose car is `special-block' and cdr is a plist
- containing `:type', `:begin', `:end', `:hiddenp',
- `:contents-begin', `:contents-end' and `:post-blank' keywords.
- Assume point is at beginning or end of the block."
- (save-excursion
- (let* ((case-fold-search t)
- (type (progn (looking-at
- "[ \t]*#\\+\\(?:begin\\|end\\)_\\([-A-Za-z0-9]+\\)")
- (org-match-string-no-properties 1)))
- (keywords (progn
- (end-of-line)
- (re-search-backward
- (concat "^[ \t]*#\\+begin_" type) nil t)
- (org-element-collect-affiliated-keywords)))
- (begin (car keywords))
- (contents-begin (progn (forward-line) (point)))
- (hidden (org-truely-invisible-p))
- (contents-end (progn (re-search-forward
- (concat "^[ \t]*#\\+end_" type) nil t)
- (point-at-bol)))
- (pos-before-blank (progn (forward-line) (point)))
- (end (progn (org-skip-whitespace)
- (if (eobp) (point) (point-at-bol)))))
- `(special-block
- (:type ,type
- :begin ,begin
- :end ,end
- :hiddenp ,hidden
- :contents-begin ,contents-begin
- :contents-end ,contents-end
- :post-blank ,(count-lines pos-before-blank end)
- ,@(cadr keywords))))))
- (defun org-element-special-block-interpreter (special-block contents)
- "Interpret SPECIAL-BLOCK element as Org syntax.
- CONTENTS is the contents of the element."
- (let ((block-type (org-element-property :type special-block)))
- (format "#+begin_%s\n%s#+end_%s" block-type contents block-type)))
- ;;; Elements
- ;; For each element, a parser and an interpreter are also defined.
- ;; Both follow the same naming convention used for greater elements.
- ;; Also, as for greater elements, adding a new element type is done
- ;; through the following steps: implement a parser and an interpreter,
- ;; tweak `org-element-current-element' so that it recognizes the new
- ;; type and add that new type to `org-element-all-elements'.
- ;; As a special case, when the newly defined type is a block type,
- ;; `org-element-non-recursive-block-alist' has to be modified
- ;; accordingly.
- ;;;; Babel Call
- (defun org-element-babel-call-parser ()
- "Parse a babel call.
- Return a list whose car is `babel-call' and cdr is a plist
- containing `:begin', `:end', `:info' and `:post-blank' as
- keywords."
- (save-excursion
- (let ((info (progn (looking-at org-babel-block-lob-one-liner-regexp)
- (org-babel-lob-get-info)))
- (beg (point-at-bol))
- (pos-before-blank (progn (forward-line) (point)))
- (end (progn (org-skip-whitespace)
- (if (eobp) (point) (point-at-bol)))))
- `(babel-call
- (:beg ,beg
- :end ,end
- :info ,info
- :post-blank ,(count-lines pos-before-blank end))))))
- (defun org-element-babel-call-interpreter (inline-babel-call contents)
- "Interpret INLINE-BABEL-CALL object as Org syntax.
- CONTENTS is nil."
- (let* ((babel-info (org-element-property :info inline-babel-call))
- (main-source (car babel-info))
- (post-options (nth 1 babel-info)))
- (concat "#+call: "
- (if (string-match "\\[\\(\\[.*?\\]\\)\\]" main-source)
- ;; Remove redundant square brackets.
- (replace-match
- (match-string 1 main-source) nil nil main-source)
- main-source)
- (and post-options (format "[%s]" post-options)))))
- ;;;; Comment
- (defun org-element-comment-parser ()
- "Parse a comment.
- Return a list whose car is `comment' and cdr is a plist
- containing `:begin', `:end', `:value' and `:post-blank'
- keywords."
- (let (beg-coms begin end end-coms keywords)
- (save-excursion
- (if (looking-at "#")
- ;; First type of comment: comments at column 0.
- (let ((comment-re "^\\([^#]\\|#\\+[a-z]\\)"))
- (save-excursion
- (re-search-backward comment-re nil 'move)
- (if (bobp) (setq keywords nil beg-coms (point))
- (forward-line)
- (setq keywords (org-element-collect-affiliated-keywords)
- beg-coms (point))))
- (re-search-forward comment-re nil 'move)
- (setq end-coms (if (eobp) (point) (match-beginning 0))))
- ;; Second type of comment: indented comments.
- (let ((comment-re "[ \t]*#\\+\\(?: \\|$\\)"))
- (unless (bobp)
- (while (and (not (bobp)) (looking-at comment-re))
- (forward-line -1))
- (unless (looking-at comment-re) (forward-line)))
- (setq beg-coms (point))
- (setq keywords (org-element-collect-affiliated-keywords))
- ;; Get comments ending. This may not be accurate if
- ;; commented lines within an item are followed by commented
- ;; lines outside of the list. Though, parser will always
- ;; get it right as it already knows surrounding element and
- ;; has narrowed buffer to its contents.
- (while (looking-at comment-re) (forward-line))
- (setq end-coms (point))))
- ;; Find position after blank.
- (goto-char end-coms)
- (org-skip-whitespace)
- (setq end (if (eobp) (point) (point-at-bol))))
- `(comment
- (:begin ,(or (car keywords) beg-coms)
- :end ,end
- :value ,(buffer-substring-no-properties beg-coms end-coms)
- :post-blank ,(count-lines end-coms end)
- ,@(cadr keywords)))))
- (defun org-element-comment-interpreter (comment contents)
- "Interpret COMMENT element as Org syntax.
- CONTENTS is nil."
- (org-element-property :value comment))
- ;;;; Comment Block
- (defun org-element-comment-block-parser ()
- "Parse an export block.
- Return a list whose car is `comment-block' and cdr is a plist
- containing `:begin', `:end', `:hiddenp', `:value' and
- `:post-blank' keywords."
- (save-excursion
- (end-of-line)
- (let* ((case-fold-search t)
- (keywords (progn
- (re-search-backward "^[ \t]*#\\+begin_comment" nil t)
- (org-element-collect-affiliated-keywords)))
- (begin (car keywords))
- (contents-begin (progn (forward-line) (point)))
- (hidden (org-truely-invisible-p))
- (contents-end (progn (re-search-forward
- "^[ \t]*#\\+end_comment" nil t)
- (point-at-bol)))
- (pos-before-blank (progn (forward-line) (point)))
- (end (progn (org-skip-whitespace)
- (if (eobp) (point) (point-at-bol))))
- (value (buffer-substring-no-properties contents-begin contents-end)))
- `(comment-block
- (:begin ,begin
- :end ,end
- :value ,value
- :hiddenp ,hidden
- :post-blank ,(count-lines pos-before-blank end)
- ,@(cadr keywords))))))
- (defun org-element-comment-block-interpreter (comment-block contents)
- "Interpret COMMENT-BLOCK element as Org syntax.
- CONTENTS is nil."
- (concat "#+begin_comment\n"
- (org-remove-indentation
- (org-element-property :value comment-block))
- "#+begin_comment"))
- ;;;; Example Block
- (defun org-element-example-block-parser ()
- "Parse an example block.
- Return a list whose car is `example-block' and cdr is a plist
- containing `:begin', `:end', `:number-lines', `:preserve-indent',
- `:retain-labels', `:use-labels', `:label-fmt', `:hiddenp',
- `:switches', `:value' and `:post-blank' keywords."
- (save-excursion
- (end-of-line)
- (let* ((case-fold-search t)
- (switches (progn
- (re-search-backward
- "^[ \t]*#\\+BEGIN_EXAMPLE\\(?: +\\(.*\\)\\)?" nil t)
- (org-match-string-no-properties 1)))
- ;; Switches analysis
- (number-lines (cond ((not switches) nil)
- ((string-match "-n\\>" switches) 'new)
- ((string-match "+n\\>" switches) 'continued)))
- (preserve-indent (and switches (string-match "-i\\>" switches)))
- ;; Should labels be retained in (or stripped from) example
- ;; blocks?
- (retain-labels
- (or (not switches)
- (not (string-match "-r\\>" switches))
- (and number-lines (string-match "-k\\>" switches))))
- ;; What should code-references use - labels or
- ;; line-numbers?
- (use-labels
- (or (not switches)
- (and retain-labels (not (string-match "-k\\>" switches)))))
- (label-fmt (and switches
- (string-match "-l +\"\\([^\"\n]+\\)\"" switches)
- (match-string 1 switches)))
- ;; Standard block parsing.
- (keywords (org-element-collect-affiliated-keywords))
- (begin (car keywords))
- (contents-begin (progn (forward-line) (point)))
- (hidden (org-truely-invisible-p))
- (contents-end (progn
- (re-search-forward "^[ \t]*#\\+END_EXAMPLE" nil t)
- (point-at-bol)))
- (value (buffer-substring-no-properties contents-begin contents-end))
- (pos-before-blank (progn (forward-line) (point)))
- (end (progn (org-skip-whitespace)
- (if (eobp) (point) (point-at-bol)))))
- `(example-block
- (:begin ,begin
- :end ,end
- :value ,value
- :switches ,switches
- :number-lines ,number-lines
- :preserve-indent ,preserve-indent
- :retain-labels ,retain-labels
- :use-labels ,use-labels
- :label-fmt ,label-fmt
- :hiddenp ,hidden
- :post-blank ,(count-lines pos-before-blank end)
- ,@(cadr keywords))))))
- (defun org-element-example-block-interpreter (example-block contents)
- "Interpret EXAMPLE-BLOCK element as Org syntax.
- CONTENTS is nil."
- (let ((options (org-element-property :options example-block)))
- (concat "#+begin_example" (and options (concat " " options)) "\n"
- (org-remove-indentation
- (org-element-property :value example-block))
- "#+end_example")))
- ;;;; Export Block
- (defun org-element-export-block-parser ()
- "Parse an export block.
- Return a list whose car is `export-block' and cdr is a plist
- containing `:begin', `:end', `:type', `:hiddenp', `:value' and
- `:post-blank' keywords."
- (save-excursion
- (end-of-line)
- (let* ((case-fold-search t)
- (contents)
- (type (progn (re-search-backward
- (concat "[ \t]*#\\+begin_"
- (org-re "\\([[:alnum:]]+\\)")))
- (downcase (org-match-string-no-properties 1))))
- (keywords (org-element-collect-affiliated-keywords))
- (begin (car keywords))
- (contents-begin (progn (forward-line) (point)))
- (hidden (org-truely-invisible-p))
- (contents-end (progn (re-search-forward
- (concat "^[ \t]*#\\+end_" type) nil t)
- (point-at-bol)))
- (pos-before-blank (progn (forward-line) (point)))
- (end (progn (org-skip-whitespace)
- (if (eobp) (point) (point-at-bol))))
- (value (buffer-substring-no-properties contents-begin contents-end)))
- `(export-block
- (:begin ,begin
- :end ,end
- :type ,type
- :value ,value
- :hiddenp ,hidden
- :post-blank ,(count-lines pos-before-blank end)
- ,@(cadr keywords))))))
- (defun org-element-export-block-interpreter (export-block contents)
- "Interpret EXPORT-BLOCK element as Org syntax.
- CONTENTS is nil."
- (let ((type (org-element-property :type export-block)))
- (concat (format "#+begin_%s\n" type)
- (org-element-property :value export-block)
- (format "#+end_%s" type))))
- ;;;; Fixed-width
- (defun org-element-fixed-width-parser ()
- "Parse a fixed-width section.
- Return a list whose car is `fixed-width' and cdr is a plist
- containing `:begin', `:end', `:value' and `:post-blank'
- keywords."
- (let ((fixed-re "[ \t]*:\\( \\|$\\)")
- beg-area begin end value pos-before-blank keywords)
- (save-excursion
- ;; Move to the beginning of the fixed-width area.
- (unless (bobp)
- (while (and (not (bobp)) (looking-at fixed-re))
- (forward-line -1))
- (unless (looking-at fixed-re) (forward-line 1)))
- (setq beg-area (point))
- ;; Get affiliated keywords, if any.
- (setq keywords (org-element-collect-affiliated-keywords))
- ;; Store true beginning of element.
- (setq begin (car keywords))
- ;; Get ending of fixed-width area. If point is in a list,
- ;; ensure to not get outside of it.
- (let* ((itemp (org-in-item-p))
- (max-pos (if itemp
- (org-list-get-bottom-point
- (save-excursion (goto-char itemp) (org-list-struct)))
- (point-max))))
- (while (and (looking-at fixed-re) (< (point) max-pos))
- (forward-line)))
- (setq pos-before-blank (point))
- ;; Find position after blank
- (org-skip-whitespace)
- (setq end (if (eobp) (point) (point-at-bol)))
- ;; Extract value.
- (setq value (buffer-substring-no-properties beg-area pos-before-blank)))
- `(fixed-width
- (:begin ,begin
- :end ,end
- :value ,value
- :post-blank ,(count-lines pos-before-blank end)
- ,@(cadr keywords)))))
- (defun org-element-fixed-width-interpreter (fixed-width contents)
- "Interpret FIXED-WIDTH element as Org syntax.
- CONTENTS is nil."
- (org-remove-indentation (org-element-property :value fixed-width)))
- ;;;; Horizontal Rule
- (defun org-element-horizontal-rule-parser ()
- "Parse an horizontal rule.
- Return a list whose car is `horizontal-rule' and cdr is
- a plist containing `:begin', `:end' and `:post-blank'
- keywords."
- (save-excursion
- (let* ((keywords (org-element-collect-affiliated-keywords))
- (begin (car keywords))
- (post-hr (progn (forward-line) (point)))
- (end (progn (org-skip-whitespace)
- (if (eobp) (point) (point-at-bol)))))
- `(horizontal-rule
- (:begin ,begin
- :end ,end
- :post-blank ,(count-lines post-hr end)
- ,@(cadr keywords))))))
- (defun org-element-horizontal-rule-interpreter (horizontal-rule contents)
- "Interpret HORIZONTAL-RULE element as Org syntax.
- CONTENTS is nil."
- "-----")
- ;;;; Keyword
- (defun org-element-keyword-parser ()
- "Parse a keyword at point.
- Return a list whose car is `keyword' and cdr is a plist
- containing `:key', `:value', `:begin', `:end' and `:post-blank'
- keywords."
- (save-excursion
- (let* ((begin (point))
- (key (progn (looking-at
- "[ \t]*#\\+\\(\\(?:[a-z]+\\)\\(?:_[a-z]+\\)*\\):")
- (org-match-string-no-properties 1)))
- (value (org-trim (buffer-substring-no-properties
- (match-end 0) (point-at-eol))))
- (pos-before-blank (progn (forward-line) (point)))
- (end (progn (org-skip-whitespace)
- (if (eobp) (point) (point-at-bol)))))
- `(keyword
- (:key ,key
- :value ,value
- :begin ,begin
- :end ,end
- :post-blank ,(count-lines pos-before-blank end))))))
- (defun org-element-keyword-interpreter (keyword contents)
- "Interpret KEYWORD element as Org syntax.
- CONTENTS is nil."
- (format "#+%s: %s"
- (org-element-property :key keyword)
- (org-element-property :value keyword)))
- ;;;; Latex Environment
- (defun org-element-latex-environment-parser ()
- "Parse a LaTeX environment.
- Return a list whose car is `latex-environment' and cdr is a plist
- containing `:begin', `:end', `:value' and `:post-blank' keywords."
- (save-excursion
- (end-of-line)
- (let* ((case-fold-search t)
- (contents-begin (re-search-backward "^[ \t]*\\\\begin" nil t))
- (keywords (org-element-collect-affiliated-keywords))
- (begin (car keywords))
- (contents-end (progn (re-search-forward "^[ \t]*\\\\end")
- (forward-line)
- (point)))
- (value (buffer-substring-no-properties contents-begin contents-end))
- (end (progn (org-skip-whitespace)
- (if (eobp) (point) (point-at-bol)))))
- `(latex-environment
- (:begin ,begin
- :end ,end
- :value ,value
- :post-blank ,(count-lines contents-end end)
- ,@(cadr keywords))))))
- (defun org-element-latex-environment-interpreter (latex-environment contents)
- "Interpret LATEX-ENVIRONMENT element as Org syntax.
- CONTENTS is nil."
- (org-element-property :value latex-environment))
- ;;;; Paragraph
- (defun org-element-paragraph-parser ()
- "Parse a paragraph.
- Return a list whose car is `paragraph' and cdr is a plist
- containing `:begin', `:end', `:contents-begin' and
- `:contents-end' and `:post-blank' keywords.
- Assume point is at the beginning of the paragraph."
- (save-excursion
- (let* ((contents-begin (point))
- (keywords (org-element-collect-affiliated-keywords))
- (begin (car keywords))
- (contents-end (progn
- (end-of-line)
- (if (re-search-forward
- org-element-paragraph-separate nil 'm)
- (progn (forward-line -1) (end-of-line) (point))
- (point))))
- (pos-before-blank (progn (forward-line) (point)))
- (end (progn (org-skip-whitespace)
- (if (eobp) (point) (point-at-bol)))))
- `(paragraph
- (:begin ,begin
- :end ,end
- :contents-begin ,contents-begin
- :contents-end ,contents-end
- :post-blank ,(count-lines pos-before-blank end)
- ,@(cadr keywords))))))
- (defun org-element-paragraph-interpreter (paragraph contents)
- "Interpret PARAGRAPH element as Org syntax.
- CONTENTS is the contents of the element."
- contents)
- ;;;; Property Drawer
- (defun org-element-property-drawer-parser ()
- "Parse a property drawer.
- Return a list whose car is `property-drawer' and cdr is a plist
- containing `:begin', `:end', `:hiddenp', `:contents-begin',
- `:contents-end', `:properties' and `:post-blank' keywords."
- (save-excursion
- (let ((case-fold-search t)
- (begin (progn (end-of-line)
- (re-search-backward org-property-start-re)
- (match-beginning 0)))
- (contents-begin (progn (forward-line) (point)))
- (hidden (org-truely-invisible-p))
- (properties (let (val)
- (while (not (looking-at "^[ \t]*:END:"))
- (when (looking-at
- (org-re
- "[ \t]*:\\([[:alpha:]][[:alnum:]_-]*\\):"))
- (push (cons (match-string 1)
- (org-trim
- (buffer-substring
- (match-end 0) (point-at-eol))))
- val))
- (forward-line))
- val))
- (contents-end (progn (re-search-forward "^[ \t]*:END:" nil t)
- (point-at-bol)))
- (pos-before-blank (progn (forward-line) (point)))
- (end (progn (org-skip-whitespace)
- (if (eobp) (point) (point-at-bol)))))
- `(property-drawer
- (:begin ,begin
- :end ,end
- :hiddenp ,hidden
- :properties ,properties
- :post-blank ,(count-lines pos-before-blank end))))))
- (defun org-element-property-drawer-interpreter (property-drawer contents)
- "Interpret PROPERTY-DRAWER element as Org syntax.
- CONTENTS is nil."
- (let ((props (org-element-property :properties property-drawer)))
- (concat
- ":PROPERTIES:\n"
- (mapconcat (lambda (p)
- (format org-property-format (format ":%s:" (car p)) (cdr p)))
- (nreverse props) "\n")
- "\n:END:")))
- ;;;; Quote Section
- (defun org-element-quote-section-parser ()
- "Parse a quote section.
- Return a list whose car is `quote-section' and cdr is a plist
- containing `:begin', `:end', `:value' and `:post-blank'
- keywords.
- Assume point is at beginning of the section."
- (save-excursion
- (let* ((begin (point))
- (end (progn (org-with-limited-levels (outline-next-heading))
- (point)))
- (pos-before-blank (progn (skip-chars-backward " \r\t\n")
- (forward-line)
- (point)))
- (value (buffer-substring-no-properties begin pos-before-blank)))
- `(quote-section
- (:begin ,begin
- :end ,end
- :value ,value
- :post-blank ,(count-lines pos-before-blank end))))))
- (defun org-element-quote-section-interpreter (quote-section contents)
- "Interpret QUOTE-SECTION element as Org syntax.
- CONTENTS is nil."
- (org-element-property :value quote-section))
- ;;;; Src Block
- (defun org-element-src-block-parser ()
- "Parse a src block.
- Return a list whose car is `src-block' and cdr is a plist
- containing `:language', `:switches', `:parameters', `:begin',
- `:end', `:hiddenp', `:contents-begin', `:contents-end',
- `:number-lines', `:retain-labels', `:use-labels', `:label-fmt',
- `:preserve-indent', `:value' and `:post-blank' keywords."
- (save-excursion
- (end-of-line)
- (let* ((case-fold-search t)
- ;; Get position at beginning of block.
- (contents-begin
- (re-search-backward
- (concat
- "^[ \t]*#\\+BEGIN_SRC"
- "\\(?: +\\(\\S-+\\)\\)?" ; language
- "\\(\\(?: +\\(?:-l \".*?\"\\|[-+][A-Za-z]\\)\\)*\\)" ; switches
- "\\(.*\\)[ \t]*$") ; parameters
- nil t))
- ;; Get language as a string.
- (language (org-match-string-no-properties 1))
- ;; Get parameters.
- (parameters (org-trim (org-match-string-no-properties 3)))
- ;; Get switches.
- (switches (org-match-string-no-properties 2))
- ;; Switches analysis
- (number-lines (cond ((not switches) nil)
- ((string-match "-n\\>" switches) 'new)
- ((string-match "+n\\>" switches) 'continued)))
- (preserve-indent (and switches (string-match "-i\\>" switches)))
- (label-fmt (and switches
- (string-match "-l +\"\\([^\"\n]+\\)\"" switches)
- (match-string 1 switches)))
- ;; Should labels be retained in (or stripped from) src
- ;; blocks?
- (retain-labels
- (or (not switches)
- (not (string-match "-r\\>" switches))
- (and number-lines (string-match "-k\\>" switches))))
- ;; What should code-references use - labels or
- ;; line-numbers?
- (use-labels
- (or (not switches)
- (and retain-labels (not (string-match "-k\\>" switches)))))
- ;; Get affiliated keywords.
- (keywords (org-element-collect-affiliated-keywords))
- ;; Get beginning position.
- (begin (car keywords))
- ;; Get position at end of block.
- (contents-end (progn (re-search-forward "^[ \t]*#\\+END_SRC" nil t)
- (forward-line)
- (point)))
- ;; Retrieve code.
- (value (buffer-substring-no-properties
- (save-excursion (goto-char contents-begin)
- (forward-line)
- (point))
- (match-beginning 0)))
- ;; Get position after ending blank lines.
- (end (progn (org-skip-whitespace)
- (if (eobp) (point) (point-at-bol))))
- ;; Get visibility status.
- (hidden (progn (goto-char contents-begin)
- (forward-line)
- (org-truely-invisible-p))))
- `(src-block
- (:language ,language
- :switches ,switches
- :parameters ,parameters
- :begin ,begin
- :end ,end
- :number-lines ,number-lines
- :preserve-indent ,preserve-indent
- :retain-labels ,retain-labels
- :use-labels ,use-labels
- :label-fmt ,label-fmt
- :hiddenp ,hidden
- :value ,value
- :post-blank ,(count-lines contents-end end)
- ,@(cadr keywords))))))
- (defun org-element-src-block-interpreter (src-block contents)
- "Interpret SRC-BLOCK element as Org syntax.
- CONTENTS is nil."
- (let ((lang (org-element-property :language src-block))
- (switches (org-element-property :switches src-block))
- (params (org-element-property :parameters src-block))
- (value (let ((val (org-element-property :value src-block)))
- (cond
- (org-src-preserve-indentation val)
- ((zerop org-edit-src-content-indentation)
- (org-remove-indentation val))
- (t
- (let ((ind (make-string
- org-edit-src-content-indentation 32)))
- (replace-regexp-in-string
- "\\(^\\)[ \t]*\\S-" ind
- (org-remove-indentation val) nil nil 1)))))))
- (concat (format "#+begin_src%s\n"
- (concat (and lang (concat " " lang))
- (and switches (concat " " switches))
- (and params (concat " " params))))
- value
- "#+end_src")))
- ;;;; Table
- (defun org-element-table-parser ()
- "Parse a table at point.
- Return a list whose car is `table' and cdr is a plist containing
- `:begin', `:end', `:contents-begin', `:contents-end', `:tblfm',
- `:type', `:raw-table' and `:post-blank' keywords."
- (save-excursion
- (let* ((table-begin (goto-char (org-table-begin t)))
- (type (if (org-at-table.el-p) 'table.el 'org))
- (keywords (org-element-collect-affiliated-keywords))
- (begin (car keywords))
- (table-end (goto-char (marker-position (org-table-end t))))
- (tblfm (when (looking-at "[ \t]*#\\+tblfm: +\\(.*\\)[ \t]*")
- (prog1 (org-match-string-no-properties 1)
- (forward-line))))
- (pos-before-blank (point))
- (end (progn (org-skip-whitespace)
- (if (eobp) (point) (point-at-bol))))
- (raw-table (org-remove-indentation
- (buffer-substring-no-properties table-begin table-end))))
- `(table
- (:begin ,begin
- :end ,end
- :type ,type
- :raw-table ,raw-table
- :tblfm ,tblfm
- :post-blank ,(count-lines pos-before-blank end)
- ,@(cadr keywords))))))
- (defun org-element-table-interpreter (table contents)
- "Interpret TABLE element as Org syntax.
- CONTENTS is nil."
- (org-element-property :raw-table table))
- ;;;; Verse Block
- (defun org-element-verse-block-parser ()
- "Parse a verse block.
- Return a list whose car is `verse-block' and cdr is a plist
- containing `:begin', `:end', `:hiddenp', `:raw-value', `:value'
- and `:post-blank' keywords.
- Assume point is at beginning or end of the block."
- (save-excursion
- (let* ((case-fold-search t)
- (keywords (progn
- (end-of-line)
- (re-search-backward
- (concat "^[ \t]*#\\+begin_verse") nil t)
- (org-element-collect-affiliated-keywords)))
- (begin (car keywords))
- (hidden (progn (forward-line) (org-truely-invisible-p)))
- (raw-val (buffer-substring-no-properties
- (point)
- (progn
- (re-search-forward (concat "^[ \t]*#\\+end_verse") nil t)
- (point-at-bol))))
- (pos-before-blank (progn (forward-line) (point)))
- (end (progn (org-skip-whitespace)
- (if (eobp) (point) (point-at-bol))))
- (value (org-element-parse-secondary-string
- (org-remove-indentation raw-val)
- (cdr (assq 'verse-block org-element-string-restrictions)))))
- `(verse-block
- (:begin ,begin
- :end ,end
- :hiddenp ,hidden
- :raw-value ,raw-val
- :value ,value
- :post-blank ,(count-lines pos-before-blank end)
- ,@(cadr keywords))))))
- (defun org-element-verse-block-interpreter (verse-block contents)
- "Interpret VERSE-BLOCK element as Org syntax.
- CONTENTS is nil."
- (format "#+begin_verse\n%s#+end_verse"
- (org-remove-indentation
- (org-element-property :raw-value verse-block))))
- ;;; Objects
- ;; Unlike to elements, interstices can be found between objects.
- ;; That's why, along with the parser, successor functions are provided
- ;; for each object. Some objects share the same successor
- ;; (i.e. `emphasis' and `verbatim' objects).
- ;; A successor must accept a single argument bounding the search. It
- ;; will return either a cons cell whose car is the object's type, as
- ;; a symbol, and cdr the position of its next occurrence, or nil.
- ;; Successors follow the naming convention:
- ;; org-element-NAME-successor, where NAME is the name of the
- ;; successor, as defined in `org-element-all-successors'.
- ;; Some object types (i.e. `emphasis') are recursive. Restrictions on
- ;; object types they can contain will be specified in
- ;; `org-element-object-restrictions'.
- ;; Adding a new type of object is simple. Implement a successor,
- ;; a parser, and an interpreter for it, all following the naming
- ;; convention. Register type in `org-element-all-objects' and
- ;; successor in `org-element-all-successors'. Maybe tweak
- ;; restrictions about it, and that's it.
- ;;;; Emphasis
- (defun org-element-emphasis-parser ()
- "Parse text markup object at point.
- Return a list whose car is `emphasis' and cdr is a plist with
- `:marker', `:begin', `:end', `:contents-begin' and
- `:contents-end' and `:post-blank' keywords.
- Assume point is at the first emphasis marker."
- (save-excursion
- (unless (bolp) (backward-char 1))
- (looking-at org-emph-re)
- (let ((begin (match-beginning 2))
- (marker (org-match-string-no-properties 3))
- (contents-begin (match-beginning 4))
- (contents-end (match-end 4))
- (post-blank (progn (goto-char (match-end 2))
- (skip-chars-forward " \t")))
- (end (point)))
- `(emphasis
- (:marker ,marker
- :begin ,begin
- :end ,end
- :contents-begin ,contents-begin
- :contents-end ,contents-end
- :post-blank ,post-blank)))))
- (defun org-element-emphasis-interpreter (emphasis contents)
- "Interpret EMPHASIS object as Org syntax.
- CONTENTS is the contents of the object."
- (let ((marker (org-element-property :marker emphasis)))
- (concat marker contents marker)))
- (defun org-element-text-markup-successor (limit)
- "Search for the next emphasis or verbatim object.
- LIMIT bounds the search.
- Return value is a cons cell whose car is `emphasis' or
- `verbatim' and cdr is beginning position."
- (save-excursion
- (unless (bolp) (backward-char))
- (when (re-search-forward org-emph-re limit t)
- (cons (if (nth 4 (assoc (match-string 3) org-emphasis-alist))
- 'verbatim
- 'emphasis)
- (match-beginning 2)))))
- ;;;; Entity
- (defun org-element-entity-parser ()
- "Parse entity at point.
- Return a list whose car is `entity' and cdr a plist with
- `:begin', `:end', `:latex', `:latex-math-p', `:html', `:latin1',
- `:utf-8', `:ascii', `:use-brackets-p' and `:post-blank' as
- keywords.
- Assume point is at the beginning of the entity."
- (save-excursion
- (looking-at "\\\\\\(frac[13][24]\\|[a-zA-Z]+\\)\\($\\|{}\\|[^[:alpha:]]\\)")
- (let* ((value (org-entity-get (match-string 1)))
- (begin (match-beginning 0))
- (bracketsp (string= (match-string 2) "{}"))
- (post-blank (progn (goto-char (match-end 1))
- (when bracketsp (forward-char 2))
- (skip-chars-forward " \t")))
- (end (point)))
- `(entity
- (:name ,(car value)
- :latex ,(nth 1 value)
- :latex-math-p ,(nth 2 value)
- :html ,(nth 3 value)
- :ascii ,(nth 4 value)
- :latin1 ,(nth 5 value)
- :utf-8 ,(nth 6 value)
- :begin ,begin
- :end ,end
- :use-brackets-p ,bracketsp
- :post-blank ,post-blank)))))
- (defun org-element-entity-interpreter (entity contents)
- "Interpret ENTITY object as Org syntax.
- CONTENTS is nil."
- (concat "\\"
- (org-element-property :name entity)
- (when (org-element-property :use-brackets-p entity) "{}")))
- (defun org-element-latex-or-entity-successor (limit)
- "Search for the next latex-fragment or entity object.
- LIMIT bounds the search.
- Return value is a cons cell whose car is `entity' or
- `latex-fragment' and cdr is beginning position."
- (save-excursion
- (let ((matchers (plist-get org-format-latex-options :matchers))
- ;; ENTITY-RE matches both LaTeX commands and Org entities.
- (entity-re
- "\\\\\\(frac[13][24]\\|[a-zA-Z]+\\)\\($\\|[^[:alpha:]\n]\\)"))
- (when (re-search-forward
- (concat (mapconcat (lambda (e) (nth 1 (assoc e org-latex-regexps)))
- matchers "\\|")
- "\\|" entity-re)
- limit t)
- (goto-char (match-beginning 0))
- (if (looking-at entity-re)
- ;; Determine if it's a real entity or a LaTeX command.
- (cons (if (org-entity-get (match-string 1)) 'entity 'latex-fragment)
- (match-beginning 0))
- ;; No entity nor command: point is at a LaTeX fragment.
- ;; Determine its type to get the correct beginning position.
- (cons 'latex-fragment
- (catch 'return
- (mapc (lambda (e)
- (when (looking-at (nth 1 (assoc e org-latex-regexps)))
- (throw 'return
- (match-beginning
- (nth 2 (assoc e org-latex-regexps))))))
- matchers)
- (point))))))))
- ;;;; Export Snippet
- (defun org-element-export-snippet-parser ()
- "Parse export snippet at point.
- Return a list whose car is `export-snippet' and cdr a plist with
- `:begin', `:end', `:back-end', `:value' and `:post-blank' as
- keywords.
- Assume point is at the beginning of the snippet."
- (save-excursion
- (looking-at "@\\([-A-Za-z0-9]+\\){")
- (let* ((begin (point))
- (back-end (org-match-string-no-properties 1))
- (before-blank (progn (goto-char (scan-sexps (1- (match-end 0)) 1))))
- (value (buffer-substring-no-properties
- (match-end 0) (1- before-blank)))
- (post-blank (skip-chars-forward " \t"))
- (end (point)))
- `(export-snippet
- (:back-end ,back-end
- :value ,value
- :begin ,begin
- :end ,end
- :post-blank ,post-blank)))))
- (defun org-element-export-snippet-interpreter (export-snippet contents)
- "Interpret EXPORT-SNIPPET object as Org syntax.
- CONTENTS is nil."
- (format "@%s{%s}"
- (org-element-property :back-end export-snippet)
- (org-element-property :value export-snippet)))
- (defun org-element-export-snippet-successor (limit)
- "Search for the next export-snippet object.
- LIMIT bounds the search.
- Return value is a cons cell whose car is `export-snippet' cdr is
- its beginning position."
- (save-excursion
- (catch 'exit
- (while (re-search-forward "@[-A-Za-z0-9]+{" limit t)
- (when (let ((end (ignore-errors (scan-sexps (1- (point)) 1))))
- (and end (eq (char-before end) ?})))
- (throw 'exit (cons 'export-snippet (match-beginning 0))))))))
- ;;;; Footnote Reference
- (defun org-element-footnote-reference-parser ()
- "Parse footnote reference at point.
- Return a list whose car is `footnote-reference' and cdr a plist
- with `:label', `:type', `:definition', `:begin', `:end' and
- `:post-blank' as keywords."
- (save-excursion
- (let* ((ref (org-footnote-at-reference-p))
- (label (car ref))
- (raw-def (nth 3 ref))
- (inline-def
- (and raw-def
- (org-element-parse-secondary-string
- raw-def
- (cdr (assq 'footnote-reference
- org-element-string-restrictions)))))
- (type (if (nth 3 ref) 'inline 'standard))
- (begin (nth 1 ref))
- (post-blank (progn (goto-char (nth 2 ref))
- (skip-chars-forward " \t")))
- (end (point)))
- `(footnote-reference
- (:label ,label
- :type ,type
- :inline-definition ,inline-def
- :begin ,begin
- :end ,end
- :post-blank ,post-blank
- :raw-definition ,raw-def)))))
- (defun org-element-footnote-reference-interpreter (footnote-reference contents)
- "Interpret FOOTNOTE-REFERENCE object as Org syntax.
- CONTENTS is nil."
- (let ((label (or (org-element-property :label footnote-reference) "fn:"))
- (def
- (let ((raw (org-element-property :raw-definition footnote-reference)))
- (if raw (concat ":" raw) ""))))
- (format "[%s]" (concat label def))))
- (defun org-element-footnote-reference-successor (limit)
- "Search for the next footnote-reference object.
- LIMIT bounds the search.
- Return value is a cons cell whose car is `footnote-reference' and
- cdr is beginning position."
- (let (fn-ref)
- (when (setq fn-ref (org-footnote-get-next-reference nil nil limit))
- (cons 'footnote-reference (nth 1 fn-ref)))))
- ;;;; Inline Babel Call
- (defun org-element-inline-babel-call-parser ()
- "Parse inline babel call at point.
- Return a list whose car is `inline-babel-call' and cdr a plist with
- `:begin', `:end', `:info' and `:post-blank' as keywords.
- Assume point is at the beginning of the babel call."
- (save-excursion
- (unless (bolp) (backward-char))
- (looking-at org-babel-inline-lob-one-liner-regexp)
- (let ((info (save-match-data (org-babel-lob-get-info)))
- (begin (match-end 1))
- (post-blank (progn (goto-char (match-end 0))
- (skip-chars-forward " \t")))
- (end (point)))
- `(inline-babel-call
- (:begin ,begin
- :end ,end
- :info ,info
- :post-blank ,post-blank)))))
- (defun org-element-inline-babel-call-interpreter (inline-babel-call contents)
- "Interpret INLINE-BABEL-CALL object as Org syntax.
- CONTENTS is nil."
- (let* ((babel-info (org-element-property :info inline-babel-call))
- (main-source (car babel-info))
- (post-options (nth 1 babel-info)))
- (concat "call_"
- (if (string-match "\\[\\(\\[.*?\\]\\)\\]" main-source)
- ;; Remove redundant square brackets.
- (replace-match
- (match-string 1 main-source) nil nil main-source)
- main-source)
- (and post-options (format "[%s]" post-options)))))
- (defun org-element-inline-babel-call-successor (limit)
- "Search for the next inline-babel-call object.
- LIMIT bounds the search.
- Return value is a cons cell whose car is `inline-babel-call' and
- cdr is beginning position."
- (save-excursion
- ;; Use a simplified version of
- ;; org-babel-inline-lob-one-liner-regexp as regexp for more speed.
- (when (re-search-forward
- "\\(?:babel\\|call\\)_\\([^()\n]+?\\)\\(\\[\\(.*\\)\\]\\|\\(\\)\\)(\\([^\n]*\\))\\(\\[\\(.*?\\)\\]\\)?"
- limit t)
- (cons 'inline-babel-call (match-beginning 0)))))
- ;;;; Inline Src Block
- (defun org-element-inline-src-block-parser ()
- "Parse inline source block at point.
- Return a list whose car is `inline-src-block' and cdr a plist
- with `:begin', `:end', `:language', `:value', `:parameters' and
- `:post-blank' as keywords.
- Assume point is at the beginning of the inline src block."
- (save-excursion
- (unless (bolp) (backward-char))
- (looking-at org-babel-inline-src-block-regexp)
- (let ((begin (match-beginning 1))
- (language (org-match-string-no-properties 2))
- (parameters (org-match-string-no-properties 4))
- (value (org-match-string-no-properties 5))
- (post-blank (progn (goto-char (match-end 0))
- (skip-chars-forward " \t")))
- (end (point)))
- `(inline-src-block
- (:language ,language
- :value ,value
- :parameters ,parameters
- :begin ,begin
- :end ,end
- :post-blank ,post-blank)))))
- (defun org-element-inline-src-block-interpreter (inline-src-block contents)
- "Interpret INLINE-SRC-BLOCK object as Org syntax.
- CONTENTS is nil."
- (let ((language (org-element-property :language inline-src-block))
- (arguments (org-element-property :parameters inline-src-block))
- (body (org-element-property :value inline-src-block)))
- (format "src_%s%s{%s}"
- language
- (if arguments (format "[%s]" arguments) "")
- body)))
- (defun org-element-inline-src-block-successor (limit)
- "Search for the next inline-babel-call element.
- LIMIT bounds the search.
- Return value is a cons cell whose car is `inline-babel-call' and
- cdr is beginning position."
- (save-excursion
- (when (re-search-forward org-babel-inline-src-block-regexp limit t)
- (cons 'inline-src-block (match-beginning 1)))))
- ;;;; Latex Fragment
- (defun org-element-latex-fragment-parser ()
- "Parse latex fragment at point.
- Return a list whose car is `latex-fragment' and cdr a plist with
- `:value', `:begin', `:end', and `:post-blank' as keywords.
- Assume point is at the beginning of the latex fragment."
- (save-excursion
- (let* ((begin (point))
- (substring-match
- (catch 'exit
- (mapc (lambda (e)
- (let ((latex-regexp (nth 1 (assoc e org-latex-regexps))))
- (when (or (looking-at latex-regexp)
- (and (not (bobp))
- (save-excursion
- (backward-char)
- (looking-at latex-regexp))))
- (throw 'exit (nth 2 (assoc e org-latex-regexps))))))
- (plist-get org-format-latex-options :matchers))
- ;; None found: it's a macro.
- (looking-at "\\\\[a-zA-Z]+\\*?\\(\\(\\[[^][\n{}]*\\]\\)\\|\\({[^{}\n]*}\\)\\)*")
- 0))
- (value (match-string-no-properties substring-match))
- (post-blank (progn (goto-char (match-end substring-match))
- (skip-chars-forward " \t")))
- (end (point)))
- `(latex-fragment
- (:value ,value
- :begin ,begin
- :end ,end
- :post-blank ,post-blank)))))
- (defun org-element-latex-fragment-interpreter (latex-fragment contents)
- "Interpret LATEX-FRAGMENT object as Org syntax.
- CONTENTS is nil."
- (org-element-property :value latex-fragment))
- ;;;; Line Break
- (defun org-element-line-break-parser ()
- "Parse line break at point.
- Return a list whose car is `line-break', and cdr a plist with
- `:begin', `:end' and `:post-blank' keywords.
- Assume point is at the beginning of the line break."
- (let ((begin (point))
- (end (save-excursion (forward-line) (point))))
- `(line-break (:begin ,begin :end ,end :post-blank 0))))
- (defun org-element-line-break-interpreter (line-break contents)
- "Interpret LINE-BREAK object as Org syntax.
- CONTENTS is nil."
- "\\\\\n")
- (defun org-element-line-break-successor (limit)
- "Search for the next line-break object.
- LIMIT bounds the search.
- Return value is a cons cell whose car is `line-break' and cdr is
- beginning position."
- (save-excursion
- (let ((beg (and (re-search-forward "[^\\\\]\\(\\\\\\\\\\)[ \t]*$" limit t)
- (goto-char (match-beginning 1)))))
- ;; A line break can only happen on a non-empty line.
- (when (and beg (re-search-backward "\\S-" (point-at-bol) t))
- (cons 'line-break beg)))))
- ;;;; Link
- (defun org-element-link-parser ()
- "Parse link at point.
- Return a list whose car is `link' and cdr a plist with `:type',
- `:path', `:raw-link', `:begin', `:end', `:contents-begin',
- `:contents-end' and `:post-blank' as keywords.
- Assume point is at the beginning of the link."
- (save-excursion
- (let ((begin (point))
- end contents-begin contents-end link-end post-blank path type
- raw-link link)
- (cond
- ;; Type 1: Text targeted from a radio target.
- ((and org-target-link-regexp (looking-at org-target-link-regexp))
- (setq type "radio"
- link-end (match-end 0)
- path (org-match-string-no-properties 0)))
- ;; Type 2: Standard link, i.e. [[http://orgmode.org][homepage]]
- ((looking-at org-bracket-link-regexp)
- (setq contents-begin (match-beginning 3)
- contents-end (match-end 3)
- link-end (match-end 0)
- ;; RAW-LINK is the original link.
- raw-link (org-match-string-no-properties 1)
- link (org-link-expand-abbrev
- (replace-regexp-in-string
- " *\n *" " " (org-link-unescape raw-link) t t)))
- ;; Determine TYPE of link and set PATH accordingly.
- (cond
- ;; File type.
- ((or (file-name-absolute-p link) (string-match "^\\.\\.?/" link))
- (setq type "file" path link))
- ;; Explicit type (http, irc, bbdb...). See `org-link-types'.
- ((string-match org-link-re-with-space3 link)
- (setq type (match-string 1 link) path (match-string 2 link)))
- ;; Id type: PATH is the id.
- ((string-match "^id:\\([-a-f0-9]+\\)" link)
- (setq type "id" path (match-string 1 link)))
- ;; Code-ref type: PATH is the name of the reference.
- ((string-match "^(\\(.*\\))$" link)
- (setq type "coderef" path (match-string 1 link)))
- ;; Custom-id type: PATH is the name of the custom id.
- ((= (aref link 0) ?#)
- (setq type "custom-id" path (substring link 1)))
- ;; Fuzzy type: Internal link either matches a target, an
- ;; headline name or nothing. PATH is the target or headline's
- ;; name.
- (t (setq type "fuzzy" path link))))
- ;; Type 3: Plain link, i.e. http://orgmode.org
- ((looking-at org-plain-link-re)
- (setq raw-link (org-match-string-no-properties 0)
- type (org-match-string-no-properties 1)
- path (org-match-string-no-properties 2)
- link-end (match-end 0)))
- ;; Type 4: Angular link, i.e. <http://orgmode.org>
- ((looking-at org-angle-link-re)
- (setq raw-link (buffer-substring-no-properties
- (match-beginning 1) (match-end 2))
- type (org-match-string-no-properties 1)
- path (org-match-string-no-properties 2)
- link-end (match-end 0))))
- ;; In any case, deduce end point after trailing white space from
- ;; LINK-END variable.
- (setq post-blank (progn (goto-char link-end) (skip-chars-forward " \t"))
- end (point))
- `(link
- (:type ,type
- :path ,path
- :raw-link ,(or raw-link path)
- :begin ,begin
- :end ,end
- :contents-begin ,contents-begin
- :contents-end ,contents-end
- :post-blank ,post-blank)))))
- (defun org-element-link-interpreter (link contents)
- "Interpret LINK object as Org syntax.
- CONTENTS is the contents of the object."
- (let ((type (org-element-property :type link))
- (raw-link (org-element-property :raw-link link)))
- (if (string= type "radio") raw-link
- (format "[[%s]%s]"
- raw-link
- (if (string= contents "") "" (format "[%s]" contents))))))
- (defun org-element-link-successor (limit)
- "Search for the next link object.
- LIMIT bounds the search.
- Return value is a cons cell whose car is `link' and cdr is
- beginning position."
- (save-excursion
- (let ((link-regexp
- (if (not org-target-link-regexp) org-any-link-re
- (concat org-any-link-re "\\|" org-target-link-regexp))))
- (when (re-search-forward link-regexp limit t)
- (cons 'link (match-beginning 0))))))
- ;;;; Macro
- (defun org-element-macro-parser ()
- "Parse macro at point.
- Return a list whose car is `macro' and cdr a plist with `:key',
- `:args', `:begin', `:end', `:value' and `:post-blank' as
- keywords.
- Assume point is at the macro."
- (save-excursion
- (looking-at "{{{\\([a-zA-Z][-a-zA-Z0-9_]*\\)\\(([ \t\n]*\\([^\000]*?\\))\\)?}}}")
- (let ((begin (point))
- (key (downcase (org-match-string-no-properties 1)))
- (value (org-match-string-no-properties 0))
- (post-blank (progn (goto-char (match-end 0))
- (skip-chars-forward " \t")))
- (end (point))
- (args (let ((args (org-match-string-no-properties 3)) args2)
- (when args
- (setq args (org-split-string args ","))
- (while args
- (while (string-match "\\\\\\'" (car args))
- ;; Repair bad splits.
- (setcar (cdr args) (concat (substring (car args) 0 -1)
- "," (nth 1 args)))
- (pop args))
- (push (pop args) args2))
- (mapcar 'org-trim (nreverse args2))))))
- `(macro
- (:key ,key
- :value ,value
- :args ,args
- :begin ,begin
- :end ,end
- :post-blank ,post-blank)))))
- (defun org-element-macro-interpreter (macro contents)
- "Interpret MACRO object as Org syntax.
- CONTENTS is nil."
- (org-element-property :value macro))
- (defun org-element-macro-successor (limit)
- "Search for the next macro object.
- LIMIT bounds the search.
- Return value is cons cell whose car is `macro' and cdr is
- beginning position."
- (save-excursion
- (when (re-search-forward
- "{{{\\([a-zA-Z][-a-zA-Z0-9_]*\\)\\(([ \t\n]*\\([^\000]*?\\))\\)?}}}"
- limit t)
- (cons 'macro (match-beginning 0)))))
- ;;;; Radio-target
- (defun org-element-radio-target-parser ()
- "Parse radio target at point.
- Return a list whose car is `radio-target' and cdr a plist with
- `:begin', `:end', `:contents-begin', `:contents-end', `raw-value'
- and `:post-blank' as keywords.
- Assume point is at the radio target."
- (save-excursion
- (looking-at org-radio-target-regexp)
- (let ((begin (point))
- (contents-begin (match-beginning 1))
- (contents-end (match-end 1))
- (raw-value (org-match-string-no-properties 1))
- (post-blank (progn (goto-char (match-end 0))
- (skip-chars-forward " \t")))
- (end (point)))
- `(radio-target
- (:begin ,begin
- :end ,end
- :contents-begin ,contents-begin
- :contents-end ,contents-end
- :raw-value ,raw-value
- :post-blank ,post-blank)))))
- (defun org-element-radio-target-interpreter (target contents)
- "Interpret TARGET object as Org syntax.
- CONTENTS is the contents of the object."
- (concat "<<<" contents ">>>"))
- (defun org-element-radio-target-successor (limit)
- "Search for the next radio-target object.
- LIMIT bounds the search.
- Return value is a cons cell whose car is `radio-target' and cdr
- is beginning position."
- (save-excursion
- (when (re-search-forward org-radio-target-regexp limit t)
- (cons 'radio-target (match-beginning 0)))))
- ;;;; Statistics Cookie
- (defun org-element-statistics-cookie-parser ()
- "Parse statistics cookie at point.
- Return a list whose car is `statistics-cookie', and cdr a plist
- with `:begin', `:end', `:value' and `:post-blank' keywords.
- Assume point is at the beginning of the statistics-cookie."
- (save-excursion
- (looking-at "\\[[0-9]*\\(%\\|/[0-9]*\\)\\]")
- (let* ((begin (point))
- (value (buffer-substring-no-properties
- (match-beginning 0) (match-end 0)))
- (post-blank (progn (goto-char (match-end 0))
- (skip-chars-forward " \t")))
- (end (point)))
- `(statistics-cookie
- (:begin ,begin
- :end ,end
- :value ,value
- :post-blank ,post-blank)))))
- (defun org-element-statistics-cookie-interpreter (statistics-cookie contents)
- "Interpret STATISTICS-COOKIE object as Org syntax.
- CONTENTS is nil."
- (org-element-property :value statistics-cookie))
- (defun org-element-statistics-cookie-successor (limit)
- "Search for the next statistics cookie object.
- LIMIT bounds the search.
- Return value is a cons cell whose car is `statistics-cookie' and
- cdr is beginning position."
- (save-excursion
- (when (re-search-forward "\\[[0-9]*\\(%\\|/[0-9]*\\)\\]" limit t)
- (cons 'statistics-cookie (match-beginning 0)))))
- ;;;; Subscript
- (defun org-element-subscript-parser ()
- "Parse subscript at point.
- Return a list whose car is `subscript' and cdr a plist with
- `:begin', `:end', `:contents-begin', `:contents-end',
- `:use-brackets-p' and `:post-blank' as keywords.
- Assume point is at the underscore."
- (save-excursion
- (unless (bolp) (backward-char))
- (let ((bracketsp (if (looking-at org-match-substring-with-braces-regexp)
- t
- (not (looking-at org-match-substring-regexp))))
- (begin (match-beginning 2))
- (contents-begin (or (match-beginning 5)
- (match-beginning 3)))
- (contents-end (or (match-end 5) (match-end 3)))
- (post-blank (progn (goto-char (match-end 0))
- (skip-chars-forward " \t")))
- (end (point)))
- `(subscript
- (:begin ,begin
- :end ,end
- :use-brackets-p ,bracketsp
- :contents-begin ,contents-begin
- :contents-end ,contents-end
- :post-blank ,post-blank)))))
- (defun org-element-subscript-interpreter (subscript contents)
- "Interpret SUBSCRIPT object as Org syntax.
- CONTENTS is the contents of the object."
- (format
- (if (org-element-property :use-brackets-p subscript) "_{%s}" "_%s")
- contents))
- (defun org-element-sub/superscript-successor (limit)
- "Search for the next sub/superscript object.
- LIMIT bounds the search.
- Return value is a cons cell whose car is either `subscript' or
- `superscript' and cdr is beginning position."
- (save-excursion
- (when (re-search-forward org-match-substring-regexp limit t)
- (cons (if (string= (match-string 2) "_") 'subscript 'superscript)
- (match-beginning 2)))))
- ;;;; Superscript
- (defun org-element-superscript-parser ()
- "Parse superscript at point.
- Return a list whose car is `superscript' and cdr a plist with
- `:begin', `:end', `:contents-begin', `:contents-end',
- `:use-brackets-p' and `:post-blank' as keywords.
- Assume point is at the caret."
- (save-excursion
- (unless (bolp) (backward-char))
- (let ((bracketsp (if (looking-at org-match-substring-with-braces-regexp)
- t
- (not (looking-at org-match-substring-regexp))))
- (begin (match-beginning 2))
- (contents-begin (or (match-beginning 5)
- (match-beginning 3)))
- (contents-end (or (match-end 5) (match-end 3)))
- (post-blank (progn (goto-char (match-end 0))
- (skip-chars-forward " \t")))
- (end (point)))
- `(superscript
- (:begin ,begin
- :end ,end
- :use-brackets-p ,bracketsp
- :contents-begin ,contents-begin
- :contents-end ,contents-end
- :post-blank ,post-blank)))))
- (defun org-element-superscript-interpreter (superscript contents)
- "Interpret SUPERSCRIPT object as Org syntax.
- CONTENTS is the contents of the object."
- (format
- (if (org-element-property :use-brackets-p superscript) "^{%s}" "^%s")
- contents))
- ;;;; Target
- (defun org-element-target-parser ()
- "Parse target at point.
- Return a list whose car is `target' and cdr a plist with
- `:begin', `:end', `:contents-begin', `:contents-end', `value' and
- `:post-blank' as keywords.
- Assume point is at the target."
- (save-excursion
- (looking-at org-target-regexp)
- (let ((begin (point))
- (value (org-match-string-no-properties 1))
- (post-blank (progn (goto-char (match-end 0))
- (skip-chars-forward " \t")))
- (end (point)))
- `(target
- (:begin ,begin
- :end ,end
- :value ,value
- :post-blank ,post-blank)))))
- (defun org-element-target-interpreter (target contents)
- "Interpret TARGET object as Org syntax.
- CONTENTS is the contents of target."
- (concat ""))
- (defun org-element-target-successor (limit)
- "Search for the next target object.
- LIMIT bounds the search.
- Return value is a cons cell whose car is `target' and cdr is
- beginning position."
- (save-excursion
- (when (re-search-forward org-target-regexp limit t)
- (cons 'target (match-beginning 0)))))
- ;;;; Time-stamp
- (defun org-element-time-stamp-parser ()
- "Parse time stamp at point.
- Return a list whose car is `time-stamp', and cdr a plist with
- `:appt-type', `:type', `:begin', `:end', `:value' and
- `:post-blank' keywords.
- Assume point is at the beginning of the time-stamp."
- (save-excursion
- (let* ((appt-type (cond
- ((looking-at (concat org-deadline-string " +"))
- (goto-char (match-end 0))
- 'deadline)
- ((looking-at (concat org-scheduled-string " +"))
- (goto-char (match-end 0))
- 'scheduled)
- ((looking-at (concat org-closed-string " +"))
- (goto-char (match-end 0))
- 'closed)))
- (begin (and appt-type (match-beginning 0)))
- (type (cond
- ((looking-at org-tsr-regexp)
- (if (match-string 2) 'active-range 'active))
- ((looking-at org-tsr-regexp-both)
- (if (match-string 2) 'inactive-range 'inactive))
- ((looking-at (concat
- "\\(<[0-9]+-[0-9]+-[0-9]+[^>\n]+?\\+[0-9]+[dwmy]>\\)"
- "\\|"
- "\\(<%%\\(([^>\n]+)\\)>\\)"))
- 'diary)))
- (begin (or begin (match-beginning 0)))
- (value (buffer-substring-no-properties
- (match-beginning 0) (match-end 0)))
- (post-blank (progn (goto-char (match-end 0))
- (skip-chars-forward " \t")))
- (end (point)))
- `(time-stamp
- (:appt-type ,appt-type
- :type ,type
- :value ,value
- :begin ,begin
- :end ,end
- :post-blank ,post-blank)))))
- (defun org-element-time-stamp-interpreter (time-stamp contents)
- "Interpret TIME-STAMP object as Org syntax.
- CONTENTS is nil."
- (concat
- (case (org-element-property :appt-type time-stamp)
- (closed (concat org-closed-string " "))
- (deadline (concat org-deadline-string " "))
- (scheduled (concat org-scheduled-string " ")))
- (org-element-property :value time-stamp)))
- (defun org-element-time-stamp-successor (limit)
- "Search for the next time-stamp object.
- LIMIT bounds the search.
- Return value is a cons cell whose car is `time-stamp' and cdr is
- beginning position."
- (save-excursion
- (when (re-search-forward
- (concat "\\(?:" org-scheduled-string " +\\|"
- org-deadline-string " +\\|" org-closed-string " +\\)?"
- org-ts-regexp-both
- "\\|"
- "\\(?:<[0-9]+-[0-9]+-[0-9]+[^>\n]+?\\+[0-9]+[dwmy]>\\)"
- "\\|"
- "\\(?:<%%\\(?:([^>\n]+)\\)>\\)")
- limit t)
- (cons 'time-stamp (match-beginning 0)))))
- ;;;; Verbatim
- (defun org-element-verbatim-parser ()
- "Parse verbatim object at point.
- Return a list whose car is `verbatim' and cdr is a plist with
- `:marker', `:begin', `:end' and `:post-blank' keywords.
- Assume point is at the first verbatim marker."
- (save-excursion
- (unless (bolp) (backward-char 1))
- (looking-at org-emph-re)
- (let ((begin (match-beginning 2))
- (marker (org-match-string-no-properties 3))
- (value (org-match-string-no-properties 4))
- (post-blank (progn (goto-char (match-end 2))
- (skip-chars-forward " \t")))
- (end (point)))
- `(verbatim
- (:marker ,marker
- :begin ,begin
- :end ,end
- :value ,value
- :post-blank ,post-blank)))))
- (defun org-element-verbatim-interpreter (verbatim contents)
- "Interpret VERBATIM object as Org syntax.
- CONTENTS is nil."
- (let ((marker (org-element-property :marker verbatim))
- (value (org-element-property :value verbatim)))
- (concat marker value marker)))
- ;;; Definitions And Rules
- ;; Define elements, greater elements and specify recursive objects,
- ;; along with the affiliated keywords recognized. Also set up
- ;; restrictions on recursive objects combinations.
- ;; These variables really act as a control center for the parsing
- ;; process.
- (defconst org-element-paragraph-separate
- (concat "\f" "\\|" "^[ \t]*$" "\\|"
- ;; Headlines and inlinetasks.
- org-outline-regexp-bol "\\|"
- ;; Comments, blocks (any type), keywords and babel calls.
- "^[ \t]*#\\+" "\\|" "^#\\( \\|$\\)" "\\|"
- ;; Lists.
- (org-item-beginning-re) "\\|"
- ;; Fixed-width, drawers (any type) and tables.
- "^[ \t]*[:|]" "\\|"
- ;; Footnote definitions.
- org-footnote-definition-re "\\|"
- ;; Horizontal rules.
- "^[ \t]*-\\{5,\\}[ \t]*$" "\\|"
- ;; LaTeX environments.
- "^[ \t]*\\\\\\(begin\\|end\\)")
- "Regexp to separate paragraphs in an Org buffer.")
- (defconst org-element-all-elements
- '(center-block comment comment-block drawer dynamic-block example-block
- export-block fixed-width footnote-definition headline
- horizontal-rule inlinetask item keyword latex-environment
- babel-call paragraph plain-list property-drawer quote-block
- quote-section section special-block src-block table
- verse-block)
- "Complete list of element types.")
- (defconst org-element-greater-elements
- '(center-block drawer dynamic-block footnote-definition headline inlinetask
- item plain-list quote-block section special-block)
- "List of recursive element types aka Greater Elements.")
- (defconst org-element-all-successors
- '(export-snippet footnote-reference inline-babel-call inline-src-block
- latex-or-entity line-break link macro radio-target
- statistics-cookie sub/superscript target text-markup
- time-stamp)
- "Complete list of successors.")
- (defconst org-element-object-successor-alist
- '((subscript . sub/superscript) (superscript . sub/superscript)
- (emphasis . text-markup) (verbatim . text-markup)
- (entity . latex-or-entity) (latex-fragment . latex-or-entity))
- "Alist of translations between object type and successor name.
- Sharing the same successor comes handy when, for example, the
- regexp matching one object can also match the other object.")
- (defconst org-element-all-objects
- '(emphasis entity export-snippet footnote-reference inline-babel-call
- inline-src-block line-break latex-fragment link macro radio-target
- statistics-cookie subscript superscript target time-stamp
- verbatim)
- "Complete list of object types.")
- (defconst org-element-recursive-objects
- '(emphasis link macro subscript superscript radio-target)
- "List of recursive object types.")
- (defconst org-element-non-recursive-block-alist
- '(("ascii" . export-block)
- ("comment" . comment-block)
- ("docbook" . export-block)
- ("example" . example-block)
- ("html" . export-block)
- ("latex" . export-block)
- ("odt" . export-block)
- ("src" . src-block)
- ("verse" . verse-block))
- "Alist between non-recursive block name and their element type.")
- (defconst org-element-affiliated-keywords
- '("attr_ascii" "attr_docbook" "attr_html" "attr_latex" "attr_odt" "caption"
- "data" "header" "headers" "label" "name" "plot" "resname" "result" "results"
- "source" "srcname" "tblname")
- "List of affiliated keywords as strings.")
- (defconst org-element-keyword-translation-alist
- '(("data" . "name") ("label" . "name") ("resname" . "name")
- ("source" . "name") ("srcname" . "name") ("tblname" . "name")
- ("result" . "results") ("headers" . "header"))
- "Alist of usual translations for keywords.
- The key is the old name and the value the new one. The property
- holding their value will be named after the translated name.")
- (defconst org-element-multiple-keywords
- '("attr_ascii" "attr_docbook" "attr_html" "attr_latex" "attr_odt" "header")
- "List of affiliated keywords that can occur more that once in an element.
- Their value will be consed into a list of strings, which will be
- returned as the value of the property.
- This list is checked after translations have been applied. See
- `org-element-keyword-translation-alist'.")
- (defconst org-element-parsed-keywords '("author" "caption" "title")
- "List of keywords whose value can be parsed.
- Their value will be stored as a secondary string: a list of
- strings and objects.
- This list is checked after translations have been applied. See
- `org-element-keyword-translation-alist'.")
- (defconst org-element-dual-keywords '("caption" "results")
- "List of keywords which can have a secondary value.
- In Org syntax, they can be written with optional square brackets
- before the colons. For example, results keyword can be
- associated to a hash value with the following:
- #+results[hash-string]: some-source
- This list is checked after translations have been applied. See
- `org-element-keyword-translation-alist'.")
- (defconst org-element-object-restrictions
- '((emphasis entity export-snippet inline-babel-call inline-src-block link
- radio-target sub/superscript target text-markup time-stamp)
- (link entity export-snippet inline-babel-call inline-src-block
- latex-fragment link sub/superscript text-markup)
- (macro macro)
- (radio-target entity export-snippet latex-fragment sub/superscript)
- (subscript entity export-snippet inline-babel-call inline-src-block
- latex-fragment sub/superscript text-markup)
- (superscript entity export-snippet inline-babel-call inline-src-block
- latex-fragment sub/superscript text-markup))
- "Alist of recursive objects restrictions.
- CAR is a recursive object type and CDR is a list of successors
- that will be called within an object of such type.
- For example, in a `radio-target' object, one can only find
- entities, export snippets, latex-fragments, subscript and
- superscript.")
- (defconst org-element-string-restrictions
- '((footnote-reference entity export-snippet footnote-reference
- inline-babel-call inline-src-block latex-fragment
- line-break link macro radio-target sub/superscript
- target text-markup time-stamp)
- (headline entity inline-babel-call inline-src-block latex-fragment link
- macro radio-target statistics-cookie sub/superscript text-markup
- time-stamp)
- (inlinetask entity inline-babel-call inline-src-block latex-fragment link
- macro radio-target sub/superscript text-markup time-stamp)
- (item entity inline-babel-call latex-fragment macro radio-target
- sub/superscript target text-markup)
- (keyword entity latex-fragment macro sub/superscript text-markup)
- (table entity latex-fragment macro target text-markup)
- (verse-block entity footnote-reference inline-babel-call inline-src-block
- latex-fragment line-break link macro radio-target
- sub/superscript target text-markup time-stamp))
- "Alist of secondary strings restrictions.
- When parsed, some elements have a secondary string which could
- contain various objects (i.e. headline's name, or table's cells).
- For association, CAR is the element type, and CDR a list of
- successors that will be called in that secondary string.
- Note: `keyword' secondary string type only applies to keywords
- matching `org-element-parsed-keywords'.")
- (defconst org-element-secondary-value-alist
- '((headline . :title)
- (inlinetask . :title)
- (item . :tag)
- (footnote-reference . :inline-definition)
- (verse-block . :value))
- "Alist between element types and location of secondary value.
- Only elements with a secondary value available at parse time are
- considered here. This is used internally by `org-element-map',
- which will look into the secondary strings of an element only if
- its type is listed here.")
- ;;; Accessors
- ;;
- ;; Provide three accessors: `org-element-type', `org-element-property'
- ;; and `org-element-contents'.
- (defun org-element-type (element)
- "Return type of element ELEMENT.
- The function returns the type of the element or object provided.
- It can also return the following special value:
- `plain-text' for a string
- `org-data' for a complete document
- nil in any other case."
- (cond
- ((not (consp element)) (and (stringp element) 'plain-text))
- ((symbolp (car element)) (car element))))
- (defun org-element-property (property element)
- "Extract the value from the PROPERTY of an ELEMENT."
- (plist-get (nth 1 element) property))
- (defun org-element-contents (element)
- "Extract contents from an ELEMENT."
- (nthcdr 2 element))
- ;;; Parsing Element Starting At Point
- ;; `org-element-current-element' is the core function of this section.
- ;; It returns the Lisp representation of the element starting at
- ;; point. It uses `org-element--element-block-re' for quick access to
- ;; a common regexp.
- (defconst org-element--element-block-re
- (format "[ \t]*#\\+begin_\\(%s\\)\\(?: \\|$\\)"
- (mapconcat
- 'regexp-quote
- (mapcar 'car org-element-non-recursive-block-alist) "\\|"))
- "Regexp matching the beginning of a non-recursive block type.
- Used internally by `org-element-current-element'. Do not modify
- it directly, set `org-element-recursive-block-alist' instead.")
- (defun org-element-current-element (&optional special structure)
- "Parse the element starting at point.
- Return value is a list like (TYPE PROPS) where TYPE is the type
- of the element and PROPS a plist of properties associated to the
- element.
- Possible types are defined in `org-element-all-elements'.
- Optional argument SPECIAL, when non-nil, can be either `item',
- `section' or `quote-section'. `item' allows to parse item wise
- instead of plain-list wise, using STRUCTURE as the current list
- structure. `section' (resp. `quote-section') will try to parse
- a section (resp. a quote section) before anything else.
- If STRUCTURE isn't provided but SPECIAL is set to `item', it will
- be computed.
- Unlike to `org-element-at-point', this function assumes point is
- always at the beginning of the element it has to parse. As such,
- it is quicker than its counterpart, albeit more restrictive."
- (save-excursion
- (beginning-of-line)
- ;; If point is at an affiliated keyword, try moving to the
- ;; beginning of the associated element. If none is found, the
- ;; keyword is orphaned and will be treated as plain text.
- (when (looking-at org-element--affiliated-re)
- (let ((opoint (point)))
- (while (looking-at org-element--affiliated-re) (forward-line))
- (when (looking-at "[ \t]*$") (goto-char opoint))))
- (let ((case-fold-search t))
- (cond
- ;; Headline.
- ((org-with-limited-levels (org-at-heading-p))
- (org-element-headline-parser))
- ;; Quote section.
- ((eq special 'quote-section) (org-element-quote-section-parser))
- ;; Section.
- ((eq special 'section) (org-element-section-parser))
- ;; Non-recursive block.
- ((when (looking-at org-element--element-block-re)
- (let ((type (downcase (match-string 1))))
- (if (save-excursion
- (re-search-forward
- (format "[ \t]*#\\+end_%s\\(?: \\|$\\)" type) nil t))
- ;; Build appropriate parser.
- (funcall
- (intern
- (format "org-element-%s-parser"
- (cdr (assoc type
- org-element-non-recursive-block-alist)))))
- (org-element-paragraph-parser)))))
- ;; Inlinetask.
- ((org-at-heading-p) (org-element-inlinetask-parser))
- ;; LaTeX Environment or paragraph if incomplete.
- ((looking-at "^[ \t]*\\\\begin{")
- (if (save-excursion
- (re-search-forward "^[ \t]*\\\\end{[^}]*}[ \t]*" nil t))
- (org-element-latex-environment-parser)
- (org-element-paragraph-parser)))
- ;; Property drawer.
- ((looking-at org-property-start-re)
- (if (save-excursion (re-search-forward org-property-end-re nil t))
- (org-element-property-drawer-parser)
- (org-element-paragraph-parser)))
- ;; Recursive block, or paragraph if incomplete.
- ((looking-at "[ \t]*#\\+begin_\\([-A-Za-z0-9]+\\)\\(?: \\|$\\)")
- (let ((type (downcase (match-string 1))))
- (cond
- ((not (save-excursion
- (re-search-forward
- (format "[ \t]*#\\+end_%s\\(?: \\|$\\)" type) nil t)))
- (org-element-paragraph-parser))
- ((string= type "center") (org-element-center-block-parser))
- ((string= type "quote") (org-element-quote-block-parser))
- (t (org-element-special-block-parser)))))
- ;; Drawer.
- ((looking-at org-drawer-regexp)
- (if (save-excursion (re-search-forward "^[ \t]*:END:[ \t]*$" nil t))
- (org-element-drawer-parser)
- (org-element-paragraph-parser)))
- ((looking-at "[ \t]*:\\( \\|$\\)") (org-element-fixed-width-parser))
- ;; Babel call.
- ((looking-at org-babel-block-lob-one-liner-regexp)
- (org-element-babel-call-parser))
- ;; Keyword, or paragraph if at an affiliated keyword.
- ((looking-at "[ \t]*#\\+\\([a-z]+\\(:?_[a-z]+\\)*\\):")
- (let ((key (downcase (match-string 1))))
- (if (or (string= key "tblfm")
- (member key org-element-affiliated-keywords))
- (org-element-paragraph-parser)
- (org-element-keyword-parser))))
- ;; Footnote definition.
- ((looking-at org-footnote-definition-re)
- (org-element-footnote-definition-parser))
- ;; Dynamic block or paragraph if incomplete.
- ((looking-at "[ \t]*#\\+begin:\\(?: \\|$\\)")
- (if (save-excursion
- (re-search-forward "^[ \t]*#\\+end:\\(?: \\|$\\)" nil t))
- (org-element-dynamic-block-parser)
- (org-element-paragraph-parser)))
- ;; Comment.
- ((looking-at "\\(#\\|[ \t]*#\\+\\(?: \\|$\\)\\)")
- (org-element-comment-parser))
- ;; Horizontal rule.
- ((looking-at "[ \t]*-\\{5,\\}[ \t]*$")
- (org-element-horizontal-rule-parser))
- ;; Table.
- ((org-at-table-p t) (org-element-table-parser))
- ;; List or item.
- ((looking-at (org-item-re))
- (if (eq special 'item)
- (org-element-item-parser (or structure (org-list-struct)))
- (org-element-plain-list-parser (or structure (org-list-struct)))))
- ;; Default element: Paragraph.
- (t (org-element-paragraph-parser))))))
- ;; Most elements can have affiliated keywords. When looking for an
- ;; element beginning, we want to move before them, as they belong to
- ;; that element, and, in the meantime, collect information they give
- ;; into appropriate properties. Hence the following function.
- ;; Usage of optional arguments may not be obvious at first glance:
- ;; - TRANS-LIST is used to polish keywords names that have evolved
- ;; during Org history. In example, even though =result= and
- ;; =results= coexist, we want to have them under the same =result=
- ;; property. It's also true for "srcname" and "name", where the
- ;; latter seems to be preferred nowadays (thus the "name" property).
- ;; - CONSED allows to regroup multi-lines keywords under the same
- ;; property, while preserving their own identity. This is mostly
- ;; used for "attr_latex" and al.
- ;; - PARSED prepares a keyword value for export. This is useful for
- ;; "caption". Objects restrictions for such keywords are defined in
- ;; `org-element-string-restrictions'.
- ;; - DUALS is used to take care of keywords accepting a main and an
- ;; optional secondary values. For example "results" has its
- ;; source's name as the main value, and may have an hash string in
- ;; optional square brackets as the secondary one.
- ;; A keyword may belong to more than one category.
- (defconst org-element--affiliated-re
- (format "[ \t]*#\\+\\(%s\\):"
- (mapconcat
- (lambda (keyword)
- (if (member keyword org-element-dual-keywords)
- (format "\\(%s\\)\\(?:\\[\\(.*\\)\\]\\)?"
- (regexp-quote keyword))
- (regexp-quote keyword)))
- org-element-affiliated-keywords "\\|"))
- "Regexp matching any affiliated keyword.
- Keyword name is put in match group 1. Moreover, if keyword
- belongs to `org-element-dual-keywords', put the dual value in
- match group 2.
- Don't modify it, set `org-element-affiliated-keywords' instead.")
- (defun org-element-collect-affiliated-keywords (&optional key-re trans-list
- consed parsed duals)
- "Collect affiliated keywords before point.
- Optional argument KEY-RE is a regexp matching keywords, which
- puts matched keyword in group 1. It defaults to
- `org-element--affiliated-re'.
- TRANS-LIST is an alist where key is the keyword and value the
- property name it should be translated to, without the colons. It
- defaults to `org-element-keyword-translation-alist'.
- CONSED is a list of strings. Any keyword belonging to that list
- will have its value consed. The check is done after keyword
- translation. It defaults to `org-element-multiple-keywords'.
- PARSED is a list of strings. Any keyword member of this list
- will have its value parsed. The check is done after keyword
- translation. If a keyword is a member of both CONSED and PARSED,
- it's value will be a list of parsed strings. It defaults to
- `org-element-parsed-keywords'.
- DUALS is a list of strings. Any keyword member of this list can
- have two parts: one mandatory and one optional. Its value is
- a cons cell whose car is the former, and the cdr the latter. If
- a keyword is a member of both PARSED and DUALS, both values will
- be parsed. It defaults to `org-element-dual-keywords'.
- Return a list whose car is the position at the first of them and
- cdr a plist of keywords and values."
- (save-excursion
- (let ((case-fold-search t)
- (key-re (or key-re org-element--affiliated-re))
- (trans-list (or trans-list org-element-keyword-translation-alist))
- (consed (or consed org-element-multiple-keywords))
- (parsed (or parsed org-element-parsed-keywords))
- (duals (or duals org-element-dual-keywords))
- ;; RESTRICT is the list of objects allowed in parsed
- ;; keywords value.
- (restrict (cdr (assq 'keyword org-element-string-restrictions)))
- output)
- (unless (bobp)
- (while (and (not (bobp))
- (progn (forward-line -1) (looking-at key-re)))
- (let* ((raw-kwd (downcase (or (match-string 2) (match-string 1))))
- ;; Apply translation to RAW-KWD. From there, KWD is
- ;; the official keyword.
- (kwd (or (cdr (assoc raw-kwd trans-list)) raw-kwd))
- ;; Find main value for any keyword.
- (value
- (save-match-data
- (org-trim
- (buffer-substring-no-properties
- (match-end 0) (point-at-eol)))))
- ;; If KWD is a dual keyword, find its secondary
- ;; value. Maybe parse it.
- (dual-value
- (and (member kwd duals)
- (let ((sec (org-match-string-no-properties 3)))
- (if (or (not sec) (not (member kwd parsed))) sec
- (org-element-parse-secondary-string sec restrict)))))
- ;; Attribute a property name to KWD.
- (kwd-sym (and kwd (intern (concat ":" kwd)))))
- ;; Now set final shape for VALUE.
- (when (member kwd parsed)
- (setq value (org-element-parse-secondary-string value restrict)))
- (when (member kwd duals)
- ;; VALUE is mandatory. Set it to nil if there is none.
- (setq value (and value (cons value dual-value))))
- (when (member kwd consed)
- (setq value (cons value (plist-get output kwd-sym))))
- ;; Eventually store the new value in OUTPUT.
- (setq output (plist-put output kwd-sym value))))
- (unless (looking-at key-re) (forward-line 1)))
- (list (point) output))))
- ;;; The Org Parser
- ;; The two major functions here are `org-element-parse-buffer', which
- ;; parses Org syntax inside the current buffer, taking into account
- ;; region, narrowing, or even visibility if specified, and
- ;; `org-element-parse-secondary-string', which parses objects within
- ;; a given string.
- ;; The (almost) almighty `org-element-map' allows to apply a function
- ;; on elements or objects matching some type, and accumulate the
- ;; resulting values. In an export situation, it also skips unneeded
- ;; parts of the parse tree.
- (defun org-element-parse-buffer (&optional granularity visible-only)
- "Recursively parse the buffer and return structure.
- If narrowing is in effect, only parse the visible part of the
- buffer.
- Optional argument GRANULARITY determines the depth of the
- recursion. It can be set to the following symbols:
- `headline' Only parse headlines.
- `greater-element' Don't recurse into greater elements. Thus,
- elements parsed are the top-level ones.
- `element' Parse everything but objects and plain text.
- `object' Parse the complete buffer (default).
- When VISIBLE-ONLY is non-nil, don't parse contents of hidden
- elements.
- Assume buffer is in Org mode."
- (save-excursion
- (goto-char (point-min))
- (org-skip-whitespace)
- (nconc (list 'org-data nil)
- (org-element-parse-elements
- (point-at-bol) (point-max)
- ;; Start is section mode so text before the first headline
- ;; belongs to a section.
- 'section nil granularity visible-only nil))))
- (defun org-element-parse-secondary-string (string restriction &optional buffer)
- "Recursively parse objects in STRING and return structure.
- RESTRICTION, when non-nil, is a symbol limiting the object types
- that will be looked after.
- Optional argument BUFFER indicates the buffer from where the
- secondary string was extracted. It is used to determine where to
- get extraneous information for an object \(i.e. when resolving
- a link or looking for a footnote definition\). It defaults to
- the current buffer."
- (with-temp-buffer
- (insert string)
- (org-element-parse-objects (point-min) (point-max) nil restriction)))
- (defun org-element-map (data types fun &optional info first-match no-recursion)
- "Map a function on selected elements or objects.
- DATA is the parsed tree, as returned by, i.e,
- `org-element-parse-buffer'. TYPES is a symbol or list of symbols
- of elements or objects types. FUN is the function called on the
- matching element or object. It must accept one arguments: the
- element or object itself.
- When optional argument INFO is non-nil, it should be a plist
- holding export options. In that case, parts of the parse tree
- not exportable according to that property list will be skipped.
- When optional argument FIRST-MATCH is non-nil, stop at the first
- match for which FUN doesn't return nil, and return that value.
- Optional argument NO-RECURSION is a symbol or a list of symbols
- representing elements or objects types. `org-element-map' won't
- enter any recursive element or object whose type belongs to that
- list. Though, FUN can still be applied on them.
- Nil values returned from FUN do not appear in the results."
- ;; Ensure TYPES and NO-RECURSION are a list, even of one element.
- (unless (listp types) (setq types (list types)))
- (unless (listp no-recursion) (setq no-recursion (list no-recursion)))
- ;; Recursion depth is determined by --CATEGORY.
- (let* ((--category
- (cond
- ((loop for type in types
- always (memq type org-element-greater-elements))
- 'greater-elements)
- ((loop for type in types
- always (memq type org-element-all-elements))
- 'elements)
- (t 'objects)))
- ;; --RESTRICTS is a list of element types whose secondary
- ;; string could possibly contain an object with a type among
- ;; TYPES.
- (--restricts
- (and (eq --category 'objects)
- (loop for el in org-element-secondary-value-alist
- when
- (loop for o in types
- thereis
- (memq o (cdr
- (assq (car el)
- org-element-string-restrictions))))
- collect (car el))))
- --acc
- (--walk-tree
- (function
- (lambda (--data)
- ;; Recursively walk DATA. INFO, if non-nil, is
- ;; a plist holding contextual information.
- (mapc
- (lambda (--blob)
- (unless (and info (member --blob (plist-get info :ignore-list)))
- (let ((--type (org-element-type --blob)))
- ;; Check if TYPE is matching among TYPES. If so,
- ;; apply FUN to --BLOB and accumulate return value
- ;; into --ACC (or exit if FIRST-MATCH is non-nil).
- (when (memq --type types)
- (let ((result (funcall fun --blob)))
- (cond ((not result))
- (first-match (throw 'first-match result))
- (t (push result --acc)))))
- ;; If --BLOB has a secondary string that can
- ;; contain objects with their type among TYPES,
- ;; look into that string.
- (when (memq --type --restricts)
- (funcall
- --walk-tree
- `(org-data
- nil
- ,@(org-element-property
- (cdr (assq --type org-element-secondary-value-alist))
- --blob))))
- ;; Now determine if a recursion into --BLOB is
- ;; possible. If so, do it.
- (unless (memq --type no-recursion)
- (when (or (and (memq --type org-element-greater-elements)
- (not (eq --category 'greater-elements)))
- (and (memq --type org-element-all-elements)
- (not (eq --category 'elements)))
- (memq --type org-element-recursive-objects))
- (funcall --walk-tree --blob))))))
- (org-element-contents --data))))))
- (catch 'first-match
- (funcall --walk-tree data)
- ;; Return value in a proper order.
- (reverse --acc))))
- ;; The following functions are internal parts of the parser.
- ;; The first one, `org-element-parse-elements' acts at the element's
- ;; level.
- ;; The second one, `org-element-parse-objects' applies on all objects
- ;; of a paragraph or a secondary string. It uses
- ;; `org-element-get-candidates' to optimize the search of the next
- ;; object in the buffer.
- ;; More precisely, that function looks for every allowed object type
- ;; first. Then, it discards failed searches, keeps further matches,
- ;; and searches again types matched behind point, for subsequent
- ;; calls. Thus, searching for a given type fails only once, and every
- ;; object is searched only once at top level (but sometimes more for
- ;; nested types).
- (defun org-element-parse-elements
- (beg end special structure granularity visible-only acc)
- "Parse elements between BEG and END positions.
- SPECIAL prioritize some elements over the others. It can set to
- `quote-section', `section' or `item', which will focus search,
- respectively, on quote sections, sections and items. Moreover,
- when value is `item', STRUCTURE will be used as the current list
- structure.
- GRANULARITY determines the depth of the recursion. It can be set
- to the following symbols:
- `headline' Only parse headlines.
- `greater-element' Don't recurse into greater elements. Thus,
- elements parsed are the top-level ones.
- `element' Parse everything but objects and plain text.
- `object' or nil Parse the complete buffer.
- When VISIBLE-ONLY is non-nil, don't parse contents of hidden
- elements.
- Elements are accumulated into ACC."
- (save-excursion
- (save-restriction
- (narrow-to-region beg end)
- (goto-char beg)
- ;; When parsing only headlines, skip any text before first one.
- (when (and (eq granularity 'headline) (not (org-at-heading-p)))
- (org-with-limited-levels (outline-next-heading)))
- ;; Main loop start.
- (while (not (eobp))
- (push
- ;; 1. Item mode is active: point must be at an item. Parse it
- ;; directly, skipping `org-element-current-element'.
- (if (eq special 'item)
- (let ((element (org-element-item-parser structure)))
- (goto-char (org-element-property :end element))
- (org-element-parse-elements
- (org-element-property :contents-begin element)
- (org-element-property :contents-end element)
- nil structure granularity visible-only (reverse element)))
- ;; 2. When ITEM is nil, find current element's type and parse
- ;; it accordingly to its category.
- (let* ((element (org-element-current-element special structure))
- (type (org-element-type element)))
- (goto-char (org-element-property :end element))
- (cond
- ;; Case 1. ELEMENT is a paragraph. Parse objects inside,
- ;; if GRANULARITY allows it.
- ((and (eq type 'paragraph)
- (or (not granularity) (eq granularity 'object)))
- (org-element-parse-objects
- (org-element-property :contents-begin element)
- (org-element-property :contents-end element)
- (reverse element) nil))
- ;; Case 2. ELEMENT is recursive: parse it between
- ;; `contents-begin' and `contents-end'. Make sure
- ;; GRANULARITY allows the recursion, or ELEMENT is an
- ;; headline, in which case going inside is mandatory, in
- ;; order to get sub-level headings. If VISIBLE-ONLY is
- ;; true and element is hidden, do not recurse into it.
- ((and (memq type org-element-greater-elements)
- (or (not granularity)
- (memq granularity '(element object))
- (eq type 'headline))
- (not (and visible-only
- (org-element-property :hiddenp element))))
- (org-element-parse-elements
- (org-element-property :contents-begin element)
- (org-element-property :contents-end element)
- ;; At a plain list, switch to item mode. At an
- ;; headline, switch to section mode. Any other
- ;; element turns off special modes.
- (case type
- (plain-list 'item)
- (headline (if (org-element-property :quotedp element)
- 'quote-section
- 'section)))
- (org-element-property :structure element)
- granularity visible-only (reverse element)))
- ;; Case 3. Else, just accumulate ELEMENT.
- (t element))))
- acc)))
- ;; Return result.
- (nreverse acc)))
- (defun org-element-parse-objects (beg end acc restriction)
- "Parse objects between BEG and END and return recursive structure.
- Objects are accumulated in ACC.
- RESTRICTION, when non-nil, is a list of object types which are
- allowed in the current object."
- (let ((get-next-object
- (function
- (lambda (cand)
- ;; Return the parsing function associated to the nearest
- ;; object among list of candidates CAND.
- (let ((pos (apply #'min (mapcar #'cdr cand))))
- (save-excursion
- (goto-char pos)
- (funcall
- (intern
- (format "org-element-%s-parser" (car (rassq pos cand))))))))))
- next-object candidates)
- (save-excursion
- (goto-char beg)
- (while (setq candidates (org-element-get-next-object-candidates
- end restriction candidates))
- (setq next-object (funcall get-next-object candidates))
- ;; 1. Text before any object. Untabify it.
- (let ((obj-beg (org-element-property :begin next-object)))
- (unless (= (point) obj-beg)
- (push (replace-regexp-in-string
- "\t" (make-string tab-width ? )
- (buffer-substring-no-properties (point) obj-beg))
- acc)))
- ;; 2. Object...
- (let ((obj-end (org-element-property :end next-object))
- (cont-beg (org-element-property :contents-begin next-object)))
- (push (if (and (memq (car next-object) org-element-recursive-objects)
- cont-beg)
- ;; ... recursive. The CONT-BEG check is for
- ;; links, as some of them might not be recursive
- ;; (i.e. plain links).
- (save-restriction
- (narrow-to-region
- cont-beg
- (org-element-property :contents-end next-object))
- (org-element-parse-objects
- (point-min) (point-max) (reverse next-object)
- ;; Restrict allowed objects. This is the
- ;; intersection of current restriction and next
- ;; object's restriction.
- (let ((new-restr
- (cdr (assq (car next-object)
- org-element-object-restrictions))))
- (if (not restriction) new-restr
- (delq nil (mapcar
- (lambda (e) (and (memq e restriction) e))
- new-restr))))))
- ;; ... not recursive.
- next-object)
- acc)
- (goto-char obj-end)))
- ;; 3. Text after last object. Untabify it.
- (unless (= (point) end)
- (push (replace-regexp-in-string
- "\t" (make-string tab-width ? )
- (buffer-substring-no-properties (point) end))
- acc))
- ;; Result.
- (nreverse acc))))
- (defun org-element-get-next-object-candidates (limit restriction objects)
- "Return an alist of candidates for the next object.
- LIMIT bounds the search, and RESTRICTION, when non-nil, bounds
- the possible object types.
- Return value is an alist whose car is position and cdr the object
- type, as a string. There is an association for the closest
- object of each type within RESTRICTION when non-nil, or for every
- type otherwise.
- OBJECTS is the previous candidates alist."
- (let ((restriction (or restriction org-element-all-successors))
- next-candidates types-to-search)
- ;; If no previous result, search every object type in RESTRICTION.
- ;; Otherwise, keep potential candidates (old objects located after
- ;; point) and ask to search again those which had matched before.
- (if (not objects) (setq types-to-search restriction)
- (mapc (lambda (obj)
- (if (< (cdr obj) (point)) (push (car obj) types-to-search)
- (push obj next-candidates)))
- objects))
- ;; Call the appropriate "get-next" function for each type to
- ;; search and accumulate matches.
- (mapc
- (lambda (type)
- (let* ((successor-fun
- (intern
- (format "org-element-%s-successor"
- (or (cdr (assq type org-element-object-successor-alist))
- type))))
- (obj (funcall successor-fun limit)))
- (and obj (push obj next-candidates))))
- types-to-search)
- ;; Return alist.
- next-candidates))
- ;;; Towards A Bijective Process
- ;; The parse tree obtained with `org-element-parse-buffer' is really
- ;; a snapshot of the corresponding Org buffer. Therefore, it can be
- ;; interpreted and expanded into a string with canonical Org
- ;; syntax. Hence `org-element-interpret-data'.
- ;;
- ;; Data parsed from secondary strings, whose shape is slightly
- ;; different than the standard parse tree, is expanded with the
- ;; equivalent function `org-element-interpret-secondary'.
- ;;
- ;; Both functions rely internally on
- ;; `org-element-interpret--affiliated-keywords'.
- (defun org-element-interpret-data (data &optional genealogy previous)
- "Interpret a parse tree representing Org data.
- DATA is the parse tree to interpret.
- Optional arguments GENEALOGY and PREVIOUS are used for recursive
- calls:
- GENEALOGY is the list of its parents types.
- PREVIOUS is the type of the element or object at the same level
- interpreted before.
- Return Org syntax as a string."
- (mapconcat
- (lambda (blob)
- ;; BLOB can be an element, an object, a string, or nil.
- (cond
- ((not blob) nil)
- ((equal blob "") nil)
- ((stringp blob) blob)
- (t
- (let* ((type (org-element-type blob))
- (interpreter
- (if (eq type 'org-data) 'identity
- (intern (format "org-element-%s-interpreter" type))))
- (contents
- (cond
- ;; Full Org document.
- ((eq type 'org-data)
- (org-element-interpret-data blob genealogy previous))
- ;; Recursive objects.
- ((memq type org-element-recursive-objects)
- (org-element-interpret-data
- blob (cons type genealogy) nil))
- ;; Recursive elements.
- ((memq type org-element-greater-elements)
- (org-element-normalize-string
- (org-element-interpret-data
- blob (cons type genealogy) nil)))
- ;; Paragraphs.
- ((eq type 'paragraph)
- (let ((paragraph
- (org-element-normalize-contents
- blob
- ;; When normalizing contents of an item,
- ;; ignore first line's indentation.
- (and (not previous)
- (memq (car genealogy)
- '(footnote-definiton item))))))
- (org-element-interpret-data
- paragraph (cons type genealogy) nil)))))
- (results (funcall interpreter blob contents)))
- ;; Update PREVIOUS.
- (setq previous type)
- ;; Build white spaces.
- (cond
- ((eq type 'org-data) results)
- ((memq type org-element-all-elements)
- (concat
- (org-element-interpret--affiliated-keywords blob)
- (org-element-normalize-string results)
- (make-string (org-element-property :post-blank blob) 10)))
- (t (concat
- results
- (make-string (org-element-property :post-blank blob) 32))))))))
- (org-element-contents data) ""))
- (defun org-element-interpret-secondary (secondary)
- "Interpret SECONDARY string as Org syntax.
- SECONDARY-STRING is a nested list as returned by
- `org-element-parse-secondary-string'.
- Return interpreted string."
- ;; Make SECONDARY acceptable for `org-element-interpret-data'.
- (let ((s (if (listp secondary) secondary (list secondary))))
- (org-element-interpret-data `(org-data nil ,@s) nil nil)))
- ;; Both functions internally use `org-element--affiliated-keywords'.
- (defun org-element-interpret--affiliated-keywords (element)
- "Return ELEMENT's affiliated keywords as Org syntax.
- If there is no affiliated keyword, return the empty string."
- (let ((keyword-to-org
- (function
- (lambda (key value)
- (let (dual)
- (when (member key org-element-dual-keywords)
- (setq dual (cdr value) value (car value)))
- (concat "#+" key (and dual (format "[%s]" dual)) ": "
- (if (member key org-element-parsed-keywords)
- (org-element-interpret-secondary value)
- value)
- "\n"))))))
- (mapconcat
- (lambda (key)
- (let ((value (org-element-property (intern (concat ":" key)) element)))
- (when value
- (if (member key org-element-multiple-keywords)
- (mapconcat (lambda (line)
- (funcall keyword-to-org key line))
- value "")
- (funcall keyword-to-org key value)))))
- ;; Remove translated keywords.
- (delq nil
- (mapcar
- (lambda (key)
- (and (not (assoc key org-element-keyword-translation-alist)) key))
- org-element-affiliated-keywords))
- "")))
- ;; Because interpretation of the parse tree must return the same
- ;; number of blank lines between elements and the same number of white
- ;; space after objects, some special care must be given to white
- ;; spaces.
- ;;
- ;; The first function, `org-element-normalize-string', ensures any
- ;; string different from the empty string will end with a single
- ;; newline character.
- ;;
- ;; The second function, `org-element-normalize-contents', removes
- ;; global indentation from the contents of the current element.
- (defun org-element-normalize-string (s)
- "Ensure string S ends with a single newline character.
- If S isn't a string return it unchanged. If S is the empty
- string, return it. Otherwise, return a new string with a single
- newline character at its end."
- (cond
- ((not (stringp s)) s)
- ((string= "" s) "")
- (t (and (string-match "\\(\n[ \t]*\\)*\\'" s)
- (replace-match "\n" nil nil s)))))
- (defun org-element-normalize-contents (element &optional ignore-first)
- "Normalize plain text in ELEMENT's contents.
- ELEMENT must only contain plain text and objects.
- If optional argument IGNORE-FIRST is non-nil, ignore first line's
- indentation to compute maximal common indentation.
- Return the normalized element that is element with global
- indentation removed from its contents. The function assumes that
- indentation is not done with TAB characters."
- (let (ind-list
- (collect-inds
- (function
- ;; Return list of indentations within BLOB. This is done by
- ;; walking recursively BLOB and updating IND-LIST along the
- ;; way. FIRST-FLAG is non-nil when the first string hasn't
- ;; been seen yet. It is required as this string is the only
- ;; one whose indentation doesn't happen after a newline
- ;; character.
- (lambda (blob first-flag)
- (mapc
- (lambda (object)
- (when (and first-flag (stringp object))
- (setq first-flag nil)
- (string-match "\\`\\( *\\)" object)
- (let ((len (length (match-string 1 object))))
- ;; An indentation of zero means no string will be
- ;; modified. Quit the process.
- (if (zerop len) (throw 'zero (setq ind-list nil))
- (push len ind-list))))
- (cond
- ((stringp object)
- (let ((start 0))
- (while (string-match "\n\\( *\\)" object start)
- (setq start (match-end 0))
- (push (length (match-string 1 object)) ind-list))))
- ((memq (org-element-type object) org-element-recursive-objects)
- (funcall collect-inds object first-flag))))
- (org-element-contents blob))))))
- ;; Collect indentation list in ELEMENT. Possibly remove first
- ;; value if IGNORE-FIRST is non-nil.
- (catch 'zero (funcall collect-inds element (not ignore-first)))
- (if (not ind-list) element
- ;; Build ELEMENT back, replacing each string with the same
- ;; string minus common indentation.
- (let ((build
- (function
- (lambda (blob mci first-flag)
- ;; Return BLOB with all its strings indentation
- ;; shortened from MCI white spaces. FIRST-FLAG is
- ;; non-nil when the first string hasn't been seen
- ;; yet.
- (nconc
- (list (org-element-type blob) (nth 1 blob))
- (mapcar
- (lambda (object)
- (when (and first-flag (stringp object))
- (setq first-flag nil)
- (setq object
- (replace-regexp-in-string
- (format "\\` \\{%d\\}" mci) "" object)))
- (cond
- ((stringp object)
- (replace-regexp-in-string
- (format "\n \\{%d\\}" mci) "\n" object))
- ((memq (org-element-type object) org-element-recursive-objects)
- (funcall build object mci first-flag))
- (t object)))
- (org-element-contents blob)))))))
- (funcall build element (apply 'min ind-list) (not ignore-first))))))
- ;;; The Toolbox
- ;; The first move is to implement a way to obtain the smallest element
- ;; containing point. This is the job of `org-element-at-point'. It
- ;; basically jumps back to the beginning of section containing point
- ;; and moves, element after element, with
- ;; `org-element-current-element' until the container is found.
- (defun org-element-at-point (&optional keep-trail)
- "Determine closest element around point.
- Return value is a list like (TYPE PROPS) where TYPE is the type
- of the element and PROPS a plist of properties associated to the
- element. Possible types are defined in
- `org-element-all-elements'.
- As a special case, if point is at the very beginning of a list or
- sub-list, element returned will be that list instead of the first
- item.
- If optional argument KEEP-TRAIL is non-nil, the function returns
- a list of of elements leading to element at point. The list's
- CAR is always the element at point. Its last item will be the
- element's parent, unless element was either the first in its
- section (in which case the last item in the list is the first
- element of section) or an headline (in which case the list
- contains that headline as its single element). Elements
- in-between, if any, are siblings of the element at point."
- (org-with-wide-buffer
- ;; If at an headline, parse it. It is the sole element that
- ;; doesn't require to know about context.
- (if (org-with-limited-levels (org-at-heading-p))
- (if (not keep-trail) (org-element-headline-parser)
- (list (org-element-headline-parser)))
- ;; Otherwise move at the beginning of the section containing
- ;; point.
- (let ((origin (point)) element type item-flag trail struct prevs)
- (org-with-limited-levels
- (if (org-before-first-heading-p) (goto-char (point-min))
- (org-back-to-heading)
- (forward-line)))
- (org-skip-whitespace)
- (beginning-of-line)
- ;; Starting parsing successively each element with
- ;; `org-element-current-element'. Skip those ending before
- ;; original position.
- (catch 'exit
- (while t
- (setq element (org-element-current-element item-flag struct)
- type (car element))
- (when keep-trail (push element trail))
- (cond
- ;; 1. Skip any element ending before point or at point.
- ((let ((end (org-element-property :end element)))
- (when (<= end origin)
- (if (> (point-max) end) (goto-char end)
- (throw 'exit (or trail element))))))
- ;; 2. An element containing point is always the element at
- ;; point.
- ((not (memq type org-element-greater-elements))
- (throw 'exit (if keep-trail trail element)))
- ;; 3. At a plain list.
- ((eq type 'plain-list)
- (setq struct (org-element-property :structure element)
- prevs (or prevs (org-list-prevs-alist struct)))
- (let ((beg (org-element-property :contents-begin element)))
- (if (= beg origin) (throw 'exit (or trail element))
- ;; Find the item at this level containing ORIGIN.
- (let ((items (org-list-get-all-items beg struct prevs)))
- (let (parent)
- (catch 'local
- (mapc
- (lambda (pos)
- (cond
- ;; Item ends before point: skip it.
- ((<= (org-list-get-item-end pos struct) origin))
- ;; Item contains point: store is in PARENT.
- ((<= pos origin) (setq parent pos))
- ;; We went too far: return PARENT.
- (t (throw 'local nil)))) items))
- ;; No parent: no item contained point, though
- ;; the plain list does. Point is in the blank
- ;; lines after the list: return plain list.
- (if (not parent) (throw 'exit (or trail element))
- (setq item-flag 'item)
- (goto-char parent)))))))
- ;; 4. At any other greater element type, if point is
- ;; within contents, move into it. Otherwise, return
- ;; that element.
- (t
- (when (eq type 'item) (setq item-flag nil))
- (let ((beg (org-element-property :contents-begin element))
- (end (org-element-property :contents-end element)))
- (if (or (> beg origin) (< end origin))
- (throw 'exit (or trail element))
- ;; Reset trail, since we found a parent.
- (when keep-trail (setq trail (list element)))
- (narrow-to-region beg end)
- (goto-char beg)))))))))))
- ;; Once the local structure around point is well understood, it's easy
- ;; to implement some replacements for `forward-paragraph'
- ;; `backward-paragraph', namely `org-element-forward' and
- ;; `org-element-backward'.
- ;; Also, `org-transpose-elements' mimics the behaviour of
- ;; `transpose-words', at the element's level, whereas
- ;; `org-element-drag-forward', `org-element-drag-backward', and
- ;; `org-element-up' generalize, respectively, functions
- ;; `org-subtree-down', `org-subtree-up' and `outline-up-heading'.
- ;; `org-element-unindent-buffer' will, as its name almost suggests,
- ;; smartly remove global indentation from buffer, making it possible
- ;; to use Org indent mode on a file created with hard indentation.
- ;; `org-element-nested-p' and `org-element-swap-A-B' are used
- ;; internally by some of the previously cited tools.
- (defsubst org-element-nested-p (elem-A elem-B)
- "Non-nil when elements ELEM-A and ELEM-B are nested."
- (let ((beg-A (org-element-property :begin elem-A))
- (beg-B (org-element-property :begin elem-B))
- (end-A (org-element-property :end elem-A))
- (end-B (org-element-property :end elem-B)))
- (or (and (>= beg-A beg-B) (<= end-A end-B))
- (and (>= beg-B beg-A) (<= end-B end-A)))))
- (defun org-element-swap-A-B (elem-A elem-B)
- "Swap elements ELEM-A and ELEM-B.
- Leave point at the end of ELEM-A.
- Assume ELEM-A is before ELEM-B and that they are not nested."
- (goto-char (org-element-property :begin elem-A))
- (let* ((beg-B (org-element-property :begin elem-B))
- (end-B-no-blank (save-excursion
- (goto-char (org-element-property :end elem-B))
- (skip-chars-backward " \r\t\n")
- (forward-line)
- (point)))
- (beg-A (org-element-property :begin elem-A))
- (end-A-no-blank (save-excursion
- (goto-char (org-element-property :end elem-A))
- (skip-chars-backward " \r\t\n")
- (forward-line)
- (point)))
- (body-A (buffer-substring beg-A end-A-no-blank))
- (body-B (buffer-substring beg-B end-B-no-blank))
- (between-A-B (buffer-substring end-A-no-blank beg-B)))
- (delete-region beg-A end-B-no-blank)
- (insert body-B between-A-B body-A)
- (goto-char (org-element-property :end elem-B))))
- (defun org-element-backward ()
- "Move backward by one element.
- Move to the previous element at the same level, when possible."
- (interactive)
- (if (save-excursion (skip-chars-backward " \r\t\n") (bobp))
- (error "Cannot move further up")
- (let* ((trail (org-element-at-point 'keep-trail))
- (element (car trail))
- (beg (org-element-property :begin element)))
- ;; Move to beginning of current element if point isn't there.
- (if (/= (point) beg) (goto-char beg)
- (let ((type (org-element-type element)))
- (cond
- ;; At an headline: move to previous headline at the same
- ;; level, a parent, or BOB.
- ((eq type 'headline)
- (let ((dest (save-excursion (org-backward-same-level 1) (point))))
- (if (= (point-min) dest) (error "Cannot move further up")
- (goto-char dest))))
- ;; At an item: try to move to the previous item, if any.
- ((and (eq type 'item)
- (let* ((struct (org-element-property :structure element))
- (prev (org-list-get-prev-item
- beg struct (org-list-prevs-alist struct))))
- (when prev (goto-char prev)))))
- ;; In any other case, find the previous element in the
- ;; trail and move to its beginning. If no previous element
- ;; can be found, move to headline.
- (t (let ((prev (nth 1 trail)))
- (if prev (goto-char (org-element-property :begin prev))
- (org-back-to-heading))))))))))
- (defun org-element-drag-backward ()
- "Drag backward element at point."
- (interactive)
- (let* ((pos (point))
- (elem (org-element-at-point)))
- (when (= (progn (goto-char (point-min))
- (org-skip-whitespace)
- (point-at-bol))
- (org-element-property :end elem))
- (error "Cannot drag element backward"))
- (goto-char (org-element-property :begin elem))
- (org-element-backward)
- (let ((prev-elem (org-element-at-point)))
- (when (or (org-element-nested-p elem prev-elem)
- (and (eq (car elem) 'headline)
- (not (eq (car prev-elem) 'headline))))
- (goto-char pos)
- (error "Cannot drag element backward"))
- ;; Compute new position of point: it's shifted by PREV-ELEM
- ;; body's length.
- (let ((size-prev (- (org-element-property :end prev-elem)
- (org-element-property :begin prev-elem))))
- (org-element-swap-A-B prev-elem elem)
- (goto-char (- pos size-prev))))))
- (defun org-element-drag-forward ()
- "Move forward element at point."
- (interactive)
- (let* ((pos (point))
- (elem (org-element-at-point)))
- (when (= (point-max) (org-element-property :end elem))
- (error "Cannot drag element forward"))
- (goto-char (org-element-property :end elem))
- (let ((next-elem (org-element-at-point)))
- (when (or (org-element-nested-p elem next-elem)
- (and (eq (car next-elem) 'headline)
- (not (eq (car elem) 'headline))))
- (goto-char pos)
- (error "Cannot drag element forward"))
- ;; Compute new position of point: it's shifted by NEXT-ELEM
- ;; body's length (without final blanks) and by the length of
- ;; blanks between ELEM and NEXT-ELEM.
- (let ((size-next (- (save-excursion
- (goto-char (org-element-property :end next-elem))
- (skip-chars-backward " \r\t\n")
- (forward-line)
- (point))
- (org-element-property :begin next-elem)))
- (size-blank (- (org-element-property :end elem)
- (save-excursion
- (goto-char (org-element-property :end elem))
- (skip-chars-backward " \r\t\n")
- (forward-line)
- (point)))))
- (org-element-swap-A-B elem next-elem)
- (goto-char (+ pos size-next size-blank))))))
- (defun org-element-forward ()
- "Move forward by one element.
- Move to the next element at the same level, when possible."
- (interactive)
- (if (eobp) (error "Cannot move further down")
- (let* ((trail (org-element-at-point 'keep-trail))
- (element (car trail))
- (type (org-element-type element))
- (end (org-element-property :end element)))
- (cond
- ;; At an headline, move to next headline at the same level.
- ((eq type 'headline) (goto-char end))
- ;; At an item. Move to the next item, if possible.
- ((and (eq type 'item)
- (let* ((struct (org-element-property :structure element))
- (prevs (org-list-prevs-alist struct))
- (beg (org-element-property :begin element))
- (next-item (org-list-get-next-item beg struct prevs)))
- (when next-item (goto-char next-item)))))
- ;; In any other case, move to element's end, unless this
- ;; position is also the end of its parent's contents, in which
- ;; case, directly jump to parent's end.
- (t
- (let ((parent
- ;; Determine if TRAIL contains the real parent of ELEMENT.
- (and (> (length trail) 1)
- (let* ((parent-candidate (car (last trail))))
- (and (memq (org-element-type parent-candidate)
- org-element-greater-elements)
- (>= (org-element-property
- :contents-end parent-candidate) end)
- parent-candidate)))))
- (cond ((not parent) (goto-char end))
- ((= (org-element-property :contents-end parent) end)
- (goto-char (org-element-property :end parent)))
- (t (goto-char end)))))))))
- (defun org-element-mark-element ()
- "Put point at beginning of this element, mark at end.
- Interactively, if this command is repeated or (in Transient Mark
- mode) if the mark is active, it marks the next element after the
- ones already marked."
- (interactive)
- (let (deactivate-mark)
- (if (or (and (eq last-command this-command) (mark t))
- (and transient-mark-mode mark-active))
- (set-mark
- (save-excursion
- (goto-char (mark))
- (goto-char (org-element-property :end (org-element-at-point)))))
- (let ((element (org-element-at-point)))
- (end-of-line)
- (push-mark (org-element-property :end element) t t)
- (goto-char (org-element-property :begin element))))))
- (defun org-narrow-to-element ()
- "Narrow buffer to current element."
- (interactive)
- (let ((elem (org-element-at-point)))
- (cond
- ((eq (car elem) 'headline)
- (narrow-to-region
- (org-element-property :begin elem)
- (org-element-property :end elem)))
- ((memq (car elem) org-element-greater-elements)
- (narrow-to-region
- (org-element-property :contents-begin elem)
- (org-element-property :contents-end elem)))
- (t
- (narrow-to-region
- (org-element-property :begin elem)
- (org-element-property :end elem))))))
- (defun org-transpose-elements ()
- "Transpose current and previous elements, keeping blank lines between.
- Point is moved after both elements."
- (interactive)
- (org-skip-whitespace)
- (let ((pos (point))
- (cur (org-element-at-point)))
- (when (= (save-excursion (goto-char (point-min))
- (org-skip-whitespace)
- (point-at-bol))
- (org-element-property :begin cur))
- (error "No previous element"))
- (goto-char (org-element-property :begin cur))
- (forward-line -1)
- (let ((prev (org-element-at-point)))
- (when (org-element-nested-p cur prev)
- (goto-char pos)
- (error "Cannot transpose nested elements"))
- (org-element-swap-A-B prev cur))))
- (defun org-element-unindent-buffer ()
- "Un-indent the visible part of the buffer.
- Relative indentation \(between items, inside blocks, etc.\) isn't
- modified."
- (interactive)
- (unless (eq major-mode 'org-mode)
- (error "Cannot un-indent a buffer not in Org mode"))
- (let* ((parse-tree (org-element-parse-buffer 'greater-element))
- unindent-tree ; For byte-compiler.
- (unindent-tree
- (function
- (lambda (contents)
- (mapc (lambda (element)
- (if (eq (org-element-type element) 'headline)
- (funcall unindent-tree
- (org-element-contents element))
- (save-excursion
- (save-restriction
- (narrow-to-region
- (org-element-property :begin element)
- (org-element-property :end element))
- (org-do-remove-indentation)))))
- (reverse contents))))))
- (funcall unindent-tree (org-element-contents parse-tree))))
- (defun org-element-up ()
- "Move to upper element."
- (interactive)
- (cond
- ((bobp) (error "No surrounding element"))
- ((org-with-limited-levels (org-at-heading-p))
- (or (org-up-heading-safe) (error "No surronding element")))
- (t
- (let* ((trail (org-element-at-point 'keep-trail))
- (element (car trail))
- (type (org-element-type element)))
- (cond
- ;; At an item, with a parent in the list: move to that parent.
- ((and (eq type 'item)
- (let* ((beg (org-element-property :begin element))
- (struct (org-element-property :structure element))
- (parents (org-list-parents-alist struct))
- (parentp (org-list-get-parent beg struct parents)))
- (and parentp (goto-char parentp)))))
- ;; Determine parent in the trail.
- (t
- (let ((parent
- (and (> (length trail) 1)
- (let ((parentp (car (last trail))))
- (and (memq (org-element-type parentp)
- org-element-greater-elements)
- (>= (org-element-property :contents-end parentp)
- (org-element-property :end element))
- parentp)))))
- (cond
- ;; When parent is found move to its beginning.
- (parent (goto-char (org-element-property :begin parent)))
- ;; If no parent was found, move to headline above, if any
- ;; or return an error.
- ((org-before-first-heading-p) (error "No surrounding element"))
- (t (org-back-to-heading))))))))))
- (defun org-element-down ()
- "Move to inner element."
- (interactive)
- (let ((element (org-element-at-point)))
- (cond
- ((eq (org-element-type element) 'plain-list)
- (forward-char))
- ((memq (org-element-type element) org-element-greater-elements)
- (goto-char (org-element-property :contents-begin element)))
- (t (error "No inner element")))))
- (provide 'org-element)
- ;;; org-element.el ends here
|