終於要接近尾聲了,上一篇基本上將文章模塊的全部功能都完成了,整個博客頁面也都完成了,本篇主要來美化幾個地方,修修補補。javascript
當咱們新增和編輯文章的時候,默認編輯器是白色的,若是點擊了頭部切換主題按鈕,我想要把編輯器主題顏色也作相應的改變該如何去實現呢?css
恰好,editor.md
是支持主題切換的,這就比較舒服了,直接按照要求調用對應的方法便可。html
在app.js
的renderEditor
函數中咱們已經自定義了三個參數theme
、editorTheme
、previewTheme
,這三個參數就是來改變編輯器主題顏色的。java
仍是將值存在localStorage中,和咱們博客的主題切換同樣,這裏叫editorTheme
。git
theme: localStorage.editorTheme || 'default', editorTheme: localStorage.editorTheme === 'dark' ? 'pastel-on-dark' : 'default', previewTheme: localStorage.editorTheme || 'default',
默認從localStorage
中取數據,若是沒取到的話,給對應的默認值。第二個參數有點特殊,用了一個三元表達式給不一樣的值。github
而後在主題切換的時候也對編輯器作相應的調整便可。緩存
打開Header.razor
頭部組件,找到SwitchTheme()
切換主題的方法,添加一句await Common.SwitchEditorTheme(currentTheme);
。markdown
/// <summary> /// 切換主題 /// </summary> private async Task SwitchTheme() { currentTheme = currentTheme == "Light" ? "Dark" : "Light"; await Common.SetStorageAsync("theme", currentTheme); await Common.InvokeAsync("window.func.switchTheme"); var uri = await Common.CurrentUri(); if (uri.AbsolutePath.StartsWith("/admin/post")) { await Common.SwitchEditorTheme(currentTheme); } }
將具體切換邏輯放到SwitchEditorTheme
中,他接收一個參數currentTheme
,用來判斷是切換黑的仍是白的。app
/// <summary> /// 切換編輯器主題 /// </summary> /// <param name="currentTheme"></param> /// <returns></returns> public async Task SwitchEditorTheme(string currentTheme) { var editorTheme = currentTheme == "Light" ? "default" : "dark"; await SetStorageAsync("editorTheme", editorTheme); await InvokeAsync("window.func.switchEditorTheme"); }
切換主題以前拿到當前URI對象,判斷當前請求的連接是不是新增和更新文章的那個頁面,即"/admin/post",纔去執行切換編輯器主題的方法,當不是這個頁面的時候,編輯器是沒有渲染出來的,若是也執行這段代碼就會報錯。async
去看看效果。
如今的文章詳情頁是沒有將markdown格式渲染出來的,這裏仍是使用editor.md
提供的方法,由於須要加載幾個js文件,而後才能渲染樣式。
因此仍是在app.js
添加一段代碼。
renderMarkdown: async function () { await this._loadScript('./editor.md/lib/zepto.min.js').then(function () { func._loadScript('./editor.md/lib/marked.min.js').then(function () { func._loadScript('./editor.md/lib/prettify.min.js').then(function () { func._loadScript('./editor.md/editormd.js').then(function () { editormd.markdownToHTML("content"); }); }); }); }); },
而後在文章詳情頁的組件Post.razor
中修改代碼,當數據加載完成後調用renderMarkdown
便可,而後還須要引用一個css文件editormd.preview.css
。
提供一下Post.razor
最終的代碼。
@page "/post/{year:int}/{month:int}/{day:int}/{name}" <link href="./editor.md/css/editormd.preview.css" rel="stylesheet" /> @if (post == null) { <Loading /> } else { @if (post.Success) { var _post = post.Result; <article class="post-wrap"> <header class="post-header"> <h1 class="post-title">@_post.Title</h1> <div class="post-meta"> Author: <a itemprop="author" rel="author" href="javascript:;">@_post.Author</a> <span class="post-time"> Date: <a href="javascript:;">@_post.CreationTime</a> </span> <span class="post-category"> Category:<a href="/category/@_post.Category.DisplayName/">@_post.Category.CategoryName</a> </span> </div> </header> <div class="post-content" id="content"> @((MarkupString)_post.Html) </div> <section class="post-copyright"> <p class="copyright-item"> <span>Author:</span> <span>@_post.Author</span> </p> <p class="copyright-item"> <span>Permalink:</span> <span><a href="/post@_post.Url">https://meowv.com/post@_post.Url</a></span> </p> <p class="copyright-item"> <span>License:</span> <span>本文采用<a target="_blank" href="http://creativecommons.org/licenses/by-nc-nd/4.0/"> 知識共享 署名-非商業性使用-禁止演繹(CC BY-NC-ND)國際許可協議 </a>進行許可</span> </p> </section> <section class="post-tags"> <div> <span>Tag(s):</span> <span class="tag"> @if (_post.Tags.Any()) { @foreach (var tag in _post.Tags) { <a href="/tag/@tag.DisplayName/"># @tag.TagName</a> } } </span> </div> <div> <a @onclick="@(async () => await Common.NavigateTo("/posts"))">back</a> <span>· </span> <a href="/">home</a> </div> </section> <section class="post-nav"> @if (_post.Previous != null) { <a class="prev" rel="prev" @onclick="@(async () => await FetchData(_post.Previous.Url))" href="/post@_post.Previous.Url">@_post.Previous.Title</a> } @if (_post.Next != null) { <a class="next" rel="next" @onclick="@(async () => await FetchData(_post.Next.Url))" href="/post@_post.Next.Url"> @_post.Next.Title </a> } </section> </article> } else { <ErrorTip /> } } @code { [Parameter] public int year { get; set; } [Parameter] public int month { get; set; } [Parameter] public int day { get; set; } [Parameter] public string name { get; set; } /// <summary> /// URL /// </summary> private string url => $"/{year}/{(month >= 10 ? month.ToString() : $"0{month}")}/{(day >= 10 ? day.ToString() : $"0{day}")}/{name}/"; /// <summary> /// 文章詳情數據 /// </summary> private ServiceResult<PostDetailDto> post; /// <summary> /// 初始化 /// </summary> protected override async Task OnInitializedAsync() { await FetchData(url); } /// <summary> /// 請求數據,渲染頁面 /// </summary> /// <param name="url"></param> /// <returns></returns> private async Task FetchData(string url, bool isPostNav = false) { post = await Http.GetFromJsonAsync<ServiceResult<PostDetailDto>>($"/blog/post?url={url}"); await Common.InvokeAsync("window.func.renderMarkdown"); } }
到這裏整個開發工做便結束了,這裏只是一個小小的實戰系列記錄,沒有深層次的剖析研究Blazor的全部使用方式。
若是本系列對你有些許幫助,即是我最大的收穫,歡迎你們關注個人公衆號:阿星Plus。