2023-08-20 19:46:37 +00:00
|
|
|
import { QuartzComponentConstructor, QuartzComponentProps } from "./types"
|
|
|
|
import { FullSlug, SimpleSlug, resolveRelative } from "../util/path"
|
|
|
|
import { QuartzPluginData } from "../plugins/vfile"
|
|
|
|
import { byDateAndAlphabetical } from "./PageList"
|
|
|
|
import style from "./styles/recentNotes.scss"
|
2023-08-24 15:56:40 +00:00
|
|
|
import { Date, getDate } from "./Date"
|
|
|
|
import { GlobalConfiguration } from "../cfg"
|
2023-08-20 19:46:37 +00:00
|
|
|
|
|
|
|
interface Options {
|
|
|
|
title: string
|
|
|
|
limit: number
|
|
|
|
linkToMore: SimpleSlug | false
|
|
|
|
filter: (f: QuartzPluginData) => boolean
|
|
|
|
sort: (f1: QuartzPluginData, f2: QuartzPluginData) => number
|
|
|
|
}
|
|
|
|
|
2023-08-24 15:56:40 +00:00
|
|
|
const defaultOptions = (cfg: GlobalConfiguration): Options => ({
|
2023-08-20 19:46:37 +00:00
|
|
|
title: "Recent Notes",
|
|
|
|
limit: 3,
|
|
|
|
linkToMore: false,
|
|
|
|
filter: () => true,
|
2023-08-24 15:56:40 +00:00
|
|
|
sort: byDateAndAlphabetical(cfg),
|
|
|
|
})
|
2023-08-20 19:46:37 +00:00
|
|
|
|
|
|
|
export default ((userOpts?: Partial<Options>) => {
|
2023-10-02 00:20:55 +00:00
|
|
|
function RecentNotes({ allFiles, fileData, displayClass, cfg }: QuartzComponentProps) {
|
2023-08-24 15:56:40 +00:00
|
|
|
const opts = { ...defaultOptions(cfg), ...userOpts }
|
2023-08-20 21:27:44 +00:00
|
|
|
const pages = allFiles.filter(opts.filter).sort(opts.sort)
|
2023-08-20 19:46:37 +00:00
|
|
|
const remaining = Math.max(0, pages.length - opts.limit)
|
|
|
|
return (
|
2023-10-02 00:20:55 +00:00
|
|
|
<div class={`recent-notes ${displayClass ?? ""}`}>
|
2023-08-20 19:46:37 +00:00
|
|
|
<h3>{opts.title}</h3>
|
|
|
|
<ul class="recent-ul">
|
2023-08-20 21:27:44 +00:00
|
|
|
{pages.slice(0, opts.limit).map((page) => {
|
2023-08-20 19:46:37 +00:00
|
|
|
const title = page.frontmatter?.title
|
|
|
|
const tags = page.frontmatter?.tags ?? []
|
|
|
|
|
|
|
|
return (
|
|
|
|
<li class="recent-li">
|
|
|
|
<div class="section">
|
|
|
|
<div class="desc">
|
|
|
|
<h3>
|
|
|
|
<a href={resolveRelative(fileData.slug!, page.slug!)} class="internal">
|
|
|
|
{title}
|
|
|
|
</a>
|
|
|
|
</h3>
|
|
|
|
</div>
|
|
|
|
{page.dates && (
|
|
|
|
<p class="meta">
|
2023-08-24 16:38:00 +00:00
|
|
|
<Date date={getDate(cfg, page)!} />
|
2023-08-20 19:46:37 +00:00
|
|
|
</p>
|
|
|
|
)}
|
|
|
|
<ul class="tags">
|
|
|
|
{tags.map((tag) => (
|
|
|
|
<li>
|
|
|
|
<a
|
|
|
|
class="internal tag-link"
|
|
|
|
href={resolveRelative(fileData.slug!, `tags/${tag}` as FullSlug)}
|
|
|
|
>
|
|
|
|
#{tag}
|
|
|
|
</a>
|
|
|
|
</li>
|
|
|
|
))}
|
|
|
|
</ul>
|
|
|
|
</div>
|
|
|
|
</li>
|
|
|
|
)
|
|
|
|
})}
|
|
|
|
</ul>
|
|
|
|
{opts.linkToMore && remaining > 0 && (
|
|
|
|
<p>
|
|
|
|
<a href={resolveRelative(fileData.slug!, opts.linkToMore)}>See {remaining} more →</a>
|
|
|
|
</p>
|
|
|
|
)}
|
|
|
|
</div>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
RecentNotes.css = style
|
|
|
|
return RecentNotes
|
|
|
|
}) satisfies QuartzComponentConstructor
|