markly 0.6.1 → 0.8.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (52) hide show
  1. checksums.yaml +4 -4
  2. checksums.yaml.gz.sig +0 -0
  3. data/conduct.md +133 -0
  4. data/ext/markly/arena.c +9 -8
  5. data/ext/markly/autolink.c +217 -134
  6. data/ext/markly/blocks.c +40 -4
  7. data/ext/markly/cmark-gfm-core-extensions.h +11 -11
  8. data/ext/markly/cmark-gfm-extension_api.h +1 -0
  9. data/ext/markly/cmark-gfm.h +18 -2
  10. data/ext/markly/cmark-gfm_version.h +2 -2
  11. data/ext/markly/cmark.c +3 -3
  12. data/ext/markly/commonmark.c +33 -38
  13. data/ext/markly/ext_scanners.c +360 -640
  14. data/ext/markly/extconf.rb +8 -1
  15. data/ext/markly/footnotes.c +23 -0
  16. data/ext/markly/footnotes.h +2 -0
  17. data/ext/markly/html.c +60 -23
  18. data/ext/markly/inlines.c +216 -61
  19. data/ext/markly/latex.c +6 -4
  20. data/ext/markly/man.c +7 -11
  21. data/ext/markly/map.c +11 -4
  22. data/ext/markly/map.h +5 -2
  23. data/ext/markly/markly.c +582 -586
  24. data/ext/markly/markly.h +1 -1
  25. data/ext/markly/node.c +76 -10
  26. data/ext/markly/node.h +49 -1
  27. data/ext/markly/parser.h +1 -0
  28. data/ext/markly/plaintext.c +12 -29
  29. data/ext/markly/references.c +1 -0
  30. data/ext/markly/render.c +15 -7
  31. data/ext/markly/scanners.c +13916 -20242
  32. data/ext/markly/scanners.h +8 -0
  33. data/ext/markly/scanners.re +47 -8
  34. data/ext/markly/strikethrough.c +1 -1
  35. data/ext/markly/table.c +143 -74
  36. data/ext/markly/xml.c +2 -1
  37. data/lib/markly/flags.rb +16 -0
  38. data/lib/markly/node/inspect.rb +59 -53
  39. data/lib/markly/node.rb +125 -58
  40. data/lib/markly/renderer/generic.rb +136 -0
  41. data/lib/markly/renderer/html.rb +301 -0
  42. data/lib/markly/version.rb +7 -1
  43. data/lib/markly.rb +38 -32
  44. data/license.md +39 -0
  45. data/readme.md +36 -0
  46. data.tar.gz.sig +0 -0
  47. metadata +63 -31
  48. metadata.gz.sig +0 -0
  49. data/bin/markly +0 -94
  50. data/lib/markly/markly.so +0 -0
  51. data/lib/markly/renderer/html_renderer.rb +0 -281
  52. data/lib/markly/renderer.rb +0 -133
data/lib/markly.rb CHANGED
@@ -1,43 +1,49 @@
1
1
  #!/usr/bin/env ruby
2
2
  # frozen_string_literal: true
3
3
 
4
- # The compiled library.
4
+ # Released under the MIT License.
5
+ # Copyright, 2014, by John MacFarlane.
6
+ # Copyright, 2015-2019, by Garen Torikian.
7
+ # Copyright, 2015, by Nick Wellnhofer.
8
+ # Copyright, 2016-2017, by Yuki Izumi.
9
+ # Copyright, 2020-2023, by Samuel Williams.
10
+
5
11
  require 'markly/markly'
6
12
 
7
13
  require_relative 'markly/flags'
8
14
  require_relative 'markly/node'
9
- require_relative 'markly/renderer'
10
- require_relative 'markly/renderer/html_renderer'
15
+ require_relative 'markly/renderer/html'
16
+
11
17
  require_relative 'markly/version'
12
18
 
13
19
  module Markly
