File: odoc_for_authors.mld

package info (click to toggle)
ocaml-odoc 3.1.0-1
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid
  • size: 14,008 kB
  • sloc: ml: 60,567; javascript: 2,572; sh: 566; makefile: 31
file content (1145 lines) | stat: -rw-r--r-- 37,269 bytes parent folder | download
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
1001
1002
1003
1004
1005
1006
1007
1008
1009
1010
1011
1012
1013
1014
1015
1016
1017
1018
1019
1020
1021
1022
1023
1024
1025
1026
1027
1028
1029
1030
1031
1032
1033
1034
1035
1036
1037
1038
1039
1040
1041
1042
1043
1044
1045
1046
1047
1048
1049
1050
1051
1052
1053
1054
1055
1056
1057
1058
1059
1060
1061
1062
1063
1064
1065
1066
1067
1068
1069
1070
1071
1072
1073
1074
1075
1076
1077
1078
1079
1080
1081
1082
1083
1084
1085
1086
1087
1088
1089
1090
1091
1092
1093
1094
1095
1096
1097
1098
1099
1100
1101
1102
1103
1104
1105
1106
1107
1108
1109
1110
1111
1112
1113
1114
1115
1116
1117
1118
1119
1120
1121
1122
1123
1124
1125
1126
1127
1128
1129
1130
1131
1132
1133
1134
1135
1136
1137
1138
1139
1140
1141
1142
1143
1144
1145
{0:top [odoc] for Authors}

This manual describes the features available and recommended for users
in order to write great documentation with [odoc]. See also the page listing the
{{!page-features}language features} [odoc] understands.

