prose_mirror_ruby 0.1.0 → 0.1.2

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.
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 9ca1e1715fe8d1e412d15a8ffddb367b6bd3e648029e62372051f343adcf0172
4
- data.tar.gz: d5e675c75e8acb88e347d1f38ee30f157d63310d886883ea838ed96e52c9b4c1
3
+ metadata.gz: 997123de42d6971c9247120b96c785e771688b0e6d0555d0ba0f6f0a2810b45e
4
+ data.tar.gz: e03ebfd450a7265dbeae8f3bc257464116a526a6921f40cd842a857ae9507459
5
5
  SHA512:
6
- metadata.gz: b02364d8907439c6ab5cab1a1c829bd8f5c437300dc72a93c81786e54a32bf0ede115a63280441fe58a2c3effeb973eb91e997e3359bae2e572e4fc1192886d0
7
- data.tar.gz: 80ccdbcef5ab0e714185707114402727fbb603d431be68c28a2ab58bb4c5bad61640d5b7b4373fefa89a7a4660f07be805f838d03ff9a5670c4201878f70c978
6
+ metadata.gz: eefad1d1c11004d5d7b1ae3586185b52b589d9a84d3c8c8d7368e48587035dc6dc981687edca8fa96e27beab326d0f5fb86ff9b3ae6c1295f38bd55a2cd45308
7
+ data.tar.gz: 160e9d5b278280935b2b5550ff8062df6c4241a6d549a13d6c4ce17b4871ccb40f1455a43cd4980db9e92fe594203ae54d2a58ae7f2cf803c14081a2d9e714db
data/README.md CHANGED
@@ -134,6 +134,18 @@ After checking out the repo, run `bin/setup` to install dependencies. Then, run
134
134
 
