v0.12.2版本以及重新修改说说&相册

前言

安装memos v0.12.2版本是因为去年部署的小程序终于被我想起来了,所以,准备重启一下该小程序.

部署memos

还是使用docker-compose的方式部署,具体的配置文件如下

services:
memos:
image: neosmemo/memos:0.12.2
container_name: memos
volumes:
- ./data:/var/opt/memos
ports:
- 5232:5230
restart: always

说说页面

新建一个页面,内容如下

<div id="bber"></div>
<script>
// 以下信息改掉
var bbMemo = {
memos : 'https://memos.loliko.cn/', //填入memos网站域名,末尾需带斜杠
limit : '10', //填入需要展示的memos数量
creatorId:'1' , //自己部署的话默认为1,不用修改
domId: '#bber', //可以不修改
username:"浪子", //修改为你自己的昵称
useravatar:"https://img.imsun.org/avatar.jpg", //修改为自己的头像链接
userlink:"https://blog.loliko.cn", //修改为你的域名
tags:"",
commentsShow:true, //没有评论功能可修改为false
commentsUrl:"https://memos.loliko.cn/m/", //修改为你的Memos域名,但保留包含m的尾巴部分
commentsTitle:"评论" //可以不修改
}
var artalkInit = {
site: "memos", //填入前面artalk设置中的站点名,没有评论功能可以不管
server:'https://artalk.loliko.cn' //填入前面artalk的网站域名,没有评论功能可以不管
}
</script>

<!-- js引用路径自己改好 -->
<link rel="stylesheet" href="https://artalk.loliko.cn/dist/Artalk.css">
<script type="text/javascript" src="https://artalk.loliko.cn/dist/Artalk.js"></script>
<script src="/memos/shuoshuo.js"></script>
<script src="https://npm.elemecdn.com/marked/marked.min.js"></script>
<script src="https://jsd.onmicrosoft.cn/gh/Tokinx/ViewImage/view-image.min.js"></script>
<script src="https://jsd.onmicrosoft.cn/gh/Tokinx/Lately/lately.min.js"></script>

shuoshuo.js的内容为

function loadCssCode(code) {
var style = document.createElement('style');
style.type = 'text/css';
style.rel = 'stylesheet';
style.innerHTML = code; // 使用 innerHTML 代替 createTextNode
var head = document.getElementsByTagName('head')[0];
head.appendChild(style);
}

