Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/fsegurai/ngx-markdown
Angular markdown component/directive/pipe/service to parse static, dynamic or remote content to HTML with syntax highlight and more...
https://github.com/fsegurai/ngx-markdown
angular clipboardjs emoji katex markdown markedjs mermaidjs ngx-markdown npm prismjs typescript-library
Last synced: about 3 hours ago
JSON representation
Angular markdown component/directive/pipe/service to parse static, dynamic or remote content to HTML with syntax highlight and more...
- Host: GitHub
- URL: https://github.com/fsegurai/ngx-markdown
- Owner: fsegurai
- License: mit
- Created: 2024-06-05T18:05:14.000Z (7 months ago)
- Default Branch: main
- Last Pushed: 2024-12-29T07:55:08.000Z (18 days ago)
- Last Synced: 2024-12-29T08:20:34.598Z (18 days ago)
- Topics: angular, clipboardjs, emoji, katex, markdown, markedjs, mermaidjs, ngx-markdown, npm, prismjs, typescript-library
- Language: TypeScript
- Homepage: https://fsegurai.github.io/ngx-markdown/
- Size: 1.54 MB
- Stars: 1
- Watchers: 2
- Forks: 0
- Open Issues: 5
-
Metadata Files:
- Readme: README.md
- License: LICENSE
Awesome Lists containing this project
README
**This is just a side project to provide additional features that fulfill my needs.**
`@fsegurai/ngx-markdown` is an [Angular](https://angular.dev/) library that combines...
- [Marked](http://marked.js.org/) to parse markdown to HTML
- [Prism.js](http://prismjs.com/) for language syntax highlight
- [Emoji-Toolkit](https://github.com/joypixels/emoji-toolkit) for emoji support
- [KaTeX](https://katex.org/) for math expression rendering
- [Mermaid](https://mermaid-js.github.io/) for diagrams and charts visualization
- [Clipboard.js](https://clipboardjs.com/) to copy code blocks to clipboard### Table of contents
- [Installation](#installation)
- [Configuration](#configuration)
- [Usage](#usage)
- [Renderer](#renderer)
- [Syntax highlight](#syntax-highlight)
- [Demo application](#demo-application)
- [License](#license)## Installation
### @fsegurai/ngx-markdown
To add `@fsegurai/ngx-markdown` along with the required marked library to your `package.json` use the following commands.
```bash
npm install @fsegurai/ngx-markdown marked@^15.0.3 --save
```### Syntax highlight
> :bell: Syntax highlight is **optional**, skip this step if you are not planning to use it
To add [Prism.js](http://prismjs.com/) library to your `package.json` use the following command.
```bash
npm install prismjs@^1.29.0 --save
```To activate [Prism.js](http://prismjs.com/) syntax highlight you will need to include...
- prism.js core library - `node_modules/prismjs/prism.js` file
- a highlight css theme - from `node_modules/prismjs/themes` directory
- desired code language syntax files - from `node_modules/prismjs/components` directory_Additional themes can be found by browsing the web such as [Prism-Themes](https://github.com/PrismJS/prism-themes) or [Mokokai](https://github.com/Ahrengot/Monokai-theme-for-Prism.js) for example._
If you are using [Angular CLI](https://cli.angular.dev/) you can follow the `angular.json` example below...
```diff
"styles": [
"styles.css",
+ "node_modules/prismjs/themes/prism-okaidia.css"
],
"scripts": [
+ "node_modules/prismjs/prism.js",
+ "node_modules/prismjs/components/prism-csharp.min.js", # c-sharp language syntax
+ "node_modules/prismjs/components/prism-css.min.js" # css language syntax
]
```#### Line Numbers plugin
To use the [line numbers plugin](http://prismjs.com/plugins/line-numbers/) that shows line numbers in code blocks, in addition to Prism.js configuration files, you will need to include the following files from `prismjs/plugins/line-numbers` directory to your application:
- CSS styling for line numbers - `prism-line-numbers.css`
- line numbers plugin script - `prism-line-numbers.js`If you are using [Angular CLI](https://cli.angular.dev/) you can follow the `angular.json` example below...
```diff
"styles": [
"src/styles.css",
"node_modules/prismjs/themes/prism-okaidia.css",
+ "node_modules/prismjs/plugins/line-numbers/prism-line-numbers.css"
],
"scripts": [
"node_modules/prismjs/prism.js",
"node_modules/prismjs/components/prism-csharp.min.js",
"node_modules/prismjs/components/prism-css.min.js",
+ "node_modules/prismjs/plugins/line-numbers/prism-line-numbers.js"
]
```Using `markdown` component and/or directive, you will be able to use the `lineNumbers` property to activate the plugin. The property can be used in combination with either `data` for variable binding, `src` for remote content or using transclusion for static markdown.
Additionally, you can use `start` input property to specify the offset number for the first display line.
```html
```
#### Line Highlight plugin
To use the [line highlight plugin](http://prismjs.com/plugins/line-highlight/) that highlights specific lines and/or line ranges in code blocks, in addition to Prism.js configuration files, you will need to include the following files from `prismjs/plugins/line-highlight` directory to your application:
- CSS styling for line highlight - `prism-line-highlight.css`
- line highlight plugin script - `prism-line-highlight.js`If you are using [Angular CLI](https://cli.angular.dev/) you can follow the `angular.json` example below...
```diff
"styles": [
"src/styles.css",
"node_modules/prismjs/themes/prism-okaidia.css",
+ "node_modules/prismjs/plugins/line-highlight/prism-line-highlight.css"
],
"scripts": [
"node_modules/prismjs/prism.js",
"node_modules/prismjs/components/prism-csharp.min.js",
"node_modules/prismjs/components/prism-css.min.js",
+ "node_modules/prismjs/plugins/line-highlight/prism-line-highlight.js"
]
```Using `markdown` component and/or directive, you will be able to use the `lineHighlight` property to activate the plugin. The property can be used in combination with either `data` for variable binding, `src` for remote content or using transclusion for static markdown.
Use `line` input property to specify the line(s) to highlight and optionally there is a `lineOffset` property to specify the starting line of code your snippet represents.
```html
```
#### Command Line Plugin
To use the [command line plugin](https://prismjs.com/plugins/command-line/) that displays a command line with a prompt and, optionally, the output/response from the commands, you will need to include the following files from `prismjs/plugins/command-line` directory to your application:
- CSS styling for command line - `prism-command-line.css`
- command line plugin script - `prism-command-line.js`If you are using [Angular CLI](https://cli.angular.dev/) you can follow the `angular.json` example below...
```diff
"styles": [
"src/styles.css",
"node_modules/prismjs/themes/prism-okaidia.css",
+ "node_modules/prismjs/plugins/command-line/prism-command-line.css"
],
"scripts": [
"node_modules/prismjs/prism.js",
"node_modules/prismjs/components/prism-csharp.min.js",
"node_modules/prismjs/components/prism-css.min.js",
+ "node_modules/prismjs/plugins/command-line/prism-command-line.js"
]
```Using `markdown` component and/or directive, you will be able to use the `commandLine` property to activate the plugin. The property can be used in combination with either `data` for variable binding, `src` for remote content or using transclusion for static markdown.
For a server command line, specify the user and host names using the `user` and `host` input properties. The resulting prompt displays a `#` for the root user and `$` for all other users. For any other command line, such as a Windows prompt, you may specify the entire prompt using the `prompt` input property.
You may also specify the lines to be presented as output (no prompt and no highlighting) through the `output` property in the following simple format:
- A single number refers to the line with that number
- Ranges are denoted by two numbers, separated with a hyphen (-)
- Multiple line numbers or ranges are separated by commas
- Whitespace is allowed anywhere and will be stripped off```html
```
Optionally, to automatically present some lines as output without providing the line numbers, you can prefix those lines with any string and specify the prefix using the `filterOutput` input property. For example, `[filterOutput]="'(out)'"` will treat lines beginning with `(out)` as output and remove the prefix.
```html
'"
[filterOutput]="'(out)'">
```powershell
Get-Date
(out)
(out)Sunday, November 7, 2021 8:19:21 PM
(out)
``````
### Emoji support
> :bell: Emoji support is **optional**, skip this step if you are not planning to use it
To add [Emoji-Toolkit](https://github.com/joypixels/emoji-toolkit) library to your `package.json` use the following command.
```bash
npm install emoji-toolkit@^9.0.1 --save
```To activate [Emoji-Toolkit](https://github.com/joypixels/emoji-toolkit) for emoji suppport you will need to include...
- Emoji-Toolkit library - `node_modules/emoji-toolkit/lib/js/joypixels.min.js`
If you are using [Angular CLI](https://cli.angular.dev/) you can follow the `angular.json` example below...
```diff
"scripts": [
+ "node_modules/emoji-toolkit/lib/js/joypixels.min.js",
]
```#### Emoji plugin
Using `markdown` component and/or directive, you will be able to use the `emoji` property to activate [Emoji-Toolkit](https://github.com/joypixels/emoji-toolkit) plugin that converts emoji shortnames such as `:heart:` to native unicode emojis.
```html
I :heart: @fsegurai/ngx-markdown
```
> :blue_book: You can refer to this [Emoji Cheat Sheet](https://github.com/ikatyang/emoji-cheat-sheet/blob/master/README.md) for a complete list of _shortnames_.
### Math rendering
> :bell: Math rendering is **optional**, skip this step if you are not planning to use it
To add [KaTeX](https://katex.org/) library to your `package.json` use the following command.
```bash
npm install katex@^0.16.11 --save
```To activate [KaTeX](https://katex.org/) math rendering you will need to include...
- KaTex JavaScript library - `node_modules/katex/dist/katex.min.js` file
- KaTex Auto-Render extension - `node_modules/katex/dist/contrib/auto-render.min.js,` file
- KaTex CSS customization - `node_modules/katex/dist/katex.min.css` fileIf you are using [Angular CLI](https://cli.angular.dev/) you can follow the `angular.json` example below...
```diff
"styles": [
"styles.css",
+ "node_modules/katex/dist/katex.min.css"
],
"scripts": [
+ "node_modules/katex/dist/katex.min.js",
+ "node_modules/katex/dist/contrib/auto-render.min.js",
]
```#### KaTeX plugin
Using `markdown` component and/or directive, you will be able to use the `katex` property to activate [KaTeX](https://katex.org/) plugin that renders mathematical expression to HTML.
```html
```
Optionally, you can use `katexOptions` property to specify both the [KaTeX options](https://katex.org/docs/options.html) and the [KaTeX Auto-Render options](https://katex.org/docs/autorender.html#api).
```typescript
import {KatexOptions} from '@fsegurai/ngx-markdown';public options: KatexOptions = {
displayMode: true,
throwOnError: false,
errorColor: '#cc0000',
delimiters: [...],
...
};
``````html
```
> :blue_book: Follow official [KaTeX options](https://katex.org/docs/options.html) and [KaTeX Auto-Render options](https://katex.org/docs/autorender.html#api) documentation for more details on the available options.
### Diagrams tool
> :bell: Diagram support is **optional**, skip this step if you are not planning to use it
To add [Mermaid](https://mermaid-js.github.io/) library to your `package.json` use the following command.
```bash
npm install mermaid@^11.4.1 --save
```To activate [Mermaid](https://mermaid-js.github.io/) diagramming and charting tool you will need to include...
- Mermaid JavaScript library - `node_modules/mermaid/dist/mermaid.min.js` file
If you are using [Angular CLI](https://cli.angular.dev/) you can follow the `angular.json` example below...
```diff
"scripts": [
+ "node_modules/mermaid/dist/mermaid.min.js",
]
```#### Mermaid plugin
Using `markdown` component and/or directive, you will be able to use the `mermaid` property to activate [Mermaid](https://mermaid-js.github.io/) plugin that renders Markdown-inspired text definitions to create and modify diagrams dynamically.
```html
```
#### Global configuration
You can provide a global configuration for mermaid [configuration options](https://mermaid.js.org/config/schema-docs/config.html#mermaid-config-properties) to use across your application with the `mermaidOptions` in the `MarkdownModuleConfig` either with `provideMarkdown` provide-function for standalone components or `MarkdownModule.forRoot()` for module configuration.
##### Using the `provideMarkdown` function
```typescript
provideMarkdown({
mermaidOptions: {
provide: MERMAID_OPTIONS,
useValue: {
darkMode: true,
look: 'handDrawn',
...
},
},
}),
```
##### Using the `MarkdownModule` import
```typescript
MarkdownModule.forRoot({
mermaidOptions: {
provide: MERMAID_OPTIONS,
useValue: {
darkMode: true,
look: 'handDrawn',
...
},
},
}),
```
#### Component configuration
Additionally, you can specify mermaid [configuration options](https://mermaid.js.org/config/schema-docs/config.html#mermaid-config-properties) on component directly using `mermaidOptions` property.```typescript
import {MermaidAPI} from '@fsegurai/ngx-markdown';public options: MermaidAPI.MermaidConfig = {
darkMode: true,
look: 'handDrawn',
...
};
``````html
```
> :blue_book: Follow official [Mermaid](https://mermaid-js.github.io/) documentation for more details on diagrams and charts syntax.
### Copy-to-clipboard
> :bell: Copy-to-clipboard support is **optional**, skip this step if you are not planning to use it
To add [Clipboard](https://clipboardjs.com/) library to your `package.json` use the following command.
```bash
npm install clipboard@^2.0.11 --save
```To activate [Clipboard](https://clipboardjs.com/) allowing copy-to-clipboard you will need to include...
- Clipboard JavaScript library - `node_modules/clipboard/dist/clipboard.min.js` file
If you are using [Angular CLI](https://cli.angular.dev/) you can follow the `angular.json` example below...
```diff
"scripts": [
+ "node_modules/clipboard/dist/clipboard.min.js",
]
```#### Clipboard plugin
Using `markdown` component and/or directive, you will be able to use the `clipboard` property to activate [Clipboard](https://clipboardjs.com/) plugin that enable copy-to-clipboard for code block from a single click.
```html
```
#### Default button
The `clipboard` plugin provide an unstyled default button with a default behavior out of the box if no alternative is used.
#### Customize button toolbar
The clipboard button is placed inside a wrapper element that can be customized using the `.markdown-clipboard-toolbar` CSS selector in your global `styles.css/scss` file.
This allows to override the default positioning of the clipboard button and play with the visibility of the button using the `.hover` CSS selector that is applied on the toolbar when the mouse cursor enters and leaves the code block element.
#### Customize default button
To customize the default button styling, use the `.markdown-clipboard-button` CSS selector in your global `styles.css/scss` file. You can also customize the "copied" state happening after the button is clicked using the `.copied` CSS selector.
#### Using global configuration
You can provide a custom component to use globaly across your application with the `clipboardOptions` in the `MarkdownModuleConfig` either with `provideMarkdown` provide-function for standalone components or `MarkdownModule.forRoot()` for module configuration.
##### Using the `provideMarkdown` function
```typescript
provideMarkdown({
clipboardOptions: {
provide: CLIPBOARD_OPTIONS,
useValue: {
buttonComponent: ClipboardButtonComponent,
},
},
})
```##### Using the `MarkdownModule` import
```typescript
MarkdownModule.forRoot({
clipboardOptions: {
provide: CLIPBOARD_OPTIONS,
useValue: {
buttonComponent: ClipboardButtonComponent,
},
},
}),
```#### Using a component
You can also provide your custom component using the `clipboardButtonComponent` input property when using the `clipboard` directive.
```typescript
import {Component} from '@angular/core';@Component({
selector: 'app-clipboard-button',
template: `Copy`,
})
export class ClipboardButtonComponent {
onClick() {
alert('Copied to clipboard!');
}
}
``````typescript
import {ClipboardButtonComponent} from './clipboard-button-component';@Component({...})
export class ExampleComponent {
readonly clipboardButton = ClipboardButtonComponent;
}
``````html
```
#### Using ng-template
Alternatively, the `clipboard` directive can be used in conjonction with `ng-template` to provide a custom button implementation via the `clipboardButtonTemplate` input property on the `markdown` component.
```html
...
```
> :blue_book: Refer to the `@fsegurai/ngx-markdown` [clipboard plugin demo](https://fsegurai.github.io/ngx-markdown/plugins#clipboard) for live examples.
## Configuration
The `@fsegurai/ngx-markdown` library can be used either with the standalone components or with modules configuration. Please follow the configuration section that matches your application.
### Standalone components
Use the `provideMarkdown` provide-function in your application configuration `ApplicationConfig` to be able to provide the `MarkdownComponent` and `MarkdownPipe` to your standalone components and/or inject the `MarkdownService`.
```diff
import { NgModule } from '@angular/core';
+ import { provideMarkdown } from '@fsegurai/ngx-markdown';export const appConfig: ApplicationConfig = {
providers: [
+ provideMarkdown(),
],
};
```### Modules configuration
You must import `MarkdownModule` inside your main application module (usually named AppModule) with `forRoot` to be able to use the `markdown` component, directive, pipe and/or `MarkdownService`.
```diff
import { NgModule } from '@angular/core';
+ import { MarkdownModule } from '@fsegurai/ngx-markdown';
import { AppComponent } from './app.component';@NgModule({
imports: [
+ MarkdownModule.forRoot(),
],
declarations: [AppComponent],
bootstrap: [AppComponent],
})
export class AppModule { }
```Use `forChild` when importing `MarkdownModule` into other application modules to allow you to use the same parser configuration across your application.
```diff
import { NgModule } from '@angular/core';
+ import { MarkdownModule } from '@fsegurai/ngx-markdown';
import { HomeComponent } from './home.component';@NgModule({
imports: [
+ MarkdownModule.forChild(),
],
declarations: [HomeComponent],
})
export class HomeModule { }
```### Remote file configuration
If you want to use the `[src]` attribute to directly load a remote file, in order to keep only one instance of `HttpClient` and avoid issues with interceptors, you also have to provide `HttpClient`:
##### Using the `provideMarkdown` function
```diff
providers: [
+ provideHttpClient(),
+ provideMarkdown({ loader: HttpClient }),
],
```##### Using the `MarkdownModule` import
```diff
imports: [
+ HttpClientModule,
+ MarkdownModule.forRoot({ loader: HttpClient }),
],
```#### Sanitization
As of `@fsegurai/[email protected]` **sanitization is enabled by default** and uses Angular `DomSanitizer` with `SecurityContext.HTML` to avoid XSS vulnerabilities. The `SecurityContext` level can be changed using the `sanitize` property when configuring `MarkdownModule`.
##### Using the `provideMarkdown` function
```typescript
import {SecurityContext} from '@angular/core';// enable default sanitization
provideMarkdown()// turn off sanitization
provideMarkdown({
sanitize: SecurityContext.NONE
})
```##### Using the `MarkdownModule` import
```typescript
import {SecurityContext} from '@angular/core';// enable default sanitization
MarkdownModule.forRoot()// turn off sanitization
MarkdownModule.forRoot({
sanitize: SecurityContext.NONE
})
```> :blue_book: Follow [Angular DomSanitizer](https://angular.io/api/platform-browser/DomSanitizer#sanitize) documentation for more information on sanitization and security contexts.
You can bypass sanitization using the markdown component, directive or pipe using the `disableSanitizer` option as follow:
```html
```#### MarkedOptions
Optionally, markdown parsing can be configured using [MarkedOptions](https://marked.js.org/#/USING_ADVANCED.md#options) that can be provided with the `MARKED_OPTIONS` injection token via the `markedOptions` property of the `forRoot` method of `MarkdownModule`.
##### Using the `provideMarkdown` function
```typescript
// imports
import {MARKED_OPTIONS, provideMarkdown} from '@fsegurai/ngx-markdown';// using default options
provideMarkdown(),// using specific options with ValueProvider and passing HttpClient
provideMarkdown({
markedOptions: {
provide: MARKED_OPTIONS,
useValue: {
gfm: true,
breaks: false,
pedantic: false,
},
},
}),
```##### Using the `MarkdownModule` import
```typescript
// imports
import {MarkdownModule, MARKED_OPTIONS} from '@fsegurai/ngx-markdown';// using default options
MarkdownModule.forRoot(),// using specific options with ValueProvider and passing HttpClient
MarkdownModule.forRoot({
loader: HttpClient, // optional, only if you use [src] attribute
markedOptions: {
provide: MARKED_OPTIONS,
useValue: {
gfm: true,
breaks: false,
pedantic: false,
},
},
}),
```#### MarkedOptions.renderer
`MarkedOptions` also exposes the `renderer` property which allows you to override token rendering for your whole application.
The example uses a factory function and override the default blockquote token rendering by adding a CSS class for custom styling when using Bootstrap CSS:
```typescript
import {MARKED_OPTIONS, MarkedOptions, MarkedRenderer} from '@fsegurai/ngx-markdown';// function that returns `MarkedOptions` with renderer override
export function markedOptionsFactory(): MarkedOptions {
const renderer = new MarkedRenderer();renderer.blockquote = (text: string) => {
return '';' + text + '
};return {
renderer: renderer,
gfm: true,
breaks: false,
pedantic: false,
};
}
```##### Using the `provideMarkdown` function
```typescript
// using specific option with FactoryProvider
provideMarkdown({
markedOptions: {
provide: MARKED_OPTIONS,
useFactory: markedOptionsFactory,
},
}),
```##### Using the `MarkdownModule` import
```typescript
// using specific option with FactoryProvider
MarkdownModule.forRoot({
markedOptions: {
provide: MARKED_OPTIONS,
useFactory: markedOptionsFactory,
},
}),
```### Marked extensions
You can provide [marked extensions](https://marked.js.org/using_advanced#extensions) using the `markedExtensions` property that accepts an array of extensions when configuring `MarkdownModule`.
##### Using the `provideMarkdown` function
```typescript
import {gfmHeadingId} from 'marked-gfm-heading-id';providemarkdown({
markedExtensions: [gfmHeadingId()],
}),
```##### Using the `MarkdownModule` import
```typescript
import {gfmHeadingId} from 'marked-gfm-heading-id';MarkdownModule.forRoot({
markedExtensions: [gfmHeadingId()],
}),
```## Usage
`@fsegurai/ngx-markdown` provides different approaches to help you parse markdown to your application depending on your needs.
> :bulb: As of Angular 6, the template compiler strips whitespace by default. Use `ngPreserveWhitespaces` directive to preserve whitespaces such as newlines in order for the markdown-formatted content to render as intended.
https://angular.io/api/core/Component#preserveWhitespaces### Component
You can use `markdown` component to either parse static markdown directly from your HTML markup, load the content from a remote URL using `src` property or bind a variable to your component using `data` property. You can get a hook on load complete using `load` output event property, on loading error using `error` output event property or when parsing is completed using `ready` output event property.
```html
# Markdown
```
### Directive
The same way the component works, you can use `markdown` directive to accomplish the same thing.
```html
# Markdown
```### Pipe
Using `markdown` pipe to transform markdown to HTML allow you to chain pipe transformations and will update the DOM when value changes. It is important to note that, because the `marked` parsing method returns a `Promise`, it requires the use of the `async` pipe.
```html
```The `markdown` pipe allow you to use all the same plugins as the component by providing the options parameters.
```html
```This is the `MarkdownPipeOptions` parameters interface, those options are the same as the ones available for the `markdown` component:
```typescript
export interface MarkdownPipeOptions {
decodeHtml?: boolean;
inline?: boolean;
emoji?: boolean;
katex?: boolean;
katexOptions?: KatexOptions;
mermaid?: boolean;
mermaidOptions?: MermaidAPI.MermaidConfig;
markedOptions?: MarkedOptions;
disableSanitizer?: boolean;
}
```### Service
You can use `MarkdownService` to have access to markdown parsing, rendering and syntax highlight methods.
```typescript
import {Component, OnInit} from '@angular/core';
import {MarkdownService} from '@fsegurai/ngx-markdown';@Component({...})
export class ExampleComponent implements OnInit {
constructor(private markdownService: MarkdownService) {
}ngOnInit() {
// outputs:I am using markdown.
console.log(this.markdownService.parse('I am using __markdown__.'));
}
}
```## Renderer
Tokens can be rendered in a custom manner by either...
- providing the `renderer` property with the `MarkedOptions` when importing `MarkdownModule.forRoot()` into your main application module (see [Configuration](#markedoptionsrenderer) section)
- using `MarkdownService` exposed `renderer`Here is an example of overriding the default heading token rendering through `MarkdownService` by adding an embedded anchor tag like on GitHub:
```typescript
import {Component, OnInit} from '@angular/core';
import {MarkdownService} from '@fsegurai/ngx-markdown';@Component({
selector: 'app-example',
template: '# Heading',
})
export class ExampleComponent implements OnInit {
constructor(private markdownService: MarkdownService) {
}ngOnInit() {
this.markdownService.renderer.heading = (text: string, level: number) => {
const escapedText = text.toLowerCase().replace(/[^\w]+/g, '-');
return '' +
'' +
'' +
'' + text +
'';
};
}
}
```This code will output the following HTML:
```html
Heading
```> :blue_book: Follow official [marked.renderer](https://marked.js.org/#/USING_PRO.md#renderer) documentation for the list of tokens that can be overriden.
## Re-render Markdown
In some situations, you might need to re-render markdown after making changes. If you've updated the text this would be done automatically, however if the changes are internal to the library such as rendering options, you will need to inform the `MarkdownService` that it needs to update.
To do so, inject the `MarkdownService` and call the `reload()` function as shown below.
```typescript
import {MarkdownService} from '@fsegurai/ngx-markdown';constructor(private markdownService: MarkdownService){
}update(){
this.markdownService.reload();
}
```> :blue_book: Refer to the `@fsegurai/ngx-markdown` [re-render demo](https://fsegurai.github.io/ngx-markdown/rerender) for a live example.
## Syntax highlight
When using static markdown you are responsible to provide the code block with related language.
```diff
+ ```typescript
const myProp: string = 'value';
+ ```When using remote URL `@fsegurai/ngx-markdown` will use the file extension to automatically resolve the code language.
```html
```
When using variable binding you can optionally use `language` pipe to specify the language of the variable content (default value is markdown when pipe is not used).
```html
```
## Demo application
A demo is available @ [https://fsegurai.github.io/ngx-markdown](https://fsegurai.github.io/ngx-markdown) and its source code can be found inside the `demo` directory.
It's important to mention that for this project I'm using:
Node.js v20.11.1 and Bun v1.1.32 (or later).The following commands will clone the repository, install npm dependencies and serve the application @ [http://localhost:4200](http://localhost:4200)
> It is advisable to use `bun` as the package manager for managing numerous dependencies, as it is faster than `npm` and generally more reliable.
```bash
git clone https://github.com/fsegurai/ngx-markdown.git
bun install
bun start
```## License
Licensed under [MIT](https://opensource.org/licenses/MIT).