|
@@ -22,44 +22,37 @@
|
|
|
<br>
|
|
|
<div class="ui text container">
|
|
|
<h2 class="ui header">
|
|
|
- WebStick Blog
|
|
|
- <div class="sub header">Welcome to this new personal blog of mine!</div>
|
|
|
+ <span id="blogtitle">WebStick Blog</span> <span onclick="editBlogTitle();" class="adminOnly"><i class="ui small edit grey icon"></i></span>
|
|
|
+ <div class="sub header"><span id="blogsubtitle">Welcome to this new personal blog of mine!</span> <span class="adminOnly" onclick="editBlogSubtitle();"><i class="ui edit icon"></i></span></div>
|
|
|
</h2>
|
|
|
<div class="ui divider"></div>
|
|
|
<a class="ui basic icon button" href="../" title="Back to Home"><i class="home icon"></i></a>
|
|
|
- <a class="ui basic icon button" href="../" title="Share"><i class="green share alternate icon"></i></a>
|
|
|
- <a class="ui basic button adminOnly"><i class="blue add icon"></i> New Post</a>
|
|
|
- <br><br>
|
|
|
- <div class="posttable">
|
|
|
- <div class="ui message">
|
|
|
- <h4 class="ui header">
|
|
|
- <i class="blue question icon"></i>
|
|
|
- <div class="content">
|
|
|
- No Post
|
|
|
- <div class="sub header">It seems this is a brand new blog site!</div>
|
|
|
- </div>
|
|
|
- </h4>
|
|
|
+ <button id="createNewPostBtn" class="ui basic button adminOnly"><i class="blue add icon"></i> New Post</button>
|
|
|
+ <button onclick="forceUpdatePostIndex()" class="ui basic green button adminOnly"><i class="green refresh icon"></i> Force Update Post Index</button>
|
|
|
+ <br>
|
|
|
+ <div id="newPostModal" class="ui basic segment" style="display:none;">
|
|
|
+ <div class="ui divider"></div>
|
|
|
+ <h3><i class="ui blue add icon"></i> Create a new Post</h3>
|
|
|
+ <p>New post in WebStick Blog Engine are markdown files strored in the posts/ folder. Create a new post with filename as title (exclude the .md extension when filling in the filename).<br>
|
|
|
+ A new window with markdown editor will be opened. After finish writing, save & close the markdown editor and refresh this page to see your post.</p>
|
|
|
+ <div class="ui action fluid input">
|
|
|
+ <input id="newPostTitle" type="text" maxlength="16">
|
|
|
+ <button id="confirmNewPostBtn" onclick="createNewPost();" class="ui green button">Create & Edit</button>
|
|
|
</div>
|
|
|
-
|
|
|
-
|
|
|
- <div class="ui basic segment postObject">
|
|
|
- <h4 class="ui header">
|
|
|
- <i class="blue map marker alternate icon"></i>
|
|
|
- <div class="content">
|
|
|
- 11 Sep 2023
|
|
|
- </div>
|
|
|
- </h4>
|
|
|
- <div class="ui divider"></div>
|
|
|
- <div class="adminOnly" style="position: absolute; top: 0.4em; right: 0.4em;">
|
|
|
- <a class="ui basic mini icon button" title="Edit Post"><i class="edit icon"></i></a>
|
|
|
- <button class="ui basic mini icon button" title="Remove Post"><i class="red trash icon"></i></button>
|
|
|
- </div>
|
|
|
- <div class="postContent">
|
|
|
-
|
|
|
+ </div>
|
|
|
+ <div id="nopost" class="ui message" style="display:none;">
|
|
|
+ <h4 class="ui header">
|
|
|
+ <i class="green check icon"></i>
|
|
|
+ <div class="content">
|
|
|
+ Seems this blog has nothing posted. Check back later!
|
|
|
+ <div class="sub header">You own the place? Try create a new post after login.</div>
|
|
|
</div>
|
|
|
- <div class="ui divider"></div>
|
|
|
+ </h4>
|
|
|
+ </div>
|
|
|
+ <div id="posttable">
|
|
|
+ <div class="ui basic segment">
|
|
|
+ <i class="ui loading spinner icon"></i> Loading Posts
|
|
|
</div>
|
|
|
-
|
|
|
</div>
|
|
|
</div>
|
|
|
|
|
@@ -69,32 +62,299 @@
|
|
|
</div>
|
|
|
</div>
|
|
|
<script>
|
|
|
+ let loggedIn = false;
|
|
|
+
|
|
|
//Check the user has logged in
|
|
|
- $.get("", function(data){
|
|
|
+ $.get("/api/auth/chk", function(data){
|
|
|
if (data == false){
|
|
|
//User cannot use admin function. Hide the buttons.
|
|
|
$(".adminOnly").hide();
|
|
|
+ loggedIn = false;
|
|
|
+ }else{
|
|
|
+ loggedIn = true;
|
|
|
}
|
|
|
+ loadValue("blog-posts", function(){
|
|
|
+ initPosts();
|
|
|
+ })
|
|
|
});
|
|
|
|
|
|
- //Load the list of post from the server side
|
|
|
- function loadPostFromServer(){
|
|
|
+
|
|
|
+
|
|
|
+ //Initialize blog info
|
|
|
+ function initBlogInfo(){
|
|
|
+ loadValue("blog-title", function(title){
|
|
|
+ if (title.error != undefined || title == ""){
|
|
|
+ title = "WebStick Blog";
|
|
|
+ }
|
|
|
+ document.title = decodeURIComponent(title);
|
|
|
+ $("#blogtitle").text(decodeURIComponent(title));
|
|
|
+ });
|
|
|
+
|
|
|
+ loadValue("blog-subtitle", function(title){
|
|
|
+ if (title.error != undefined || title == ""){
|
|
|
+ title = "Welcome to my personal blog!";
|
|
|
+ }
|
|
|
+ $("#blogsubtitle").text(decodeURIComponent(title));
|
|
|
+ });
|
|
|
+ }
|
|
|
+ initBlogInfo();
|
|
|
+
|
|
|
+
|
|
|
+ //Edit blog title and subtitles
|
|
|
+ function editBlogSubtitle(){
|
|
|
+ let newtitle = prompt("New Blog Subtitle", "");
|
|
|
+ if (newtitle != null) {
|
|
|
+ setValue("blog-subtitle", encodeURIComponent(newtitle), function(){
|
|
|
+ initBlogInfo();
|
|
|
+ })
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ function editBlogTitle(){
|
|
|
+ let newtitle = prompt("New Blog Title", "");
|
|
|
+ if (newtitle != null) {
|
|
|
+ setValue("blog-title", encodeURIComponent(newtitle), function(){
|
|
|
+ initBlogInfo();
|
|
|
+ })
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ //Storage and loader utils
|
|
|
+ function setValue(key, value, callback){
|
|
|
+ $.get("/api/pref/set?key=" + key + "&value=" + value, function(data){
|
|
|
+ callback(data);
|
|
|
+ });
|
|
|
+ }
|
|
|
+
|
|
|
+ function loadValue(key, callback){
|
|
|
+ $.get("/api/pref/get?key=" + key, function(data){
|
|
|
+ callback(data);
|
|
|
+ });
|
|
|
+ }
|
|
|
+
|
|
|
+ /*
|
|
|
+ New Post
|
|
|
+
|
|
|
+ New post is created via creating a markdown file in the server
|
|
|
+ side and open it with the markdown editor
|
|
|
+ */
|
|
|
+
|
|
|
+ $("#createNewPostBtn").on("click", function(){
|
|
|
+ $("#newPostModal").toggle("fast");
|
|
|
+ });
|
|
|
+
|
|
|
+ function createNewPost(){
|
|
|
+ let filename = $("#newPostTitle").val().trim();
|
|
|
+ if (filename == ""){
|
|
|
+ alert("Post title cannot be empty.");
|
|
|
+ return;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (filename.indexOf("/") >= 0){
|
|
|
+ //Contains /. Reject
|
|
|
+ alert("File name cannot contain path seperator");
|
|
|
+ return;
|
|
|
+ }
|
|
|
+
|
|
|
+ $("#confirmNewPostBtn").addClass("loading").addClass("disabled");
|
|
|
+ //Create the markdown file at the /blog/posts folder
|
|
|
+ const blob = new Blob(["# Hello World\n"], { type: 'text/plain' });
|
|
|
+ let storeFilename = parseInt(Date.now()/1000) + "_" + filename+'.md';
|
|
|
+ const file = new File([blob], storeFilename);
|
|
|
+ handleFile(file, "/blog/posts", function(){
|
|
|
+ //Update the post index
|
|
|
+ updatePostIndex();
|
|
|
+
|
|
|
+ $("#confirmNewPostBtn").removeClass("loading").removeClass("disabled");
|
|
|
+
|
|
|
+ //Open the markdown file in new tab
|
|
|
+ let hash = encodeURIComponent(JSON.stringify({
|
|
|
+ "filename": storeFilename,
|
|
|
+ "filepath": "/blog/posts/" + storeFilename
|
|
|
+ }))
|
|
|
+ window.open("/admin/mde/index.html#" + hash);
|
|
|
|
|
|
+ $("#newPostModal").hide();
|
|
|
+ });
|
|
|
+
|
|
|
+
|
|
|
+ }
|
|
|
+
|
|
|
+ function handleFile(file, dir=currentPath, callback=undefined) {
|
|
|
+ // Perform actions with the selected file
|
|
|
+ var formdata = new FormData();
|
|
|
+ formdata.append("file1", file);
|
|
|
+ var ajax = new XMLHttpRequest();
|
|
|
+ ajax.addEventListener("load", function(event){
|
|
|
+ let responseText = event.target.responseText;
|
|
|
+ try{
|
|
|
+ responseText = JSON.parse(responseText);
|
|
|
+ if (responseText.error != undefined){
|
|
|
+ alert(responseText.error);
|
|
|
+ }
|
|
|
+ }catch(ex){
|
|
|
+
|
|
|
+ }
|
|
|
+ if (callback != undefined){
|
|
|
+ callback();
|
|
|
+ }
|
|
|
+ }, false); // doesnt appear to ever get called even upon success
|
|
|
+ ajax.addEventListener("error", errorHandler, false);
|
|
|
+ //ajax.addEventListener("abort", abortHandler, false);
|
|
|
+ ajax.open("POST", "/upload?dir=" + dir);
|
|
|
+ ajax.send(formdata);
|
|
|
+ }
|
|
|
+
|
|
|
+ function errorHandler(event) {
|
|
|
+ aelrt("New Post creation failed");
|
|
|
+ $("#pasteButton").removeClass("disabled");
|
|
|
+ }
|
|
|
+
|
|
|
+ /*
|
|
|
+ Post Edit functions
|
|
|
+ */
|
|
|
+
|
|
|
+ function editPost(btn){
|
|
|
+ let postFilename = $(btn).attr("filename");
|
|
|
+ let hash = encodeURIComponent(JSON.stringify({
|
|
|
+ "filename": postFilename,
|
|
|
+ "filepath": "/blog/posts/" + postFilename
|
|
|
+ }))
|
|
|
+ window.open("/admin/mde/index.html#" + hash);
|
|
|
+ }
|
|
|
+
|
|
|
+ function deletePost(btn){
|
|
|
+ let postFilename = $(btn).attr("filename");
|
|
|
+ let postTitle = $(btn).attr("ptitle");
|
|
|
+ if (confirm("Confirm remove post titled: " + postTitle + "?")){
|
|
|
+ $.ajax({
|
|
|
+ url: "/api/fs/del?target=/blog/posts/" + postFilename,
|
|
|
+ method: "POST",
|
|
|
+ success: function(data){
|
|
|
+ if (data.error != undefined){
|
|
|
+ alert("Post delete failed. See console for more info.");
|
|
|
+ console.log(data.error);
|
|
|
+ }else{
|
|
|
+ //Deleted
|
|
|
+ initPosts();
|
|
|
+ }
|
|
|
+ }
|
|
|
+ });
|
|
|
+ }
|
|
|
}
|
|
|
|
|
|
+ /*
|
|
|
+ Rendering for Posts
|
|
|
+ */
|
|
|
//Load a markdown file from URL and render it to target element
|
|
|
function loadMarkdownToHTML(markdownURL, targetElement){
|
|
|
fetch(markdownURL).then( r => r.text() ).then( text =>{
|
|
|
var converter = new showdown.Converter();
|
|
|
-
|
|
|
let targetHTML = converter.makeHtml(text);
|
|
|
console.log(targetHTML);
|
|
|
$(targetElement).html(targetHTML);
|
|
|
});
|
|
|
}
|
|
|
|
|
|
- loadMarkdownToHTML("../../README.md", $(".postContent"));
|
|
|
- loadPostFromServer();
|
|
|
+ function initPosts(){
|
|
|
+ $("#posttable").html("<div class='ui basic segment'><p><i class='ui loading spinner icon'></i> Loading Blog Posts</p></div>");
|
|
|
+ loadValue("blog-posts", function(data){
|
|
|
+ $("#posttable").html("");
|
|
|
+ try{
|
|
|
+ let postList = JSON.parse(decodeURIComponent(atob(data)));
|
|
|
+
|
|
|
+ //From latest to oldest
|
|
|
+ postList.reverse();
|
|
|
+ console.log("Post listed loaded: ", postList);
|
|
|
+ if (postList.length == 0){
|
|
|
+ $("#nopost").show();
|
|
|
+ }else{
|
|
|
+ $("#nopost").hide();
|
|
|
+ postList.forEach(postFilename => {
|
|
|
+ renderPost(postFilename);
|
|
|
+ })
|
|
|
+ }
|
|
|
+ }catch(ex){
|
|
|
+ $("#nopost").show();
|
|
|
+ }
|
|
|
+
|
|
|
+ })
|
|
|
+ }
|
|
|
+
|
|
|
+ function forceUpdatePostIndex(){
|
|
|
+ updatePostIndex(function(){
|
|
|
+ window.location.reload();
|
|
|
+ });
|
|
|
+ }
|
|
|
+
|
|
|
+ function updatePostIndex(callback=undefined){
|
|
|
+ let postList = [];
|
|
|
+ $.ajax({
|
|
|
+ url: "/api/fs/list?dir=/blog/posts",
|
|
|
+ success: function(data){
|
|
|
+ data.forEach(file => {
|
|
|
+ let filename = file.Filename;
|
|
|
+ let ext = filename.split(".").pop();
|
|
|
+ if (ext == "md" && file.IsDir == false){
|
|
|
+ //Markdown file. Render it
|
|
|
+ postList.push(filename);
|
|
|
+ }
|
|
|
+ });
|
|
|
+
|
|
|
+ setValue("blog-posts", btoa(encodeURIComponent(JSON.stringify(postList))), function(data){
|
|
|
+ console.log(data);
|
|
|
+ if (callback != undefined){
|
|
|
+ callback();
|
|
|
+ }
|
|
|
+ });
|
|
|
+ }
|
|
|
+ });
|
|
|
+
|
|
|
+
|
|
|
+ }
|
|
|
+
|
|
|
+ //Render post
|
|
|
+ function renderPost(filename){
|
|
|
+ //Remove the timestamp
|
|
|
+ let postTitle = filename.split("_");
|
|
|
+ let timeStamp = postTitle.shift();
|
|
|
+ postTitle = postTitle.join("_");
|
|
|
+
|
|
|
+ //Pop the file extension
|
|
|
+ postTitle = postTitle.split(".");
|
|
|
+ postTitle.pop();
|
|
|
+ postTitle = postTitle.join(".");
|
|
|
+
|
|
|
+ var postTime = new Date(parseInt(timeStamp) * 1000).toLocaleDateString("en-US")
|
|
|
+ let postEditFeature = `<div class="adminOnly" style="position: absolute; top: 3em; right: 0.4em;">
|
|
|
+ <a class="ui basic mini icon button" onclick="editPost(this);" filename="${filename}" title="Edit Post"><i class="edit icon"></i></a>
|
|
|
+ <button class="ui basic mini icon button" onclick="deletePost(this);" ptitle="${postTitle}" filename="${filename}" title="Remove Post"><i class="red trash icon"></i></button>
|
|
|
+ </div>`;
|
|
|
+
|
|
|
+ if (!loggedIn){
|
|
|
+ postEditFeature = "";
|
|
|
+ }
|
|
|
+ //Create a wrapper element
|
|
|
+ $("#posttable").append(`
|
|
|
+ <div class="ui basic segment postObject" id="${timeStamp}">
|
|
|
+ <div class="ui divider"></div>
|
|
|
+ <h4 class="ui header">
|
|
|
+ <i class="blue paperclip icon"></i>
|
|
|
+ <div class="content">
|
|
|
+ ${postTitle}
|
|
|
+ </div>
|
|
|
+ </h4>
|
|
|
+ ${postEditFeature}
|
|
|
+ <div class="postContent">
|
|
|
+
|
|
|
+ </div>
|
|
|
+ <small><i class="calendar alternate outline icon"></i> ${postTime}</small>
|
|
|
+ </div>
|
|
|
+ `);
|
|
|
+ let targetElement = $("#" + timeStamp).find(".postContent");
|
|
|
+ loadMarkdownToHTML("/blog/posts/" + filename,targetElement);
|
|
|
+ }
|
|
|
+
|
|
|
</script>
|
|
|
</body>
|
|
|
</html>
|