'use strict' const { truncateContent, postDesc } = require('../common/postDesc') const { prettyUrls } = require('hexo-util') const crypto = require('crypto') const moment = require('moment-timezone') const absoluteUrlPattern = /^(?:[a-z][a-z\d+.-]*:)?\/\//i const relativeUrlPattern = /^(\.\/|\.\.\/|\/|[^/]+\/).*$/ const colorPattern = /^(#|rgb|rgba|hsl|hsla)/i const simpleFilePattern = /\.(png|jpg|jpeg|gif|bmp|webp|svg|tiff)$/i const archiveRegex = /\/archives\// hexo.extend.helper.register('truncate', truncateContent) hexo.extend.helper.register('postDesc', data => { return postDesc(data, hexo) }) hexo.extend.helper.register('cloudTags', function (options = {}) { const env = this let { source, minfontsize, maxfontsize, limit, unit = 'px', orderby, order, page = 'tags', custom_colors } = options if (limit > 0) { source = source.limit(limit) } const sizes = [...new Set(source.map(tag => tag.length).sort((a, b) => a - b))] const sizeMap = new Map(sizes.map((size, index) => [size, index])) const length = sizes.length - 1 const getRandomColor = () => { const r = Math.floor(Math.random() * 201) const g = Math.floor(Math.random() * 201) const b = Math.floor(Math.random() * 201) return `rgb(${Math.max(r, 50)}, ${Math.max(g, 50)}, ${Math.max(b, 50)})` } const normalizeColors = input => { if (!input) return null if (typeof input === 'string') { const color = input.trim() return color ? [color] : null } if (Array.isArray(input)) { const result = [] for (let i = 0; i < input.length; i++) { const value = input[i] if (value === null || value === undefined) continue const color = String(value).trim() if (!color) continue result.push(color) } return result.length ? result : null } return null } const userColors = normalizeColors(custom_colors) const resolveColorClass = (idx) => `tag-color-${idx % userColors.length}` const generateStyle = (size, unit, page, color) => { const colorStyle = page === 'tags' ? `background-color: ${color};` : `color: ${color};` return `font-size: ${parseFloat(size.toFixed(2))}${unit}; ${colorStyle}` } return source.sort(orderby, order).map((tag, idx) => { const ratio = length ? sizeMap.get(tag.length) / length : 0 const size = minfontsize + ((maxfontsize - minfontsize) * ratio) if (userColors && userColors.length) { const colorClass = resolveColorClass(idx) const color = userColors[idx % userColors.length] const style = generateStyle(size, unit, page, color) return `${tag.name}` } const color = getRandomColor() const style = generateStyle(size, unit, page, color) return `${tag.name}` }).join('') }) hexo.extend.helper.register('urlNoIndex', function (url = null, trailingIndex = false, trailingHtml = false) { return prettyUrls(url || this.url, { trailing_index: trailingIndex, trailing_html: trailingHtml }) }) hexo.extend.helper.register('md5', function (path) { return crypto.createHash('md5').update(decodeURI(this.url_for(path, { relative: false }))).digest('hex') }) hexo.extend.helper.register('injectHtml', data => { return data ? data.join('') : '' }) hexo.extend.helper.register('findArchivesTitle', function (page, menu, date) { if (page.year) { const dateStr = page.month ? `${page.year}-${page.month}` : `${page.year}` const dateFormat = page.month ? hexo.theme.config.aside.card_archives.format : 'YYYY' return date(dateStr, dateFormat) } const defaultTitle = this._p('page.archives') if (!menu) return defaultTitle const loop = m => { for (const key in m) { if (typeof m[key] === 'object') { const result = loop(m[key]) if (result) return result } if (archiveRegex.test(m[key])) { return key } } } return loop(menu) || defaultTitle }) hexo.extend.helper.register('getBgPath', function (path) { if (!path) return '' if (colorPattern.test(path)) { return `background-color: ${path};` } else if (absoluteUrlPattern.test(path) || relativeUrlPattern.test(path) || simpleFilePattern.test(path)) { return `background-image: url(${this.url_for(path)});` } else { return `background: ${path};` } }) hexo.extend.helper.register('shuoshuoFN', (data, page) => { const { limit } = page // Shallow copy to avoid mutating original data let processedData = data.map(item => ({ ...item })) // Check if limit.value is a valid date const isValidDate = date => !isNaN(Date.parse(date)) // order by date processedData.sort((a, b) => Date.parse(b.date) - Date.parse(a.date)) // Apply number limit or time limit conditionally if (limit && limit.type === 'num' && limit.value > 0) { processedData = processedData.slice(0, limit.value) } else if (limit && limit.type === 'date' && isValidDate(limit.value)) { const limitDate = Date.parse(limit.value) processedData = processedData.filter(item => Date.parse(item.date) >= limitDate) } // This is a hack method, because hexo treats time as UTC time // so you need to manually convert the time zone processedData.forEach(item => { const utcDate = moment.utc(item.date).format('YYYY-MM-DD HH:mm:ss') item.date = moment.tz(utcDate, hexo.config.timezone).format('YYYY-MM-DD HH:mm:ss') // markdown item.content = hexo.render.renderSync({ text: item.content, engine: 'markdown' }) }) return processedData }) hexo.extend.helper.register('getPageType', (page, isHome) => { const { layout, tag, category, type, archive } = page if (layout) return layout if (tag) return 'tag' if (category) return 'category' if (archive) return 'archive' if (type) { if (type === 'tags' || type === 'categories') return type else return 'page' } if (isHome) return 'home' return 'post' }) hexo.extend.helper.register('getVersion', () => { const { version } = require('../../package.json') return { hexo: hexo.version, theme: version } }) hexo.extend.helper.register('safeJSON', data => { // Safely serialize JSON for embedding in