Browse Source

optimized web server ui

Toby Chui 1 year ago
parent
commit
246f9b5517

+ 370 - 360
sd_card/www/blog/index.html

@@ -1,360 +1,370 @@
-<!DOCTYPE html>
-<html>
-<head>
-
-    <!-- HTML Meta Tags -->
-    <title>Welcome to My Blog</title>
-    <meta name="description" content="A personal blog hosted on WebStick!">
-    <meta name="viewport" content="width=device-width, initial-scale=1" >
-
-    <script src="https://code.jquery.com/jquery-3.6.0.min.js"></script>
-    <link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/semantic-ui/2.4.1/semantic.min.css" />
-    <script src="https://cdnjs.cloudflare.com/ajax/libs/semantic-ui/2.4.1/semantic.min.js"></script>
-    <script src="https://cdnjs.cloudflare.com/ajax/libs/showdown/2.1.0/showdown.min.js"></script>
-    <style>
-        .imgwrapper{
-            max-height: 200px;
-            overflow: hidden;
-        }
-  </style>
-</head>
-<body>
-    <br>
-    <div class="ui text container">
-        <h2 class="ui header">
-           <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>
-        <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>
-        <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>
-              </h4>
-        </div>
-        <div id="posttable">
-            <div class="ui basic segment">
-                <i class="ui loading spinner icon"></i> Loading Posts
-            </div>
-        </div>
-    </div>
-    
-    <div class="ui text container center aligned">
-        <div class="ui divider"></div>
-        <small>Proudly powered by <a href="https://hackaday.io/project/192618-instant-webstick-esp8266-web-server-nas">WebStick</a></small>
-    </div>
-	</div>
-    <script>
-        let loggedIn = false;
-
-        //Check the user has logged in
-        $.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();
-            })
-        });
-
-       
-
-        //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);
-            });
-        }
-
-        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>
+<!DOCTYPE html>
+<html>
+<head>
+
+    <!-- HTML Meta Tags -->
+    <title>Welcome to My Blog</title>
+    <meta name="description" content="A personal blog hosted on WebStick!">
+    <meta name="viewport" content="width=device-width, initial-scale=1" >
+
+    <script src="https://code.jquery.com/jquery-3.6.0.min.js"></script>
+    <link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/semantic-ui/2.4.1/semantic.min.css" />
+    <script src="https://cdnjs.cloudflare.com/ajax/libs/semantic-ui/2.4.1/semantic.min.js"></script>
+    <script src="https://cdnjs.cloudflare.com/ajax/libs/showdown/2.1.0/showdown.min.js"></script>
+    <style>
+        .imgwrapper{
+            max-height: 200px;
+            overflow: hidden;
+        }
+		
+		.homebtn{
+			cursor: pointer;
+			color: grey;
+		}
+		.homebtn:hover{
+			opacity: 0.5;
+		}
+  </style>
+</head>
+<body>
+    <br>
+    <div class="ui text container">
+        <h2 class="ui header">
+           <span class="homebtn" title="Homepage" onclick="window.location.href = '../';"><i class="ui chevron left circle icon"></i></span>
+		   <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="adminOnly">
+        <div class="ui divider"></div>
+        <button id="createNewPostBtn" class="ui basic button"><i class="blue add icon"></i> New Post</button>
+        <button onclick="forceUpdatePostIndex()" class="ui basic button"><i class="green refresh icon"></i> Force Update Post Index</button>
+        <br>
+		</div>
+        <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>
+        <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>
+              </h4>
+        </div>
+        <div id="posttable">
+            <div class="ui basic segment">
+                <i class="ui loading spinner icon"></i> Loading Posts
+            </div>
+        </div>
+    </div>
+    
+    <div class="ui text container center aligned">
+        <div class="ui divider"></div>
+        <small>Proudly powered by <a href="https://hackaday.io/project/192618-instant-webstick-esp8266-web-server-nas">WebStick</a></small>
+    </div>
+	</div>
+    <script>
+        let loggedIn = false;
+
+        //Check the user has logged in
+        $.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();
+            })
+        });
+
+       
+
+        //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);
+            });
+        }
+
+        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>

