mirror of
				https://github.com/lingble/chatwoot.git
				synced 2025-10-31 02:57:57 +00:00 
			
		
		
		
	 d7e928df26
			
		
	
	d7e928df26
	
	
	
		
			
			* feat: Add loom support for helpcenter articles * fix: responsiveness * fix: style issues * fix: review comments * Update custom_markdown_renderer.rb
		
			
				
	
	
		
			122 lines
		
	
	
		
			3.3 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
			
		
		
	
	
			122 lines
		
	
	
		
			3.3 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
| class CustomMarkdownRenderer < CommonMarker::HtmlRenderer
 | |
|   YOUTUBE_REGEX = %r{https?://(?:www\.)?(?:youtube\.com/watch\?v=|youtu\.be/)([^&/]+)}
 | |
|   LOOM_REGEX = %r{https?://(?:www\.)?loom\.com/share/([^&/]+)}
 | |
|   VIMEO_REGEX = %r{https?://(?:www\.)?vimeo\.com/(\d+)}
 | |
|   MP4_REGEX = %r{https?://(?:www\.)?.+\.(mp4)}
 | |
| 
 | |
|   def text(node)
 | |
|     content = node.string_content
 | |
| 
 | |
|     if content.include?('^')
 | |
|       split_content = parse_sup(content)
 | |
|       out(split_content.join)
 | |
|     else
 | |
|       out(escape_html(content))
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   def link(node)
 | |
|     return if surrounded_by_empty_lines?(node) && render_embedded_content(node)
 | |
| 
 | |
|     # If it's not YouTube or Vimeo link, render normally
 | |
|     super
 | |
|   end
 | |
| 
 | |
|   private
 | |
| 
 | |
|   def surrounded_by_empty_lines?(node)
 | |
|     prev_node_empty?(node.previous) && next_node_empty?(node.next)
 | |
|   end
 | |
| 
 | |
|   def prev_node_empty?(prev_node)
 | |
|     prev_node.nil? || node_empty?(prev_node)
 | |
|   end
 | |
| 
 | |
|   def next_node_empty?(next_node)
 | |
|     next_node.nil? || node_empty?(next_node)
 | |
|   end
 | |
| 
 | |
|   def node_empty?(node)
 | |
|     (node.type == :text && node.string_content.strip.empty?) || (node.type != :text)
 | |
|   end
 | |
| 
 | |
|   def render_embedded_content(node)
 | |
|     link_url = node.url
 | |
|     embedding_methods = {
 | |
|       YOUTUBE_REGEX => :make_youtube_embed,
 | |
|       VIMEO_REGEX => :make_vimeo_embed,
 | |
|       MP4_REGEX => :make_video_embed,
 | |
|       LOOM_REGEX => :make_loom_embed
 | |
|     }
 | |
| 
 | |
|     embedding_methods.each do |regex, method|
 | |
|       match = link_url.match(regex)
 | |
|       if match
 | |
|         out(send(method, match))
 | |
|         return true
 | |
|       end
 | |
|     end
 | |
| 
 | |
|     false
 | |
|   end
 | |
| 
 | |
|   def parse_sup(content)
 | |
|     content.split(/(\^[^\^]+\^)/).map do |segment|
 | |
|       if segment.start_with?('^') && segment.end_with?('^')
 | |
|         "<sup>#{escape_html(segment[1..-2])}</sup>"
 | |
|       else
 | |
|         escape_html(segment)
 | |
|       end
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   def make_youtube_embed(youtube_match)
 | |
|     video_id = youtube_match[1]
 | |
|     %(
 | |
|       <div style="position: relative; padding-bottom: 62.5%; height: 0;">
 | |
|        <iframe
 | |
|         src="https://www.youtube.com/embed/#{video_id}"
 | |
|         frameborder="0"
 | |
|         style="position: absolute; top: 0; left: 0; width: 100%; height: 100%;"
 | |
|         allow="accelerometer; autoplay; clipboard-write; encrypted-media; gyroscope; picture-in-picture"
 | |
|         allowfullscreen></iframe>
 | |
|       </div>
 | |
|     )
 | |
|   end
 | |
| 
 | |
|   def make_loom_embed(loom_match)
 | |
|     video_id = loom_match[1]
 | |
|     %(
 | |
|       <div style="position: relative; padding-bottom: 62.5%; height: 0;">
 | |
|         <iframe
 | |
|          src="https://www.loom.com/embed/#{video_id}"
 | |
|          frameborder="0" webkitallowfullscreen mozallowfullscreen allowfullscreen
 | |
|          style="position: absolute; top: 0; left: 0; width: 100%; height: 100%;"></iframe>
 | |
|       </div>
 | |
|     )
 | |
|   end
 | |
| 
 | |
|   def make_vimeo_embed(vimeo_match)
 | |
|     video_id = vimeo_match[1]
 | |
|     %(
 | |
|       <div style="position: relative; padding-bottom: 62.5%; height: 0;">
 | |
|        <iframe
 | |
|         src="https://player.vimeo.com/video/#{video_id}"
 | |
|         frameborder="0"
 | |
|         allow="autoplay; fullscreen; picture-in-picture"
 | |
|         allowfullscreen
 | |
|         style="position: absolute; top: 0; left: 0; width: 100%; height: 100%;"></iframe>
 | |
|        </div>
 | |
|     )
 | |
|   end
 | |
| 
 | |
|   def make_video_embed(link_url)
 | |
|     %(
 | |
|       <video width="640" height="360" controls>
 | |
|         <source src="#{link_url}" type="video/mp4">
 | |
|         Your browser does not support the video tag.
 | |
|       </video>
 | |
|     )
 | |
|   end
 | |
| end
 |