var allCSS = `
#bber { margin-top: 2em; width: 90% !important; }
@media screen and (max-width: 1000px) {
#bber { margin-top: 2em; width: 100% !important; }
.entry-content li, .comment-content li, .mu_register li { margin: 0; }
}
.bb-timeline pre { color: #aaa; }
.bb-timeline ul { margin: 0; padding: 0; }
.bb-timeline ul li { list-style-type: none; margin-bottom: 1rem; }
.bb-timeline ul li .datacont ul li { margin-bottom: 0; }
.bb-timeline ul li .bb-div { padding: 0.3rem 1rem 1.5rem 1rem; border-bottom: 1px solid #E7E9EF; }
.bb-load button { width: 100% !important; font-size: 1rem; background: none; border-radius: 10px; border: 0px solid #E7E9EF; padding: 10px 30px; letter-spacing: 0.8rem; text-align: center; color: #475671; }
.bb-timeline ul li .datatime { display: flex; overflow: hidden; max-height: 100%; }
.bb-timeline ul li .datacont { margin: 10px 0px 0px 35px; padding-left: 1.2rem; min-height: 50px; }
.bb-timeline ul li .datacont img[src*='emotion'] { display: inline-block; width: auto; }
.bb-timeline ul li .datafrom { color: #aaa; font-size: 0.75em !important; font-style: italic; }
.bb-timeline ul li p { margin: 0; font-size: 16px; letter-spacing: 1px; line-height: 28px; min-height: 18px; }
.bb-timeline pre p { display: inline-block; }
.bb-timeline pre p:empty { display: none; }
.dark .bb-timeline ul li .datatime { border-color: #666; }
.dark .bb-timeline ul li .bb-div p, .dark .bb-timeline .bb-load button { color: #fafafa; }
.dark .bb-timeline ul li .bb-div p svg { fill: #fafafa; }
.dark .bb-timeline ul li .datafrom { color: #aaa; }
.datacont p { margin: 0; }
.datacont blockquote { font-family: KaiTi, STKaiti, STFangsong !important; margin: 0 0 0 1rem; padding: .25rem 2rem; position: relative; border-left: 0 none; }
.datacont blockquote::before { line-height: 2rem; content: '“'; font-family: Georgia, serif; font-size: 28px; font-weight: bold; position: absolute; left: 10px; top: 5px; }
.datacont .tag-span { color: #98c1d9; }
.datasource a { color: #fafafa; background: #3b3d42; padding: 2px 8px; margin: 0 6px 0 0; border-radius: 5px; font-size: .9rem; font-weight: 400; }
.datacont .img { cursor: pointer; max-width: 250px; max-height: 400px; border-radius: 4px; }
.datacont .img.square { height: 180px; width: 180px; object-fit: cover; }
.resimg.grid { display: grid; grid-template-columns: repeat(3, 1fr); grid-template-rows: auto; gap: 4px; width: calc(100% * 2 / 3); box-sizing: border-box; margin: 4px 0 0; }
.resimg.grid-2 { grid-template-columns: repeat(2, 1fr); width: 80%; }
.resimg.grid-4 { grid-template-columns: repeat(2, 1fr); width: calc(80% * 2 / 3); }
.resimg.grid figure.gallery-thumbnail { position: relative; width: 100%; height: 0; padding-top: 100%; cursor: zoom-in; }
.resimg figure { text-align: left; max-height: 50%; margin: 0 !important; }
.resimg figure img { max-height: 50vh; }
.resimg.grid figure, figcaption { margin: 0 !important; }
.resimg.grid figure.gallery-thumbnail > img.thumbnail-image { position: absolute; left: 0; top: 0; display: block; width: 100%; height: 100%; object-fit: cover; object-position: 50% 50%; }
#bb-footer { margin: 5rem auto 1rem; text-align: center; }
#bb-footer p { margin: 0 0 0.6rem; }
.bb-allnums { letter-spacing: 2px; }
.bb-allpub { text-decoration: none; font-style: italic; }
.bb-timeline ul li::before { content: none; }
.post-preview { max-width: 680px; height: 210px; margin: 1em auto; position: relative; display: flex; background: #fff; border-radius: 4px; box-shadow: 0 1px 2px rgba(0,0,0,.25), 0 0 1px rgba(0,0,0,.25); }
.post-preview--meta { width: 75%; padding: 25px; overflow: hidden; }
.post-preview--middle { line-height: 28px; }
.post-preview--title { font-size: 18px; margin: 0 !important; }
.post-preview--title a { text-decoration: none; }
.post-preview--date { font-size: 14px; color: #999; }
.post-preview--excerpt { font-size: 14px; line-height: 1.825; }
.post-preview--excerpt p { display: inline; margin: 0; }
.post-preview--image { object-fit: cover; height: auto; width: 25%; float: right; border-top-right-radius: 2px !important; border-bottom-right-radius: 2px !important; border-top-left-radius: 0 !important; border-bottom-left-radius: 0 !important; }
@media (max-width: 550px) {
.post-preview { width: 95%; height: 120px; }
.post-preview--meta { padding: 15px; }
.post-preview--image { height: 120px !important; }
.post-preview--excerpt { display: none; }
.post-preview--middle { line-height: 19px; }
}
.rating { display: block; line-height: 15px; }
.rating-star { display: inline-block; width: 75px; height: 15px; background-repeat: no-repeat; background-image: url(); overflow: hidden; }
.allstar10 { background-position: 0px 0px; }
.allstar9 { background-position: 0px -15px; }
.allstar8 { background-position: 0px -30px; }
.allstar7 { background-position: 0px -45px; }
.allstar6 { background-position: 0px -60px; }
.allstar5 { background-position: 0px -75px; }
.allstar4 { background-position: 0px -90px; }
.allstar3 { background-position: 0px -105px; }
.allstar2 { background-position: 0px -120px; }
.allstar1 { background-position: 0px -135px; }
.allstar0 { background-position: 0px -150px; }
.rating-average { color: #777; display: inline-block; font-size: 13px; margin-left: 10px; }
.dark .post-preview { background: #3b3d42; }
.video-wrapper { position: relative; padding-bottom: 55%; width: 100%; height: 0; }
.video-wrapper iframe { position: absolute; height: 100%; width: 100%; }
.hy-avatar { max-height: 35px; max-width: 35px; float: left; border-radius: 50%; }
.hy-intro { margin-left: 5px; margin: 0 0 0 1.2rem; }
.hy-name { font-size: 0.8em; color: #44507b !important; font-weight: 600; }
.hy-tag .hy-text-muted { color: #9b9b9b; text-align: right; font-size: 0.65em; }
.hy-time .hy-text-muted { color: #9b9b9b; text-align: right; font-size: 0.65em; }
.hy-time { display: flex; justify-content: space-between; flex-wrap: wrap; color: #9b9b9b; text-align: right; font-size: 0.65em; margin-top: 0px; }
.hy-tag { color: #9b9b9b; text-align: right; font-size: 0.65em; margin-top: 20px; }
.hy-location { color: #576b95; margin-right: 10px; }
.hy-astyle { text-decoration: none; cursor: pointer; }
.hy-tags-item { margin-right: 10px; }
.commentsLink { text-decoration: none; color: #475671; text-align: right; font-size: 12px; border-radius: 3px; padding: 4px 10px; font-weight: 400; }
.d-none { display: none; }
.comment { margin-top: 19px; }
`;

