2023-12-17 12:27:19 -08:00
import { slug as slugAnchor } from "github-slugger"
2023-12-02 16:55:38 -08:00
import type { Element as HastElement } from "hast"
2023-12-20 09:52:17 -08:00
import rfdc from "rfdc"
2023-12-27 16:44:14 -08:00
export const clone = rfdc ( )
2023-12-20 09:52:17 -08:00
2023-08-13 17:47:07 -07:00
// this file must be isomorphic so it can't use node libs (e.g. path)
2023-05-30 08:02:20 -07:00
2023-08-12 21:16:34 -07:00
export const QUARTZ = "quartz"
2023-07-13 00:19:35 -07:00
/// Utility type to simulate nominal types in TypeScript
type SlugLike < T > = string & { __brand : T }
2023-08-19 15:52:25 -07:00
/** Cannot be relative and must have a file extension. */
export type FilePath = SlugLike < "filepath" >
export function isFilePath ( s : string ) : s is FilePath {
const validStart = ! s . startsWith ( "." )
return validStart && _hasFileExtension ( s )
2023-07-13 00:19:35 -07:00
}
2023-08-19 15:52:25 -07:00
/** Cannot be relative and may not have leading or trailing slashes. It can have `index` as it's last segment. Use this wherever possible is it's the most 'general' interpretation of a slug. */
export type FullSlug = SlugLike < "full" >
export function isFullSlug ( s : string ) : s is FullSlug {
2023-07-15 23:02:12 -07:00
const validStart = ! ( s . startsWith ( "." ) || s . startsWith ( "/" ) )
2023-08-19 15:52:25 -07:00
const validEnding = ! s . endsWith ( "/" )
2024-02-11 10:57:24 -08:00
return validStart && validEnding && ! containsForbiddenCharacters ( s )
2023-08-19 15:52:25 -07:00
}
/** Shouldn't be a relative path and shouldn't have `/index` as an ending or a file extension. It _can_ however have a trailing slash to indicate a folder path. */
export type SimpleSlug = SlugLike < "simple" >
export function isSimpleSlug ( s : string ) : s is SimpleSlug {
2023-12-02 16:52:44 -08:00
const validStart = ! ( s . startsWith ( "." ) || ( s . length > 1 && s . startsWith ( "/" ) ) )
2024-02-11 10:57:24 -08:00
const validEnding = ! endsWith ( s , "index" )
return validStart && ! containsForbiddenCharacters ( s ) && validEnding && ! _hasFileExtension ( s )
2023-07-13 00:19:35 -07:00
}
2023-08-19 15:52:25 -07:00
/** Can be found on `href`s but can also be constructed for client-side navigation (e.g. search and graph) */
2023-07-13 00:19:35 -07:00
export type RelativeURL = SlugLike < "relative" >
export function isRelativeURL ( s : string ) : s is RelativeURL {
const validStart = /^\.{1,2}/ . test ( s )
2024-02-11 10:57:24 -08:00
const validEnding = ! endsWith ( s , "index" )
2023-08-17 00:55:52 -07:00
return validStart && validEnding && ! [ ".md" , ".html" ] . includes ( _getFileExtension ( s ) ? ? "" )
2023-07-13 00:19:35 -07:00
}
2023-08-19 15:52:25 -07:00
export function getFullSlug ( window : Window ) : FullSlug {
const res = window . document . body . dataset . slug ! as FullSlug
2023-07-15 23:02:12 -07:00
return res
2023-07-13 00:19:35 -07:00
}
2023-12-17 12:27:19 -08:00
function sluggify ( s : string ) : string {
return s
. split ( "/" )
2024-01-13 09:22:27 -08:00
. map ( ( segment ) = >
2024-02-04 21:22:57 -08:00
segment
. replace ( /\s/g , "-" )
. replace ( /&/g , "-and-" )
. replace ( /%/g , "-percent" )
. replace ( /\?/g , "" )
. replace ( /#/g , "" ) ,
2024-01-28 22:12:01 -08:00
)
2023-12-17 12:27:19 -08:00
. join ( "/" ) // always use / as sep
. replace ( /\/$/ , "" )
}
2023-08-19 15:52:25 -07:00
export function slugifyFilePath ( fp : FilePath , excludeExt? : boolean ) : FullSlug {
2024-02-11 10:57:24 -08:00
fp = stripSlashes ( fp ) as FilePath
2023-08-17 00:55:28 -07:00
let ext = _getFileExtension ( fp )
const withoutFileExt = fp . replace ( new RegExp ( ext + "$" ) , "" )
if ( excludeExt || [ ".md" , ".html" , undefined ] . includes ( ext ) ) {
ext = ""
}
2023-12-17 12:27:19 -08:00
let slug = sluggify ( withoutFileExt )
2023-07-13 00:19:35 -07:00
2023-07-15 23:33:06 -07:00
// treat _index as index
2024-02-11 10:57:24 -08:00
if ( endsWith ( slug , "_index" ) ) {
2023-07-15 23:33:06 -07:00
slug = slug . replace ( /_index$/ , "index" )
}
2023-08-19 15:52:25 -07:00
return ( slug + ext ) as FullSlug
}
export function simplifySlug ( fp : FullSlug ) : SimpleSlug {
2024-02-11 10:57:24 -08:00
const res = stripSlashes ( trimSuffix ( fp , "index" ) , true )
2023-12-02 16:50:55 -08:00
return ( res . length === 0 ? "/" : res ) as SimpleSlug
2023-07-13 00:19:35 -07:00
}
export function transformInternalLink ( link : string ) : RelativeURL {
2023-07-15 23:02:12 -07:00
let [ fplike , anchor ] = splitAnchor ( decodeURI ( link ) )
2023-08-16 22:04:15 -07:00
2024-02-11 10:57:24 -08:00
const folderPath = isFolderPath ( fplike )
2023-07-22 17:27:41 -07:00
let segments = fplike . split ( "/" ) . filter ( ( x ) = > x . length > 0 )
2024-02-11 10:57:24 -08:00
let prefix = segments . filter ( isRelativeSegment ) . join ( "/" )
let fp = segments . filter ( ( seg ) = > ! isRelativeSegment ( seg ) && seg !== "" ) . join ( "/" )
2023-07-15 23:02:12 -07:00
2023-08-17 00:55:28 -07:00
// manually add ext here as we want to not strip 'index' if it has an extension
2023-08-19 15:52:25 -07:00
const simpleSlug = simplifySlug ( slugifyFilePath ( fp as FilePath ) )
2024-02-11 10:57:24 -08:00
const joined = joinSegments ( stripSlashes ( prefix ) , stripSlashes ( simpleSlug ) )
2023-08-16 22:04:15 -07:00
const trail = folderPath ? "/" : ""
2023-08-17 00:55:28 -07:00
const res = ( _addRelativeToStart ( joined ) + trail + anchor ) as RelativeURL
2023-07-15 23:02:12 -07:00
return res
2023-07-13 00:19:35 -07:00
}
2023-11-18 18:46:58 -08:00
// from micromorph/src/utils.ts
// https://github.com/natemoo-re/micromorph/blob/main/src/utils.ts#L5
2023-12-02 16:50:55 -08:00
const _rebaseHtmlElement = ( el : Element , attr : string , newBase : string | URL ) = > {
const rebased = new URL ( el . getAttribute ( attr ) ! , newBase )
el . setAttribute ( attr , rebased . pathname + rebased . hash )
}
2023-11-18 18:46:58 -08:00
export function normalizeRelativeURLs ( el : Element | Document , destination : string | URL ) {
el . querySelectorAll ( '[href^="./"], [href^="../"]' ) . forEach ( ( item ) = >
2023-12-02 16:50:55 -08:00
_rebaseHtmlElement ( item , "href" , destination ) ,
2023-11-18 18:46:58 -08:00
)
el . querySelectorAll ( '[src^="./"], [src^="../"]' ) . forEach ( ( item ) = >
2023-12-02 16:50:55 -08:00
_rebaseHtmlElement ( item , "src" , destination ) ,
2023-11-18 18:46:58 -08:00
)
}
2023-12-02 16:50:55 -08:00
const _rebaseHastElement = (
el : HastElement ,
attr : string ,
curBase : FullSlug ,
newBase : FullSlug ,
) = > {
if ( el . properties ? . [ attr ] ) {
if ( ! isRelativeURL ( String ( el . properties [ attr ] ) ) ) {
return
}
const rel = joinSegments ( resolveRelative ( curBase , newBase ) , ".." , el . properties [ attr ] as string )
el . properties [ attr ] = rel
}
}
2023-12-20 09:52:17 -08:00
export function normalizeHastElement ( rawEl : HastElement , curBase : FullSlug , newBase : FullSlug ) {
const el = clone ( rawEl ) // clone so we dont modify the original page
2023-12-02 16:50:55 -08:00
_rebaseHastElement ( el , "src" , curBase , newBase )
_rebaseHastElement ( el , "href" , curBase , newBase )
if ( el . children ) {
el . children = el . children . map ( ( child ) = >
normalizeHastElement ( child as HastElement , curBase , newBase ) ,
)
}
return el
}
2023-08-19 15:52:25 -07:00
// resolve /a/b/c to ../..
export function pathToRoot ( slug : FullSlug ) : RelativeURL {
2023-07-13 00:19:35 -07:00
let rootPath = slug
2023-07-22 17:27:41 -07:00
. split ( "/" )
. filter ( ( x ) = > x !== "" )
2023-08-19 15:52:25 -07:00
. slice ( 0 , - 1 )
2023-07-22 17:27:41 -07:00
. map ( ( _ ) = > ".." )
. join ( "/" )
2023-07-13 00:19:35 -07:00
2023-08-19 15:52:25 -07:00
if ( rootPath . length === 0 ) {
rootPath = "."
}
return rootPath as RelativeURL
2023-07-15 23:02:12 -07:00
}
2023-08-19 15:52:25 -07:00
export function resolveRelative ( current : FullSlug , target : FullSlug | SimpleSlug ) : RelativeURL {
const res = joinSegments ( pathToRoot ( current ) , simplifySlug ( target as FullSlug ) ) as RelativeURL
2023-07-15 23:02:12 -07:00
return res
}
export function splitAnchor ( link : string ) : [ string , string ] {
let [ fp , anchor ] = link . split ( "#" , 2 )
2024-03-25 00:23:25 +01:00
if ( fp . endsWith ( ".pdf" ) ) {
return [ fp , anchor === undefined ? "" : ` # ${ anchor } ` ]
}
2023-07-22 17:27:41 -07:00
anchor = anchor === undefined ? "" : "#" + slugAnchor ( anchor )
2023-07-15 23:02:12 -07:00
return [ fp , anchor ]
}
2023-07-25 21:10:37 -07:00
export function slugTag ( tag : string ) {
return tag
. split ( "/" )
2023-12-17 12:27:19 -08:00
. map ( ( tagSegment ) = > sluggify ( tagSegment ) )
2023-07-25 21:10:37 -07:00
. join ( "/" )
}
2023-07-15 23:02:12 -07:00
export function joinSegments ( . . . args : string [ ] ) : string {
2023-09-12 21:29:57 -07:00
return args
. filter ( ( segment ) = > segment !== "" )
. join ( "/" )
. replace ( /\/\/+/g , "/" )
2023-07-13 00:19:35 -07:00
}
2023-07-25 21:10:37 -07:00
export function getAllSegmentPrefixes ( tags : string ) : string [ ] {
const segments = tags . split ( "/" )
const results : string [ ] = [ ]
for ( let i = 0 ; i < segments . length ; i ++ ) {
results . push ( segments . slice ( 0 , i + 1 ) . join ( "/" ) )
}
return results
}
2023-08-12 21:16:34 -07:00
export interface TransformOptions {
strategy : "absolute" | "relative" | "shortest"
2023-08-19 15:52:25 -07:00
allSlugs : FullSlug [ ]
2023-08-12 21:16:34 -07:00
}
2023-08-19 15:52:25 -07:00
export function transformLink ( src : FullSlug , target : string , opts : TransformOptions ) : RelativeURL {
let targetSlug = transformInternalLink ( target )
2023-08-12 21:16:34 -07:00
if ( opts . strategy === "relative" ) {
2023-08-19 15:52:25 -07:00
return targetSlug as RelativeURL
2023-08-12 21:16:34 -07:00
} else {
2024-02-11 10:57:24 -08:00
const folderTail = isFolderPath ( targetSlug ) ? "/" : ""
const canonicalSlug = stripSlashes ( targetSlug . slice ( "." . length ) )
2023-08-17 00:55:28 -07:00
let [ targetCanonical , targetAnchor ] = splitAnchor ( canonicalSlug )
2023-08-12 21:16:34 -07:00
if ( opts . strategy === "shortest" ) {
// if the file name is unique, then it's just the filename
const matchingFileNames = opts . allSlugs . filter ( ( slug ) = > {
const parts = slug . split ( "/" )
const fileName = parts . at ( - 1 )
return targetCanonical === fileName
} )
// only match, just use it
if ( matchingFileNames . length === 1 ) {
2023-08-19 15:52:25 -07:00
const targetSlug = matchingFileNames [ 0 ]
2023-08-12 21:16:34 -07:00
return ( resolveRelative ( src , targetSlug ) + targetAnchor ) as RelativeURL
}
}
// if it's not unique, then it's the absolute path from the vault root
2023-08-17 00:55:52 -07:00
return ( joinSegments ( pathToRoot ( src ) , canonicalSlug ) + folderTail ) as RelativeURL
2023-08-12 21:16:34 -07:00
}
}
2023-07-13 00:19:35 -07:00
2024-02-11 10:57:24 -08:00
// path helpers
function isFolderPath ( fplike : string ) : boolean {
2023-08-19 15:52:25 -07:00
return (
fplike . endsWith ( "/" ) ||
2024-02-11 10:57:24 -08:00
endsWith ( fplike , "index" ) ||
endsWith ( fplike , "index.md" ) ||
endsWith ( fplike , "index.html" )
2023-08-19 15:52:25 -07:00
)
2023-07-02 13:08:29 -07:00
}
2024-02-11 10:57:24 -08:00
export function endsWith ( s : string , suffix : string ) : boolean {
2023-07-22 17:27:41 -07:00
return s === suffix || s . endsWith ( "/" + suffix )
2023-07-15 23:33:06 -07:00
}
2024-02-11 10:57:24 -08:00
function trimSuffix ( s : string , suffix : string ) : string {
if ( endsWith ( s , suffix ) ) {
2023-07-22 17:27:41 -07:00
s = s . slice ( 0 , - suffix . length )
2023-07-15 23:33:06 -07:00
}
return s
}
2024-02-11 10:57:24 -08:00
function containsForbiddenCharacters ( s : string ) : boolean {
return s . includes ( " " ) || s . includes ( "#" ) || s . includes ( "?" ) || s . includes ( "&" )
2023-07-13 00:19:35 -07:00
}
2023-06-16 19:41:59 -07:00
2023-07-13 00:19:35 -07:00
function _hasFileExtension ( s : string ) : boolean {
2023-07-15 23:02:12 -07:00
return _getFileExtension ( s ) !== undefined
}
function _getFileExtension ( s : string ) : string | undefined {
2023-08-08 21:31:36 -07:00
return s . match ( /\.[A-Za-z0-9]+$/ ) ? . [ 0 ]
2023-06-16 19:41:59 -07:00
}
2024-02-11 10:57:24 -08:00
function isRelativeSegment ( s : string ) : boolean {
2023-07-13 00:19:35 -07:00
return /^\.{0,2}$/ . test ( s )
2023-05-30 08:02:20 -07:00
}
2024-02-11 10:57:24 -08:00
export function stripSlashes ( s : string , onlyStripPrefix? : boolean ) : string {
2023-07-13 00:19:35 -07:00
if ( s . startsWith ( "/" ) ) {
s = s . substring ( 1 )
}
2023-05-30 08:02:20 -07:00
2023-08-19 15:52:25 -07:00
if ( ! onlyStripPrefix && s . endsWith ( "/" ) ) {
2023-07-13 00:19:35 -07:00
s = s . slice ( 0 , - 1 )
2023-06-03 15:07:19 -04:00
}
2023-07-13 00:19:35 -07:00
return s
2023-05-31 17:01:23 -04:00
}
2023-07-13 00:19:35 -07:00
function _addRelativeToStart ( s : string ) : string {
if ( s === "" ) {
s = "."
}
2023-05-31 17:01:23 -04:00
2023-07-13 00:19:35 -07:00
if ( ! s . startsWith ( "." ) ) {
2023-07-15 23:02:12 -07:00
s = joinSegments ( "." , s )
2023-07-13 00:19:35 -07:00
}
2023-05-31 17:01:23 -04:00
2023-07-13 00:19:35 -07:00
return s
}