+ 9 - 0
sd_card/www/blog/posts/1710676940_Hello World!.md

@@ -0,0 +1,9 @@
+# Hello World
+Welcome to my web page. Fun facts: This is hosted on an ESP8266!
+
+## WebSticks
+The WebStick is an Arduino programmed, mini web server stick power in a USB drive form factor.
+
+You might be wonder how it is possible to work right? Well, it is not simple. The make it work, I first
+need to write a web server in C++ and make it serve html files from an SD card formated as FAT.
+Next, I need to handle all kind of headers and werid things to make it happens.

+ 8 - 8
sd_card/www/blog/posts/index.html

@@ -1,9 +1,9 @@
-<!DOCTYPE html>
-<html>
-  <head>
-    <meta http-equiv="refresh" content="0; url='../index.html'" />
-  </head>
-  <body>
-    <p>Redirecitng to blog UI</p>
-  </body>
+<!DOCTYPE html>
+<html>
+  <head>
+    <meta http-equiv="refresh" content="0; url='../index.html'" />
+  </head>
+  <body>
+    <p>Redirecitng to blog UI</p>
+  </body>
 </html>

+ 5 - 5
sd_card/www/index.html

@@ -102,7 +102,7 @@
 						<img src="img/selfie.jpg" style="border-radius: 50%; width: 100%; user-select: none; pointer-events: none;">
 					</div>    
 				</div>    
-				<h1 style="font-weight: bolder;">Your Name</h1>
+				<h1 style="font-weight: bolder;">Hello visitor!</h1>
 				<div class="dark divider"></div>
 				<div class="ui breadcrumb">
 					<div class="section">Software Engineer</div>
@@ -110,13 +110,12 @@
 					<div class="section">Electronics Maker</div>
 				</div>
 				<div class="linkbuttons">
-					<p>Public Pages</p>
 					<a class="ui fluid white button" href="about.html">About WebStick</a>
 					<a class="ui fluid white button" href="blog/index.html">My Blog</a>
 					<a class="ui fluid white button" href="down/"><i class="ui download icon"></i> Downloads</a>
+					<a class="ui fluid white button" href="tool/qr.html"><i class="ui qrcode icon"></i> QR Code Generator</a>
 					<div class="ui divider"></div>
-					<p>Private Services</p>
-					<a class="ui fluid white button" href="admin/"><i class="ui user icon"></i> Admin Panel</a>
+					<a class="ui fluid white button" href="admin/"><i class="ui sign in icon"></i> Admin Panel</a>
 				</div>
 		</div>
 		<div class="content" align="center">
@@ -127,8 +126,9 @@
 			<a class="blacklink" href="https://www.facebook.com/ImusLaboratory"><i class="big facebook icon"></i></a>
 	  </div>
 	</div>
-		<br><br>
+		<br>
 		<p>This site is hosted on a WebStick designed by <a href="https://imuslab.com">imuslab</a></p>
+		<br>
 	</div>
 </body>
 </html>

+ 123 - 0
sd_card/www/tool/qr.html