loadCssCode(allCSS);

var limit = bbMemo.limit;
var memos = bbMemo.memos;
var page = 1,
offset = 0,
nextLength = 0,
nextDom = '';
var bbDom = document.querySelector(bbMemo.domId);
var load = '<div class="bb-load"><button class="load-btn button-load">加载中…</button></div>';

if (bbDom) {
getFirstList();
var btn = document.querySelector("button.button-load");
btn.addEventListener("click", function () {
btn.textContent = '加载中…';
updateHTMl(nextDom);
if (nextLength < limit) {
btn.remove(); // 使用 btn 引用而不是查询选择器
return;
}
getNextList();
});
}

function getFirstList() {
bbDom.insertAdjacentHTML('afterend', load);
var bbUrl = memos + "api/memo?creatorId=" + bbMemo.creatorId + "&rowStatus=NORMAL&limit=" + limit;

fetch(bbUrl)
.then(res => {
if (!res.ok) throw new Error('Network response was not ok');
return res.json();
})
.then(resdata => {
updateHTMl(resdata.data);
var nowLength = resdata.data.length;
if (nowLength < limit) {
document.querySelector("button.button-load").remove();
return;
}
page++;
offset = limit * (page - 1);
getNextList();
})
.catch(error => {
console.error('Fetch error:', error);
});
}

function getNextList() {
var bbUrl = memos + "api/memo?creatorId=" + bbMemo.creatorId + "&rowStatus=NORMAL&limit=" + limit + "&offset=" + offset;

fetch(bbUrl)
.then(res => {
if (!res.ok) throw new Error('Network response was not ok');
return res.json();
})
.then(resdata => {
// 检查数据结构
if (resdata.data && Array.isArray(resdata.data)) {
nextDom = resdata.data;
} else {
throw new Error('Unexpected data structure');
}

nextLength = nextDom.length;
page++;
offset = limit * (page - 1);

if (nextLength < 1) {
document.querySelector("button.button-load").remove();
return;
}

// 处理 nextDom 数据
updateHTMl(nextDom);
})
.catch(error => {
console.error('Fetch error:', error);
});

Artalk.loadCountWidget({
server: artalkInit.server,
site: artalkInit.site,
countEl: '#artalkCount'
});
}