+ {{!getting_started}Getting Started}
+ {{:#interfaces} Documenting Your Interfaces}
+ {{:#doc-pages} Writing Documentation Pages}

{1:getting_started Getting Started}

To generate documentation for your project, you will almost always be using
[odoc] indirectly rather than executing it yourself. There are currently several
'drivers' of [odoc] at time of writing, each with their own strengths and
weaknesses:

- {{!/odoc-driver/page-index}The reference driver} is the most up-to-date driver,
  and incorporates all of the features of odoc. It is written to build the docs
  locally as well as those published on ocaml.org. It does not feature fast rebuilds,
  nor keeping a switch-wide cache of documentation. Install the driver and the package to document via
  [opam] and run
{v
$ opam install odoc-driver <pkg>
$ odoc_driver <pkg>
v}
  to build the docs. The output will appear in the directory `_html` by default.
  Full documentation is available in the {{!/odoc-driver/page-index}reference driver package}.
- {{:https://dune.build/}Dune} is the best way to view docs while
  developing your project. We have a {{!page-dune}page} describing
  how to use it, but the short version is:
{v
$ dune build @doc
v}
  and the results can be found here: [_build/default/_doc/_html/index.html]
- {{:https://erratique.ch/software/odig}Odig} is the best way to view docs
  of your installed packages. After installing, try:
{v
$ odig doc odig
v}

{1:packagedocs Documenting a package}

Documentation in OCaml is grouped together by {i package}. Each package may contain
a number of documentation pages, organised into a hierarchy, and if the
package contains {i libraries}, their interfaces form the rest of the package documentation.
The pages are written in a subset of the usual {{!page-ocamldoc_differences}odoc markup},
and the libraries are documented via in-line special comments in the source.
Conceptually, the docs for every library are {i contained within} the docs of its package.

{2 Quickstart}

When a package is built with [dune], the simplest way to include docs in your package is
to create a [doc] dir, write [doc/index.mld] file containing the documentation for your
package:

{v
{0 My package documentation}

This is the documentation for my package!
v}

then create the following [doc/dune] file:

{v
(documentation)
v}

More mld files can be added as necessary to the [doc] directory, and they will be included
also.

This will work if you've only got one package defined in your dune project. If you've got
more than one, you'll need to let dune know which package the docs are associated with:

{v
(documentation
  (package my_package))
v}

Any assets you have will also need to be handled, and as this is a new feature in odoc 3,
at time of writing dune does not have a convenient syntax for this. However, it still can
be done, so if, for example, you have an image [asset.jpg] that you'd like to include in your
doc, you can add the following to your [doc/dune] file:

{v
(install
  (section doc)
  (files (asset.jpg as odoc-pages/asset.jpg)))
v}

Odoc 3.0 introduces a per-package configuration file, {{!section-"config-file"}odoc-config.sexp}, which is important if you'd like your
documentation to link to the docs of other packages or libraries. If your documentation needs
this configuration file, this will also need to be installed as follows:

{v
(install
 (section doc)
 (files odoc-config.sexp))
v}

{1:interfaces Documenting Your Interfaces}

As well as doc pages, [odoc] is built to produce documentation for your {e libraries},
and the unit of organisation for these is the {e module}. Documentation is written by putting
{{!special_comments}special comments} into the source of the {e module interface} or,
less commonly, your {e module implementation}. 

For the HTML output, [odoc] will produce one page for each module, module type,
class, and class type. This includes any submodules, so if a module [A] contains
module [B] then two HTML pages will be created. [A/index.html] will have a
link to [A/B/index.html] at the point that module [B] is defined. So each
module, module type, class, and class type should be documented, keeping in mind
that it will end up as a single page with a table of contents. 

For each module, [odoc] will describe all of the values, types, module types,
classes, modules, and other elements found in module signatures. Their types or
descriptions are documented along with any comments delineated by [(** ... *)],
known as docstrings, found in the source code. For any element that references
another, for example a [value] that has a [type], [odoc] will create links that
point to the definitions.

Therefore, the job of the library author is to organise the module interface
in a logical manner and write comments explaining each type, value, exception,
type constructor, and everything else. The comments are written in a rich markup
language that allows the usual formatting constructs: bold, italic, sub- and
super-script, lists, verbatim and code sections, along with section headings, 
a very rich cross-referencing mechanism, and tags to add specific information 
to individual elements.

All the OCamldoc documentation comments described in {{:https://ocaml.org/manual/ocamldoc.html#s:ocamldoc-comments}the OCaml manual}
are supported, with a few important {{!page-ocamldoc_differences}differences}. In addition, 
it is intended that all features of the OCaml language are supported by [odoc].
For examples of how the language features are handled, see the
{{!page-features}Features} page.

{2:special_comments Special Comments}

Comments containing documentation are known as {e special comments}. 
They are like normal comments except they have precisely
two asterisks at the start:

{[
(* Normal comment *)
(** Special comment containing documentation *)
(*** Normal comment *)
]}

From here forward, 'comment' will refer to the special comments.
Most comments will be associated with particular elements, and this requires
the comment to be immediately before or after an element with no blank lines
in between (although non-special comments are allowed). Comments that
are not associated with a particular element are known as 'floating' comments.

If there is ambiguity, which can happen if there are two elements with a
comment directly in between, the comment will be associated with {e both}
elements. This is an example of where [odoc] differs from OCamldoc. Read more
about that on the {{!page-ocamldoc_differences}OCamldoc Differences} page.

{[
type x
(** Ambiguous comment, associated with {e both} {!x} and {!y} *)
type y
]}

Although inherited from the compiler, this behaviour is unlikely to be desired,
so a blank line should be inserted to clarify with which element the comment
should be associated. Note that Dune will raise an error if there are ambiguous
comments in the source files.

The first comment of a module is special, as it is associated with the module
as a whole. This is discussed in more detail in the section on
{{!page_structure}page structure}.

The OCaml manual has a helpful example of comment placement, reproduced below.
Note that there is an additional line inserted to avoid an ambiguous special
comment.

{[
(** The first special comment of the file is the comment associated
  with the whole module.*)

(** Special comments can be placed between elements and are kept
  by the OCamldoc tool, but are not associated to any element.
  [@]-tags in these comments are ignored.*)

(*******************************************************************)
(** Comments like the one above, with more than two asterisks,
  are ignored. *)

(** The comment for function f. *)
val f : int -> int -> int
(** The continuation of the comment for function f. *)

exception My_exception of (int -> int) * int
(* Hello, I'm a simple comment :-) *)
(** Comment for exception My_exception, even with a simple comment
  between the special comment and the exception.*)

(** Comment for type weather  *)
type weather =
  | Rain of int  (** The comment for constructor Rain *)
  | Sun  (** The comment for constructor Sun *)

(** Comment for type weather2  *)
type weather2 =
  | Rain of int  (** The comment for constructor Rain *)
  | Sun  (** The comment for constructor Sun *)
(** I can continue the comment for type weather2 here
because there is already a comment associated to the last constructor.*)

(** The comment for type my_record *)
type my_record = {
  foo : int;  (** Comment for field foo *)
  bar : string;  (** Comment for field bar *)
}
(** Continuation of comment for type my_record *)

(** Comment for foo *)
val foo : string
(** This comment is associated to foo and not to bar. *)

val bar : string
(** This comment is associated to bar. *)

class cl :
  object

    (** Interesting information about cl *)
  end

(** The comment for class my_class *)
class my_class :
  object
    inherit cl
    (** A comment to describe inheritance from cl *)

    val mutable tutu : string
    (** The comment for attribute tutu *)

    val toto : int
    (** The comment for attribute toto. *)

    (** This comment is not attached to titi since
      there is a blank line before titi, but is kept
      as a comment in the class. *)

    val titi : string

    method toto : string
    (** Comment for method toto *)

    method m : float -> int
    (** Comment for method m *)
  end

(** The comment for the class type my_class_type *)
class type my_class_type =
  object
    val mutable x : int
    (** The comment for variable x. *)

    method m : int -> int
    (** The comment for method m. *)
  end

(** The comment for module Foo *)
module Foo : sig
  val x : int
  (** The comment for x *)

  (** A special comment that is kept but not associated to any element *)
end

(** The comment for module type my_module_type. *)
module type my_module_type = sig
  val x : int
  (** The comment for value x. *)

  (** The comment for module M. *)
  module M : sig
    val y : int
    (** The comment for value y. *)

    (* ... *)
  end
end
]}

The result of [odoc] documenting this interface can be seen on the examples
page {{!Odoc_examples.Markup.Foo}here}.

There are no differences in how [odoc] handles comment placement between [.ml]
and [.mli] files, which is {{!page-ocamldoc_differences}another difference}
from OCamldoc.

{2:basics Basic Markup}

Text within the comments can be formatted using the following markup. Firstly,
the simple typesetting markup:

- [{b <text>}] bold
- [{i <text>}] italic
- [{e <text>}] emphasis
- [{^ <text>}] superscript
- [{_ <text>}] subscript
 
{3:lists Lists}
Unordered lists:
{v
{ul {- <item>}
    {- <item>}
    {- <item>}}
v}
and ordered lists:
{v
{ol {- <item 1>}
    {- <item 2>}
    {- <item 3>}}
v}

Lists can contain block elements, apart from {{!sections}headings} and
{{!tags}tags}.

There is also an abbreviated syntax for lists. The above could be written:
{v
- <item>
- <item>
- <item>
v}
and
{v
+ <item 1>
+ <item 2>
+ <item 3>
v}

In the abbreviated synyax, lists are ended by a blank line, or anything that
cannot be part of a list item (a heading or a tag).

{3:code_blocks Code Blocks}

There are various ways of inserting code elements in your documentation.

{4:inline_code_blocks Inline Code Blocks}

For inline, language agnostic source code style, use square brackets: [ [ ... ] ].

{[
(** Here, [f 0] is [None] *)
]}

Two consecutive newlines in an inline codeblock are forbidden, but in the interest
of the 80-char rule, a single newline followed by horizontal space in an inline
codeblock is considered as a single space:

{[
(**
  A very loooooooooooooooooooooooong line and [List.map (fun x -> x +1)].

  is equivalent to:

  A very loooooooooooooooooooooooong line and [List.map
  (fun x -> x +1)].

*)
]}

All the other existing ways to insert code are meant to be used for code blocks,
not inline code fragments. Those will be formatted as a code block and will end
the paragraph. Attempting to use them inline will trigger a warning.

{4:ocaml_code_blocks OCaml Code Blocks}

OCaml code blocks can be written using the enclosing tags [ {[ ... ]} ].
The code inside these blocks will be properly styled as OCaml in the generated
doc.

{delim@ocaml[
(** This how one binds a variable in OCaml:
    {[
    let x = 0
    ]}
*)
]delim}

Note that the content of the code block above is ["let x = 0"]. How is the initial content (["\n    let x = 0\n    "]) turned into that?

{4:indentation_code_blocks Indentation in code blocks}

Odoc strips whitespace from the content of a codeblock, so that it contains only what is in the following box:

{v
  {[
  +-----------------------
  |
  |    content
  |
  +-----------------------
  ]}
v}

In other words, [odoc], in order:
{ol
{- Does not remove any content if [{\[] and [\]}] are on the same line.}
{- Removes the content of the line after [{\[] if it contains only whitespace.}
{- Removes the content of the line before [\]}] if it contains only whitespace.}
{- Removes spaces and tabs (counted as a single spaces) to reach the column of the opening curly brace of the [{\[] marker. If there is non-whitespace on the left of this column, the content is deindented only as much as it can without stripping non-whitespace, and a warning is raised.}}

Also, note that if point 2 above does not remove a line, the content of the code
block is considered as if starting at the upper [+] in the box above:

{delim@ocaml[
(**
  The following two code blocks have the same content:
  {[let x = 0]}
  {[
  let x = 0]}
*)
]delim}

{4:code_blocks_example Examples}

Here are a few examples to illustrate the explanation above:

{v
  {[ ]}
v}
  {[ ]}
{v
  {[let x = 0]}
v}
  {[let x = 0]}
{v
  {[ let x = 0 ]}
v}
  {[ let x = 0 ]}
{v
  {[let x = function
    | [] -> None
    | h :: _ -> Some h ]}
v}
  {[let x = function
    | [] -> None
    | h :: _ -> Some h ]}
{v
  {[
  let x = function
    | [] -> None
    | h :: _ -> Some h ]}
v}
  {[
  let x = function
    | [] -> None
    | h :: _ -> Some h ]}
{v
  {[let x = function]}
  Some comment
  {[
    | [] -> None
    | h :: _ -> Some h ]}
v}
  {[let x = function]}
  Some comment
  {[
    | [] -> None
    | h :: _ -> Some h ]}

{4:language_headers Language Headers}

As of [odoc.2.2], it is possible to write blocks with explicit language headers.
These blocks can be written using the enclosing tags [{@<language>[ ... ]}].
The content of those block should be properly styled if [highlight.js] supports the
styling for the given language.

{v
(** Here is some Python code:
    {@python[
    def f():
      return 0
    ]}
*)
v}

And rendered this becomes:

Here is some Python code:
{@python[
  def f():
    return 0
]}

{4:code_block_tags Code Block Tags}
Code blocks may have arbitrary metadata associated with them. This metadata is
used, for example, for selecting code blocks when extracting them from a source file via the
[odoc extract-code] command. The metadata may also be used by other tools that
operate on code blocks, for example, {{:https://github.com/realworldocaml/mdx}mdx}.

The metadata follows immediately after the language header, and is a list of tags and bindings, separated by whitespace. Tags are simple
keywords, and bindings are key-value pairs separated by an equals sign. If whitespace
is needed in keys, values or tags, it should be enclosed in double quotes. For example:

{v
{@ocaml tag1 "tag 2" key1="value 1" "key 2"="value 2"[
   ... code goes here ...
]}
v}

{4:verbatim_blocks Verbatim Blocks}

It is possible to write language agnostic code blocks, also called verbatim
blocks, using the enclosing tags [{v ... v}]. The content of these blocks will be
formatted as code but with no particular style applied.

{[
(** Here is some code formatted text:
    {v
    Some text
    v}
*)
]}

Verbatims' indentation is treated as for code blocks, see the
{{!indentation_code_blocks}explanation} there!

{3:escaping Escaping}

In most contexts, the characters [{ [ ] } @] all need to be escaped with a backslash.
In inline source code style, only square brackets need to be escaped. However, as a
convenience, {e matched} square brackets need not be escaped to aid in typesetting
code. For example, the following would be acceptable in a documentation comment:

{v
The list [ [1;2;3] ] needs no escaping
v}

In a code block section, the section is ended with a [ \]} ]. In a
verbatim formatted section, the section ends with a whitespace character
followed by [ v} ]. It is not currently possible to escape this in either case.

{2:links_and_references Links and References}

A link to a URL may be put into the text as follows:

{[
(** See {{: https://ocaml.org/ }the OCaml website} for news about OCaml *)
]}

This will render as a link to [https://ocaml.org/] with the text "the OCaml website"

References are links to other elements, e.g., comments might wish to refer to 
a module or type elsewhere as follows:

{[
(** See the module {!Stdlib.Buffer} for more details *)
]}

While [odoc] supports the syntax for references used by OCamldoc, it has an
improved syntax that allows for disambiguating in the face of clashing names.
See the section {!page-features.reference_resolution} for an example of this.

The supported methods for referring to elements are:

- Bare: [ {!Foo.bar} ] - this works well if there are no ambiguities in what's being referred to.
- OCamldoc: [ {!type:Foo.bar} ] - here the [type] prefix applies to the last element, i.e., [bar].
This is useful if there are several identifiers [bar] in module [Foo], e.g., a type and a value.
- [odoc]: [ {!module-Foo.type-bar} ] - each element in the path may be prefixed by its type. This is
useful if there are ambiguous elements in any part of the path, e.g., in this case perhaps there is a
module type [Foo].
- [odoc3]: [ {!/library_name/Module} ] - this is a path to them module [M] in library [library_name].
This is useful if the module name is ambiguous, e.g., if there are two modules [M] in different libraries.
You may need to add the library to the per-package config file.
- [odoc3]: [ {!/package/path/to/page-foo} ] - this is a path to a page [foo] in the package [package].
The package will need to be added to the per-package config file.
- [odoc3]: [ {!//path/to/page-foo} ] - this is a path to a page [foo] in the {i current} package.

The prefixes supported are:
- [module]
- [module-type] (and the equivalent deprecated prefix [modtype])
- [class]
- [class-type] (and the equivalent deprecated prefix [classtype])
- [val] (and the equivalent deprecated prefix [value])
- [type]
- [exception] (and the equivalent deprecated prefix [exn])
- [method]
- [constructor] (and the equivalent deprecated prefix [const]) both for normal
  constructors and polymorphic constructors defined in a type alias. The
  backtick [`] in polumorphic constructor is optional.
- [extension]
- [extension-decl] for refering to the declaration point of an extension constructor
- [field] (and the equivalent deprecated prefix [recfield])
- [instance-variable]
- [section] (and the equivalent deprecated prefix [label]) - for referring to headings
- [page] - for referring to [.mld] pages

In some cases the element being referenced might have a hyphen, a dot, or a space in the name,
e.g., if trying to refer to a page from a [.mld] file [1.2.3.mld]. In this case, the
element name should be quoted with double quote marks:

{v
{!page-"1.2.3"}
v}


{4 Module Lists}

[odoc] supports a special reference type for referring to a list of modules.
The markup is:

{v
{!modules: A B C}
v}

This will generate a list of links to these modules. If the module has a synopsis
(see {{!synopsis}later}), this will be inserted into the list.

{3 Reference Scope}

[odoc] uses the same scoping as OCaml when resolving references,
but with one major difference. In a particular
signature, {e all} elements are in scope, even those later in the signature.
Consider the following example:

{[
(** In this floating comment, I can refer to type {!t} and value {!v}
    declared later in the signature *)

type t

val v : t
]}

Elements from parent modules are also in scope in child modules. Therefore
the following will also work:

{[
val x : int
val y : int

module A : sig
  (** In this module, I can refer to val {!x}, declared above, as well as
      type {!u}, declared later in the parent module. Elements declared
      in this signature take priority, so {!y} refers to {!A.y} as
      opposed to the [y] declared in the parent signature.  *)

  val y : string
end

type u
]}

The above example can be seen rendered in the module {!Odoc_examples.Markup.Scope}.

[odoc] allows modules to be 'opened' for the purposes of resolving references.
By default, the module [Stdlib] is 'opened', allowing references like [{!List.t}]
to work. This feature is enabled via the command-line flag [--open]. Currently
inline [open] statements do {e not} bring other elements into scope.

In order for [odoc] to resolve links to other compilation units or [.mld] pages,
the referenced
unit or page must be {e compiled} and available to [odoc]. That is, when performing the
[odoc link] command, one of the include paths passed via the command-line argument
[-I] must contain the relevant [.odoc] file. This is normally the responsibility of
the {{!page-driver}driver}.

{2:tags Tags}

Tags are used to provide specific information for individual elements, such
as author, version, parameters, etc. Tags start with an [@] symbol.
They should appear on their own lines with nothing but whitespace before them.

There are three types of tags. Those with:
- no associated data (simple tags),
- a single line of text (line tags), and 
- a block of marked-up text (block tags).

Some tags can only be used on specific contexts: specific items ([include]s,
[module]s, ...) or pages.

{3 Simple Tags}

The three tags without data are hints to the HTML renderer to do with [include]s.
These are:

- [@open] - the contents of the include will be expanded by default in the HTML.
- [@closed] - the contents of the include will be collapsed by default in the HTML.
- [@inline] - the contents of the include will be rendered as if they were part of 
the signature.

{3 Line Tags}

These tags have a single line of data associated with them, [string] in the
examples below. They are:

- [@author string] - allows the element's author to be specified
- [@since string] - declares from which version the element has been available
- [@version string] - declares the version of the element itself
- [@canonical string] - declares the path to this element's canonical instance.
It can be applied to modules, module types, and types. See the {{!page-features.canonical}
Language Features} page for more details

The tag's content is read from the rest of the line. It is also uninterpreted,
i.e., there shouldn't be any [odoc] markup.

{3 Block Tags}

These tags have a block of potentially marked-up text associated with them, and
occasionally some more data too.

The content of a block tag can be any markup, apart from {{!sections}headings}
and other tags. Note that compared to ocamldoc, block tags do not extend to the
end of the docstring. Instead, they are ended by a blank line, or a block that
cannot be included in (a heading or another tag).

{4 Signature tags}

- [@deprecated <text>] - marks the element as deprecated. [text] should describe
when the element was deprecated, what to use as a replacement, and possibly
the reason for the deprecation.
- [@param <id> <text>] - associates the the given description (text) to the given 
parameter name ID. OCamldoc uses this tag for functions, methods, classes, 
and functors, but [odoc] does not currently enforce this restriction.
- [@raise <exn> <text>] - indicates that the element may raise [exn]. The text should
describe when this might occur. [@raises] is a synonym for this tag.
- [@return <text>] - describes the return value. [@returns] is a synonym for this.
- [@before <version> <text>] - allows differences in past behaviour to be described.
This is intended to be used to document compatibility issues.
- [@see <<URL>> <text>] - adds a reference to the URL (written between [<] and
[>] delimiters), with [text] as a comment.
- [@see '<filename>' <text>] - adds a reference to the given file (written between
single quotes), with the given text as comment. {e Note:} [odoc] currently
doesn't turn this into a link in the output HTML.
- [@see "<document-name>" <text>] - adds a reference to the given document name
(written between double quotes), with the given text as comment. {e Note:} 
As with the file reference, [odoc] doesn't turn this into a link.

{4 Page tags}

These tags are the only tags that can be used on pages.

- [@children_order <order>] - defines the order, in the sidebar, of the content
  of a directory. It can only be used on [index.mld] pages. [<order>] must be a
  space-separated list of content. Pages are referred by filename, and modules
  are prefixed with [module-]. Directories are suffixed with a [/]. Here is an
  example:
  {[
    @children_order content module-Unit dir1/
  ]}

- [@toc_status <status>] determines the behaviour of the entry in the sidebar
  and breadcrumbs. It can only be used on [index.mld] pages. [<status>] can be
  either [open] or [hidden]. If it is [open], the content of the directory will
  always be displayed in the sidebar. If it is [hidden], it will be [opened] but
  the directory entry, in the sidebar and breadcrumbs, will not be clickable.

{2:math Mathematics}

[odoc] 2.2 introduced new markup for maths, available both for inline and block
content. The syntax for the maths itself is LaTeX, and it is rendered by {{:https://katex.org/}KaTeX}
in HTML output, dropped inline in the LaTeX output, and ignored in the man-page
renderer.

To render maths inline, use [{m <latex>}], and for block mode use [{math <latex>}].

This is an inline equation: {m x=\frac{-b \pm \sqrt{b^2-4ac}}{2a}}. When rendered
in block form, this becomes:

{math x=\frac{-b \pm \sqrt{b^2-4ac}}{2a}}

See the {{:https://katex.org/docs/supported.html}KaTeX documentation} for the
HTML mode LaTeX support status.

{2:tables Tables}

[odoc] 2.3 introduced new markup for tables. This markup comes in two flavors: the light syntax and the heavy syntax.

The heavy syntax uses several markup: [{table ...}] to define a table, [{tr ...}] to define a row, and [{th ...}] and [{td ...}] to respectively define a header cell and a data cell.
Direct children of tables have to be rows, and direct children of rows have to be cells. Similarly, rows have to be direct children of tables, and cells direct children of row. Cells can contain any markup.

For instance, the following table:

{[
  {table
    {tr
      {th Header 1}
      {th Header 2}
      {th Header 3}
    }
    {tr
      {td Cell 1}
      {td Cell with {e emphasized content}}
      {td {v a block v} }
    }
  }
]}

would render as

{table
  {tr
    {th Header 1}
    {th Header 2}
    {th Header 3}
  }
  {tr
    {td Cell 1}
    {td Cell with {e emphasized content}}
    {td {v a block v} }
  }
}


The light syntax has the advantages of being simple to read, even as plain text. It is very similar to the {{:https://github.github.com/gfm/#tables-extension-}GFM Markdown syntax}, with the exception that it has to be enclosed in [{t ...}], and the inline markup is the OCamldoc one. It supports alignment for columns using the [:] notation from the GFM syntax. [---] is the default alignment, [:--] left-aligned, [--:] right-aligned, and [:---:] centered.

The following table, in light syntax:

{[
{t
  | Header 1 | Header 2 | Header 3 | Header 4|
  | :------: | --------:|:---------|---------|
  | centered | right    | left     | default |
    omitted  | bar at   | start and| finish
  | {e emph} | and | unaligned | bars |
}
]}

would render as

{t
  | Header 1 | Header 2 | Header 3 | Header 4|
  | :------: | --------:|:---------|---------|
  | centered | right    | left     | default |
    omitted  | bar at   | start and| finish
  | {e emph} | and | unaligned | bars |
}

The light syntax has the advantages of being arguably more readable for small tables when viewing the source file directly. However, its content is restricted (for instance, no new line is allowed).
The heavy syntax is easier to write, can be more readable for big tables, and supports having any kind of content inside. It does not support alignment (yet).

{2:media Media}

Odoc 3.0 introduced new markup for media. Media are nestable blocks,
so they can be put inside lists and tables, but they cannot be
inlined, for instance in a link.

There are currently three kinds of media: image, audio, and
video. Each of them can refer to the file either using an asset
reference, or a direct link.

The markup for images is [{image:<link>}], [{image!<path>}],
[{{image:<link>}Replacement text}] and [{{image!<path>}Replacement text}], where
[<link>] is a link and [<path>] is a path to an asset. This path can be either
relative or absolute.

The markup for videos and audios is similar, replacing [image] by respectively
[video] and [audio].

The replacement text is used for backends that do not support media
(latex and man), and for when a reference is unresolved. In the case
of an image, it is also used to generate an alternative text.

Images are clickable and link to the image file.

The following source:

{[
  renders as: {image:https://picsum.photos/200/300}
]}

renders as: {image:https://picsum.photos/200/300}

{2 Stop Comments}

The special comment:

{v
(**/**)
v}

is a {e stop comment}. It acts as a toggle, causing subsequent elements to be
omitted from the documentation. If the stop comment is repeated, the subsequent
items will be visible once more.

The OCaml manual provides an instructive example:

{[
  class type foo =
    object
      (** comment for method m *)
      method m : string

      (**/**)

      (** This method won't appear in the documentation *)
      method bar : int
    end

  (** This value appears in the documentation, since the Stop special comment
      in the class does not affect the parent module of the class.*)
  val foo : string

  (**/**)
  (** The value bar does not appear in the documentation.*)
  val bar : string
  (**/**)

  (** The type [t] appears, since in the documentation since the previous stop comment
  toggled off the "no documentation mode". *)
  type t = string
]}

[odoc] renders the output as {{!Odoc_examples.Markup.Stop}here}.


{1:page_structure Page Structure}

Producing good documentation for your library is more than simply annotating
the various modules, type, and functions that are contained. 
[odoc] expects the documentation to be structured in a logical way, and it will work
best if the following conventions are applied.

The overall structure is that modules start with a {{!preamble} preamble} or
'Lead Section' that serves as an overview of the most important information
about the module. This is followed by the module's content, organised into {e
sections} and {e subsections}. Its structure will be used to populate
a {e table of contents} that will be placed in the HTML immediately
after the preamble and rendered by default as a sidebar.

The preamble's first paragraph will be treated as the module {{!synopsis}
synopsis}, and it will be used as a short description of the module when it
appears in a list of modules elsewhere in the documentation of the library.

{2:top_comment Top-Comment}

The top-comment is the first element of a signature, if it is a documentation
comment. For example, in an [.mli] file:

{[
(** This is the top-comment of the current module. *)

module M : sig
  (** This is the top-comment of [M]. *)

    (* ... *)
  end
]}

As an exception, [open] statements and attributes can be placed before the top-comment.
For example:

{[
[@@@ocaml.warning "-6"]

(* Copyright header *)

open Base

(** This is the top-comment *)

(* ... *)
]}

Note that the top-comment can't be attached to a declaration, for example:

{[
(** This is {e not} the top-comment because it's attached to [t]. *)
type t
]}

{2 Preamble}

The preamble is composed of the comment attached to a declaration and the
top-comment of the corresponding signature, if there is one.
It is special only because it will be placed in the [header] part of the page,
just before the table of contents (if any), and is used to compute the {e synopsis}.

{[
(** This is the comment attached to the declaration. This paragraph will be the
    first of the preamble. *)
module M : sig
  (** This is the top-comment of the expansion. This paragraph will be the
      second of the preamble. *)

  (* ... *)
end
]}

The preamble stops at the first heading. The rest is moved into the [content]
part of the page. For example, the next two snippets will {e render} the same
way:

{[
module M : sig
  (** Preamble.

      {1 Heading}

      This paragraph is not part of the preamble. *)
end
]}

{[
module M : sig
  (** Preamble. *)

  (** {1 Heading}

      This paragraph is not part of the preamble. *)
end
]}

Note: A comment attached to a declaration shouldn't contain any heading.

{2:synopsis Synopsis}

A module's synopsis is the first paragraph of the {!preamble}, {e if} the preamble starts with a paragraph.
This also applies to module types, classes, etc.

It is rendered in [{!modules:...}] lists and after {{!page-features.expansion_aliases}expanded aliases}.

Note that the synopsis is computed on top of the {e preamble}. In these two
examples, the synopsis is the same:

{[
(** This paragraph is the synopsis of the module [M].

    This paragraph is no longer the synopsis and won't be rendered in the
    current page near the declaration of [M]. This paragraph will be part of
    [M]'s preamble. *)
module M : sig
  (* ... *)
end
]}

{[
module M : sig
  (** This paragraph is the synopsis of the module [M]. *)

  (* ... *)
end
]}

{1:sections Sections and Headings}

Both API references and documentation pages can be split into sections that can
be introduced with level-1 headings. Each section can also have subsections
(level-2) and subsubsections (level-3).

Additionally, paragraphs can be annotated with level-4 or level-5 headings.
Note that paragraph headings are {e not} be included in the generated table of
contents and thus should be used to introduce examples, comments, or other
complementary notes. An alternative would be to consider splitting into
multiple files.

Finally, documentation pages should start with a level-0 heading:

{v
{0 Title of the page}
v}

Level-0 headings should not be used elsewhere.

The syntax for declaring sections is as follows:

{v
{[0-5] <text>}
v}
or
{v
{[0-5]:<label> <text>}
v}

where the number represents the sectioning level. [<label>] is an optional label
for the section, allowing it to be referenced via the [{!section-<label>}] reference.
For example:

{v
{2:foobar Foo Bar}
...
See {!section-foobar} for details
v}

In this case, the reference text would be "Foo Bar," so the paragraph would read
"See Foo Bar for details."

Sections with a label cannot be referenced via the section title, only
using the label.

Note that label names can contain [-], but if they do, you have to quote the label
name in the reference link as follows:

{v
{2:foo-bar Foo Bar}
...
See {!section-"foo-bar"} for details
v}

You can use [_] instead of [-] to avoid this syntax.

Sections can also be cross-referenced from another page.
If you define the following section in a [a.mld] file:

{v
{1:foo Foo}
v}

and then want to reference it from another [b.mld] page,
you can use the following syntax:

{v
See {!page-a.foo} for details
v}

{1:config-file Configuration file}

[odoc] 3.0 introduced a per-package configuration file, [odoc-config.sexp]. This file is required if you
wish to make use of the cross-package linking feature, or sometimes if you wish to link to other libraries.

As an example, here is [odoc]'s configuration file:

{v
(libraries fmt)
(packages odoc-driver cmdliner odig)
v}

This file applies to all pages and all modules of all libraries in a package. With this config file, the
modules of the [fmt] library, and the {e modules and pages} of the [odoc-driver], [cmdliner] and [odig] packages can be
linked to via the following syntax:

{v
{!/fmt/module-Fmt}
{!/odoc-driver/page-index}
v}

Important note: In order for this to work on ocaml.org, it is necessary that the tool used to build the
docs is aware that extra packages are required. If the package is already a dependency, then nothing needs
to be done. If it can be added without introducing dependency cycles, it can be added to the normal
dependencies field. If it would otherwise introduce a cycle into the dependencies, then it must be added
to an extra field. In the case of odoc, because the docs of [odoc] wish to link to the docs of [odoc-driver],
but [odoc-driver] depends on odoc, the following must be added to the [odoc]'s opam file:

{v
x-extra-doc-deps: [
  "odoc-driver" {>= "3.0.0"}
]
v}