14
- # Public: Parses a Markdown string into a `document` node.
15
- #
16
- # string - {String} to be parsed
17
- # option - A {Symbol} or {Array of Symbol}s indicating the parse options
18
- # extensions - An {Array of Symbol}s indicating the extensions to use
19
- #
20
- # Returns the `parser` node.
21
- def self.parse(text, flags: DEFAULT, extensions: nil)
22
- parser = Parser.new(flags)
23
-
24
- extensions&.each do |extension|
25
- parser.enable(extension)
26
- end
27
-
28
- return parser.parse(text.encode(Encoding::UTF_8))
29
- end
30
-
31
- # Public: Parses a Markdown string into an HTML string.
32
- #
33
- # text - A {String} of text
34
- # option - Either a {Symbol} or {Array of Symbol}s indicating the render options
35
- # extensions - An {Array of Symbol}s indicating the extensions to use
36
- #
37
- # Returns a {String} of converted HTML.
38
- def self.render_html(text, flags: DEFAULT, parse_flags: flags, render_flags: flags, extensions: [])
39
- root = self.parse(text, flags: parse_flags, extensions: extensions)
40
-
41
- return root.to_html(flags: render_flags, extensions: extensions)
42
- end
20
+ # Public: Parses a Markdown string into a `document` node.
21
+ #
22
+ # string - {String} to be parsed
23
+ # option - A {Symbol} or {Array of Symbol}s indicating the parse options
24
+ # extensions - An {Array of Symbol}s indicating the extensions to use
25
+ #
26
+ # Returns the `parser` node.
27
+ def self.parse(text, flags: DEFAULT, extensions: nil)
28
+ parser = Parser.new(flags)
29
+
30
+ extensions&.each do |extension|
31
+ parser.enable(extension)
32
+ end
33
+
34
+ return parser.parse(text.encode(Encoding::UTF_8))
35
+ end
36
+
37
+ # Public: Parses a Markdown string into an HTML string.
38
+ #
39
+ # text - A {String} of text
40
+ # option - Either a {Symbol} or {Array of Symbol}s indicating the render options
41
+ # extensions - An {Array of Symbol}s indicating the extensions to use
42
+ #
43
+ # Returns a {String} of converted HTML.
44
+ def self.render_html(text, flags: DEFAULT, parse_flags: flags, render_flags: flags, extensions: [])
45
+ root = self.parse(text, flags: parse_flags, extensions: extensions)
46
+
47
+ return root.to_html(flags: render_flags, extensions: extensions)
48
+ end
43
49
  end