function updateHTMl(data) {
var result = "",
resultAll = "";
const TAG_REG = /#([^\s#]+?) /g,
BILIBILI_REG = /<a.*?href="https:\/\/www\.bilibili\.com\/video\/((av[\d]{1,10})|(BV([\w]{10})))\/?".*?>.*<\/a>/g,
NETEASE_MUSIC_REG = /<a.*?href="https:\/\/music\.163\.com\/.*id=([0-9]+)".*?>.*<\/a>/g,
QQMUSIC_REG = /<a.*?href="https\:\/\/y\.qq\.com\/.*(\/[0-9a-zA-Z]+)(\.html)?".*?>.*?<\/a>/g,
QQVIDEO_REG = /<a.*?href="https:\/\/v\.qq\.com\/.*\/([a-z|A-Z|0-9]+)\.html".*?>.*<\/a>/g,
YOUKU_REG = /<a.*?href="https:\/\/v\.youku\.com\/.*\/id_([a-z|A-Z|0-9|==]+)\.html".*?>.*<\/a>/g,
YOUTUBE_REG = /<a.*?href="https:\/\/www\.youtube\.com\/watch\?v\=([a-z|A-Z|0-9]{11})\".*?>.*<\/a>/g;

marked.setOptions({
breaks: true,
smartypants: true,
langPrefix: 'language-'
});

for (var i = 0; i < data.length; i++) {
var bbContREG = data[i].content.replace(TAG_REG, "<span class='tag-span'>#$1</span> ");
bbContREG = marked.parse(bbContREG)
.replace(BILIBILI_REG, "<div class='video-wrapper'><iframe src='//player.bilibili.com/player.html?bvid=$1&as_wide=1&high_quality=1&danmaku=0' scrolling='no' border='0' frameborder='no' framespacing='0' allowfullscreen='true'></iframe></div>")
.replace(NETEASE_MUSIC_REG, "<meting-js auto='https://music.163.com/#/song?id=$1'></meting-js>")
.replace(QQMUSIC_REG, "<meting-js auto='https://y.qq.com/n/yqq/song$1.html'></meting-js>")
.replace(QQVIDEO_REG, "<div class='video-wrapper'><iframe src='//v.qq.com/iframe/player.html?vid=$1' allowFullScreen='true' frameborder='no'></iframe></div>")
.replace(YOUKU_REG, "<div class='video-wrapper'><iframe src='https://player.youku.com/embed/$1' frameborder=0 'allowfullscreen'></iframe></div>")
.replace(YOUTUBE_REG, "<div class='video-wrapper'><iframe src='https://www.youtube.com/embed/$1' title='YouTube video player' frameborder='0' allow='accelerometer; autoplay; clipboard-write; encrypted-media; gyroscope; picture-in-picture' allowfullscreen title='YouTube Video'></iframe></div>");

// 提取内容中的图片
var contentImages = bbContREG.match(/<img.*?src="([^"]+)".*?>/g) || [];
var imgUrl = '';
var resImgLength = 0;

// 提取附件列表中的图片
if (data[i].resourceList && data[i].resourceList.length > 0) {
var resourceList = data[i].resourceList;
for (var j = 0; j < resourceList.length; j++) {
var restype = resourceList[j].type.slice(0, 5);
if (restype == 'image') {
// 检查 externalLink 是否为空
var imageUrl = resourceList[j].externalLink || (memos + 'o/r/' + resourceList[j].id + '/' + resourceList[j].publicId + '/' + resourceList[j].filename);
imgUrl += '<figure class="gallery-thumbnail"><img class="img thumbnail-image" src="' + imageUrl + '"/></figure>';
resImgLength++;
}
}
}

// 将内容中的图片添加到图片列表中
contentImages.forEach(imgTag => {
var imageUrl = imgTag.match(/src="([^"]+)"/)[1];
imgUrl += '<figure class="gallery-thumbnail"><img class="img thumbnail-image" src="' + imageUrl + '"/></figure>';
resImgLength++;
});

// 清除 content 中的图片
var cleanedContent = bbContREG.replace(/<img.*?>/g, '');

// 生成图片列表
if (imgUrl) {
var resImgGrid = resImgLength !== 1 ? "grid grid-" + resImgLength : "";
cleanedContent += '<div class="resimg ' + resImgGrid + '">' + imgUrl + '</div>';
}

