How did i create a directory for my blogs
Motivation
I did not find any useful directory page for static sites like Jekyll. And I definitely find listing all blogs In chronological order is not handy, so I decided to implement one.
Idea
First generate html page from structures of the folders of blogs. This is possible since Jekyll supports iterating through subfolders and finding all posts now. Then use JavaScript to modify html page to what I want.
Result
You can find the demo here. I use jQuery and Bootstrap to ease my coding, so don’t forget to import them.
Code
Only three files were created for the directory, one html file, one JavaScript file and one CSS file.
Reading the posts in the folder _posts and put the path of them in the html page:
<div id="loadfiles">
<span class="post-direct">/posts/Learning/Review/Advertising/2024-10-17-Game%20Theory/</span>
<span class="post-direct">/posts/Learning/Review/Advertising/2024-08-05-Computational%20Advertising/</span>
<span class="post-direct">/posts/Learning/Review/2024-05-27-%E9%A9%BE%E8%80%83%E6%80%BB%E7%BB%93/</span>
<span class="post-direct">/posts/Coding/Python/2024-5-19-%E4%B8%AD%E6%96%87%E5%B0%8F%E8%AF%B4%20tts%20%E7%94%9F%E6%88%90%E9%9F%B3%E8%A7%86%E9%A2%91/</span>
<span class="post-direct">/posts/Learning/Review/Recommendation/2024-04-28-Some%20Key%20Points%20in%20Recommendation/</span>
<span class="post-direct">/posts/Learning/Review/Recommendation/2024-04-02-Multi-objective%20Fusion%20in%20Recommendation%20System/</span>
<span class="post-direct">/posts/Coding/Python/2020-8-11-SPAV%20algorithm%20(Python%E4%BB%A3%E7%A0%81%E5%AE%9E%E7%8E%B0)/</span>
<span class="post-direct">/posts/Learning/Review/2020-07-24-Math/</span>
<span class="post-direct">/posts/Coding/Python/2020-7-24-%E6%9D%A8%E8%B6%85%E8%B6%8A%E5%90%A72018%E5%88%91%E4%BE%A6%E6%8E%A8%E7%90%86%E9%A2%98%E7%9A%84Python%E8%A7%A3%E6%B3%95/</span>
<span class="post-direct">/posts/Coding/Python/2020-07-24-Python%20cheat%20sheet%202/</span>
<span class="post-direct">/posts/Coding/Python/2020-6-30-%E8%8E%B7%E5%8F%96B%E7%AB%99%E8%A7%86%E9%A2%91%E5%90%88%E9%9B%86%E5%8F%8A%E7%95%AA%E5%89%A7%E7%9A%84%E5%BC%B9%E5%B9%95%E5%88%86%E5%B8%83%E7%BB%9F%E8%AE%A1/</span>
<span class="post-direct">/posts/Coding/Python/2020-5-22-%E6%A0%B9%E6%8D%AEB%E7%AB%99%E8%AF%84%E5%88%86%E6%8E%92%E5%BA%8F%E7%94%B5%E8%A7%86%E5%89%A7%EF%BC%8C%E7%94%B5%E5%BD%B1%EF%BC%8C%E7%BA%AA%E5%BD%95%E7%89%87/</span>
<span class="post-direct">/posts/Coding/Python/2020-1-10-finetune%20gpt2%E7%94%9F%E6%88%90%E4%B8%AD%E6%96%87%E6%AD%8C%E8%AF%8D/</span>
<span class="post-direct">/posts/Handbook/2019-11-27-Pycharm%20common%20shortcuts/</span>
<span class="post-direct">/posts/Handbook/2019-11-07-Docker%20and%20docker-airflow/</span>
<span class="post-direct">/posts/Learning/Reading/2019-11-01-%E6%89%80%E8%AF%BB%E4%B9%A6%E7%B1%8D%E7%9A%84%E8%AF%84%E5%88%86/</span>
<span class="post-direct">/posts/Handbook/2019-10-23-Python%20and%20pip/</span>
<span class="post-direct">/posts/Coding/JS-HTML-CSS/2019-9-19-%E6%88%91%E5%86%99%E4%BA%86%E4%B8%80%E4%B8%AA%E7%BE%8E%E5%9B%BE%E6%89%93%E5%88%86%EF%BC%8C%E5%AE%A1%E7%BE%8E%E8%AF%84%E6%B5%8B%E7%9A%84%E7%BD%91%E7%AB%99/</span>
<span class="post-direct">/posts/Coding/Python/2019-8-30-%E5%A6%82%E4%BD%95%E7%94%A8%E5%BE%AE%E5%8D%9Aapi%E6%97%A0%E4%BC%A4%E6%80%BC%E9%BB%91%E5%AD%90/</span>
<span class="post-direct">/posts/Coding/Python/2019-7-16-How%20I%20create%20gists%20and%20write%20them%20to%20markdown%20table/</span>
<span class="post-direct">/posts/Learning/Review/2019-06-03-Machine%20learning/</span>
<span class="post-direct">/posts/Learning/Review/2019-06-01-Algorithms/</span>
<span class="post-direct">/posts/Coding/Python/2019-1-24-Python%20One%20Liners%20for%20Coding%20Problems/</span>
<span class="post-direct">/posts/Coding/Python/2019-1-18-A%20Solution%20to%20the%20Egg%20Drop%20Problem/</span>
<span class="post-direct">/posts/Coding/Python/2018-7-6-Ascii%20Art%20-%20from%20image%20to%20characters/</span>
<span class="post-direct">/posts/Coding/Python/2018-5-24-Python%20cheat%20sheet/</span>
<span class="post-direct">/posts/Coding/Python/2017-11-29-Solution%20to%20LeetCode%20No.52%20N-Queens%20II/</span>
<span class="post-direct">/posts/Learning/RWTH/2017-11-03-Summaries%20for%20Courses%20I've%20Taken/</span>
<span class="post-direct">/posts/Coding/Python/2017-10-27-How%20to%20select%20exam%20dates%20automatically/</span>
<span class="post-direct">/posts/Coding/Python/2017-8-17-A%20Spell%20Checker%20for%20German%20Courses/</span>
<span class="post-direct">/posts/Coding/JS-HTML-CSS/2017-6-14-How%20Did%20I%20Create%20a%20Directory%20for%20My%20Blogs/</span>
</div>
We will get something like:
<div id="loadfiles">
<span class="post-direct">/posts/Learning/RWTH/semester1/2017-4-6-Test4/</span>
<span class="post-direct">/posts/Coding/Python/2017-4-6-Test4/</span>
<span class="post-direct">/posts/Coding/Python/2017-4-6-Test4%20-%20Copy/</span>
<span class="post-direct">/posts/Coding/Python/2017-4-6-Test4%20-%20Copy%20(2)/</span>
</div>
The paths are shown in <span>
s, the last element is the filename while the others are the subfolders’ name. Spaces in the filename and folder name are changed into %20.
Note that empty subfolders in the _posts will not show up, since Jekyll only iterates through md files.
Then we use JavaScript to modify the page when document is ready, the first step is to construct a tree from the URL of posts:
function getDirectoryStructure() {
var collection = [];
var tree = new Tree();
tree.add('0posts');
$(".post-direct").each(function(i, obj) {
var path = $(this).text();
path = path.substring(1, path.length - 1);
var strArray = path.split('/');
for (i = 1; i < strArray.length; i++) {
if (i == (strArray.length - 1)) {
tree.add(i + path, (i - 1) + strArray[i - 1]);
} else {
//since add to tree does not check multiples, so may result in many duplicate nodes
if (!collection.includes(i + strArray[i])) {
tree.add(i + strArray[i], (i - 1) + strArray[i - 1]);
collection.push(i + strArray[i]);
}
}
}
});
return tree;
}
I did not implement the tree myself, just using a piece of code from GitHub.
However, since I have to visit the nodes in a recursive way, so I can’t use BFS or DFS provided by the tree structure. But visit the nodes as below:
function recurseTree(node) {
var out = "";
if (node) {
if (node.children.length == 0) {
var strArray = node.data.split('/');
out = '<li><a href=\"../' + node.data.substring(1) + '\">' + decodeURI(strArray[strArray.length - 1]).replace(/[0-9]+-[0-9]+-[0-9]+-/g, "") + '</a></li>';
} else {
for (var i = 0; i < node.children.length; i++) {
out += recurseTree(node.children[i]);
}
if (node.data.substring(1) != 'posts') {
out = '<li class=\"indent-li\"><label class=\"tree-toggler nav-header\">' + decodeURI(node.data.substring(1)) + '</label><ul class=\"nav nav-list tree\">' + out + '</ul></li>';
}
}
}
return out;
}
A little refinement
Since the visitors can notice changing of content of html, I added a loader to the page.
It is a simple div in html page:
<div id="loader"></div>
The corresponding CSS is :
#loader {
position: absolute;
left: 50%;
top: 50%;
z-index: 1;
margin: -40px 0 0 -40px;
border: 16px solid #f3f3f3;
border-radius: 50%;
border-top: 16px solid #3498db;
width: 80px;
height: 80px;
-webkit-animation: spin 0.8s linear infinite;
animation: spin 0.8s linear infinite;
}
@-webkit-keyframes spin {
0% {
-webkit-transform: rotate(0deg);
}
100% {
-webkit-transform: rotate(360deg);
}
}
@keyframes spin {
0% {
transform: rotate(0deg);
}
100% {
transform: rotate(360deg);
}
}
Add CSS style and class to the <div>
for path:
<div id="loadfiles" style="display:none;" class="animate-bottom">
<span class="post-direct">/posts/Learning/Review/Advertising/2024-10-17-Game%20Theory/</span>
<span class="post-direct">/posts/Learning/Review/Advertising/2024-08-05-Computational%20Advertising/</span>
<span class="post-direct">/posts/Learning/Review/2024-05-27-%E9%A9%BE%E8%80%83%E6%80%BB%E7%BB%93/</span>
<span class="post-direct">/posts/Coding/Python/2024-5-19-%E4%B8%AD%E6%96%87%E5%B0%8F%E8%AF%B4%20tts%20%E7%94%9F%E6%88%90%E9%9F%B3%E8%A7%86%E9%A2%91/</span>
<span class="post-direct">/posts/Learning/Review/Recommendation/2024-04-28-Some%20Key%20Points%20in%20Recommendation/</span>
<span class="post-direct">/posts/Learning/Review/Recommendation/2024-04-02-Multi-objective%20Fusion%20in%20Recommendation%20System/</span>
<span class="post-direct">/posts/Coding/Python/2020-8-11-SPAV%20algorithm%20(Python%E4%BB%A3%E7%A0%81%E5%AE%9E%E7%8E%B0)/</span>
<span class="post-direct">/posts/Learning/Review/2020-07-24-Math/</span>
<span class="post-direct">/posts/Coding/Python/2020-7-24-%E6%9D%A8%E8%B6%85%E8%B6%8A%E5%90%A72018%E5%88%91%E4%BE%A6%E6%8E%A8%E7%90%86%E9%A2%98%E7%9A%84Python%E8%A7%A3%E6%B3%95/</span>
<span class="post-direct">/posts/Coding/Python/2020-07-24-Python%20cheat%20sheet%202/</span>
<span class="post-direct">/posts/Coding/Python/2020-6-30-%E8%8E%B7%E5%8F%96B%E7%AB%99%E8%A7%86%E9%A2%91%E5%90%88%E9%9B%86%E5%8F%8A%E7%95%AA%E5%89%A7%E7%9A%84%E5%BC%B9%E5%B9%95%E5%88%86%E5%B8%83%E7%BB%9F%E8%AE%A1/</span>
<span class="post-direct">/posts/Coding/Python/2020-5-22-%E6%A0%B9%E6%8D%AEB%E7%AB%99%E8%AF%84%E5%88%86%E6%8E%92%E5%BA%8F%E7%94%B5%E8%A7%86%E5%89%A7%EF%BC%8C%E7%94%B5%E5%BD%B1%EF%BC%8C%E7%BA%AA%E5%BD%95%E7%89%87/</span>
<span class="post-direct">/posts/Coding/Python/2020-1-10-finetune%20gpt2%E7%94%9F%E6%88%90%E4%B8%AD%E6%96%87%E6%AD%8C%E8%AF%8D/</span>
<span class="post-direct">/posts/Handbook/2019-11-27-Pycharm%20common%20shortcuts/</span>
<span class="post-direct">/posts/Handbook/2019-11-07-Docker%20and%20docker-airflow/</span>
<span class="post-direct">/posts/Learning/Reading/2019-11-01-%E6%89%80%E8%AF%BB%E4%B9%A6%E7%B1%8D%E7%9A%84%E8%AF%84%E5%88%86/</span>
<span class="post-direct">/posts/Handbook/2019-10-23-Python%20and%20pip/</span>
<span class="post-direct">/posts/Coding/JS-HTML-CSS/2019-9-19-%E6%88%91%E5%86%99%E4%BA%86%E4%B8%80%E4%B8%AA%E7%BE%8E%E5%9B%BE%E6%89%93%E5%88%86%EF%BC%8C%E5%AE%A1%E7%BE%8E%E8%AF%84%E6%B5%8B%E7%9A%84%E7%BD%91%E7%AB%99/</span>
<span class="post-direct">/posts/Coding/Python/2019-8-30-%E5%A6%82%E4%BD%95%E7%94%A8%E5%BE%AE%E5%8D%9Aapi%E6%97%A0%E4%BC%A4%E6%80%BC%E9%BB%91%E5%AD%90/</span>
<span class="post-direct">/posts/Coding/Python/2019-7-16-How%20I%20create%20gists%20and%20write%20them%20to%20markdown%20table/</span>
<span class="post-direct">/posts/Learning/Review/2019-06-03-Machine%20learning/</span>
<span class="post-direct">/posts/Learning/Review/2019-06-01-Algorithms/</span>
<span class="post-direct">/posts/Coding/Python/2019-1-24-Python%20One%20Liners%20for%20Coding%20Problems/</span>
<span class="post-direct">/posts/Coding/Python/2019-1-18-A%20Solution%20to%20the%20Egg%20Drop%20Problem/</span>
<span class="post-direct">/posts/Coding/Python/2018-7-6-Ascii%20Art%20-%20from%20image%20to%20characters/</span>
<span class="post-direct">/posts/Coding/Python/2018-5-24-Python%20cheat%20sheet/</span>
<span class="post-direct">/posts/Coding/Python/2017-11-29-Solution%20to%20LeetCode%20No.52%20N-Queens%20II/</span>
<span class="post-direct">/posts/Learning/RWTH/2017-11-03-Summaries%20for%20Courses%20I've%20Taken/</span>
<span class="post-direct">/posts/Coding/Python/2017-10-27-How%20to%20select%20exam%20dates%20automatically/</span>
<span class="post-direct">/posts/Coding/Python/2017-8-17-A%20Spell%20Checker%20for%20German%20Courses/</span>
<span class="post-direct">/posts/Coding/JS-HTML-CSS/2017-6-14-How%20Did%20I%20Create%20a%20Directory%20for%20My%20Blogs/</span>
</div>
Note that when you want to change the size of the loader, change “margin-top” and “margin-left” to half of its size, so as to make the loader exactly in the center of the screen.
Also, do not forget to change the visibility of the <div>
s afterwards:
function showPage() {
document.getElementById("loader").style.display = "none";
document.getElementById("placeHolder").style.display = "none";
document.getElementById("loadfiles").style.display = "block";
}