mirror of
https://github.com/oqtane/oqtane.framework.git
synced 2025-05-25 06:14:22 +00:00
251 lines
7.4 KiB
Plaintext
251 lines
7.4 KiB
Plaintext
@using System.Text.RegularExpressions
|
|
@using Microsoft.AspNetCore.Components.Rendering
|
|
@using Microsoft.Extensions.DependencyInjection
|
|
@namespace Oqtane.Modules.Controls
|
|
@inherits ModuleControlBase
|
|
@inject IServiceProvider ServiceProvider
|
|
@inject ISettingService SettingService
|
|
@inject IStringLocalizer<RichTextEditor> Localizer
|
|
|
|
<div class="row" style="margin-bottom: 50px;">
|
|
<div class="col">
|
|
<TabStrip ActiveTab="@_activetab">
|
|
@if (AllowRichText)
|
|
{
|
|
<TabPanel Name="Rich" Heading="Rich Text Editor" ResourceKey="RichTextEditor">
|
|
@_textEditorComponent
|
|
</TabPanel>
|
|
}
|
|
@if (AllowRawHtml)
|
|
{
|
|
<TabPanel Name="Raw" Heading="Raw HTML Editor" ResourceKey="HtmlEditor">
|
|
@if (_rawfilemanager)
|
|
{
|
|
<FileManager @ref="_fileManager" Filter="@PageState.Site.ImageFiles" />
|
|
<ModuleMessage Message="@_message" Type="MessageType.Warning"></ModuleMessage>
|
|
<br />
|
|
}
|
|
<div class="d-flex justify-content-center mb-2">
|
|
@if (AllowFileManagement)
|
|
{
|
|
<button type="button" class="btn btn-primary" @onclick="InsertRawImage">@Localizer["InsertImage"]</button>
|
|
}
|
|
@if (_rawfilemanager)
|
|
{
|
|
@((MarkupString)" ")
|
|
<button type="button" class="btn btn-secondary" @onclick="CloseRawFileManager">@Localizer["Close"]</button>
|
|
}
|
|
</div>
|
|
@if (ReadOnly)
|
|
{
|
|
<textarea id="@_rawhtmlid" class="form-control" placeholder="@Placeholder" @bind="@_rawhtml" rows="10" readonly></textarea>
|
|
}
|
|
else
|
|
{
|
|
<textarea id="@_rawhtmlid" class="form-control" placeholder="@Placeholder" @bind="@_rawhtml" rows="10"></textarea>
|
|
}
|
|
</TabPanel>
|
|
}
|
|
</TabStrip>
|
|
</div>
|
|
</div>
|
|
|
|
@code {
|
|
private string _activetab = "Rich";
|
|
|
|
private bool _rawfilemanager = false;
|
|
private FileManager _fileManager;
|
|
private string _message = string.Empty;
|
|
private string _rawhtmlid = "RawHtmlEditor_" + Guid.NewGuid().ToString("N");
|
|
private string _rawhtml = string.Empty;
|
|
private string _originalrawhtml = string.Empty;
|
|
|
|
private ITextEditorProvider _textEditorProvider;
|
|
private RenderFragment _textEditorComponent;
|
|
private ITextEditor _textEditor;
|
|
|
|
[Parameter]
|
|
public string Content { get; set; }
|
|
|
|
[Parameter]
|
|
public bool ReadOnly { get; set; } = false;
|
|
|
|
[Parameter]
|
|
public string Placeholder { get; set; }
|
|
|
|
[Parameter]
|
|
public bool AllowFileManagement { get; set; } = true;
|
|
|
|
[Parameter]
|
|
public bool AllowRichText { get; set; } = true;
|
|
|
|
[Parameter]
|
|
public bool AllowRawHtml { get; set; } = true;
|
|
|
|
// parameters only applicable to rich text editor
|
|
[Parameter]
|
|
public RenderFragment ToolbarContent { get; set; }
|
|
|
|
[Parameter]
|
|
public string Theme { get; set; } = "snow";
|
|
|
|
[Parameter]
|
|
public string DebugLevel { get; set; } = "info";
|
|
|
|
public override List<Resource> Resources { get; set; } = new List<Resource>();
|
|
|
|
protected override async Task OnInitializedAsync()
|
|
{
|
|
if (string.IsNullOrEmpty(Placeholder))
|
|
{
|
|
Placeholder = Localizer["Placeholder"];
|
|
}
|
|
|
|
if(AllowRichText)
|
|
{
|
|
_textEditorProvider = await GetTextEditorProvider();
|
|
}
|
|
}
|
|
|
|
protected override void OnParametersSet()
|
|
{
|
|
_rawhtml = Content;
|
|
_originalrawhtml = _rawhtml; // preserve for comparison later
|
|
|
|
if (!AllowRichText)
|
|
{
|
|
_activetab = "Raw";
|
|
}
|
|
|
|
_textEditorComponent = (builder) =>
|
|
{
|
|
CreateTextEditor(builder);
|
|
};
|
|
}
|
|
|
|
protected override async Task OnAfterRenderAsync(bool firstRender)
|
|
{
|
|
if(_textEditor != null)
|
|
{
|
|
_textEditor.Initialize(Content, Content != _originalrawhtml);
|
|
}
|
|
|
|
await base.OnAfterRenderAsync(firstRender);
|
|
}
|
|
|
|
|
|
public void CloseRawFileManager()
|
|
{
|
|
_rawfilemanager = false;
|
|
_message = string.Empty;
|
|
StateHasChanged();
|
|
}
|
|
|
|
public async Task<string> GetHtml()
|
|
{
|
|
// evaluate raw html content as first priority
|
|
if (_rawhtml != _originalrawhtml)
|
|
{
|
|
return _rawhtml;
|
|
}
|
|
else
|
|
{
|
|
var richhtml = string.Empty;
|
|
if (AllowRichText && _textEditor != null)
|
|
{
|
|
richhtml = await _textEditor.GetContent();
|
|
}
|
|
|
|
return richhtml != null ? richhtml : _originalrawhtml;
|
|
}
|
|
}
|
|
|
|
|
|
public async Task InsertRawImage()
|
|
{
|
|
_message = string.Empty;
|
|
if (_rawfilemanager)
|
|
{
|
|
var file = _fileManager.GetFile();
|
|
if (file != null)
|
|
{
|
|
var interop = new Interop(JSRuntime);
|
|
int pos = await interop.GetCaretPosition(_rawhtmlid);
|
|
var image = "<img src=\"" + file.Url + "\" alt=\"" + ((!string.IsNullOrEmpty(file.Description)) ? file.Description : file.Name) + "\" class=\"img-fluid\">";
|
|
_rawhtml = _rawhtml.Substring(0, pos) + image + _rawhtml.Substring(pos);
|
|
_rawfilemanager = false;
|
|
}
|
|
else
|
|
{
|
|
_message = Localizer["Message.Require.Image"];
|
|
}
|
|
}
|
|
else
|
|
{
|
|
_rawfilemanager = true;
|
|
}
|
|
StateHasChanged();
|
|
}
|
|
|
|
private void CreateTextEditor(RenderTreeBuilder builder)
|
|
{
|
|
if(_textEditorProvider != null)
|
|
{
|
|
var editorType = Type.GetType(_textEditorProvider.EditorType);
|
|
if (editorType != null)
|
|
{
|
|
builder.OpenComponent(0, editorType);
|
|
|
|
//set editor parameters if available.
|
|
var attributes = new Dictionary<string, object>
|
|
{
|
|
{ "AllowFileManagement", AllowFileManagement },
|
|
{ "ReadOnly", ReadOnly },
|
|
{ "Placeholder", Placeholder },
|
|
{ "Theme", Theme },
|
|
{ "DebugLevel", DebugLevel },
|
|
{ "ToolbarContent", ToolbarContent }
|
|
};
|
|
|
|
var index = 1;
|
|
foreach(var name in attributes.Keys)
|
|
{
|
|
if (editorType.GetProperty(name) != null)
|
|
{
|
|
builder.AddAttribute(index++, name, attributes[name]);
|
|
}
|
|
}
|
|
|
|
builder.AddComponentReferenceCapture(index, (c) =>
|
|
{
|
|
_textEditor = (ITextEditor)c;
|
|
});
|
|
builder.CloseComponent();
|
|
}
|
|
}
|
|
}
|
|
|
|
private async Task<ITextEditorProvider> GetTextEditorProvider()
|
|
{
|
|
const string DefaultEditorName = "Quill";
|
|
|
|
var editorName = await GetTextEditorName(DefaultEditorName);
|
|
var editorProviders = ServiceProvider.GetServices<ITextEditorProvider>();
|
|
var editorProvider = editorProviders.FirstOrDefault(i => i.Name == editorName);
|
|
if(editorProvider == null)
|
|
{
|
|
editorProvider = editorProviders.FirstOrDefault(i => i.Name == DefaultEditorName);
|
|
}
|
|
|
|
return editorProvider;
|
|
}
|
|
|
|
private async Task<string> GetTextEditorName(string defaultName)
|
|
{
|
|
const string EditorSettingName = "TextEditor";
|
|
|
|
var settings = await SettingService.GetSiteSettingsAsync(PageState.Site.SiteId);
|
|
return SettingService.GetSetting(settings, EditorSettingName, defaultName);
|
|
}
|
|
}
|