// 原有的列表项生成逻辑
result += '<li class="bb-list-li"><div class="bb-div"><div class="datatime"><div class="hy-avatar-block"><a href="' + bbMemo.userlink + '" class="hy-astyle"><img src="' + bbMemo.useravatar + '" class="hy-avatar"></a></div><div class="hy-intro"><div class="hy-name">' + bbMemo.username + '</div><div><span class="hy-time hy-text-muted">' + new Date(data[i].createdTs * 1000).toLocaleString() + '</span></div></div></div><div class="datacont"><div>' + cleanedContent + '</div></div><div class="hy-tag hy-text-muted"><span class="hy-location">' + (bbMemo.location == undefined ? "" : bbMemo.location) + '</span><span class="hy-tags-item">' + (bbMemo.tags == undefined ? "" : bbMemo.tags) + '</span><span><a data-id="' + data[i].id + '" data-site="' + artalkInit.site + '" data-server="' + artalkInit.server + '" class="commentsLink" onclick="loadArtalk(this)">' + (bbMemo.commentsShow ? bbMemo.commentsTitle : "") + ' ' + '<span id="artalkCount" data-page-key="/m/' + data[i].id + '"></span></a ></span></div><div id="' + data[i].id + '" class="comment d-none"></div></div></li>';
}

var bbBefore = "<section class='bb-timeline'><ul class='bb-list-ul'>";
var bbAfter = "</ul></section>";
resultAll = bbBefore + result + bbAfter;
bbDom.insertAdjacentHTML('beforeend', resultAll);
fetchDB();
document.querySelector('button.button-load').innerHTML = '<div class="post-button"><span class="btn">加载更多</span></div>';
window.ViewImage && ViewImage.init('.datacont img');
window.Lately && Lately.init({
target: '.datatime'
});
}

function loadArtalk(e) {
let id = e.getAttribute("data-id"),
site = e.getAttribute("data-site"),
server = e.getAttribute("data-server");
let artalkDom = document.getElementById(`${id}`);
let artalkCon = "<div id='artalk'></div>";

if (artalkDom && artalkDom.classList.contains('d-none')) {
document.querySelectorAll('.comment').forEach((item) => {
item.classList.add('d-none');
});

if (document.getElementById("artalk")) {
document.getElementById("artalk").remove();
}

artalkDom.insertAdjacentHTML('beforeend', artalkCon);
artalkDom.classList.remove('d-none');

Artalk.init({
el: '#artalk',
pageKey: '/m/' + id,
pageTitle: '',
site: site,
server: server,
emoticons: false
});
} else {
artalkDom.classList.add('d-none');
if (document.getElementById("artalk")) {
document.getElementById("artalk").remove();
}
}
}

function fetchDB() {
var dbAPI = "https://cors.ow3.cn/https://api.loliko.cn/";
var dbA = document.querySelectorAll(".bb-timeline a[href*='douban.com/subject/']:not([rel='noreferrer'])") || '';

if (dbA) {
for (let i = 0; i < dbA.length; i++) {
let _this = dbA[i];
var dbHref = _this.href;
var db_reg = /^https\:\/\/(movie|book)\.douban\.com\/subject\/([0-9]+)\/?/;
var db_type = dbHref.replace(db_reg, "$1");
var db_id = dbHref.replace(db_reg, "$2").toString();

if (db_type == 'movie') {
var this_item = 'movie' + db_id;
var url = dbAPI + "movies/" + db_id;

if (localStorage.getItem(this_item) === null) {
fetch(url)
.then(res => res.json())
.then(data => {
let fetch_item = 'movies' + data.sid;
let fetch_href = "https://movie.douban.com/subject/" + data.sid + "/";
localStorage.setItem(fetch_item, JSON.stringify(data));
movieShow(fetch_href, fetch_item);
});
} else {
movieShow(dbHref, this_item);
}
} else if (db_type == 'book') {
var this_item = 'book' + db_id;
var url = dbAPI + "v2/book/id/" + db_id;

if (localStorage.getItem(this_item) === null) {
fetch(url)
.then(res => res.json())
.then(data => {
let fetch_item = 'book' + data.id;
let fetch_href = "https://book.douban.com/subject/" + data.id + "/";
localStorage.setItem(fetch_item, JSON.stringify(data));
bookShow(fetch_href, fetch_item);
});
} else {
bookShow(dbHref, this_item);
}
}
}
}
}