135
135
  To install this gem onto your local machine, run `bundle exec rake install`. To release a new version, update the version number in `version.rb`, and then run `bundle exec rake release`, which will create a git tag for the version, push git commits and the created tag, and push the `.gem` file to [rubygems.org](https://rubygems.org).
136
136
 
137
+ For developers looking to contribute or extend this gem, the test suite includes examples of various complex ProseMirror document structures including:
138
+
139
+ - Blockquotes with embedded lists
140
+ - Linked images
141
+ - Nested lists (ordered and unordered)
142
+ - Mixed formatting (bold and italic combined)
143
+ - Code blocks with language specification
144
+ - Horizontal rules
145
+ - Tables (pending implementation)
146
+
147
+ These tests serve as documentation for how different structures are (or should be) handled.
148
+
137
149
  ## Contributing
138
150
 
139
151
  Bug reports and pull requests are welcome on GitHub at https://github.com/firehydrant/prose_mirror.
@@ -141,6 +153,19 @@ Bug reports and pull requests are welcome on GitHub at https://github.com/firehy
141
153
  ## Known Issues
142
154
 
143
155
  - Markdown serialization for lists and code blocks may not produce perfect output in all cases. Contributions to improve this are welcome.
156
+ - Complex document structures like nested lists, linked images, and tables have varying levels of support:
157
+ - Basic formatting (bold, italic, links) works well
158
+ - Blockquotes with lists now have improved formatting
159
+ - Nested lists have significantly improved indentation and spacing
160
+ - Complex alternating list types (switching between ordered and bullet lists) now render with better formatting
161
+ - Tables are not currently supported
162
+ - Custom mark types (beyond the standard strong, em, link, code) may not be properly rendered
163
+
164
+ ## Recent Improvements
165
+
166
+ - **Better Nested List Handling**: The library now supports improved indentation for nested lists (e.g., ordered lists inside bullet lists or vice versa), making the generated Markdown more readable.
167
+ - **CamelCase Support**: The library now automatically converts camelCase node types (like "orderedList") to snake_case ("ordered_list"), making it compatible with a wider range of ProseMirror documents and schemas.
168
+ - **Enhanced List Rendering**: Completely refactored list rendering with better indentation, spacing, and handling of complex nested structures.
144
169
 
145
170
  ## License
146
171
 
@@ -15,7 +15,8 @@ module ProseMirror
15
15
  # @param node_data [Hash] The node data from JSON
16
16
  # @return [Node] The parsed node
17
17
  def self.parse_node(node_data)
18
- type = node_data["type"]
18
+ # Convert camelCase node types to snake_case (e.g., "orderedList" -> "ordered_list")
19
+ type = node_data["type"].underscore
19
20
  attrs = parse_attrs(node_data["attrs"] || {})
20
21
  marks = parse_marks(node_data["marks"] || [])
21
22
 
@@ -35,7 +36,8 @@ module ProseMirror
35
36
  def self.parse_attrs(attrs_data)
36
37
  result = {}
37
38
  attrs_data.each do |key, value|
38
- result[key.to_sym] = value
39
+ # Convert camelCase attr keys to snake_case as well
40
+ result[key.underscore.to_sym] = value
39
41
  end
40
42
  result
41
43
  end
@@ -45,7 +47,9 @@ module ProseMirror
45
47
  # @return [Array<Mark>] Array of Mark objects
46
48
  def self.parse_marks(marks_data)
47
49
  marks_data.map do |mark_data|
48
- Mark.new(mark_data["type"], parse_attrs(mark_data["attrs"] || {}))
50
+ # Convert camelCase mark types to snake_case
51
+ mark_type = mark_data["type"].underscore
52
+ Mark.new(mark_type, parse_attrs(mark_data["attrs"] || {}))
49
53
  end
50
54
  end
51
55
  end
@@ -10,7 +10,15 @@ module ProseMirror
10
10
  # Default serializers for various node types
11
11
  DEFAULT_NODE_SERIALIZERS = {
12
12
  blockquote: ->(state, node, parent = nil, index = nil) {
13
+ # Track that we're in a blockquote to handle lists within blockquotes properly
14
+ old_in_blockquote = state.instance_variable_get(:@in_blockquote) || false
15
+ state.instance_variable_set(:@in_blockquote, true)
16
+
17
+ # Use the standard blockquote prefix for all content
13
18
  state.wrap_block("> ", nil, node) { state.render_content(node) }
19
+
20
+ # Restore the blockquote state
21
+ state.instance_variable_set(:@in_blockquote, old_in_blockquote)
14
22
  },
15
23
 
16
24
  code_block: ->(state, node, parent = nil, index = nil) {
@@ -38,27 +46,86 @@ module ProseMirror
38
46
  },
39
47
 
40
48
  bullet_list: ->(state, node, parent = nil, index = nil) {
41
- state.render_list(node, " ", ->(_) { (node.attrs[:bullet] || "*") + " " })
49
+ # Special handling for lists inside blockquotes
50
+ if state.instance_variable_get(:@in_blockquote)
51
+ # For lists in blockquotes, add the blockquote prefix to each line
52
+ # Each list item should start with "> * "
53
+ state.render_list(node, "", ->(_) { "* " }, true)
54
+ else
55
+ # Standard list rendering
56
+ state.render_list(node, " ", ->(_) { "* " })
57
+ end
42
58
  },
43
59
 
44
60
  ordered_list: ->(state, node, parent = nil, index = nil) {
45
61
  start = node.attrs[:order] || 1
46
- max_w = (start + node.child_count - 1).to_s.length
47
- space = state.repeat(" ", max_w + 2)
48
62
 
49
- state.render_list(node, space, ->(i) {
50
- n_str = (start + i).to_s
51
- state.repeat(" ", max_w - n_str.length) + n_str + ". "
52
- })
63
+ # Special handling for ordered lists inside blockquotes
64
+ if state.instance_variable_get(:@in_blockquote)
65
+ # For lists in blockquotes, add the blockquote prefix to each line
66
+ # Each list item should start with "> 1. " etc.
67
+ state.render_list(node, "", ->(i) {
68
+ "#{start + i}. "
69
+ }, true)
70
+ else
71
+ # Standard ordered list rendering
72
+ state.render_list(node, " ", ->(i) {
73
+ "#{start + i}. "
74
+ })
75
+ end
53
76
  },
54
77
 
55
78
  list_item: ->(state, node, parent = nil, index = nil) {
56
- state.render_content(node)
79
+ # Track that we're processing a list item to handle nested lists
80
+ old_in_list_item = state.instance_variable_get(:@in_list_item) || false
81
+ state.instance_variable_set(:@in_list_item, true)
82
+
83
+ # Special handling for list items in blockquotes
84
+ if state.instance_variable_get(:@in_blockquote)
85
+ # Process the list item content with special handling
86
+ node.each_with_index do |child, i|
87
+ if child.type.name == "paragraph"
88
+ # Render paragraph content directly
89
+ state.render_inline(child)
90
+ else
91
+ # Render other content normally
92
+ state.render(child, node, i)
93
+ end
94
+ end
95
+ else
96
+ # Process the list item content normally
97
+ state.render_content(node)
98
+ end
99
+
100
+ # Restore the previous state
101
+ state.instance_variable_set(:@in_list_item, old_in_list_item)
57
102
  },
58
103
 
59
104
  paragraph: ->(state, node, parent = nil, index = nil) {
60
- state.render_inline(node)
61
- state.close_block(node)
105
+ # Special handling for paragraphs inside list items to avoid extra whitespace
106
+ if state.instance_variable_get(:@in_list_item)
107
+ # Create a clean paragraph renderer for list items
108
+ old_at_block_start = state.instance_variable_get(:@at_block_start)
109
+
110
+ # Render the paragraph content directly with minimal whitespace
111
+ state.render_inline(node)
112
+
113
+ # Restore state
114
+ state.instance_variable_set(:@at_block_start, old_at_block_start)
115
+ elsif state.instance_variable_get(:@in_blockquote) && parent&.type&.name == "bullet_list"
116
+ # Special handling for paragraphs in bullet lists inside blockquotes
117
+ old_at_block_start = state.instance_variable_get(:@at_block_start)
118
+
119
+ # Render with blockquote prefix
120
+ state.render_inline(node)
121
+
122
+ # Restore state
123
+ state.instance_variable_set(:@at_block_start, old_at_block_start)
124
+ else
125
+ # Normal paragraph rendering for non-list items
126
+ state.render_inline(node)
127
+ state.close_block(node)
128
+ end
62
129
  },
63
130
 
64
131
  image: ->(state, node, parent = nil, index = nil) {
@@ -79,6 +146,62 @@ module ProseMirror
79
146
 
80
147
  text: ->(state, node, parent = nil, index = nil) {
81
148
  state.text(node.text, !state.in_autolink)
149
+ },
150
+
151
+ # Table serialization support
152
+ table: ->(state, node, parent = nil, index = nil) {
153
+ # Track that we're in a table
154
+ old_in_table = state.instance_variable_get(:@in_table) || false
155
+ state.instance_variable_set(:@in_table, true)
156
+
157
+ # Render table content
158
+ state.render_content(node)
159
+
160
+ # Restore table state
161
+ state.instance_variable_set(:@in_table, old_in_table)
162
+
163
+ # Only add newline if not at the end of the document
164
+ if parent && index < parent.child_count - 1
165
+ state.close_block(node)
166
+ end
167
+ },
168
+
169
+ table_row: ->(state, node, parent = nil, index = nil) {
170
+ # Write row separator after headers
171
+ if index == 1 && parent && parent.child(0).content.any? { |cell| cell.type.name == "table_header" }
172
+ state.write("|")
173
+ node.content.each do |_|
174
+ state.write(" --- |")
175
+ end
176
+ state.write("\n")
177
+ end
178
+
179
+ # Write row content
180
+ state.write("|")
181
+ state.render_content(node)
182
+
183
+ # Add newline unless this is the last row
184
+ if parent && index < parent.child_count - 1
185
+ state.write("\n")
186
+ end
187
+ },
188
+
189
+ table_header: ->(state, node, parent = nil, index = nil) {
190
+ state.write(" ")
191
+ # Render content with marks
192
+ node.content.each do |cell_content|
193
+ state.render_inline(cell_content)
194
+ end
195
+ state.write(" |")
196
+ },
197
+
198
+ table_cell: ->(state, node, parent = nil, index = nil) {
199
+ state.write(" ")
200
+ # Render content with marks
201
+ node.content.each do |cell_content|
202
+ state.render_inline(cell_content)
203
+ end
204
+ state.write(" |")
82
205
  }
83
206
  }
84
207
 
@@ -174,7 +297,7 @@ module ProseMirror
174
297
 
175
298
  # This class is used to track state and expose methods related to markdown serialization
176
299
  class MarkdownSerializerState
177
- attr_accessor :delim, :out, :closed, :in_autolink, :at_block_start, :in_tight_list
300
+ attr_accessor :delim, :out, :closed, :in_autolink, :at_block_start, :in_tight_list, :in_blockquote
178
301
  attr_reader :nodes, :marks, :options
179
302
 
180
303
  # Initialize a new state object
@@ -188,6 +311,7 @@ module ProseMirror
188
311
  @in_autolink = nil
189
312
  @at_block_start = false
190
313
  @in_tight_list = false
314
+ @in_blockquote = false
191
315
 
192
316
  @options[:tight_lists] = false if @options[:tight_lists].nil?
193
317
  @options[:hard_break_node_name] = "hard_break" if @options[:hard_break_node_name].nil?
@@ -428,23 +552,62 @@ module ProseMirror
428
552
  end
429
553
 
430
554
  # Render a list
431
- def render_list(node, delim, first_delim)
555
+ def render_list(node, indent, marker_gen, in_blockquote = false)
556
+ # Clear any closed block
432
557
  if @closed && @closed.type == node.type
433
558
  @closed = nil
434
559
  else
435
560
  ensure_new_line
436
561
  end
437
562
 
438
- starting = @delim
563
+ # Remember current indentation level
564
+ old_indent = @delim
565
+ is_nested = !old_indent.empty?
566
+
567
+ # Set up tight list handling
568
+ old_tight = @in_tight_list
569
+ @in_tight_list = node.attrs[:tight]
439
570
 
571
+ # Process each list item
440
572
  node.each_with_index do |child, i|
441
- old_tight = @in_tight_list
442
- @in_tight_list = node.attrs[:tight]
443
- @delim = starting + first_delim.call(i)
573
+ # For blockquote lists, add the blockquote prefix
574
+ @delim = if in_blockquote
575
+ "> "
576
+ # For nested lists, add standard indentation
577
+ elsif is_nested
578
+ " "
579
+ else
580
+ ""
581
+ end
582
+
583
+ # Write the marker with proper spacing
584
+ write(marker_gen.call(i))
585
+
586
+ # Store the current position after the marker
587
+ old_delim = @delim
588
+
589
+ # Add exactly one space after the marker for content
590
+ @delim = old_delim + " "
591
+
592
+ # Render the item's content
444
593
  render(child, node, i)
445
- @in_tight_list = old_tight
594
+
595
+ # Restore delimiter for next item
596
+ @delim = old_delim
597
+
598
+ # Add a newline after each list item unless it's the last one
599
+ unless i == node.child_count - 1
600
+ ensure_new_line
601
+ end
446
602
  end
447
603
 
604
+ # Restore the tight list setting
605
+ @in_tight_list = old_tight
606
+
607
+ # Restore the original indentation
608
+ @delim = old_indent
609
+
610
+ # Handle spacing between list items and the next content
448
611
  size = (@closed && @closed.type.name == "paragraph" && !node.attrs[:tight]) ? 2 : 1
449
612
  flush_close(size)
450
613
  end
@@ -1,3 +1,3 @@
1
1
  module ProseMirror
2
- VERSION = "0.1.0"
2
+ VERSION = "0.1.2"
3
3
  end
data/lib/prose_mirror.rb CHANGED
@@ -1,5 +1,6 @@
1
1
  require "json"
2
2
  require "ostruct"
3
+ require "active_support/core_ext/string/inflections" # For underscore
3
4
 
4
5
  # Require all ProseMirror components
5
6
  require_relative "prose_mirror/version"
metadata CHANGED
@@ -1,7 +1,7 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: prose_mirror_ruby
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.1.0
4
+ version: 0.1.2
5
5
  platform: ruby
6
6
  authors:
7
7
  - Robert Ross
@@ -10,6 +10,20 @@ bindir: bin
10
10
  cert_chain: []
11
11
  date: 2025-03-07 00:00:00.000000000 Z
12
12
  dependencies:
13
+ - !ruby/object:Gem::Dependency
14
+ name: activesupport
15
+ requirement: !ruby/object:Gem::Requirement
16
+ requirements:
17
+ - - ">="
18
+ - !ruby/object:Gem::Version
19
+ version: '5.0'
20
+ type: :runtime
21
+ prerelease: false
22
+ version_requirements: !ruby/object:Gem::Requirement
23
+ requirements:
24
+ - - ">="
25
+ - !ruby/object:Gem::Version
26
+ version: '5.0'
13
27
  - !ruby/object:Gem::Dependency
14
28
  name: rake
15
29
  requirement: !ruby/object:Gem::Requirement