I want to use SCEditor in my Blazor page.
For example I create a new Blazor WASM project and I did these steps:
According to documentation I add this codes and references to
index.html
:- https://cdn.jsdelivr.net/npm/sceditor@3/minified/themes/default.min.css
- https://cdn.jsdelivr.net/npm/sceditor@3/minified/sceditor.min.js
- https://cdn.jsdelivr.net/npm/sceditor@3/minified/formats/bbcode.min.js
window.InitEditor = () => { var textarea = document.getElementById('myTextArea'); sceditor.create(textarea, { format: 'bbcode', style: 'https://cdn.jsdelivr.net/npm/sceditor@3/minified/themes/content/default.min.css' }); };
Add a
textarea
incounter
page:<textarea id="myTextArea" style="width:100%; height:200px;"></textarea>
add follow code to code section of
counter
page:protected override Task OnAfterRenderAsync(bool firstRender) { JsRuntime.InvokeVoidAsync("InitEditor"); return base.OnAfterRenderAsync(firstRender); }
Now I run the project and I see this editor:
then I click on fetch data menu and I see this:
Surprisingly, editor has shown in fetch data page. if I click on counter page again I see this:
There are 2 Editors O-O. and if I click on this menus then editors are increasing...
I changed the code this way:
protected override Task OnAfterRenderAsync(bool firstRender)
{
if (firstRender)
JsRuntime.InvokeVoidAsync("InitEditor");
return base.OnAfterRenderAsync(firstRender);
}
the editor is shown once in counter
page and fetch data
page. Again I don't have any textarea
in fetch data
page.
How can I solve this problem?
Blazor documentation warns:
SCEditor is exactly one of those DOM-mutating libraries, and the effects of failure to observe that guidance you can see for yourself. (The ‘security risks’ bit is rather nonsensical: if your app can be made insecure merely by modifying client-side code, then it wasn’t very secure to begin with. But it’s otherwise good advice.)
Blazor does provide some interoperability with external DOM mutation in the form of element references. The documentation again warns:
Heeding that warning, you should probably write something like below (not tested). In the component file (
.razor
):And in JavaScript:
If the SCEditor library is sufficiently well-behaved, it should only modify the DOM tree at most at the level of the parent of the textarea node you give it. You may think it would be enough to place a
<textarea>
in your component markup and capture a reference to that, but as it happens SCEditor adds siblings to that node, which may keep messing up rendering. For that reason, it is safer to put everything in an initially-empty wrapper element, which can act as a sandbox in which SCEditor has free rein.Ideally, you would encapsulate everything SCEditor-related into a dedicated component that deals only with SCEditor, while the rest of your app would use that component like any other Blazor component.