function movieShow(fetch_href, fetch_item) {
var storage = localStorage.getItem(fetch_item);
var data = JSON.parse(storage);
var db_star = Math.ceil(data.rating);
var db_html = "<div class='post-preview'><div class='post-preview--meta'><div class='post-preview--middle'><h4 class='post-preview--title'><a target='_blank' rel='noreferrer' href='" + fetch_href + "'>《" + data.name + "》</a></h4><div class='rating'><div class='rating-star allstar" + db_star + "'></div><div class='rating-average'>" + data.rating + "</div></div><time class='post-preview--date'>导演:" + data.director + " / 类型:" + data.genre + " / " + data.year + "</time><section style='max-height:75px;overflow:hidden;' class='post-preview--excerpt'>" + data.intro.replace(/\s*/g, "") + "</section></div></div><img referrer-policy='no-referrer' loading='lazy' class='post-preview--image' src=" + data.img + "></div>";
var db_div = document.createElement("div");
var qs_href = ".bb-timeline a[href='" + fetch_href + "']";
var qs_dom = document.querySelector(qs_href);

if (qs_dom) {
qs_dom.parentNode.replaceChild(db_div, qs_dom);
db_div.innerHTML = db_html;
}
}

function bookShow(fetch_href, fetch_item) {
var storage = localStorage.getItem(fetch_item);
var data = JSON.parse(storage);
var db_star = Math.ceil(data.rating.average);
var db_html = "<div class='post-preview'><div class='post-preview--meta'><div class='post-preview--middle'><h4 class='post-preview--title'><a target='_blank' rel='noreferrer' href='" + fetch_href + "'>《" + data.title + "》</a></h4><div class='rating'><div class='rating-star allstar" + db_star + "'></div><div class='rating-average'>" + data.rating.average + "</div></div><time class='post-preview--date'>作者:" + data.author + " </time><section style='max-height:75px;overflow:hidden;' class='post-preview--excerpt'>" + data.summary.replace(/\s*/g, "") + "</section></div></div><img referrer-policy='no-referrer' loading='lazy' class='post-preview--image' src=" + data.images.medium + "></div>";
var db_div = document.createElement("div");
var qs_href = ".bb-timeline a[href='" + fetch_href + "']";
var qs_dom = document.querySelector(qs_href);

if (qs_dom) {
qs_dom.parentNode.replaceChild(db_div, qs_dom);
db_div.innerHTML = db_html;
}
}

在memos中使用Aratalk评论

memos的自定义代码中插入

// 用 JS 向页面中插入 JS
function addArtalkJS() {
var memosArtalk = document.createElement("script");
memosArtalk.src = `https://artalk.loliko.cn/dist/Artalk.js`;
//memosArtalk.defer = true;
var artakPos = document.getElementsByTagName("script")[0];
artakPos.parentNode.insertBefore(memosArtalk, artakPos);
};
// div
function startArtalk() {
start = setInterval(function(){
var artalkDom = document.getElementById('Comments') || '';
var memoAt = document.querySelector('.memo-container') || '';
var memoLoading = document.querySelector('.action-button-container') || '';
var memoLoadingA = document.querySelector('.action-button-container a') || '';
if(window.location.href.replace(/^.*\/(m)\/.*$/,'$1') == "m" && memoLoadingA){
memoLoading.innerHTML = "评论加载中……"
}
if(window.location.href.replace(/^.*\/(m)\/.*$/,'$1') == "m" && !artalkDom){
addArtalkJS()
if(memoAt){
clearInterval(start)
var cssLink = document.createElement("link");
cssLink.rel = "stylesheet";
cssLink.href = "https://artalk.loliko.cn/dist/Artalk.css";
document.head.appendChild(cssLink);
memoAt.insertAdjacentHTML('afterend', '<div id="Comments"></div>');
setTimeout(function() {
Artalk.init({
el: '#Comments',
server: 'https://artalk.loliko.cn',
site: 'memos',
darkMode: 'auto'
});
Artalk.on('list-loaded', function() {
// console.log('评论加载完成');
memoLoading.innerHTML = ''
startArtalk();
});
}, 1000);
}
}
//console.log(window.location.href);
}, 1000)
}
startArtalk();

相册页面

新建js文件

let currentPage = 1;
const imagesPerPage = 6;
let allImages = [];

