mirror of
				https://github.com/lingble/chatwoot.git
				synced 2025-10-31 11:08:04 +00:00 
			
		
		
		
	![dependabot[bot]](/assets/img/avatar_default.png) c76b588850
			
		
	
	c76b588850
	
	
	
		
			
			* chore(deps): bump marked from 2.0.3 to 4.0.10 Bumps [marked](https://github.com/markedjs/marked) from 2.0.3 to 4.0.10. - [Release notes](https://github.com/markedjs/marked/releases) - [Changelog](https://github.com/markedjs/marked/blob/master/.releaserc.json) - [Commits](https://github.com/markedjs/marked/compare/v2.0.3...v4.0.10) --- updated-dependencies: - dependency-name: marked dependency-type: direct:production ... Signed-off-by: dependabot[bot] <support@github.com> * use {marked} Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> Co-authored-by: Pranav Raj S <pranav@chatwoot.com>
		
			
				
	
	
		
			67 lines
		
	
	
		
			2.0 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			67 lines
		
	
	
		
			2.0 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| import { marked } from 'marked';
 | |
| import DOMPurify from 'dompurify';
 | |
| import { escapeHtml } from './HTMLSanitizer';
 | |
| 
 | |
| const TWITTER_USERNAME_REGEX = /(^|[^@\w])@(\w{1,15})\b/g;
 | |
| const TWITTER_USERNAME_REPLACEMENT =
 | |
|   '$1<a href="http://twitter.com/$2" target="_blank" rel="noreferrer nofollow noopener">@$2</a>';
 | |
| 
 | |
| const TWITTER_HASH_REGEX = /(^|\s)#(\w+)/g;
 | |
| const TWITTER_HASH_REPLACEMENT =
 | |
|   '$1<a href="https://twitter.com/hashtag/$2" target="_blank" rel="noreferrer nofollow noopener">#$2</a>';
 | |
| 
 | |
| const USER_MENTIONS_REGEX = /mention:\/\/(user|team)\/(\d+)\/(.+)/gm;
 | |
| 
 | |
| class MessageFormatter {
 | |
|   constructor(message, isATweet = false) {
 | |
|     this.message = DOMPurify.sanitize(escapeHtml(message || ''));
 | |
|     this.isATweet = isATweet;
 | |
|     this.marked = marked;
 | |
| 
 | |
|     const renderer = {
 | |
|       heading(text) {
 | |
|         return `<strong>${text}</strong>`;
 | |
|       },
 | |
|       link(url, title, text) {
 | |
|         const mentionRegex = new RegExp(USER_MENTIONS_REGEX);
 | |
|         if (url.match(mentionRegex)) {
 | |
|           return `<span class="prosemirror-mention-node">${text}</span>`;
 | |
|         }
 | |
|         return `<a rel="noreferrer noopener nofollow" href="${url}" class="link" title="${title ||
 | |
|           ''}" target="_blank">${text}</a>`;
 | |
|       },
 | |
|     };
 | |
|     this.marked.use({ renderer });
 | |
|   }
 | |
| 
 | |
|   formatMessage() {
 | |
|     if (this.isATweet) {
 | |
|       const withUserName = this.message.replace(
 | |
|         TWITTER_USERNAME_REGEX,
 | |
|         TWITTER_USERNAME_REPLACEMENT
 | |
|       );
 | |
|       const withHash = withUserName.replace(
 | |
|         TWITTER_HASH_REGEX,
 | |
|         TWITTER_HASH_REPLACEMENT
 | |
|       );
 | |
|       const markedDownOutput = marked(withHash);
 | |
|       return markedDownOutput;
 | |
|     }
 | |
|     return marked(this.message, { breaks: true, gfm: true });
 | |
|   }
 | |
| 
 | |
|   get formattedMessage() {
 | |
|     return this.formatMessage();
 | |
|   }
 | |
| 
 | |
|   get plainText() {
 | |
|     const strippedOutHtml = new DOMParser().parseFromString(
 | |
|       this.formattedMessage,
 | |
|       'text/html'
 | |
|     );
 | |
|     return strippedOutHtml.body.textContent || '';
 | |
|   }
 | |
| }
 | |
| 
 | |
| export default MessageFormatter;
 |