@@ -0,0 +1,123 @@
+<!DOCTYPE html>
+<html lang="en">
+<head>
+    <meta charset="UTF-8">
+    <meta name="viewport" content="width=device-width, initial-scale=1.0">
+    <title>QR Code Generator</title>
+	<link rel="icon" type="image/png" href="/favicon.png" />
+    <script src="https://cdnjs.cloudflare.com/ajax/libs/qrcodejs/1.0.0/qrcode.min.js"></script>
+	<script src="https://cdnjs.cloudflare.com/ajax/libs/jquery/3.7.1/jquery.min.js" integrity="sha512-v2CJ7UaYy4JwqLDIrZUI/4hqeoQieOmAZNXBeQyjo21dadnwR+8ZaIJVT8EE2iyI61OV8e6M8PP2/4hpQINQ/g==" crossorigin="anonymous" referrerpolicy="no-referrer"></script>
+	<script src="https://cdnjs.cloudflare.com/ajax/libs/fomantic-ui/2.9.3/semantic.min.js" integrity="sha512-gnoBksrDbaMnlE0rhhkcx3iwzvgBGz6mOEj4/Y5ZY09n55dYddx6+WYc72A55qEesV8VX2iMomteIwobeGK1BQ==" crossorigin="anonymous" referrerpolicy="no-referrer"></script>
+	<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/fomantic-ui/2.9.3/semantic.min.css" integrity="sha512-3quBdRGJyLy79hzhDDcBzANW+mVqPctrGCfIPosHQtMKb3rKsCxfyslzwlz2wj1dT8A7UX+sEvDjaUv+WExQrA==" crossorigin="anonymous" referrerpolicy="no-referrer" />
+	 <link rel="preconnect" href="https://fonts.googleapis.com">
+	<link rel="preconnect" href="https://fonts.gstatic.com" crossorigin>
+	<link href="https://fonts.googleapis.com/css2?family=Noto+Sans+TC:wght@100;200&display=swap" rel="stylesheet">
+
+    <style>
+		body {
+            
+        }
+
+		.main{
+			display: flex;
+            align-items: center;
+            justify-content: center;
+            height: calc(100vh - 60px);
+            margin: 0;
+		}
+		
+        #qr-code {
+            max-width: 300px;
+			height: 300px;
+        }
+
+        #generate-btn {
+            cursor: pointer;
+        }
+		
+		.ui.header,button,input{
+		    font-family: 'Noto Sans TC', sans-serif !important;
+		}
+		
+		@media screen and (min-width: 768px) {
+			#title{
+				padding-top: 15em;
+				border-right: 1px solid #dedede;
+				padding-right: 5em;
+			}
+			
+			#title .ui.header{
+				width: 300px;
+			}
+			
+			#qrgrid{
+				padding-left: 5em;
+			}
+		}
+    </style>
+</head>
+<body>
+	<div class="main">
+		<div class="ui stackable grid">
+		<div id="title" class="eight wide column" align="center">
+			<h2 class="ui header">
+			QR Code Generator
+			<div class="sub header">No ads and it just works</div>
+			</h2>
+			<br>
+			<a href="../">Back to homepage</a>
+		</div>
+		<div id="qrgrid" class="eight wide column"  align="center">
+		<div style="width: 300px">
+			<div class="ui fluid input" style="margin-bottom: 1em;">
+			<input type="text" id="text-input" placeholder="Enter text for QR code" onkeypress="handleKeyPress(event)">
+			</div>
+			<button class="ui fluid basic button" id="generate-btn" onclick="generateQRCode()">Generate QR Code</button>
+			<div class="ui divider"></div>
+			<div id="qr-code"></div>
+			</div>
+		</div>
+		</div>
+		
+	</div>
+    <script>
+		function initQrCode(){
+			var qrcode = new QRCode(document.getElementById('qr-code'), {
+                text: "https://imuslab.com",
+                width: 300,
+                height: 300
+            });
+		}
+		initQrCode();
+        function generateQRCode() {
+            var textInput = document.getElementById('text-input').value;
+            var qrCodeDiv = document.getElementById('qr-code');
+
+            if (textInput.trim() === "") {
+                alert("Please enter text for the QR code.");
+                return;
+            }
+
+            qrCodeDiv.innerHTML = ""; // Clear previous QR code
+
+            var qrcode = new QRCode(qrCodeDiv, {
+                text: textInput,
+                width: 300,
+                height: 300
+            });
+			
+			$(qrCodeDiv).transition('fade down', function(){
+				$(qrCodeDiv).transition('fade down');
+			});
+        }
+		
+		 function handleKeyPress(event) {
+            // Check if the pressed key is Enter (key code 13)
+            if (event.key === 'Enter') {
+                generateQRCode();
+            }
+        }
+		
+    </script>
+</body>
+</html>