function loadImages(page) {
const url = 'https://memos.loliko.cn'; // 修改api
const startIndex = (page - 1) * imagesPerPage;
const endIndex = startIndex + imagesPerPage;

fetch(url + '/api/memo?creatorId=1&tag=照片')
.then(res => res.json())
.then(data => {
if (!Array.isArray(data.data)) {
throw new Error('Expected data.data to be an array');
}
let imgs = [];
data.data.forEach(item => {
let ls = item.content.match(/\!\[.*?\]\(.*?\)/g);
if (ls) imgs = imgs.concat(ls);
if (item.resourceList && item.resourceList.length) {
item.resourceList.forEach(t => {
if (t.externalLink) imgs.push(`![](${t.externalLink})`);
else imgs.push(`![](${url}/o/r/${t.id}/${t.publicId}/${t.filename})`);
});
}
});

allImages = imgs;
displayImages(startIndex, endIndex);
})
.catch(error => console.error('Error fetching images:', error));
}

function displayImages(startIndex, endIndex) {
let html = '';
const imagesToDisplay = allImages.slice(startIndex, endIndex);

imagesToDisplay.forEach(item => {
let img = item.replace(/!\[.*?\]\((.*?)\)/g, '$1'),
time, title, tat = item.replace(/!\[(.*?)\]\(.*?\)/g, '$1');
if (tat.indexOf(' ') != -1) {
[time, title] = tat.split(' ');
} else {
title = tat;
}

html += `<div class="picture-container">
<a target="_blank" rel="noopener"
href="${img}" data-lightbox="${time}" data-title="${title}">
<picture>
<img src="${img}" class="img-thumbnail" loading="lazy" alt="${title}">
</picture>
</a>
</div>`;
});

document.querySelector('.gallery').innerHTML += html;
if (endIndex < allImages.length) {
if (!document.getElementById('load-more')) {
const loadMoreButton = document.createElement('button');
loadMoreButton.id = 'load-more';
loadMoreButton.className = 'post-button';
loadMoreButton.innerHTML = '<span class="btn">加载更多</span>';
loadMoreButton.onclick = loadNextPage;
document.querySelector('.gallery').after(loadMoreButton);
}
} else {
const loadMoreButton = document.getElementById('load-more');
if (loadMoreButton) {
loadMoreButton.remove();
}
}
}

function loadNextPage() {
const loadMoreButton = document.getElementById('load-more');
loadMoreButton.classList.add('loading');

currentPage++;
const startIndex = (currentPage - 1) * imagesPerPage;
const endIndex = startIndex + imagesPerPage;

// 模拟加载延迟
setTimeout(() => {
displayImages(startIndex, endIndex);
loadMoreButton.classList.remove('loading');
}, 1000); // 1秒延迟,你可以根据实际加载时间调整
}

// 初始加载
document.addEventListener('DOMContentLoaded', () => {
loadImages(currentPage);
});

新建CSS文件,内容为

.inner .gallery {
width: 100%;
display: flex;
flex-flow: wrap;
gap: 10px 20px;
padding-bottom: 30px;
}

.inner .gallery .picture-container {
min-width: 200px;
flex: 0 0 calc(33.333% - 13.333px);
aspect-ratio: 1 / 1; /* 使容器保持正方形 */
}

@media screen and (max-width: 736px) {
.inner .gallery .picture-container {
flex: 0 0 calc(50% - 20px);
}
}

@media screen and (max-width: 480px) {
.inner .gallery .picture-container {
flex: 0 0 100%;
}
}

.inner .gallery .picture-container a {
border: none;
display: block;
width: 100%;
height: 100%;
}

.inner .gallery .picture-container a .img-thumbnail {
border-radius: 4px;
width: 100%;
height: 100%;
object-fit: cover; /* 确保图片填充整个容器 */
object-position: center; /* 居中裁剪 */
}


@keyframes spin {
to { transform: rotate(360deg); }
}

#load-more {
display: block;
margin: 20px auto; /* 根据需要调整上边距 */
outline: none; /* 移除外框线 */
border: 0px;
width: 100% !important;
background-color: transparent; /* 完全透明 */
}

新建index.md 文件

引用

<link href="https://cdn.staticfile.org/lightbox2/2.11.3/css/lightbox.min.css" rel="stylesheet">
<script src="https://cdn.staticfile.org/lightbox2/2.11.3/js/lightbox-plus-jquery.min.js" charset="utf-8"></script>
<div class="inner">
<div class="gallery"></div>
</div>
<link href="/photos/photos.css" rel="stylesheet">
<script defer src="/photos/photos.js"></script>