data/license.md ADDED
@@ -0,0 +1,39 @@
1
+ # MIT License
2
+
3
+ Copyright, 2014, by John MacFarlane.
4
+ Copyright, 2015-2020, by Garen Torikian.
5
+ Copyright, 2015, by Nick Wellnhofer.
6
+ Copyright, 2015-2017, by Andrew Anderson.
7
+ Copyright, 2016-2017, by Yuki Izumi.
8
+ Copyright, 2017, by Mu-An Chiou.
9
+ Copyright, 2017-2018, by Goro Fuji.
10
+ Copyright, 2017, by Ben Woosley.
11
+ Copyright, 2017, by Roberto Hidalgo.
12
+ Copyright, 2017-2020, by Ashe Connor.
13
+ Copyright, 2018, by Jerry van Leeuwen.
14
+ Copyright, 2018, by Vitaliy Klachkov.
15
+ Copyright, 2018, by Michael Camilleri.
16
+ Copyright, 2018, by Akira Matsuda.
17
+ Copyright, 2018, by Danny Iachini.
18
+ Copyright, 2019-2020, by Tomoya Chiba.
19
+ Copyright, 2019, by Brett Walker.
20
+ Copyright, 2020, by Olle Jonsson.
21
+ Copyright, 2020-2023, by Samuel Williams.
22
+
23
+ Permission is hereby granted, free of charge, to any person obtaining a copy
24
+ of this software and associated documentation files (the "Software"), to deal
25
+ in the Software without restriction, including without limitation the rights
26
+ to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
27
+ copies of the Software, and to permit persons to whom the Software is
28
+ furnished to do so, subject to the following conditions:
29
+
30
+ The above copyright notice and this permission notice shall be included in all
31
+ copies or substantial portions of the Software.
32
+
33
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
34
+ IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
35
+ FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
36
+ AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
37
+ LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
38
+ OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
39
+ SOFTWARE.
data/readme.md ADDED
@@ -0,0 +1,36 @@
1
+ # Markly
2
+
3
+ A parser and abstract syntax tree for Markdown documents (CommonMark compatible) in Ruby. Originally forked from
4
+ [CommonMarker](https://github.com/gjtorikian/commonmarker). It also includes extensions to the CommonMark spec as
5
+ documented in the [GitHub Flavored Markdown spec](http://github.github.com/gfm/), such as support for tables,
6
+ strikethroughs, and autolinking.
7
+
8
+ [![Development
9
+ Status](https://github.com/ioquatix/markly/workflows/Test/badge.svg)](https://github.com/ioquatix/markly/actions?workflow=Test)
10
+
11
+ ## Motivation
12
+
13
+ This code base was originally forked from [Commonmarker](https://github.com/gjtorikian/commonmarker) before they
14
+ switched from `cmark-gfm` (C) to `comrak` (Rust). The original implementation provided access to the abstract syntax
15
+ tree (AST), which is useful for building tools on top of Markdown. The Rust implementation does not provide this
16
+ functionality, and so this fork was created to continue to provide these (and more) features.
17
+
18
+ ## Usage
19
+
20
+ Please see the [project documentation](https://ioquatix.github.io/markly/) for more details.
21
+
22
+ - [Getting Started](https://ioquatix.github.io/markly/guides/getting-started/index) - This guide explains now to
23
+ install and use Markly.
24
+
25
+ - [Abstract Syntax Tree](https://ioquatix.github.io/markly/guides/abstract-syntax-tree/index) - This guide explains
26
+ how to use Markly's abstract syntax tree (AST) to parse and manipulate Markdown documents.
27
+
28
+ ## Contributing
29
+
30
+ We welcome contributions to this project.
31
+
32
+ 1. Fork it.
33
+ 2. Create your feature branch (`git checkout -b my-new-feature`).
34
+ 3. Commit your changes (`git commit -am 'Add some feature'`).
35
+ 4. Push to the branch (`git push origin my-new-feature`).
36
+ 5. Create new Pull Request.
data.tar.gz.sig ADDED
Binary file
metadata CHANGED
@@ -1,16 +1,61 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: markly
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.6.1
4
+ version: 0.8.0
5
5
  platform: ruby
6
6
  authors:
7
7
  - Garen Torikian
8
- - Ashe Connor
8
+ - Yuki Izumi
9
9
  - Samuel Williams
10
+ - John MacFarlane
11
+ - Ashe Connor
12
+ - Nick Wellnhofer
13
+ - Brett Walker
14
+ - Andrew Anderson
15
+ - Ben Woosley
16
+ - Goro Fuji
17
+ - Tomoya Chiba
18
+ - Akira Matsuda
19
+ - Danny Iachini
20
+ - Jerry van Leeuwen
21
+ - Michael Camilleri
22
+ - Mu-An Chiou
23
+ - Olle Jonsson
24
+ - Roberto Hidalgo
25
+ - Vitaliy Klachkov
10
26
  autorequire:
11
27
  bindir: bin
12
- cert_chain: []
13
- date: 2021-09-03 00:00:00.000000000 Z
28
+ cert_chain:
29
+ - |
30
+ -----BEGIN CERTIFICATE-----
31
+ MIIE2DCCA0CgAwIBAgIBATANBgkqhkiG9w0BAQsFADBhMRgwFgYDVQQDDA9zYW11
32
+ ZWwud2lsbGlhbXMxHTAbBgoJkiaJk/IsZAEZFg1vcmlvbnRyYW5zZmVyMRIwEAYK
33
+ CZImiZPyLGQBGRYCY28xEjAQBgoJkiaJk/IsZAEZFgJuejAeFw0yMjA4MDYwNDUz
34
+ MjRaFw0zMjA4MDMwNDUzMjRaMGExGDAWBgNVBAMMD3NhbXVlbC53aWxsaWFtczEd
35
+ MBsGCgmSJomT8ixkARkWDW9yaW9udHJhbnNmZXIxEjAQBgoJkiaJk/IsZAEZFgJj
36
+ bzESMBAGCgmSJomT8ixkARkWAm56MIIBojANBgkqhkiG9w0BAQEFAAOCAY8AMIIB
37
+ igKCAYEAomvSopQXQ24+9DBB6I6jxRI2auu3VVb4nOjmmHq7XWM4u3HL+pni63X2
38
+ 9qZdoq9xt7H+RPbwL28LDpDNflYQXoOhoVhQ37Pjn9YDjl8/4/9xa9+NUpl9XDIW
39
+ sGkaOY0eqsQm1pEWkHJr3zn/fxoKPZPfaJOglovdxf7dgsHz67Xgd/ka+Wo1YqoE
40
+ e5AUKRwUuvaUaumAKgPH+4E4oiLXI4T1Ff5Q7xxv6yXvHuYtlMHhYfgNn8iiW8WN
41
+ XibYXPNP7NtieSQqwR/xM6IRSoyXKuS+ZNGDPUUGk8RoiV/xvVN4LrVm9upSc0ss
42
+ RZ6qwOQmXCo/lLcDUxJAgG95cPw//sI00tZan75VgsGzSWAOdjQpFM0l4dxvKwHn
43
+ tUeT3ZsAgt0JnGqNm2Bkz81kG4A2hSyFZTFA8vZGhp+hz+8Q573tAR89y9YJBdYM
44
+ zp0FM4zwMNEUwgfRzv1tEVVUEXmoFCyhzonUUw4nE4CFu/sE3ffhjKcXcY//qiSW
45
+ xm4erY3XAgMBAAGjgZowgZcwCQYDVR0TBAIwADALBgNVHQ8EBAMCBLAwHQYDVR0O
46
+ BBYEFO9t7XWuFf2SKLmuijgqR4sGDlRsMC4GA1UdEQQnMCWBI3NhbXVlbC53aWxs
47
+ aWFtc0BvcmlvbnRyYW5zZmVyLmNvLm56MC4GA1UdEgQnMCWBI3NhbXVlbC53aWxs
48
+ aWFtc0BvcmlvbnRyYW5zZmVyLmNvLm56MA0GCSqGSIb3DQEBCwUAA4IBgQB5sxkE
49
+ cBsSYwK6fYpM+hA5B5yZY2+L0Z+27jF1pWGgbhPH8/FjjBLVn+VFok3CDpRqwXCl
50
+ xCO40JEkKdznNy2avOMra6PFiQyOE74kCtv7P+Fdc+FhgqI5lMon6tt9rNeXmnW/
51
+ c1NaMRdxy999hmRGzUSFjozcCwxpy/LwabxtdXwXgSay4mQ32EDjqR1TixS1+smp
52
+ 8C/NCWgpIfzpHGJsjvmH2wAfKtTTqB9CVKLCWEnCHyCaRVuKkrKjqhYCdmMBqCws
53
+ JkxfQWC+jBVeG9ZtPhQgZpfhvh+6hMhraUYRQ6XGyvBqEUe+yo6DKIT3MtGE2+CP
54
+ eX9i9ZWBydWb8/rvmwmX2kkcBbX0hZS1rcR593hGc61JR6lvkGYQ2MYskBveyaxt
55
+ Q2K9NVun/S785AP05vKkXZEFYxqG6EW012U4oLcFl5MySFajYXRYbuUpH6AY+HP8
56
+ voD0MPg1DssDLKwXyt1eKD/+Fq0bFWhwVM/1XiAXL7lyYUyOq24KHgQ2Csg=
57
+ -----END CERTIFICATE-----
58
+ date: 2023-06-17 00:00:00.000000000 Z
14
59
  dependencies:
15
60
  - !ruby/object:Gem::Dependency
16
61
  name: bake
@@ -27,21 +72,7 @@ dependencies:
27
72
  - !ruby/object:Gem::Version
28
73
  version: '0'
29
74
  - !ruby/object:Gem::Dependency
30
- name: minitest
31
- requirement: !ruby/object:Gem::Requirement
32
- requirements:
33
- - - "~>"
34
- - !ruby/object:Gem::Version
35
- version: '5.6'
36
- type: :development
37
- prerelease: false
38
- version_requirements: !ruby/object:Gem::Requirement
39
- requirements:
40
- - - "~>"
41
- - !ruby/object:Gem::Version
42
- version: '5.6'
43
- - !ruby/object:Gem::Dependency
44
- name: rake
75
+ name: covered
45
76
  requirement: !ruby/object:Gem::Requirement
46
77
  requirements:
47
78
  - - ">="
@@ -55,28 +86,27 @@ dependencies:
55
86
  - !ruby/object:Gem::Version
56
87
  version: '0'
57
88
  - !ruby/object:Gem::Dependency
58
- name: rake-compiler
89
+ name: sus
59
90
  requirement: !ruby/object:Gem::Requirement
60
91
  requirements:
61
- - - "~>"
92
+ - - ">="
62
93
  - !ruby/object:Gem::Version
63
- version: '0.9'
94
+ version: '0'
64
95
  type: :development
65
96
  prerelease: false
66
97
  version_requirements: !ruby/object:Gem::Requirement
67
98
  requirements:
68
- - - "~>"
99
+ - - ">="
69
100
  - !ruby/object:Gem::Version
70
- version: '0.9'
101
+ version: '0'
71
102
  description:
72
103
  email:
73
- executables:
74
- - markly
104
+ executables: []
75
105
  extensions:
76
106
  - ext/markly/extconf.rb
77
107
  extra_rdoc_files: []
78
108
  files:
79
- - bin/markly
109
+ - conduct.md
80
110
  - ext/markly/arena.c
81
111
  - ext/markly/autolink.c
82
112
  - ext/markly/autolink.h
@@ -150,17 +180,19 @@ files:
150
180
  - ext/markly/xml.c
151
181
  - lib/markly.rb
152
182
  - lib/markly/flags.rb
153
- - lib/markly/markly.so
154
183
  - lib/markly/node.rb
155
184
  - lib/markly/node/inspect.rb
156
- - lib/markly/renderer.rb
157
- - lib/markly/renderer/html_renderer.rb
185
+ - lib/markly/renderer/generic.rb
186
+ - lib/markly/renderer/html.rb
158
187
  - lib/markly/version.rb
188
+ - license.md
189
+ - readme.md
159
190
  homepage: https://github.com/ioquatix/markly
160
191
  licenses:
161
192
  - MIT
162
193
  metadata:
163
194
  funding_uri: https://github.com/sponsors/ioquatix/
195
+ documentation_uri: https://ioquatix.github.io/markly/
164
196
  post_install_message:
165
197
  rdoc_options: []
166
198
  require_paths:
@@ -176,7 +208,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
176
208
  - !ruby/object:Gem::Version
177
209
  version: '0'
178
210
  requirements: []
179
- rubygems_version: 3.1.6
211
+ rubygems_version: 3.4.7
180
212
  signing_key:
181
213
  specification_version: 4
182
214
  summary: CommonMark parser and renderer. Written in C, wrapped in Ruby.
metadata.gz.sig ADDED
Binary file
data/bin/markly DELETED
@@ -1,94 +0,0 @@
1
- #!/usr/bin/env ruby
2
- # frozen_string_literal: true
3
-
4
- require 'optparse'
5
- require 'ostruct'
6
-
7
- require_relative '../lib/markly'
8
-
9
- def parse_options
10
- options = OpenStruct.new
11
- extensions = Markly.extensions
12
- parse_flags = Markly::PARSE_FLAGS
13
- render_flags = Markly::RENDER_FLAGS
14
-
15
- options.active_extensions = []
16
- options.active_parse_flags = Markly::DEFAULT
17
- options.active_render_flags = Markly::DEFAULT
18
-
19
- option_parser = OptionParser.new do |opts|
20
- opts.banner = 'Usage: markly [--html-renderer] [--extension=EXTENSION]'
21
- opts.separator ' [--parse-option=OPTION]'
22
- opts.separator ' [--render-option=OPTION]'
23
- opts.separator ' [FILE..]'
24
- opts.separator ''
25
- opts.separator 'Convert one or more CommonMark files to HTML and write to standard output.'
26
- opts.separator 'If no FILE argument is provided, text will be read from STDIN.'
27
- opts.separator ''
28
-
29
- opts.on('--extension=EXTENSION', Array, 'Use EXTENSION for parsing and HTML output (unless --html-renderer is specified)') do |values|
30
- values.each do |value|
31
- if extensions.include?(value)
32
- options.active_extensions << value.to_sym
33
- else
34
- abort("extension '#{value}' not found")
35
- end
36
- end
37
- end
38
-
39
- opts.on('-h', '--help', 'Prints this help') do
40
- puts opts
41
- puts
42
- puts "Available extentions: #{extensions.join(', ')}"
43
- puts "Available parse flags: #{parser_flags.keys.join(', ')}"
44
- puts "Available render flags: #{render_flags.keys.join(', ')}"
45
- puts
46
- puts 'See the README for more information on these.'
47
- exit
48
- end
49
-
50
- opts.on('--html-renderer', 'Use the HtmlRenderer renderer rather than the native C renderer') do
51
- options.renderer = true
52
- end
53
-
54
- opts.on('--parse-option=OPTION', Array, 'OPTION passed during parsing') do |values|
55
- values.each do |value|active_parser_flags
56
- if parser_flags.key?(value.to_sym)
57
- options.active_parser_flags << value.to_sym
58
- else
59
- abort("parse-option '#{value}' not found")
60
- end
61
- end
62
- end
63
-
64
- opts.on('--render-option=OPTION', Array, 'OPTION passed during rendering') do |values|
65
- values.each do |value|
66
- if render_flags.key?(value.to_sym)
67
- options.active_render_flags << value.to_sym
68
- else
69
- abort("render-option '#{value}' not found")
70
- end
71
- end
72
- end
73
-
74
- opts.on('-v', '--version', 'Version information') do
75
- puts "markly #{Markly::VERSION}"
76
- exit
77
- end
78
- end
79
-
80
- option_parser.parse!
81
-
82
- options
83
- end
84
-
85
- options = parse_options
86
-
87
- doc = Markly.parse(ARGF.read, flags: options.active_parse_flags, extensions: options.active_extensions)
88
-
89
- if options.renderer
90
- renderer = Markly::HtmlRenderer.new(extensions: options.active_extensions)
91
- STDOUT.write(renderer.render(doc))
92
- else
93
- STDOUT.write(doc.to_html(flags: options.active_render_flags, extensions: options.active_extensions))
94
- end
data/lib/markly/markly.so DELETED
Binary file
@@ -1,281 +0,0 @@
1
- # frozen_string_literal: true
2
-
3
- require 'cgi'
4
-
5
- module Markly
6
- class HTMLRenderer < Renderer
7
- def initialize(ids: false, tight: false, **options)
8
- super(**options)
9
-
10
- @ids = ids
11
- @section = nil
12
- @tight = tight
13
- end
14
-
15
- def document(_)
16
- @section = false
17
- super
18
- out("</ol>\n</section>\n") if @written_footnote_ix
19
- out("</section>") if @section
20
- end
21
-
22
- def id_for(node)
23
- if @ids
24
- id = node.to_plaintext.chomp.downcase.gsub(/\s+/, '-')
25
-
26
- return " id=\"#{CGI.escape_html id}\""
27
- end
28
- end
29
-
30
- def header(node)
31
- block do
32
- if @ids
33
- out('</section>') if @section
34
- @section = true
35
- out("<section#{id_for(node)}>")
36
- end
37
-
38
- out('<h', node.header_level, "#{source_position(node)}>", :children,
39
- '</h', node.header_level, '>')
40
- end
41
- end
42
-
43
- def paragraph(node)
44
- if @tight && node.parent.type != :blockquote
45
- out(:children)
46
- else
47
- block do
48
- container("<p#{source_position(node)}>", '</p>') do
49
- out(:children)
50
- if node.parent.type == :footnote_definition && node.next.nil?
51
- out(' ')
52
- out_footnote_backref
53
- end
54
- end
55
- end
56
- end
57
- end
58
-
59
- def list(node)
60
- old_tight = @tight
61
- @tight = node.list_tight
62
-
63
- block do
64
- if node.list_type == :bullet_list
65
- container("<ul#{source_position(node)}>\n", '</ul>') do
66
- out(:children)
67
- end
68
- else
69
- start = if node.list_start == 1
70
- "<ol#{source_position(node)}>\n"
71
- else
72
- "<ol start=\"#{node.list_start}\"#{source_position(node)}>\n"
73
- end
74
- container(start, '</ol>') do
75
- out(:children)
76
- end
77
- end
78
- end
79
-
80
- @tight = old_tight
81
- end
82
-
83
- def list_item(node)
84
- block do
85
- tasklist_data = tasklist(node)
86
- container("<li#{source_position(node)}#{tasklist_data}>#{' ' if tasklist?(node)}", '</li>') do
87
- out(:children)
88
- end
89
- end
90
- end
91
-
92
- def tasklist(node)
93
- return '' unless tasklist?(node)
94
-
95
- state = if checked?(node)
96
- 'checked="" disabled=""'
97
- else
98
- 'disabled=""'
99
- end
100
- "><input type=\"checkbox\" #{state} /"
101
- end
102
-
103
- def blockquote(node)
104
- block do
105
- container("<blockquote#{source_position(node)}>\n", '</blockquote>') do
106
- out(:children)
107
- end
108
- end
109
- end
110
-
111
- def hrule(node)
112
- block do
113
- out("<hr#{source_position(node)} />")
114
- end
115
- end
116
-
117
- def code_block(node)
118
- block do
119
- if flag_enabled?(GITHUB_PRE_LANG)
120
- out("<pre#{source_position(node)}")
121
- out(' lang="', node.fence_info.split(/\s+/)[0], '"') if node.fence_info && !node.fence_info.empty?
122
- out('><code>')
123
- else
124
- out("<pre#{source_position(node)}><code")
125
- if node.fence_info && !node.fence_info.empty?
126
- out(' class="language-', node.fence_info.split(/\s+/)[0], '">')
127
- else
128
- out('>')
129
- end
130
- end
131
- out(escape_html(node.string_content))
132
- out('</code></pre>')
133
- end
134
- end
135
-
136
- def html(node)
137
- block do
138
- if flag_enabled?(UNSAFE)
139
- out(tagfilter(node.string_content))
140
- else
141
- out('<!-- raw HTML omitted -->')
142
- end
143
- end
144
- end
145
-
146
- def inline_html(node)
147
- if flag_enabled?(UNSAFE)
148
- out(tagfilter(node.string_content))
149
- else
150
- out('<!-- raw HTML omitted -->')
151
- end
152
- end
153
-
154
- def emph(_)
155
- out('<em>', :children, '</em>')
156
- end
157
-
158
- def strong(_)
159
- out('<strong>', :children, '</strong>')
160
- end
161
-
162
- def link(node)
163
- out('<a href="', node.url.nil? ? '' : escape_href(node.url), '"')
164
- out(' title="', escape_html(node.title), '"') if node.title && !node.title.empty?
165
- out('>', :children, '</a>')
166
- end
167
-
168
- def image(node)
169
- out('<img src="', escape_href(node.url), '"')
170
- plain do
171
- out(' alt="', :children, '"')
172
- end
173
- out(' title="', escape_html(node.title), '"') if node.title && !node.title.empty?
174
- out(' />')
175
- end
176
-
177
- def text(node)
178
- out(escape_html(node.string_content))
179
- end
180
-
181
- def code(node)
182
- out('<code>')
183
- out(escape_html(node.string_content))
184
- out('</code>')
185
- end
186
-
187
- def linebreak(_node)
188
- out("<br />\n")
189
- end
190
-
191
- def softbreak(_)
192
- if flag_enabled?(HARD_BREAKS)
193
- out("<br />\n")
194
- elsif flag_enabled?(NO_BREAKS)
195
- out(' ')
196
- else
197
- out("\n")
198
- end
199
- end
200
-
201
- def table(node)
202
- @alignments = node.table_alignments
203
- @needs_close_tbody = false
204
- out("<table#{source_position(node)}>\n", :children)
205
- out("</tbody>\n") if @needs_close_tbody
206
- out("</table>\n")
207
- end
208
-
209
- def table_header(node)
210
- @column_index = 0
211
-
212
- @in_header = true
213
- out("<thead>\n<tr#{source_position(node)}>\n", :children, "</tr>\n</thead>\n")
214
- @in_header = false
215
- end
216
-
217
- def table_row(node)
218
- @column_index = 0
219
- if !@in_header && !@needs_close_tbody
220
- @needs_close_tbody = true
221
- out("<tbody>\n")
222
- end
223
- out("<tr#{source_position(node)}>\n", :children, "</tr>\n")
224
- end
225
-
226
- def table_cell(node)
227
- align = case @alignments[@column_index]
228
- when :left then ' align="left"'
229
- when :right then ' align="right"'
230
- when :center then ' align="center"'
231
- else; ''
232
- end
233
- out(@in_header ? "<th#{align}#{source_position(node)}>" : "<td#{align}#{source_position(node)}>", :children, @in_header ? "</th>\n" : "</td>\n")
234
- @column_index += 1
235
- end
236
-
237
- def strikethrough(_)
238
- out('<del>', :children, '</del>')
239
- end
240
-
241
- def footnote_reference(node)
242
- out("<sup class=\"footnote-ref\"><a href=\"#fn#{node.string_content}\" id=\"fnref#{node.string_content}\">#{node.string_content}</a></sup>")
243
- end
244
-
245
- def footnote_definition(_)
246
- unless @footnote_ix
247
- out("<section class=\"footnotes\">\n<ol>\n")
248
- @footnote_ix = 0
249
- end
250
-
251
- @footnote_ix += 1
252
- out("<li id=\"fn#{@footnote_ix}\">\n", :children)
253
- out("\n") if out_footnote_backref
254
- out("</li>\n")
255
- # </ol>
256
- # </section>
257
- end
258
-
259
- private
260
-
261
- def out_footnote_backref
262
- return false if @written_footnote_ix == @footnote_ix
263
-
264
- @written_footnote_ix = @footnote_ix
265
-
266
- out("<a href=\"#fnref#{@footnote_ix}\" class=\"footnote-backref\">↩</a>")
267
- true
268
- end
269
-
270
- def tasklist?(node)
271
- node.type_string == 'tasklist'
272
- end
273
-
274
- def checked?(node)
275
- node.tasklist_item_checked?
276
- end
277
- end
278
-
279
- # Legacy.
280
- HtmlRenderer = HTMLRenderer
281
- end